From f1d67663dab2fd084c7e3996a83d17e9a6c5e037 Mon Sep 17 00:00:00 2001
From: Catalin Mititiuc
Date: Thu, 25 Jul 2024 12:25:44 -0700
Subject: WIP: add multiple counters to troopers and cells
---
src/index.js | 5 +++
src/modules/game/soldier.js | 8 +++-
src/modules/gameboard.js | 101 ++++++++++++++++++++++++--------------------
3 files changed, 66 insertions(+), 48 deletions(-)
(limited to 'src')
diff --git a/src/index.js b/src/index.js
index d2c1b01..df48274 100644
--- a/src/index.js
+++ b/src/index.js
@@ -84,6 +84,11 @@ async function buildScenario(req) {
// recordSheet.start(svg.querySelector('.start-locations'), gameboard.getUnits());
recordSheet.start(null, scenarioUnits);
+
+ const [trooper] = gameboard.getUnits();
+ Observable.notify('select', trooper);
+ gameboard.setCounter('prone');
+ gameboard.setCounter('1st-floor');
}
function updateTurnCounter() {
diff --git a/src/modules/game/soldier.js b/src/modules/game/soldier.js
index a024f9f..81dce74 100644
--- a/src/modules/game/soldier.js
+++ b/src/modules/game/soldier.js
@@ -34,7 +34,7 @@ function updatePlacement(cell, selected, clone) {
const { q, r, s, t } = clone.parentElement.dataset;
selected.dataset.previous = [q, r, s, t];
- cell.appendChild(selected);
+ placeIn(cell, selected);
Array.from(selected.children).forEach(n => {
if (n.classList.contains('removed')) {
@@ -56,6 +56,10 @@ function createTrace(previous, current, selected) {
return trace;
}
+function placeIn(location, target) {
+ location.querySelector('use[href="#hex"]').after(target);
+}
+
export function createCounter(selected, weapon = 'rifle') {
const g = document.createElementNS(svgns, 'g');
const weaponCounter = document.createElementNS(svgns, 'use');
@@ -126,7 +130,7 @@ export function place(svg, selected, cell) {
handleTrace(svg, selected, clone, getCellPosition(cell));
} else {
selected.removeAttribute('data-x');
- cell.appendChild(selected);
+ placeIn(cell, selected);
}
}
diff --git a/src/modules/gameboard.js b/src/modules/gameboard.js
index 0c33f05..530a7f5 100644
--- a/src/modules/gameboard.js
+++ b/src/modules/gameboard.js
@@ -34,8 +34,9 @@ function getActiveSightLine(svg) {
return svg.querySelector('line.sight-line.active');
}
-function isGrenade(el) {
- return el && el.getAttribute('href') === '#counter-grenade';
+function isCounter(el) {
+ const regex = new RegExp('^#counter-')
+ return el && regex.test(el.getAttribute('href'));
}
function isMechTemplate(el) {
@@ -196,10 +197,10 @@ function endMove() {
}
}
-function returnPieces(collection) {
- [...svg.querySelector('.pieces').children].forEach(piece => {
- collection.get(piece).parent.append(piece);
- collection.delete(piece);
+function returnPieces(top) {
+ [...top.container.children].forEach(piece => {
+ top.collection.get(piece).parent.append(piece);
+ top.collection.delete(piece);
});
}
@@ -209,19 +210,19 @@ export function start(el) {
const startingLocations = svg.querySelector('.start-locations');
startingLocations && getUnits(startingLocations).forEach(unit => unit.addEventListener('click', selectOffBoard));
-
- const pieces = svg.querySelector('.pieces');
- const inFront = new Map();
- let inFrontParent;
- //addEventListener('pointerout', e => { returnPieces(inFront) });
+ const top = {
+ container: svg.querySelector('.grid-top'),
+ collection: new Map()
+ };
getCells(svg).forEach(cell => {
cell.addEventListener('click', e => {
const occupant = getCellOccupant(cell);
let toPlace = placing.pop();
- if (isGrenade(toPlace) || isMechTemplate(toPlace)) {
+ if (isCounter(toPlace) || isMechTemplate(toPlace)) {
getHex(cell).after(toPlace);
+ if (isCounter(toPlace)) arrangeCounters(cell);
removeEventListener("keydown", handleMechTemplateRotation);
} else if (toPlace && !occupant) {
soldier.place(svg, toPlace, cell);
@@ -282,12 +283,12 @@ export function start(el) {
});
cell.addEventListener('pointerover', () => {
- if (!pieces.contains(cell)) {
- returnPieces(inFront);
- inFront.set(cell, { parent: cell.parentElement });
- pieces.append(cell);
+ if (!top.container.contains(cell)) {
+ returnPieces(top);
+ top.collection.set(cell, { parent: cell.parentElement });
+ top.container.append(cell);
}
-
+ console.log(top.container.children[0].children);
const selected = getSelected();
if (placing[0]?.getAttributeNS(null, 'class') == 'mech-template') {
@@ -307,22 +308,6 @@ export function start(el) {
});
cell.addEventListener('pointerout', () => {
- //if (inFront && inFrontParent) {
- // inFrontParent.append(inFront);
- // inFront = null;
- // inFrontParent = null;
- //}
-
- //for (const [el, parent] of inFront) {
- // parent.append(el);
- // inFront.delete(el);
- //}
- //[...pieces.children].forEach(piece => {
- // inFront.get(piece).parent.append(piece);
- // inFront.delete(piece);
- //});
- //returnPieces(inFront);
-
getActiveSightLine(svg) && clearSightLine();
const occupant = getCellOccupant(cell);
@@ -334,27 +319,27 @@ export function start(el) {
});
// debug //
- const attacker = { dataset: { allegiance: 'attacker', number: 1, squad: 1 }};
+ // Add a trooper counter
+ const attacker = { dataset: { allegiance: 'attacker', number: 1, squad: 1 }};
// const defender = { dataset: { allegiance: 'defender', number: 1, squad: 2 }};
-
- const cell = getCell(2, 0, -2, 0);
+ const cell = getCell(0, 0, 0, 0);
const trooper = soldier.createCounter(attacker, 'blazer');
soldier.place(svg, trooper, cell);
+
+ // Add some counters in an unoccupied cell
+ const countersCell = getCell(-1, 1, 0, 0);
+ setCounter('grenade');
+ setCounter('prone');
+ setCounter('1st-floor');
+ const e = new PointerEvent('click');
+ countersCell.dispatchEvent(e);
+ countersCell.dispatchEvent(e);
+ countersCell.dispatchEvent(e);
///////////
Observable.subscribe('select', select);
Observable.subscribe('endmove', endMove);
- Observable.notify('select', trooper);
-
- //Array(1).fill(null).forEach(() => {
- // const counter = document.createElementNS(svgns, 'use');
- // counter.setAttributeNS(null, 'href', '#counter-grenade');
- // counter.classList.add('counter-grenade');
- // trooper.appendChild(counter);
- //});
- //
- //setGrenade();
console.log('gameboard.js loaded');
}
@@ -427,6 +412,30 @@ export function setCounter(name) {
placing.push(counter);
}
+function arrangeCounters(cell) {
+ const counters = cell.querySelectorAll('[class^="counter-"]');
+ const length = 12;
+ const gravity = 1;
+ const lateralForce = gravity;
+ const rads = Math.atan(lateralForce / gravity);
+ const bestFitCount = 8;
+ const deflection = counters.length > bestFitCount ? 2 * Math.PI / counters.length : Math.atan(lateralForce / gravity);
+
+ counters.forEach((counter, index, arr) => {
+ const mult = index - arr.length / 2 + 0.5;
+ const theta = deflection * mult;
+ const x = length * Math.sin(theta);
+ const y = length * Math.cos(theta);
+ //counter.setAttributeNS(null, 'x', 0);
+ //counter.setAttributeNS(null, 'y', 10);
+ //counter.setAttributeNS(null, 'transform', `rotate(${theta * 180 / Math.PI})`);
+ //counter.setAttributeNS(null, 'style', `y: 10px; transform-origin: 5px 0; transform: translateX(-5px) rotate(${theta * 180 / Math.PI}deg)`);
+ counter.setAttributeNS(null, 'style', `--x: ${-x}px; --y: ${y}px`);
+ //counter.setAttributeNS(null, 'style', `x: 0; y: 10px;`);
+ });
+ console.log(counters);
+}
+
function handleMechTemplateRotation(event) {
const counter = placing[0];
const upper = placing[0].querySelector('use[href="#mech-template-upper"]');
--
cgit v1.2.3