From: Benjamin Auder <benjamin.auder@somewhere> Date: Thu, 23 Jan 2020 09:31:07 +0000 (+0100) Subject: Add basic analyze view from FEN X-Git-Url: https://git.auder.net/game/doc/html/current/%24%7BgetWhatsApp%28link%29%7D?a=commitdiff_plain;h=652f40de91b2694093ba9755f24b76b81caff232;p=vchess.git Add basic analyze view from FEN --- diff --git a/client/src/router.js b/client/src/router.js index a4d32369..1eb88f91 100644 --- a/client/src/router.js +++ b/client/src/router.js @@ -68,7 +68,7 @@ const router = new Router({ { path: "/analyze/:vname([a-zA-Z0-9]+)", name: "analyze", - component: loadView("Game"), + component: loadView("Analyze"), }, { path: "/about", diff --git a/client/src/views/Analyze.vue b/client/src/views/Analyze.vue index 4ec370c5..cecb7737 100644 --- a/client/src/views/Analyze.vue +++ b/client/src/views/Analyze.vue @@ -1,522 +1,54 @@ <template lang="pug"> .row .col-sm-12.col-md-10.col-md-offset-1.col-lg-8.col-lg-offset-2 - input#modalAbort.modal(type="checkbox") - div(role="dialog" aria-labelledby="abortBoxTitle") - .card.smallpad.small-modal.text-center - label.modal-close(for="modalAbort") - h3#abortBoxTitle.section {{ st.tr["Terminate game?"] }} - button(@click="abortGame") {{ st.tr["Sorry I have to go"] }} - button(@click="abortGame") {{ st.tr["Game seems over"] }} - button(@click="abortGame") {{ st.tr["Opponent is gone"] }} - BaseGame(:game="game" :vr="vr" ref="basegame" - @newmove="processMove" @gameover="gameOver") - div Names: {{ game.players[0].name }} - {{ game.players[1].name }} - div(v-if="game.score=='*'") Time: {{ virtualClocks[0] }} - {{ virtualClocks[1] }} - .button-group(v-if="game.mode!='analyze' && game.score=='*'") - button(@click="offerDraw") Draw - button(@click="() => abortGame()") Abort - button(@click="resign") Resign - textarea(v-if="game.score=='*'" v-model="corrMsg") - Chat(:players="game.players") + BaseGame(:game="game" :vr="vr" ref="basegame") </template> <script> import BaseGame from "@/components/BaseGame.vue"; -import Chat from "@/components/Chat.vue"; import { store } from "@/store"; -import { GameStorage } from "@/utils/gameStorage"; -import { ppt } from "@/utils/datetime"; -import { extractTime } from "@/utils/timeControl"; import { ArrayFun } from "@/utils/array"; export default { - name: 'my-game', + name: 'my-analyze', components: { BaseGame, - Chat, }, // gameRef: to find the game in (potentially remote) storage data: function() { return { st: store.state, gameRef: { //given in URL (rid = remote ID) - id: "", - rid: "" + vname: "", + fen: "" + }, + game: { + players:[{name:"Analyze"},{name:"Analyze"}], + mode: "analyze" }, - game: {players:[{name:""},{name:""}]}, //passed to BaseGame - corrMsg: "", //to send offline messages in corr games - virtualClocks: [0, 0], //initialized with true game.clocks vr: null, //"variant rules" object initialized from FEN - drawOffer: "", //TODO: use for button style - people: [], //players + observers + //people: [], //players + observers //TODO later: interactive analyze... }; }, watch: { "$route": function(to, from) { - this.gameRef.id = to.params["id"]; - this.gameRef.rid = to.query["rid"]; + this.gameRef.fen = to.query["fen"].replace(/_/g, " "); + this.gameRef.vname = to.params["vname"]; this.loadGame(); }, - "game.clocks": function(newState) { - if (this.game.moves.length < 2) - { - // 1st move not completed yet: freeze time - this.virtualClocks = newState.map(s => ppt(s)); - return; - } - const currentTurn = this.vr.turn; - const colorIdx = ["w","b"].indexOf(currentTurn); - let countdown = newState[colorIdx] - - (Date.now() - this.game.initime[colorIdx])/1000; - this.virtualClocks = [0,1].map(i => { - const removeTime = i == colorIdx - ? (Date.now() - this.game.initime[colorIdx])/1000 - : 0; - return ppt(newState[i] - removeTime); - }); - let clockUpdate = setInterval(() => { - if (countdown < 0 || this.vr.turn != currentTurn || this.game.score != "*") - { - clearInterval(clockUpdate); - if (countdown < 0) - { - this.$refs["basegame"].endGame( - this.vr.turn=="w" ? "0-1" : "1-0", "Time"); - } - } - else - { - // TODO: with Vue 3, just do this.virtualClocks[colorIdx] = ppt(--countdown) - this.$set(this.virtualClocks, colorIdx, ppt(Math.max(0, --countdown))); - } - }, 1000); - }, }, - // TODO: redundant code with Hall.vue (related to people array) created: function() { - // Always add myself to players' list - const my = this.st.user; - this.people.push({sid:my.sid, id:my.id, name:my.name}); - this.gameRef.id = this.$route.params["id"]; - this.gameRef.rid = this.$route.query["rid"]; //may be undefined - if (!this.gameRef.rid) - this.loadGame(); //local or corr: can load from now - // 0.1] Ask server for room composition: - const initialize = () => { - // Poll clients + load game if stored remotely - this.st.conn.send(JSON.stringify({code:"pollclients"})); - if (!!this.gameRef.rid) - this.loadGame(); - }; - if (!!this.st.conn && this.st.conn.readyState == 1) //1 == OPEN state - initialize(); - else //socket not ready yet (initial loading) - this.st.conn.onopen = initialize; - this.st.conn.onmessage = this.socketMessageListener; - const socketCloseListener = () => { - store.socketCloseListener(); //reinitialize connexion (in store.js) - this.st.conn.addEventListener('message', this.socketMessageListener); - this.st.conn.addEventListener('close', socketCloseListener); - }; - this.st.conn.onclose = socketCloseListener; + this.gameRef.fen = this.$route.query["fen"].replace(/_/g, " "); + this.gameRef.vname = this.$route.params["vname"]; + this.loadGame(); }, methods: { - getOppSid: function() { - if (!!this.game.oppsid) - return this.game.oppsid; - const opponent = this.people.find(p => p.id == this.game.oppid); - return (!!opponent ? opponent.sid : null); - }, - socketMessageListener: function(msg) { - const data = JSON.parse(msg.data); - switch (data.code) - { - // 0.2] Receive clients list (just socket IDs) - case "pollclients": - { - data.sockIds.forEach(sid => { - this.people.push({sid:sid, id:0, name:""}); - // Ask only identity - this.st.conn.send(JSON.stringify({code:"askidentity", target:sid})); - }); - break; - } - case "askidentity": - { - // Request for identification: reply if I'm not anonymous - if (this.st.user.id > 0) - { - this.st.conn.send(JSON.stringify( - // people[0] instead of st.user to avoid sending email - {code:"identity", user:this.people[0], target:data.from})); - } - break; - } - case "identity": - { - let player = this.people.find(p => p.sid == data.user.sid); - // NOTE: sometimes player.id fails because player is undefined... - // Probably because the event was meant for Hall? - if (!player) - return; - player.id = data.user.id; - player.name = data.user.name; - // Sending last state only for live games: corr games are complete - if (this.game.type == "live" && this.game.oppsid == player.sid) - { - // Send our "last state" informations to opponent - const L = this.game.moves.length; - this.st.conn.send(JSON.stringify({ - code: "lastate", - target: player.sid, - state: - { - lastMove: (L>0 ? this.game.moves[L-1] : undefined), - score: this.game.score, - movesCount: L, - drawOffer: this.drawOffer, - clocks: this.game.clocks, - } - })); - } - break; - } - case "askgame": - // Send current (live) game - const myGame = - { - // Minimal game informations: - id: this.game.id, - players: this.game.players.map(p => { return {name:p.name}; }), - vid: this.game.vid, - timeControl: this.game.timeControl, - }; - this.st.conn.send(JSON.stringify({code:"game", - game:myGame, target:data.from})); - break; - case "newmove": - // NOTE: this call to play() will trigger processMove() - this.$refs["basegame"].play(data.move, - "receive", this.game.vname!="Dark" ? "animate" : null); - break; - case "lastate": //got opponent infos about last move - { - const L = this.game.moves.length; - if (data.movesCount > L) - { - // Just got last move from him - this.$refs["basegame"].play(data.lastMove, - "receive", this.game.vname!="Dark" ? "animate" : null); - if (data.score != "*" && this.game.score == "*") - { - // Opponent resigned or aborted game, or accepted draw offer - // (this is not a stalemate or checkmate) - this.$refs["basegame"].endGame(data.score, "Opponent action"); - } - this.game.clocks = data.clocks; //TODO: check this? - this.drawOffer = data.drawOffer; //does opponent offer draw? - } - break; - } - case "resign": - this.$refs["basegame"].endGame( - this.game.mycolor=="w" ? "1-0" : "0-1", "Resign"); - break; - case "abort": - this.$refs["basegame"].endGame("?", "Abort: " + data.msg); - break; - case "draw": - this.$refs["basegame"].endGame("1/2", "Mutual agreement"); - break; - case "drawoffer": - this.drawOffer = "received"; - break; - case "askfullgame": - // TODO: use data.id to retrieve game in indexedDB (but for now only one running game so OK) - this.st.conn.send(JSON.stringify({code:"fullgame", game:this.game, target:data.from})); - break; - case "fullgame": - this.loadGame(data.game); - break; - // TODO: drawaccepted (click draw button before sending move - // ==> draw offer in move) - // ==> on "newmove", check "drawOffer" field - case "connect": - { - this.people.push({name:"", id:0, sid:data.from}); - this.st.conn.send(JSON.stringify({code:"askidentity", target:data.from})); - break; - } - case "disconnect": - ArrayFun.remove(this.people, p => p.sid == data.from); - break; - } - }, - offerDraw: function() { - // TODO: also for corr games - if (this.drawOffer == "received") - { - if (!confirm("Accept draw?")) - return; - const oppsid = this.getOppSid(); - if (!!oppsid) - this.st.conn.send(JSON.stringify({code:"draw", target:oppsid})); - this.$refs["basegame"].endGame("1/2", "Mutual agreement"); - } - else if (this.drawOffer == "sent") - this.drawOffer = ""; - else - { - if (!confirm("Offer draw?")) - return; - const oppsid = this.getOppSid(); - if (!!oppsid) - this.st.conn.send(JSON.stringify({code:"drawoffer", target:oppsid})); - } - }, - // + conn handling: "draw" message ==> agree for draw (if we have "drawOffered" at true) - receiveDrawOffer: function() { - //if (...) - // TODO: ignore if preventDrawOffer is set; otherwise show modal box with option "prevent future offers" - // if accept: send message "draw" - }, - abortGame: function(event) { - let modalBox = document.getElementById("modalAbort"); - if (!event) - { - // First call show options: - modalBox.checked = true; - } - else - { - modalBox.checked = false; //decision made: box disappear - const message = event.target.innerText; - // Next line will trigger a "gameover" event, bubbling up till here - this.$refs["basegame"].endGame("?", "Abort: " + message); - const oppsid = this.getOppSid(); - if (!!oppsid) - { - this.st.conn.send(JSON.stringify({ - code: "abort", - msg: message, - target: oppsid, - })); - } - } - }, - resign: function(e) { - if (!confirm("Resign the game?")) - return; - const oppsid = this.getOppSid(); - if (!!oppsid) - { - this.st.conn.send(JSON.stringify({ - code: "resign", - target: oppsid, - })); - } - // Next line will trigger a "gameover" event, bubbling up till here - this.$refs["basegame"].endGame( - this.game.mycolor=="w" ? "0-1" : "1-0", "Resign"); - }, - // 3 cases for loading a game: - // - from indexedDB (running or completed live game I play) - // - from server (one correspondance game I play[ed] or not) - // - from remote peer (one live game I don't play, finished or not) - loadGame: function(game) { - const afterRetrieval = async (game) => { - const vModule = await import("@/variants/" + game.vname + ".js"); - window.V = vModule.VariantRules; - this.vr = new V(game.fen); - const gtype = (game.timeControl.indexOf('d') >= 0 ? "corr" : "live"); - const tc = extractTime(game.timeControl); - if (gtype == "corr") - { - if (game.players[0].color == "b") - { - // Adopt the same convention for live and corr games: [0] = white - [ game.players[0], game.players[1] ] = - [ game.players[1], game.players[0] ]; - } - // corr game: needs to compute the clocks + initime - // NOTE: clocks in seconds, initime in milliseconds - game.clocks = [tc.mainTime, tc.mainTime]; - game.initime = [0, 0]; - const L = game.moves.length; - game.moves.sort((m1,m2) => m1.idx - m2.idx); //in case of - if (L >= 3) - { - let addTime = [0, 0]; - for (let i=2; i<L; i++) - { - addTime[i%2] += tc.increment - - (game.moves[i].played - game.moves[i-1].played) / 1000; - } - for (let i=0; i<=1; i++) - game.clocks[i] += addTime[i]; - } - if (L >= 1) - game.initime[L%2] = game.moves[L-1].played; - // Now that we used idx and played, re-format moves as for live games - game.moves = game.moves.map( (m) => { - const s = m.squares; - return { - appear: s.appear, - vanish: s.vanish, - start: s.start, - end: s.end, - message: m.message, - }; - }); - } - const myIdx = game.players.findIndex(p => { - return p.sid == this.st.user.sid || p.uid == this.st.user.id; - }); - if (gtype == "live" && game.clocks[0] < 0) //game unstarted - { - game.clocks = [tc.mainTime, tc.mainTime]; - game.initime[0] = Date.now(); - if (myIdx >= 0) - { - // I play in this live game; corr games don't have clocks+initime - GameStorage.update(game.id, - { - clocks: game.clocks, - initime: game.initime, - }); - } - } - this.game = Object.assign({}, - game, - // NOTE: assign mycolor here, since BaseGame could also be VS computer - { - type: gtype, - increment: tc.increment, - mycolor: [undefined,"w","b"][myIdx+1], - // opponent sid not strictly required (or available), but easier - // at least oppsid or oppid is available anyway: - oppsid: (myIdx < 0 ? undefined : game.players[1-myIdx].sid), - oppid: (myIdx < 0 ? undefined : game.players[1-myIdx].uid), - } - ); - }; - if (!!game) - return afterRetrieval(game); - if (!!this.gameRef.rid) - { - // Remote live game - // (TODO: send game ID as well, and receiver would pick the corresponding - // game in his current games; if allowed to play several) - this.st.conn.send(JSON.stringify( - {code:"askfullgame", target:this.gameRef.rid})); - // (send moves updates + resign/abort/draw actions) - } - else - { - // Local or corr game - GameStorage.get(this.gameRef.id, afterRetrieval); - } - }, - // Post-process a move (which was just played) - processMove: function(move) { - if (!this.game.mycolor) - return; //I'm just an observer - // Update storage (corr or live) - const colorIdx = ["w","b"].indexOf(move.color); - // https://stackoverflow.com/a/38750895 - const allowed_fields = ["appear", "vanish", "start", "end"]; - const filtered_move = Object.keys(move) - .filter(key => allowed_fields.includes(key)) - .reduce((obj, key) => { - obj[key] = move[key]; - return obj; - }, {}); - // Send move ("newmove" event) to people in the room (if our turn) - let addTime = 0; - if (move.color == this.game.mycolor) - { - if (this.game.moves.length >= 2) //after first move - { - const elapsed = Date.now() - this.game.initime[colorIdx]; - // elapsed time is measured in milliseconds - addTime = this.game.increment - elapsed/1000; - } - let sendMove = Object.assign({}, filtered_move, {addTime: addTime}); - if (this.game.type == "corr") - sendMove.message = this.corrMsg; - const oppsid = this.getOppSid(); - this.people.forEach(p => { - if (p.sid != this.st.user.sid) - { - this.st.conn.send(JSON.stringify({ - code: "newmove", - target: p.sid, - move: sendMove, - })); - } - }); - if (this.game.type == "corr" && this.corrMsg != "") - { - // Add message to last move in BaseGame: - // TODO: not very good style... - this.$refs["basegame"].setCurrentMessage(this.corrMsg); - } - } - else - addTime = move.addTime; //supposed transmitted - const nextIdx = ["w","b"].indexOf(this.vr.turn); - // Since corr games are stored at only one location, update should be - // done only by one player for each move: - if (this.game.type == "live" || move.color == this.game.mycolor) - { - if (this.game.type == "corr") - { - GameStorage.update(this.gameRef.id, - { - fen: move.fen, - move: - { - squares: filtered_move, - message: this.corrMsg, - played: Date.now(), //TODO: on server? - idx: this.game.moves.length, - }, - }); - } - else //live - { - GameStorage.update(this.gameRef.id, - { - fen: move.fen, - move: filtered_move, - clocks: this.game.clocks.map((t,i) => i==colorIdx - ? this.game.clocks[i] + addTime - : this.game.clocks[i]), - initime: this.game.initime.map((t,i) => i==nextIdx - ? Date.now() - : this.game.initime[i]), - }); - } - } - // Also update current game object: - this.game.moves.push(move); - this.game.fen = move.fen; - //TODO: just this.game.clocks[colorIdx] += addTime; - this.$set(this.game.clocks, colorIdx, this.game.clocks[colorIdx] + addTime); - this.game.initime[nextIdx] = Date.now(); - // Finally reset curMoveMessage if needed - if (this.game.type == "corr" && move.color == this.game.mycolor) - this.corrMsg = ""; - }, - gameOver: function(score) { - this.game.mode = "analyze"; - this.game.score = score; - const myIdx = this.game.players.findIndex(p => { - return p.sid == this.st.user.sid || p.uid == this.st.user.id; - }); - if (myIdx >= 0) //OK, I play in this game - GameStorage.update(this.gameRef.id, { score: score }); + loadGame: async function() { + this.game.vname = this.gameRef.vname; + this.game.fen = this.gameRef.fen; + const vModule = await import("@/variants/" + this.game.vname + ".js"); + window.V = vModule.VariantRules; + this.vr = new V(this.game.fen); }, }, };