diff options
Diffstat (limited to 'rules.js')
-rw-r--r-- | rules.js | 133 |
1 files changed, 121 insertions, 12 deletions
@@ -1,5 +1,11 @@ "use strict" +// TODO: game.selected - singleton instead of array + +// TODO: austria/pragmatic action step - show both sides cards and interleave movement on flanders map +// PLAN: move all austria on bohemia first, then alternate pragmatic and austria activations on flanders map +// TODO: confirm mixed stack creation on flanders map (force "undo" to previous location if denied?) + const R_LOUIS_XV = "Louis XV" const R_FREDERICK = "Frederick" const R_MARIA_THERESA = "Maria Theresa" @@ -16,6 +22,18 @@ var states = {} const data = require("./data") +function is_bohemia_space(s) { + return s >= 0 && s <= 401 +} + +function is_flanders_space(s) { + return s >= 402 && s <= 618 +} + +function is_map_space(s) { + return s >= 0 && s <= 618 +} + function find_city(city) { let n = data.cities.name.length let x = -1 @@ -218,6 +236,20 @@ function all_non_coop_powers(pow) { } } +function all_coop_generals(pow) { + switch (pow) { + case P_FRANCE: + case P_BAVARIA: + return all_france_bavaria_generals + case P_PRUSSIA: + case P_SAXONY: + return all_prussia_saxony_generals + case P_PRAGMATIC: + case P_AUSTRIA: + return all_austria_pragmatic_generals + } +} + function all_controlled_generals(pow) { switch (pow) { case P_FRANCE: @@ -662,7 +694,7 @@ function goto_start_turn() { // MARIA: politics // MARIA: hussars - goto_action_stage() + goto_place_hussars() } function goto_action_stage() { @@ -693,6 +725,83 @@ function check_victory() { return false } +/* HUSSARS */ + +function goto_place_hussars() { + set_active_to_power(P_AUSTRIA) + game.state = "place_hussars" +} + +function end_place_hussars() { + set_clear(game.moved) + goto_action_stage() +} + +states.place_hussars = { + inactive: "place Hussars", + prompt() { + prompt("Place the Hussars.") + for (let p of all_hussars) + if (!set_has(game.moved, p)) + gen_action_piece(p) + view.actions.next = 1 + }, + piece(p) { + push_undo() + set_add(game.moved, p) + game.selected = p + game.state = "place_hussars_where" + }, + next() { + end_place_hussars() + }, +} + +states.place_hussars_where = { + inactive: "place Hussars", + prompt() { + prompt("Place the Hussar in a city.") + view.selected = game.selected + + // bohemia + // within 4 of an austrian general + // not occupied by any piece + for (let p of all_power_generals[P_AUSTRIA]) { + let s = game.pos[p] + if (is_bohemia_space(s)) + for (let x of search_hussar_bfs(s)) + gen_action_space(x) + } + }, + space(to) { + game.state = "place_hussars" + game.pos[game.selected] = to + game.selected = -1 + }, +} + +function search_hussar_bfs(from) { + let seen = [ from ] + let queue = [ from << 4 ] + while (queue.length > 0) { + let item = queue.shift() + let here = item >> 4 + let dist = (item & 15) + 1 + for (let next of data.cities.adjacent[here]) { + if (set_has(seen, next)) + continue + if (!is_bohemia_space(next)) + continue + if (!has_any_piece(next)) + set_add(seen, next) + if (dist < 4) + queue.push((next << 4) | dist) + } + } + set_delete(seen, from) + return seen +} + /* TACTICAL CARDS */ function find_largest_discard(u) { @@ -1071,16 +1180,6 @@ function move_general_to(to) { game.pos[p] = to } - // uniting stacks (turn all oos if one is oos) - let oos = false - for (let p of all_controlled_generals(game.power)) - if (game.pos[p] === to && is_out_of_supply(p)) - oos = true - if (oos) - for (let p of all_controlled_generals(game.power)) - if (game.pos[p] === to) - set_out_of_supply(p) - // conquer space if (is_conquest_space(pow, from) && !set_has(game.conquest, from)) { if (is_protected_from_conquest(from)) { @@ -1113,13 +1212,23 @@ function move_general_to(to) { } // uniting stacks: flag all as moved and stop moving - for (let p of all_controlled_generals(pow)) { + for (let p of all_coop_generals(pow)) { if (game.pos[p] === to && !set_has(game.selected, p)) { set_add(game.moved, p) stop = true } } + // remove hussars + for (let p of all_hussars) { + if (game.pos[p] === to) { + if (!game.move_elim) + game.move_elim = [] + set_add(game.move_elim, p) + game.pos[p] = ELIMINATED + } + } + return stop } |