summaryrefslogtreecommitdiff
path: root/rules.js
diff options
context:
space:
mode:
Diffstat (limited to 'rules.js')
-rw-r--r--rules.js470
1 files changed, 353 insertions, 117 deletions
diff --git a/rules.js b/rules.js
index 165b16e..d9e6bd7 100644
--- a/rules.js
+++ b/rules.js
@@ -119,6 +119,14 @@ function find_locale(name) {
return ix
}
+function find_vassal(name) {
+ let ix = data.vassals.findIndex((x) => x.name === name)
+ if (ix < 0)
+ throw "CANNOT FIND VASSAL: " + name
+ return ix
+}
+
+
const lord_name = data.lords.map((lord) => lord.name)
const lord_count = data.lords.length
@@ -134,6 +142,8 @@ const last_lancaster_locale = 73
const first_locale = 0
const last_locale = data.locales.length - 1
+const first_vassal = 0
+const last_vassal = vassal_count - 1
const first_york_card = 0
@@ -323,6 +333,25 @@ const LOC_SCARBOROUGH = find_locale("Scarborough")
const LOC_RAVENSPUR = find_locale("Ravenspur")
+const VASSAL_BEAUMONT = find_vassal('Beaumont')
+const VASSAL_BONVILLE = find_vassal('Bonville')
+const VASSAL_DEVON = find_vassal('Devon')
+const VASSAL_DUDLEY = find_vassal('Dudley')
+const VASSAL_ESSEX = find_vassal('Essex')
+const VASSAL_FAUCONBERG = find_vassal('Fauconberg')
+const VASSAL_NORFOLK = find_vassal('Norfolk')
+const VASSAL_OXFORD = find_vassal('Oxford')
+const VASSAL_SHREWSBURY = find_vassal('Shrewsbury')
+const VASSAL_STANLEY1 = find_vassal('Stanley1')
+const VASSAL_SUFFOLK = find_vassal('Suffolk')
+const VASSAL_WESTMORLD = find_vassal('Westmorld')
+const VASSAL_WORCESTER = find_vassal('Worcester')
+const VASSAL_CLIFFORD = find_vassal('Clifford')
+const VASSAL_EDWARD = find_vassal('Edward')
+const VASSAL_HASTINGS = find_vassal('Hastings')
+const VASSAL_MONTAGU = find_vassal('Montagu')
+const VASSAL_STANLEY2 = find_vassal('Stanley2')
+const VASSAL_TROLLOPE = find_vassal('Trollope')
const AOW_LANCASTER_CULVERINS_AND_FALCONETS = [ L1, L2 ] // TODO
const AOW_LANCASTER_MUSTERD_MY_SOLDIERS = L3 // TODO
@@ -473,9 +502,9 @@ const EVENT_YORK_PATRICK_DE_LA_MOTE = Y37 // TODO
// Check all push/clear_undo
-const VASSAL_UNAVAILABLE = 0
-const VASSAL_READY = 1
-const VASSAL_MUSTERED = 2
+const VASSAL_UNAVAILABLE = -1
+const VASSAL_READY = -2
+//const VASSAL_MUSTERED = -3
const NOBODY = -1
const NOWHERE = -1
@@ -718,11 +747,18 @@ function get_lord_routed_forces(lord, n) {
}
function lord_has_unrouted_units(lord) {
- return game.pieces.forces[lord] !== 0
+ return game.pieces.forces[lord] !== 0 || get_vassals_with_lord(lord).some(v => game.battle.routed_vassals[lord] === 0 || !game.battle.routed_vassals[lord].includes(v))
}
function lord_has_routed_units(lord) {
- return game.pieces.routed[lord] !== 0
+ return game.pieces.routed[lord] !== 0 || (game.battle.routed_vassals[lord] !== 0 && game.battle.routed_vassals[lord].length > 0)
+}
+
+function rout_vassal(lord, vassal) {
+ if (game.battle.routed_vassals[lord] == 0)
+ game.battle.routed_vassals[lord] = []
+
+ set_add(game.battle.routed_vassals[lord], vassal)
}
function set_lord_locale(lord, locale) {
@@ -732,6 +768,8 @@ function set_lord_locale(lord, locale) {
function get_force_name(lord, n, x) {
if (n === RETINUE) {
return `${lord_name[lord]}'s Retinue`
+ } else if (n === VASSAL) {
+ return `Vassal ${data.vassals[x].name}`
}
return FORCE_TYPE_NAME[n]
}
@@ -1061,12 +1099,92 @@ function is_lord_ready(lord) {
return loc >= CALENDAR && loc <= CALENDAR + (game.turn >> 1)
}
+function setup_vassals(excludes = []) {
+ for (let x = first_vassal; x < last_vassal; x++) {
+ if (!excludes.includes(x) && data.vassals[x].capability === undefined) {
+ set_vassal_ready(x)
+ set_vassal_on_map(data.vassals[x].seat[0])
+ }
+ }
+}
+
+function set_vassal_on_map(vassal, loc) {
+ game.pieces.vassal[vassal] = pack8_set(game.pieces.vassal[vassal], 1, loc)
+}
+
+function get_vassal_locale(vassal) {
+ return pack8_get(game.pieces.vassal[vassal], 1)
+}
+
+function get_vassals_with_lord(lord) {
+ let results = []
+ for (let x = first_vassal; x < last_vassal; x++) {
+ if (pack8_get(game.pieces.vassal[x], 0) === lord) {
+ results.push(x)
+ }
+ }
+
+ return results
+}
+
+function set_vassal_ready(vassal) {
+ game.pieces.vassal[vassal] = pack8_set(game.pieces.vassals[vassal], 0, VASSAL_READY)
+}
+
+function set_vassal_on_calendar(vassal, turn) {
+ game.pieces.vassal[vassal] = pack8_set(game.pieces.vassal[vassal], 1, turn + CALENDAR)
+}
+
+function set_vassal_with_lord(vassal, lord) {
+ game.pieces.vassal[vassal] = pack8_set(game.pieces.vassal[vassal], 0, lord)
+}
+
+function set_vassal_unavailable(vassal) {
+ game.pieces.vassal[vassal] = pack8_set(game.pieces.vassals[vassal], 0, VASSAL_UNAVAILABLE)
+}
+
+function muster_vassal(vassal, lord) {
+ set_vassal_with_lord(vassal, lord)
+ if (data.vassals[vassal].service !== 0)
+ set_vassal_on_calendar(vassal, current_turn() + (6 - data.vassals[vassal].service))
+}
+
+function disband_vassal(vassal) {
+ let new_turn = current_turn() + (6 - data.vassals[vassal].service)
+ set_vassal_unavailable(vassal)
+ set_vassal_on_calendar(vassal, new_turn)
+ log(`Disbanded V${vassal} to turn ${current_turn() + (6 - data.vassals[vassal].service)}.`)
+
+}
+
+function pay_vassal(vassal) {
+ if (current_turn() < 15)
+ set_vassal_on_calendar(vassal, current_turn() + 1)
+}
+
+function get_ready_vassals() {
+ let favor = game.active === YORK ? game.pieces.favoury : game.pieces.favourl
+ let results = []
+
+ for (let x = first_vassal; x < last_vassal; x++) {
+ if (is_vassal_ready(x) && favor.includes(x)) {
+ results.push(x)
+ }
+ }
+
+ return results
+}
+
+function is_vassal_unavailable(vassal) {
+ return pack8_get(game.pieces.vassals[vassal], 0) === VASSAL_UNAVAILABLE
+}
+
function is_vassal_ready(vassal) {
- return game.pieces.vassals[vassal] === VASSAL_READY
+ return pack8_get(game.pieces.vassals[vassal], 0) === VASSAL_READY
}
function is_vassal_mustered(vassal) {
- return game.pieces.vassals[vassal] === VASSAL_MUSTERED
+ return pack8_get(game.pieces.vassals[vassal]) >= 0
}
function is_york_lord(lord) {
@@ -1504,30 +1622,6 @@ function muster_lord(lord, locale) {
muster_lord_forces(lord)
}
-/*
-function disband_vassal(vassal) {
- let info = data.vassals[vassal]
- let lord = data.vassals[vassal].lord
-
- add_lord_forces(lord, KNIGHTS, -(info.forces.knights | 0))
- add_lord_forces(lord, SERGEANTS, -(info.forces.sergeants | 0))
- add_lord_forces(lord, LIGHT_HORSE, -(info.forces.light_horse | 0))
- add_lord_forces(lord, ASIATIC_HORSE, -(info.forces.asiatic_horse | 0))
- add_lord_forces(lord, MEN_AT_ARMS, -(info.forces.men_at_arms | 0))
- add_lord_forces(lord, MILITIA, -(info.forces.militia | 0))
- add_lord_forces(lord, SERFS, -(info.forces.serfs | 0))
-
- game.pieces.vassals[vassal] = VASSAL_READY
-
- if (!lord_has_unrouted_units(lord)) {
- disband_lord(lord)
- }
-} */
-
-function muster_vassal(lord, vassal) {
- game.pieces.vassals[vassal] = VASSAL_MUSTERED
- muster_vassal_forces(lord, vassal)
-}
function draw_card(deck) {
let i = random(deck.length)
@@ -1674,6 +1768,8 @@ function setup_Ia() {
add_favoury_marker(LOC_LUDLOW)
add_favoury_marker(LOC_BURGUNDY)
add_favoury_marker(LOC_IRELAND)
+
+ setup_vassals()
}
@@ -1720,6 +1816,9 @@ function setup_Ib() {
add_favoury_marker(LOC_SOUTHAMPTON)
add_favoury_marker(LOC_BURGUNDY)
add_favoury_marker(LOC_IRELAND)
+
+ setup_vassals([VASSAL_FAUCONBERG, VASSAL_NORFOLK])
+ muster_vassal(VASSAL_FAUCONBERG, LORD_MARCH)
}
function setup_Ic() {
@@ -1768,6 +1867,7 @@ function setup_Ic() {
add_favoury_marker(LOC_BURGUNDY)
add_favoury_marker(LOC_IRELAND)
+ setup_vassals()
}
@@ -1808,6 +1908,8 @@ function setup_II() {
add_favoury_marker(LOC_EXETER)
add_favoury_marker(LOC_BURGUNDY)
+ setup_vassals([VASSAL_DEVON, VASSAL_OXFORD])
+
}
@@ -1836,6 +1938,8 @@ function setup_III() {
add_favoury_marker(LOC_ARUNDEL)
add_favoury_marker(LOC_YORK)
add_favoury_marker(LOC_GLOUCESTER)
+
+ setup_vassals([VASSAL_OXFORD, VASSAL_NORFOLK])
}
@@ -1858,6 +1962,8 @@ function setup_ItoIII() {
set_lord_cylinder_on_calendar(LORD_WARWICK_Y, 3)
set_lord_cylinder_on_calendar(LORD_RUTLAND, 5)
+ setup_vassals()
+
}
// setup will be used in some scenarios
@@ -2449,10 +2555,11 @@ states.levy_muster_lord = {
}
// Muster Ready Vassal Forces
- /* for (let vassal of data.lords[game.who].vassals) {
- if (is_vassal_ready(vassal))
+ if (is_friendly_locale(get_lord_locale(game.who))) {
+ for (let vassal of get_ready_vassals()) {
gen_action_vassal(vassal)
- }*/
+ }
+ }
// Add Transport
if (is_seaport(get_lord_locale(game.who)) && get_lord_assets(game.who, SHIP) < 2)
@@ -2493,8 +2600,7 @@ states.levy_muster_lord = {
vassal(vassal) {
push_undo()
- muster_vassal(game.who, vassal)
- resume_levy_muster_lord()
+ goto_muster_vassal(vassal)
},
take_ship() {
@@ -3315,6 +3421,38 @@ states.parley = {
// 2) The other vassal marker is placed, face down, on the calendar, a number of boxes right to current turn + 6 - service
// (a service 3 disbanding in turn 8 will come back turn 11)
+function goto_muster_vassal(vassal) {
+ game.what = vassal
+ push_state("levy_vassal")
+ init_influence_check(game.who)
+ game.check.push({cost: 0, modifier: data.vassals[vassal].influence * (game.active === LANCASTER? -1 : 1), source: "vassal"})
+}
+
+function end_muster_vassal() {
+ pop_state()
+ end_influence_check()
+ resume_levy_muster_lord()
+}
+
+states.levy_vassal = {
+ inactive: "Levy Vassal",
+ prompt() {
+ view.prompt = `Levy Vassal V${game.what}. `
+
+ prompt_influence_check()
+ },
+ spend1:add_influence_check_modifier_1,
+ spend3:add_influence_check_modifier_2,
+ check() {
+ let results = do_influence_check()
+ log(`Attempt to levy V${game.what} ${results.success ? "Successful" : "Failed"}: (${range(results.rating)}) ${results.success ? HIT[results.roll] : MISS[results.roll]}`)
+
+ if (success) {
+ muster_vassal(game.who, game.what)
+ }
+ end_muster_vassal()
+ }
+}
// === ACTION: MARCH ===
@@ -3459,19 +3597,16 @@ function march_with_group_2() {
}
function march_with_group_3() {
- let here = get_lord_locale(game.command)
// Disbanded in battle!
- if (here === NOWHERE) {
+ if (!is_lord_on_map(game.command)) {
game.march = 0
spend_all_actions()
resume_command()
return
}
-
game.march = 0
-
resume_command()
}
@@ -3988,6 +4123,7 @@ function init_battle(here) {
-1, -1, -1,
],
valour: Array(lord_count).fill(0),
+ routed_vassals: Array(lord_count).fill([]),
engagements: [],
reserves: [],
retreated: 0,
@@ -4896,9 +5032,9 @@ function prompt_hit_forces() {
if (get_lord_forces(lord, MILITIA) > 0)
gen_action_militia(lord)
-// get_vassals_with_lord(lord)
-// .filter(v => !game.battle.routed_vassals[lord].includes(v))
-// .forEach(gen_action_vassal)
+ get_vassals_with_lord(lord)
+ .filter(v => !game.battle.routed_vassals[lord].includes(v))
+ .forEach(gen_action_vassal)
})
}
@@ -4972,8 +5108,12 @@ function will_lord_rout(lord) {
}
function rout_unit(lord, type) {
- add_lord_forces(lord, type, -1)
- add_lord_routed_forces(lord, type, 1)
+ if (type === VASSAL) {
+ rout_vassal(lord, special)
+ } else {
+ add_lord_forces(lord, type, -1)
+ add_lord_routed_forces(lord, type, 1)
+ }
}
function which_lord_capability(lord, list) {
@@ -5181,9 +5321,8 @@ function goto_battle_influence() {
if (game.battle.loser !== BOTH) {
set_active_loser()
- // Need to add Vassals to this calculation
let influence = get_defeated_lords()
- .map(l => data.lords[l].influence)
+ .map(l => data.lords[l].influence + get_vassals_with_lord(l).length)
.reduce((p, c) => p+c,0)
reduce_influence(influence)
@@ -5427,6 +5566,15 @@ states.death_or_disband = {
function goto_battle_aftermath() {
set_active(game.battle.attacker)
+ // Routed Vassals get disbanded
+ for (let lord = first_lord; lord <= last_lord; lord++) {
+ if (is_lord_on_map(lord) && game.battle.routed_vassals[lord] !== 0) {
+ for (let vassal of game.battle.routed_vassals[lord]) {
+ disband_vassal(vassal)
+ }
+ }
+ }
+
// Events
discard_events("hold")
@@ -5691,32 +5839,165 @@ states.pay = {
}
function end_pay() {
-
- // NOTE: We can combine Pay & Disband steps because disband is mandatory only.
game.who = NOBODY
set_active_enemy()
if (game.active === P2) {
goto_pay()
}
else
- goto_levy_muster()
-
-// goto_disband()
-}
+ goto_pay_lords()
-// === LEVY & CAMPAIGN: DISBAND ===
+}
-function has_friendly_lord_who_must_disband() {
+function has_friendly_lord_who_must_pay_troops() {
for (let lord = first_friendly_lord; lord <= last_friendly_lord; ++lord)
+ if (is_lord_unfed(lord))
+ return true
return false
}
-function goto_disband() {
- game.state = "disband"
- if (!has_friendly_lord_who_must_disband())
- end_disband()
+function goto_pay_lords() {
+ for (let lord = first_friendly_lord; lord < last_friendly_lord; lord++) {
+ if (is_lord_on_map(lord))
+ set_lord_unfed(lord, 1)
+ }
+
+ if (has_friendly_lord_who_must_pay_troops()) {
+ game.who = NOBODY
+ game.state = "pay_lords"
+ } else {
+ end_pay_lords()
+ }
+}
+
+function end_pay_lords() {
+ set_active_enemy()
+
+ if (game.active === P2)
+ goto_pay_lords()
+ else
+ goto_pay_vassals()
+}
+
+states.pay_lords = {
+ inactive: "Pay Lords",
+ prompt() {
+ prompt_held_event()
+ let done = true
+
+ if (game.who === NOBODY) {
+ for (let lord = first_friendly_lord; lord < last_friendly_lord; lord++) {
+ if (is_lord_on_map(lord) && is_lord_unfed(lord)) {
+ gen_action_lord(lord)
+ done = false
+ }
+ }
+
+ if (done) {
+ view.actions.done = 1
+ }
+ } else {
+ view.actions.disband = 1
+ view.actions.pay = 1
+ }
+ },
+ lord(lord) {
+ game.who = lord
+ },
+ disband() {
+ disband_lord(game.who)
+ game.who = NOBODY
+ },
+ pay() {
+ reduce_influence(is_exile_box(get_lord_locale(game.who)) ? 2 : 1)
+ set_lord_moved(game.who, 0)
+ game.who = NOBODY
+ },
+ done() {
+ end_pay_lords()
+ }
+}
+
+
+function goto_pay_vassals() {
+ let vassal_to_pay = false
+
+ for (let v = first_vassal; v < last_vassal; v++) {
+ if (is_vassal_mustered(v) && is_friendly_lord(get_lord_with_vassal(v)) && get_vassal_locale(v) === CALENDAR + current_turn()) {
+ vassal_to_pay = true
+ }
+ }
+ if (vassal_to_pay) {
+ game.state = "pay_vassals"
+ game.what = NOTHING
+ } else {
+ end_pay_vassals()
+ }
+}
+
+function end_pay_vassals() {
+ set_active_enemy()
+
+ if (game.active === P1) {
+ //goto_muster_exiles()
+ goto_ready_vassals()
+ } else {
+ goto_pay_vassals()
+ }
+}
+
+states.pay_vassals = {
+ inactive: "Pay Vassals",
+ prompt() {
+ let done = true
+ view.prompt = "You may pay or disband vassals in the next calendar box."
+ if (game.what === NOTHING) {
+
+ for (let v = first_vassal; v < last_vassal; v++) {
+ if (is_vassal_mustered(v) && is_friendly_lord(get_lord_with_vassal(v)) && get_vassal_locale(v) === CALENDAR + current_turn()) {
+ gen_action_vassal(v)
+ done = false
+ }
+ }
+
+ if (done) {
+ view.actions.done = 1
+ }
+ } else {
+ view.actions.pay = 1
+ view.actions.disband = 1
+ }
+ },
+ vassal(v) {
+ game.what = v
+ },
+ pay() {
+ pay_vassal(game.what)
+ reduce_influence(1)
+ game.what = NOBODY
+ },
+ disband() {
+ disband_vassal(game.what)
+ game.what = NOBODY
+ },
+ done() {
+ end_pay_vassals()
+ }
+}
+
+function goto_ready_vassals() {
+ for (let vassal = first_vassal; vassal <= last_vassal; vassal++) {
+ if (is_vassal_unavailable(vassal) && get_vassal_locale(vassal) === CALENDAR + current_turn()) {
+ set_vassal_ready(vassal)
+ set_vassal_on_map(vassal, data.vassals[vassal].seat[0])
+ }
+ }
+
+ goto_levy_muster()
}
+// === LEVY & CAMPAIGN: DISBAND ===
+
function disband_lord(lord, permanently = false) {
let here = get_lord_locale(lord)
let turn = current_turn()
@@ -5742,67 +6023,12 @@ function disband_lord(lord, permanently = false) {
set_lord_moved(lord, 0)
- // for (let v of data.lords[lord].vassals)
- // game.pieces.vassals[v] = VASSAL_UNAVAILABLE
-}
-
-states.disband = {
- inactive: "Disband",
- prompt() {
- view.prompt = "Disband: You must Disband Lords at their Service limit."
-
- prompt_held_event()
-
- let done = true
- for (let lord = first_friendly_lord; lord <= last_friendly_lord; ++lord) {
- }
- if (done)
- view.actions.end_disband = 1
- },
- service_bad(lord) {
- this.lord(lord)
- },
- lord(lord) {
- if (is_lord_besieged(lord) && can_ransom_lord_siege(lord)) {
- clear_undo()
- goto_ransom(lord)
- } else {
- push_undo()
- disband_lord(lord)
- }
- },
- end_disband() {
- end_disband()
- },
- card: action_held_event,
-}
-
-function end_ransom_disband() {
- // do nothing
-}
-
-function end_disband() {
- clear_undo()
-
- if (is_campaign_phase()) {
- if (check_campaign_victory())
- return
- }
-
- set_active_enemy()
- if (is_campaign_phase()) {
- if (is_active_command())
- goto_remove_markers()
- else
- goto_feed()
- } else {
- if (game.active === P1)
- goto_levy_muster()
- else
- goto_feed()
+ for (let v of get_vassals_with_lord(lord)) {
+ disband_vassal(v)
}
}
+
// === CAMPAIGN: REMOVE MARKERS ===
function goto_remove_markers() {
@@ -6427,6 +6653,16 @@ function pack4_set(word, n, x) {
return (word & ~(15 << n)) | (x << n)
}
+function pack8_get(word, n) {
+ n = n << 4
+ return (word >>> n) & 255
+}
+
+function pack8_set(word, n, x) {
+ n = n << 4
+ return (word & ~(255 << n)) | (x << n)
+}
+
// === COMMON LIBRARY ===
function clear_undo() {