From 0090d348d3122468b15e0571f09c849b627ecbed Mon Sep 17 00:00:00 2001 From: Mischa Untaga <99098079+MischaU8@users.noreply.github.com> Date: Sat, 23 Sep 2023 08:09:47 +0200 Subject: show counters --- play.html | 99 ++++++++++++++++++------------------- play.js | 167 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++---- rules.js | 36 ++++++++------ 3 files changed, 225 insertions(+), 77 deletions(-) diff --git a/play.html b/play.html index 12f58f7..798fc85 100644 --- a/play.html +++ b/play.html @@ -37,46 +37,58 @@ main { background-color: dimgray } background-image: url(map.svg); } -#map div { +/* #map div { position: absolute; transition-property: top, left; transition-duration: 1s; transition-timing-function: ease; -} - -#boxes { position: absolute; } -#cities { position: absolute; } -#units { position: absolute; } +} */ /* SPACES */ .box{position:absolute;box-sizing:border-box;border:4px solid transparent} -.box.city{border-radius:50%} .box.loc{border-radius:50%} .box.action{border-color:lemonchiffon;} -.box.city.action{background-color:#6495ed55;border-color:deepskyblue;} .box.loc.action{background-color:#fff5;border-color:white;} .box.selected{border-color:yellow;} .box.tip { border-color: white; } +.stack { + display: grid; + grid-gap: 0.1rem; + grid-template-columns: repeat(8, 2px); + transition: grid-template-columns 0.5s; +} + +.stack:hover { + grid-template-columns: repeat(8, 36px); +} + +.stack:hover div { + z-index: 100; +} + +#tracker div { + box-sizing: border-box; +} /* COUNTERS */ -#map .counter { +/* #map .counter { position: absolute; -} +} */ .counter { background-size: 1000%; background-repeat: no-repeat; border-style: solid; - transition: top 200ms, left 200ms, transform 200ms; + /* transition: top 200ms, left 200ms, transform 200ms; */ } .counter { width: 39px; height: 39px; - border-width: 3px; + border-width: 2px; border-radius: 8px; background-image: url(counters.svg); } @@ -226,47 +238,23 @@ main { background-color: dimgray } - +
+
-
-
-
-
-
-
-
-
-
- - - -
- - -
- - -
- -
- -
+
+
+
+
+
+
+
+
+ +
@@ -278,7 +266,7 @@ main { background-color: dimgray }
Government Supply
-
+
@@ -319,10 +307,17 @@ main { background-color: dimgray }
FLN Supply
-
+ + +
+ + +
+
Eliminated
+
diff --git a/play.js b/play.js index 463b50b..d668379 100644 --- a/play.js +++ b/play.js @@ -9,6 +9,9 @@ const URBAN = 1 const REMOTE = 2 const COUNTRY = 3 +const DEPLOY = 1 +const ELIMINATED = 2 + const UG = 0 const OPS = 1 const PTL = 2 @@ -16,8 +19,81 @@ const OC = 3 // const BOXES = [UG, OPS, PTL, OC] const BOX_NAMES = ["UG", "OPS", "PTL", "OC"] +const unit_count = 120 + let ui = { board: document.getElementById("map"), + map: document.getElementById("map"), + favicon: document.getElementById("favicon"), + header: document.querySelector("header"), + status: document.getElementById("status"), + player: [ + document.getElementById("role_FLN"), + document.getElementById("role_Government"), + ], + markers: { + turn: document.getElementById("turn_now"), + fln_psl: document.getElementById("fln_psl"), + fln_ap: document.getElementById("fln_ap"), + gov_psl: document.getElementById("gov_psl"), + air_avail: document.getElementById("air_avail"), + air_max: document.getElementById("air_max"), + helo_avail: document.getElementById("helo_avail"), + helo_max: document.getElementById("helo_max"), + naval: document.getElementById("naval"), + border_zone: document.getElementById("border_zone"), + + }, + tracker: [], + drm: [], + areas: [], + locations: [], + zones: [], + units: [], + units_holder: document.getElementById("units"), + fln_supply: document.getElementById("fln_supply"), + gov_supply: document.getElementById("gov_supply"), + eliminated: document.getElementById("eliminated"), +} + +// === UNIT STATE === + +// location (8 bits), dispersed (1 bit), airmobile (1 bit), neutralized (1 bit) + +const UNIT_NEUTRALIZED_SHIFT = 0 +const UNIT_NEUTRALIZED_MASK = 1 << UNIT_NEUTRALIZED_SHIFT + +const UNIT_AIRMOBILE_SHIFT = 1 +const UNIT_AIRMOBILE_MASK = 1 << UNIT_AIRMOBILE_SHIFT + +const UNIT_DISPERSED_SHIFT = 2 +const UNIT_DISPERSED_MASK = 1 << UNIT_DISPERSED_SHIFT + +const UNIT_LOC_SHIFT = 3 +const UNIT_LOC_MASK = 255 << UNIT_LOC_SHIFT + +function is_unit_neutralized(u) { + return (game.units[u] & UNIT_NEUTRALIZED_MASK) === UNIT_NEUTRALIZED_MASK +} + +function unit_loc(u) { + return (view.units[u] & UNIT_LOC_MASK) >> UNIT_LOC_SHIFT +} + +function is_unit_airmobile(u) { + return (view.units[u] & UNIT_AIRMOBILE_MASK) === UNIT_AIRMOBILE_MASK +} + +function is_unit_not_airmobile(u) { + return (view.units[u] & UNIT_AIRMOBILE_MASK) !== UNIT_AIRMOBILE_MASK +} + +function is_unit_dispersed(u) { + return (view.units[u] & UNIT_DISPERSED_MASK) === UNIT_DISPERSED_MASK +} + +function is_unit_not_dispersed(u) { + return (view.units[u] & UNIT_DISPERSED_MASK) !== UNIT_DISPERSED_MASK } let action_register = [] @@ -58,6 +134,36 @@ function create_item(p) { let on_init_once = false +function build_units() { + function build_unit(u) { + let elt = ui.units[u] = document.createElement("div") + let klass = data.units[u].class + elt.className = `counter unit u${u} ${klass}` + elt.addEventListener("mousedown", on_click_unit) + // elt.addEventListener("mouseenter", on_focus_unit) + // elt.addEventListener("mouseleave", on_blur) + elt.unit = u + } + for (let u = 0; u < unit_count; ++u) { + build_unit(u) + } +} + +function on_click_loc(evt) { + if (evt.button === 0) { + console.log('loc', evt.target.loc) + if (send_action('loc', evt.target.loc)) + evt.stopPropagation() + } +} + +function on_click_unit(evt) { + if (evt.button === 0) { + console.log('unit', evt.target.unit) + send_action('unit', evt.target.unit) + } +} + function on_init() { if (on_init_once) return @@ -67,14 +173,14 @@ function on_init() { let x = 5 let y = 5 for (let i = 0; i < 100; ++i) { - let e = document.createElement("div") - e.id = `tracker-${i}` - e.className = "box" + let e = ui.tracker[i] = document.createElement("div") + e.my_id = i + e.className = "box stack" e.style.left = x / SCALE + "px" e.style.top = y / SCALE + "px" e.style.width = 85 / SCALE + "px" e.style.height = 85 / SCALE + "px" - document.getElementById("boxes").appendChild(e) + document.getElementById("tracker").appendChild(e) if (i < 29) { x += 90 @@ -87,10 +193,10 @@ function on_init() { } } - // DRM + // Border Zone DRM for (let i = 0; i < 4; ++i) { - let e = document.createElement("div") - e.id = `drm-${i}` + let e = ui.drm[i] = document.createElement("div") + e.my_id = i e.className = "box" e.style.left = (288.2 + (i * 99)) / SCALE + "px" e.style.top = 396 / SCALE + "px" @@ -99,7 +205,6 @@ function on_init() { document.getElementById("boxes").appendChild(e) } - // Areas for (let i = 0; i < data.areas.length; ++i) { let name = data.areas[i].name @@ -120,10 +225,12 @@ function on_init() { if (type !== COUNTRY) { for (let j = 0; j < 4; ++j) { - let e = document.createElement("div") + let e = ui.locations[i * 4 + j] = document.createElement("div") let box_name = BOX_NAMES[j] e.id = `ops-${name}-${box_name}` - e.className = "box" + e.className = "box stack loc" + e.addEventListener("mousedown", on_click_loc) + e.loc = i * 4 + j e.style.left = (data.areas[i].x + (j % 2) * 99) / SCALE + "px" e.style.top = (data.areas[i].y + Math.floor(j / 2) * 99) / SCALE + "px" e.style.width = 94 / SCALE + "px" @@ -132,11 +239,51 @@ function on_init() { } } } + + build_units() +} + +function update_map() { + + ui.tracker[view.turn].appendChild(ui.markers.turn) + ui.tracker[view.fln_ap].appendChild(ui.markers.fln_ap) + ui.tracker[view.fln_psl].appendChild(ui.markers.fln_psl) + ui.tracker[view.gov_psl].appendChild(ui.markers.gov_psl) + ui.tracker[view.air_avail].appendChild(ui.markers.air_avail) + ui.tracker[view.air_max].appendChild(ui.markers.air_max) + ui.tracker[view.helo_avail].appendChild(ui.markers.helo_avail) + ui.tracker[view.helo_max].appendChild(ui.markers.helo_max) + ui.tracker[view.naval].appendChild(ui.markers.naval) + ui.drm[view.border_zone].appendChild(ui.markers.border_zone) + + for (let u = 0; u < unit_count; ++u) { + let e = ui.units[u] + let loc = unit_loc(u) + + if (loc) { + e.loc = loc + if (loc === DEPLOY) { + if (!ui.fln_supply.contains(e)) + ui.fln_supply.appendChild(e) + + } else if (loc === ELIMINATED) { + if (!ui.eliminated.contains(e)) + ui.eliminated.appendChild(e) + } else { + if (!ui.locations[loc].contains(e)) + ui.locations[loc].appendChild(e) + } + } else { + e.remove() + } + } } function on_update() { on_init() + update_map() + for (let e of action_register) e.classList.toggle("action", is_action(e.my_action, e.my_id)) diff --git a/rules.js b/rules.js index 4a4c44b..5bdd717 100644 --- a/rules.js +++ b/rules.js @@ -233,12 +233,16 @@ exports.view = function(state, player) { fln_ap: game.fln_ap, fln_psl: game.fln_psl, gov_psl: game.gov_psl, - gov_air: game.gov_air, - gov_helo: game.gov_helo, - gov_naval: game.gov_naval, + air_avail: game.air_avail, + air_max: game.air_max, + helo_avail: game.helo_avail, + helo_max: game.helo_max, + naval: game.naval, is_morocco_tunisia_independent: game.is_morocco_tunisia_independent, border_zone: game.border_zone, + + units: game.units, } if (game.state === "game_over") { @@ -305,9 +309,11 @@ exports.setup = function (seed, scenario, options) { fln_ap: 0, fln_psl: 0, gov_psl: 0, - gov_air: 0, - gov_helo: 0, - gov_naval: 0, + air_avail: 0, + air_max: 0, + helo_avail: 0, + helo_max: 0, + naval: 0, units: new Array(unit_count).fill(0), moved: [], @@ -336,26 +342,26 @@ exports.setup = function (seed, scenario, options) { const SCENARIOS = { "1954": { gov_psl: 65, - gov_air: 0, - gov_helo: 0, - gov_naval: 0, + air_max: 0, + helo_max: 0, + naval: 0, fln_psl: 50, is_morocco_tunisia_independent: false }, "1958": { gov_psl: 50, - gov_air: 6, - gov_helo: 4, - gov_naval: 2, + air_avail: 6, + helo_avail: 4, + naval: 2, fln_psl: 60, is_morocco_tunisia_independent: true, border_zone: -2 }, "1960": { gov_psl: 45, - gov_air: 7, - gov_helo: 5, - gov_naval: 3, + air_avail: 7, + helo_avail: 5, + naval: 3, fln_psl: 45, is_morocco_tunisia_independent: true, border_zone: -3 -- cgit v1.2.3