Files
2026-04-07 03:25:10 +02:00

141 lines
4.3 KiB
JavaScript

class GameState {
constructor({ gameBoard, thinkingPanel, gamesTable, sliderEl, turnLabelEl }) {
this._gameBoard = gameBoard;
this._thinkingPanel = thinkingPanel;
this._gamesTable = gamesTable;
this._sliderEl = sliderEl;
this._turnLabelEl = turnLabelEl;
this._webSocket = null;
this.replay = null;
this.turnIndex = 0;
this.activeGameId = "";
this.selectedSnakeId = null;
this._timer = null;
this._hasLoadedReplayOnce = false;
}
setWebSocket(webSocket) {
this._webSocket = webSocket;
}
get isPlaying() { return Boolean(this._timer); }
async loadReplay(gameId) {
let nextReplay = null;
try {
nextReplay = await this._webSocket.requestReplay(gameId);
} catch {
if (!this._hasLoadedReplayOnce) {
this._thinkingPanel.render(
{ my_move: "-", my_thinking: { error: `Replay websocket unavailable for ${gameId}` } },
null,
);
return;
}
const response = await fetch(`/dashboard/game/${gameId}`);
if (!response.ok) {
this._thinkingPanel.render(
{ my_move: "-", my_thinking: { error: `Replay load failed for ${gameId}` } },
null,
);
return;
}
nextReplay = await response.json();
}
this.replay = nextReplay;
this._hasLoadedReplayOnce = true;
this.activeGameId = String(gameId || "");
await this._gameBoard.preloadSvgs(this.replay);
this.turnIndex = 0;
const count = Array.isArray(this.replay.turns) ? this.replay.turns.length : 0;
this._sliderEl.max = String(Math.max(0, count - 1));
this._sliderEl.value = "0";
this._gamesTable.setActive(gameId);
this.renderTurn();
}
renderTurn() {
if (!this.replay || !Array.isArray(this.replay.turns) || this.replay.turns.length === 0) {
this._turnLabelEl.textContent = "Turn -";
this._gameBoard.clearBoard();
this._thinkingPanel.render(null, null);
return;
}
const game = this.replay.game || {};
const turns = this.replay.turns;
const turn = turns[this.turnIndex];
this._turnLabelEl.textContent = `Turn ${turn.turn} / ${turns[turns.length - 1].turn}`;
this._sliderEl.value = String(this.turnIndex);
this._gameBoard.paintBoard(turn, game.width, game.height, this.selectedSnakeId, this.replay);
this._thinkingPanel.render(turn, this.replay);
if (this.selectedSnakeId) {
this._thinkingPanel.highlightSnake(this.selectedSnakeId);
}
}
stopPlayback() {
if (this._timer) {
clearInterval(this._timer);
this._timer = null;
}
const playBtn = document.getElementById("play-btn");
playBtn.textContent = "▶";
playBtn.setAttribute("title", "Play");
playBtn.setAttribute("aria-label", "Play");
}
startPlayback() {
if (!this.replay || !Array.isArray(this.replay.turns) || this.replay.turns.length < 2) return;
if (this.turnIndex >= this.replay.turns.length - 1) {
this.turnIndex = 0;
this.renderTurn();
}
this.stopPlayback();
const interval = Number(document.getElementById("speed").value || 650);
this._timer = setInterval(() => {
if (!this.replay || this.turnIndex >= this.replay.turns.length - 1) {
this.stopPlayback();
return;
}
this.turnIndex += 1;
this.renderTurn();
}, interval);
const playBtn = document.getElementById("play-btn");
playBtn.textContent = "❚❚";
playBtn.setAttribute("title", "Pause");
playBtn.setAttribute("aria-label", "Pause");
}
stepBackward() {
this.stopPlayback();
if (!this.replay || this.turnIndex <= 0) return;
this.turnIndex -= 1;
this.renderTurn();
}
stepForward() {
this.stopPlayback();
if (!this.replay || !Array.isArray(this.replay.turns) || this.turnIndex >= this.replay.turns.length - 1) return;
this.turnIndex += 1;
this.renderTurn();
}
adjustSpeed(direction) {
const speedEl = document.getElementById("speed");
const optionCount = speedEl.options.length;
if (optionCount <= 1) return;
const currentIndex = speedEl.selectedIndex;
const nextIndex = Math.max(0, Math.min(optionCount - 1, currentIndex + direction));
if (nextIndex === currentIndex) return;
speedEl.selectedIndex = nextIndex;
speedEl.dispatchEvent(new Event("change"));
}
setSelectedSnakeId(id) {
this.selectedSnakeId = id;
}
}