From: Benjamin Auder Date: Mon, 2 May 2022 07:52:20 +0000 (+0200) Subject: Add Madrasi, fix Absorption, better pieces management X-Git-Url: https://git.auder.net/variants/Chakart/%7B%7B%20asset%28%27mixstore/css/static/common.css?a=commitdiff_plain;h=57b8015b5c22ccfd419df11b9d0174484397c417;p=xogo.git Add Madrasi, fix Absorption, better pieces management --- diff --git a/base_pieces.css b/base_pieces.css index 207228c..6e9938e 100644 --- a/base_pieces.css +++ b/base_pieces.css @@ -1,67 +1,70 @@ /** Embedded SVGs for all chess pieces */ -piece.pawn.white { - background-image: url(''); +piece.black.pawn { + background-image: url('/pieces/black_pawn.svg'); } -piece.bishop.white { - background-image: url(''); +piece.black.rook { + background-image: url('/pieces/black_rook.svg'); } -piece.knight.white { - background-image: url(''); +piece.black.knight { + background-image: url('/pieces/black_knight.svg'); } -piece.rook.white { - background-image: url(''); +piece.black.bishop { + background-image: url('/pieces/black_bishop.svg'); } -piece.queen.white { - background-image: url(''); +piece.black.queen { + background-image: url('/pieces/black_queen.svg'); } -piece.king.white { - background-image: url(''); +piece.black.king { + background-image: url('/pieces/black_king.svg'); } -piece.pawn.black { - background-image: url(''); + +piece.white.pawn { + background-image: url('/pieces/white_pawn.svg'); } -piece.bishop.black { - background-image: url(''); +piece.white.rook { + background-image: url('/pieces/white_rook.svg'); } -piece.knight.black { - background-image: url(''); +piece.white.knight { + background-image: url('/pieces/white_knight.svg'); } -piece.rook.black { - background-image: url(''); +piece.white.bishop { + background-image: url('/pieces/white_bishop.svg'); } -piece.queen.black { - background-image: url(''); +piece.white.queen { + background-image: url('/pieces/white_queen.svg'); } -piece.king.black { - background-image: url(''); +piece.white.king { + background-image: url('/pieces/white_king.svg'); } -piece.king-pawn.black { - background-image: url(''); + +piece.black.king-pawn { + background-image: url('/pieces/black_king_pawn.svg'); } -piece.king-rook.black { - background-image: url(''); +piece.black.king-rook { + background-image: url('/pieces/black_king_rook.svg'); } -piece.king-knight.black { - background-image: url(''); +piece.black.king-knight { + background-image: url('/pieces/black_king_knight.svg'); } -piece.king-bishop.black { - background-image: url(''); +piece.black.king-bishop { + background-image: url('/pieces/black_king_bishop.svg'); } -piece.king-queen.black { - background-image: url(''); +piece.black.king-queen { + background-image: url('/pieces/black_king_queen.svg'); } -piece.king-pawn.white { - background-image: url(''); + +piece.white.king-pawn { + background-image: url('/pieces/white_king_pawn.svg'); } -piece.king-rook.white { - background-image: url(''); +piece.white.king-rook { + background-image: url('/pieces/white_king_rook.svg'); } -piece.king-knight.white { - background-image: url(''); +piece.white.king-knight { + background-image: url('/pieces/white_king_knight.svg'); } -piece.king-bishop.white { - background-image: url(''); +piece.white.king-bishop { + background-image: url('/pieces/white_king_bishop.svg'); } -piece.king-queen.white { - background-image: url(''); +piece.white.king-queen { + background-image: url('/pieces/white_king_queen.svg'); } diff --git a/base_rules.js b/base_rules.js index efeb078..5d91c72 100644 --- a/base_rules.js +++ b/base_rules.js @@ -880,7 +880,7 @@ export default class ChessRules { touchLocation = e.changedTouches[0]; if (touchLocation) return {x: touchLocation.clientX, y: touchLocation.clientY}; - return [0, 0]; //shouldn't reach here =) + return {x: 0, y: 0}; //shouldn't reach here =) } const centerOnCursor = (piece, e) => { @@ -1240,7 +1240,7 @@ export default class ChessRules { getPotentialMovesFrom(sq, color) { if (typeof sq[0] == "string") return this.getDropMovesFrom(sq); - if (this.options["madrasi"] && this.isImmobilized(sq)) + if (this.isImmobilized(sq)) return []; const piece = this.getPieceType(sq[0], sq[1]); let moves = this.getPotentialMovesOf(piece, sq); @@ -1267,138 +1267,151 @@ export default class ChessRules { const color = this.getColor(moves[0].start.x, moves[0].start.y); const oppCol = C.GetOppCol(color); - if (this.options["capture"] && this.atLeastOneCapture()) { - // Filter out non-capturing moves (not using m.vanish because of - // self captures of Recycle and Teleport). - moves = moves.filter(m => { - return ( - this.board[m.end.x][m.end.y] != "" && - this.getColor(m.end.x, m.end.y) == oppCol - ); - }); - } + if (this.options["capture"] && this.atLeastOneCapture()) + moves = this.capturePostProcess(moves, oppCol); - if (this.options["atomic"]) { - moves.forEach(m => { - if ( - this.board[m.end.x][m.end.y] != "" && - this.getColor(m.end.x, m.end.y) == oppCol - ) { - // Explosion! - let steps = [ - [-1, -1], - [-1, 0], - [-1, 1], - [0, -1], - [0, 1], - [1, -1], - [1, 0], - [1, 1] - ]; - for (let step of steps) { - let x = m.end.x + step[0]; - let y = this.computeY(m.end.y + step[1]); - if ( - this.onBoard(x, y) && - this.board[x][y] != "" && - this.getPieceType(x, y) != "p" - ) { - m.vanish.push( - new PiPo({ - p: this.getPiece(x, y), - c: this.getColor(x, y), - x: x, - y: y - }) - ); - } - } - if (!this.options["rifle"]) - m.appear.pop(); //nothin appears - } - }); - } + if (this.options["atomic"]) + this.atomicPostProcess(moves, oppCol); if ( moves.length > 0 && this.getPieceType(moves[0].start.x, moves[0].start.y) == "p" ) { - let moreMoves = []; - const lastRank = (color == "w" ? 0 : this.size.x - 1); - const initPiece = this.getPiece(moves[0].start.x, moves[0].start.y); - moves.forEach(m => { - let finalPieces = ["p"]; - const [x1, y1] = [m.start.x, m.start.y]; - const [x2, y2] = [m.end.x, m.end.y]; - const promotionOk = ( - x2 == lastRank && - (!this.options["rifle"] || this.board[x2][y2] == "") - ); - if (!promotionOk) - return; //nothing to do - if (!this.options["pawnfall"]) { - if ( - this.options["cannibal"] && - this.board[x2][y2] != "" && - this.getColor(x2, y2) == oppCol - ) { - finalPieces = [this.getPieceType(x2, y2)]; - } - else - finalPieces = this.pawnPromotions; - } - m.appear[0].p = finalPieces[0]; - if (initPiece == "!") //cannibal king-pawn - m.appear[0].p = C.CannibalKingCode[finalPieces[0]]; - for (let i=1; i { + this.riflePromotePostProcess(moves); + } + + return moves; + } + + capturePostProcess(moves, oppCol) { + // Filter out non-capturing moves (not using m.vanish because of + // self captures of Recycle and Teleport). + return moves.filter(m => { + return ( + this.board[m.end.x][m.end.y] != "" && + this.getColor(m.end.x, m.end.y) == oppCol + ); + }); + } + + atomicPostProcess(moves, oppCol) { + moves.forEach(m => { + if ( + this.board[m.end.x][m.end.y] != "" && + this.getColor(m.end.x, m.end.y) == oppCol + ) { + // Explosion! + let steps = [ + [-1, -1], + [-1, 0], + [-1, 1], + [0, -1], + [0, 1], + [1, -1], + [1, 0], + [1, 1] + ]; + for (let step of steps) { + let x = m.end.x + step[0]; + let y = this.computeY(m.end.y + step[1]); + if ( + this.onBoard(x, y) && + this.board[x][y] != "" && + this.getPieceType(x, y) != "p" + ) { + m.vanish.push( + new PiPo({ + p: this.getPiece(x, y), + c: this.getColor(x, y), + x: x, + y: y + }) + ); + } + } + if (!this.options["rifle"]) + m.appear.pop(); //nothin appears + } + }); + } + + pawnPostProcess(moves, color, oppCol) { + let moreMoves = []; + const lastRank = (color == "w" ? 0 : this.size.x - 1); + const initPiece = this.getPiece(moves[0].start.x, moves[0].start.y); + moves.forEach(m => { + let finalPieces = ["p"]; + const [x1, y1] = [m.start.x, m.start.y]; + const [x2, y2] = [m.end.x, m.end.y]; + const promotionOk = ( + x2 == lastRank && + (!this.options["rifle"] || this.board[x2][y2] == "") + ); + if (!promotionOk) + return; //nothing to do + if (!this.options["pawnfall"]) { if ( - m.start.x == lastRank && - m.appear.length >= 1 && - m.appear[0].p == "p" && - m.appear[0].x == m.start.x && - m.appear[0].y == m.start.y + this.options["cannibal"] && + this.board[x2][y2] != "" && + this.getColor(x2, y2) == oppCol ) { - const promotionPiece0 = this.pawnSpecs.promotions[0]; - m.appear[0].p = this.pawnPromotions[0]; - for (let i=1; i { + if ( + m.start.x == lastRank && + m.appear.length >= 1 && + m.appear[0].p == "p" && + m.appear[0].x == m.start.x && + m.appear[0].y == m.start.y + ) { + const promotionPiece0 = this.pawnSpecs.promotions[0]; + m.appear[0].p = this.pawnPromotions[0]; + for (let i=1; i 1e-7) - continue; - distance = Math.round(distance); //in case of (numerical...) - if (a.range < distance) + if (!C.CompatibleStep([i, j], [x, y], s, a.range)) continue; // Finally verify that nothing stand in-between let [ii, jj] = [i + s[0], this.computeY(j + s[1])]; @@ -1568,6 +1574,25 @@ export default class ChessRules { return moves; } + static CompatibleStep([x1, y1], [x2, y2], step, range) { + const rx = (x2 - x1) / step[0], + ry = (y2 - y1) / step[1]; + if ( + (!Number.isFinite(rx) && !Number.isNaN(rx)) || + (!Number.isFinite(ry) && !Number.isNaN(ry)) + ) { + return false; + } + let distance = (Number.isNaN(rx) ? ry : rx); + // TODO: 1e-7 here is totally arbitrary + if (Math.abs(distance - Math.round(distance)) > 1e-7) + return false; + distance = Math.round(distance); //in case of (numerical...) + if (range < distance) + return false; + return true; + } + // Build a regular move from its initial and destination squares. // tr: transformation getBasicMove([sx, sy], [ex, ey], tr) { @@ -2204,8 +2229,8 @@ export default class ChessRules { const r = container.querySelector(".chessboard").getBoundingClientRect(); const animateRec = i => { this.animate(moves[i], () => { - this.playVisual(moves[i], r); this.play(moves[i]); + this.playVisual(moves[i], r); if (i < moves.length - 1) setTimeout(() => animateRec(i+1), 300); else diff --git a/pieces/black_amazon.svg b/pieces/black_amazon.svg new file mode 100644 index 0000000..0215b87 --- /dev/null +++ b/pieces/black_amazon.svg @@ -0,0 +1,232 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + diff --git a/pieces/black_bishop.svg b/pieces/black_bishop.svg new file mode 100644 index 0000000..f0a098c --- /dev/null +++ b/pieces/black_bishop.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/pieces/black_empress.svg b/pieces/black_empress.svg new file mode 100644 index 0000000..3b29027 --- /dev/null +++ b/pieces/black_empress.svg @@ -0,0 +1,208 @@ + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/pieces/black_king.svg b/pieces/black_king.svg new file mode 100644 index 0000000..f3936d1 --- /dev/null +++ b/pieces/black_king.svg @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/pieces/black_king_bishop.svg b/pieces/black_king_bishop.svg new file mode 100644 index 0000000..508381b --- /dev/null +++ b/pieces/black_king_bishop.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/pieces/black_king_knight.svg b/pieces/black_king_knight.svg new file mode 100644 index 0000000..537dc09 --- /dev/null +++ b/pieces/black_king_knight.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/black_king_pawn.svg b/pieces/black_king_pawn.svg new file mode 100644 index 0000000..718a4a3 --- /dev/null +++ b/pieces/black_king_pawn.svg @@ -0,0 +1,3 @@ + + + diff --git a/pieces/black_king_queen.svg b/pieces/black_king_queen.svg new file mode 100644 index 0000000..52fcc2d --- /dev/null +++ b/pieces/black_king_queen.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/pieces/black_king_rook.svg b/pieces/black_king_rook.svg new file mode 100644 index 0000000..e4051e4 --- /dev/null +++ b/pieces/black_king_rook.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/pieces/black_knight.svg b/pieces/black_knight.svg new file mode 100644 index 0000000..261eec4 --- /dev/null +++ b/pieces/black_knight.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/black_pawn.svg b/pieces/black_pawn.svg new file mode 100644 index 0000000..3eaa9f5 --- /dev/null +++ b/pieces/black_pawn.svg @@ -0,0 +1,3 @@ + + + diff --git a/pieces/black_princess.svg b/pieces/black_princess.svg new file mode 100644 index 0000000..c34593d --- /dev/null +++ b/pieces/black_princess.svg @@ -0,0 +1,158 @@ + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + diff --git a/pieces/black_queen.svg b/pieces/black_queen.svg new file mode 100644 index 0000000..49bfbe3 --- /dev/null +++ b/pieces/black_queen.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/pieces/black_rook.svg b/pieces/black_rook.svg new file mode 100644 index 0000000..c95631c --- /dev/null +++ b/pieces/black_rook.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/pieces/white_amazon.svg b/pieces/white_amazon.svg new file mode 100644 index 0000000..beeaaea --- /dev/null +++ b/pieces/white_amazon.svg @@ -0,0 +1,25 @@ + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + diff --git a/pieces/white_bishop.svg b/pieces/white_bishop.svg new file mode 100644 index 0000000..2dfcf08 --- /dev/null +++ b/pieces/white_bishop.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/white_empress.svg b/pieces/white_empress.svg new file mode 100644 index 0000000..cc1099c --- /dev/null +++ b/pieces/white_empress.svg @@ -0,0 +1,178 @@ + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/pieces/white_king.svg b/pieces/white_king.svg new file mode 100644 index 0000000..a31f36b --- /dev/null +++ b/pieces/white_king.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/white_king_bishop.svg b/pieces/white_king_bishop.svg new file mode 100644 index 0000000..4af5154 --- /dev/null +++ b/pieces/white_king_bishop.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/white_king_knight.svg b/pieces/white_king_knight.svg new file mode 100644 index 0000000..20541c8 --- /dev/null +++ b/pieces/white_king_knight.svg @@ -0,0 +1,4 @@ + + + + diff --git a/pieces/white_king_pawn.svg b/pieces/white_king_pawn.svg new file mode 100644 index 0000000..bc21051 --- /dev/null +++ b/pieces/white_king_pawn.svg @@ -0,0 +1,4 @@ + + + + diff --git a/pieces/white_king_queen.svg b/pieces/white_king_queen.svg new file mode 100644 index 0000000..185be0e --- /dev/null +++ b/pieces/white_king_queen.svg @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + + + diff --git a/pieces/white_king_rook.svg b/pieces/white_king_rook.svg new file mode 100644 index 0000000..6c6528f --- /dev/null +++ b/pieces/white_king_rook.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/pieces/white_knight.svg b/pieces/white_knight.svg new file mode 100644 index 0000000..8bf5d89 --- /dev/null +++ b/pieces/white_knight.svg @@ -0,0 +1,4 @@ + + + + diff --git a/pieces/white_pawn.svg b/pieces/white_pawn.svg new file mode 100644 index 0000000..eccfc27 --- /dev/null +++ b/pieces/white_pawn.svg @@ -0,0 +1,4 @@ + + + + diff --git a/pieces/white_princess.svg b/pieces/white_princess.svg new file mode 100644 index 0000000..ee56aff --- /dev/null +++ b/pieces/white_princess.svg @@ -0,0 +1,153 @@ + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + diff --git a/pieces/white_queen.svg b/pieces/white_queen.svg new file mode 100644 index 0000000..b98b76f --- /dev/null +++ b/pieces/white_queen.svg @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + + + diff --git a/pieces/white_rook.svg b/pieces/white_rook.svg new file mode 100644 index 0000000..d2a6ea2 --- /dev/null +++ b/pieces/white_rook.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/server.js b/server.js index 5661136..0382e37 100644 --- a/server.js +++ b/server.js @@ -63,9 +63,9 @@ wss.on("connection", (socket, req) => { switch (obj.code) { // Send challenge (may trigger game creation) case "seekgame": { - let opponent = undefined, - choice = undefined; - const vname = obj.vname, + let oppIndex = undefined, //variant name + choice = undefined; //variant finally played + const vname = obj.vname, //variant requested randvar = (obj.vname == "_random"); if (vname == "_random") { // Pick any current challenge if possible @@ -73,22 +73,28 @@ wss.on("connection", (socket, req) => { if (currentChalls.length >= 1) { choice = currentChalls[Math.floor(Math.random() * currentChalls.length)]; - opponent = challenges[choice]; + oppIndex = choice; } } else if (challenges[vname]) { - opponent = challenges[vname]; + // Anyone wanting to play the same variant ? choice = vname; + oppIndex = vname; } - if (opponent) { - delete challenges[choice]; + else if (challenges["_random"]) { + // Anyone accepting any variant (including vname) ? + choice = vname; + oppIndex = "_random"; + } + if (oppIndex) { if (choice == "_random") choice = getRandomVariant(); // Launch game let players = [ {sid: sid, name: obj.name, randvar: randvar}, - opponent + Object.assign({}, challenges[oppIndex]) ]; + delete challenges[oppIndex]; if (Math.random() < 0.5) players = players.reverse(); // Empty options = default diff --git a/variants.js b/variants.js index a5d62ce..9111e07 100644 --- a/variants.js +++ b/variants.js @@ -87,7 +87,7 @@ const variants = [ // { name: 'Koth', desc: 'King of the Hill', disp:'King of the Hill' }, // { name: 'Losers', desc: 'Get strong at self-mate' }, // { name: 'Madhouse', desc: 'Rearrange enemy pieces' }, -// { name: 'Madrasi', desc: 'Paralyzed pieces' }, + { name: 'Madrasi', desc: 'Paralyzed pieces' }, // { name: 'Magnetic', desc: 'Laws of attraction' }, // { name: 'Maharajah', desc: 'Augmented Queens' }, // { name: 'Makpong', desc: 'Thai Chess (v2)' }, @@ -113,11 +113,11 @@ const variants = [ // { name: 'Pawnsking', desc: 'Reach the last rank (v2)', disp: 'Pawns & King' }, // { name: 'Perfect', desc: 'Powerful pieces' }, // { name: 'Pocketknight', desc: 'Knight in pocket', disp: 'Pocket Knight' }, -// { name: 'Progressive', desc: 'Play more and more moves' }, + { name: 'Progressive', desc: 'Play more and more moves' }, // { name: 'Racingkings', desc: 'Kings cross the 8x8 board', disp: 'Racing Kings' }, // { name: 'Rampage', desc: 'Move under cover' }, // { name: 'Relayup', desc: 'Upgrade pieces', disp: 'Relay-up' }, -// { name: 'Rifle', desc: 'Shoot pieces' }, + { name: 'Rifle', desc: 'Shoot pieces' }, { name: 'Recycle', desc: 'Reuse pieces' }, // { name: 'Refusal', desc: 'Do not play that!' }, // { name: 'Rollerball', desc: 'As in the movie' }, diff --git a/variants/Absorption/class.js b/variants/Absorption/class.js index 9869414..3979662 100644 --- a/variants/Absorption/class.js +++ b/variants/Absorption/class.js @@ -14,14 +14,14 @@ export default class AbsorptionRules extends ChessRules { "doublemove", "progressive", "recycle", - "rifle", + //"rifle", //TODO "teleport", "zen" ] }; } - pieces(color) { + pieces(color, x, y) { let fusions = { // amazon 'a': { @@ -45,145 +45,105 @@ export default class AbsorptionRules extends ChessRules { // empress 'e': { "class": "empress", - steps: [ - [1, 2], [1, -2], [-1, 2], [-1, -2], - [2, 1], [-2, 1], [2, -1], [-2, -1] - ], + moves: [ + { + steps: [ + [1, 0], [-1, 0], [0, 1], [0, -1] + ] + }, + { + steps: [ + [1, 2], [1, -2], [-1, 2], [-1, -2], + [2, 1], [-2, 1], [2, -1], [-2, -1] + ], + range: 1 + } + ] }, // princess - 'b': { - "class": "bishop", - steps: [[1, 1], [1, -1], [-1, 1], [-1, -1]] + 's': { + "class": "princess", + moves: [ + { + steps: [ + [1, 1], [1, -1], [-1, 1], [-1, -1] + ] + }, + { + steps: [ + [1, 2], [1, -2], [-1, 2], [-1, -2], + [2, 1], [-2, 1], [2, -1], [-2, -1] + ], + range: 1 + } + ] } }; - return Object.assign(fusions, super.pieces(color)); + return Object.assign(fusions, super.pieces(color, x, y)); } static get MergeComposed() { return { "be": "a", + "bq": "q", + "br": "q", "bs": "s", + "eq": "a", "er": "e", + "es": "a", "rs": "a", - "eq": "a", + "qr": "q", "qs": "a", - "ee": "e", - "es": "a", - "ss": "s" + "rs": "a" }; } + // Assumption p1 != p2 static Fusion(p1, p2) { - if (p1 == V.KING) return p1; - if (p1 == V.PAWN) return p2; - if (p2 == V.PAWN) return p1; - if ([p1, p2].includes(V.KNIGHT)) { - if ([p1, p2].includes(V.QUEEN)) return V.QN; - if ([p1, p2].includes(V.ROOK)) return V.RN; - if ([p1, p2].includes(V.BISHOP)) return V.BN; + if (p1 == "k") + return p1; + if (p1 == "p") + return p2; + if (p2 == "p") + return p1; + if ([p1, p2].includes("n")) { + if ([p1, p2].includes("q")) + return "a"; + if ([p1, p2].includes("r")) + return "e"; + if ([p1, p2].includes("b")) + return "s"; // p1 or p2 already have knight + other piece - return (p1 == V.KNIGHT ? p2 : p1); - } - if ([p1, p2].includes(V.QN)) return V.QN; - for (let p of [p1, p2]) { - if ([V.BN, V.RN].includes(p)) - return V.MergeComposed[[p1, p2].sort().join("")]; + return (p1 == "n" ? p2 : p1); } - // bishop + rook, or queen + [bishop or rook] - return V.QUEEN; + if ([p1, p2].includes("a")) + return "a"; + // No king, no pawn, no knight or amazon => 5 remaining pieces + return V.MergeComposed[[p1, p2].sort().join("")]; } - getPotentialMovesFrom(sq) { - let moves = []; - const piece = this.getPiece(sq[0], sq[1]); - switch (piece) { - case V.RN: - moves = - super.getPotentialRookMoves(sq).concat( - super.getPotentialKnightMoves(sq)); - break; - case V.BN: - moves = - super.getPotentialBishopMoves(sq).concat( - super.getPotentialKnightMoves(sq)); - break; - case V.QN: - moves = - super.getPotentialQueenMoves(sq).concat( - super.getPotentialKnightMoves(sq)); - break; - default: - moves = super.getPotentialMovesFrom(sq); - } + // TODO: interaction with rifle ? + postProcessPotentialMoves(moves) { // Filter out capturing promotions (except one), // because they are all the same. moves = moves.filter(m => { return ( m.vanish.length == 1 || - m.vanish[0].p != V.PAWN || - [V.PAWN, V.QUEEN].includes(m.appear[0].p) + m.vanish[0].p != "p" || + ["p", "q"].includes(m.appear[0].p) ); }); moves.forEach(m => { if ( m.vanish.length == 2 && m.appear.length == 1 && - piece != m.vanish[1].p + m.vanish[0].p != m.vanish[1].p ) { // Augment pieces abilities in case of captures - m.appear[0].p = V.Fusion(piece, m.vanish[1].p); + m.appear[0].p = V.Fusion(m.vanish[0].p, m.vanish[1].p); } }); - return moves; - } - - isAttacked(sq, color) { - return ( - super.isAttacked(sq, color) || - this.isAttackedByBN(sq, color) || - this.isAttackedByRN(sq, color) || - this.isAttackedByQN(sq, color) - ); - } - - isAttackedByBN(sq, color) { - return ( - this.isAttackedBySlideNJump(sq, color, V.BN, V.steps[V.BISHOP]) || - this.isAttackedBySlideNJump( - sq, color, V.BN, V.steps[V.KNIGHT], 1) - ); - } - - isAttackedByRN(sq, color) { - return ( - this.isAttackedBySlideNJump(sq, color, V.RN, V.steps[V.ROOK]) || - this.isAttackedBySlideNJump( - sq, color, V.RN, V.steps[V.KNIGHT], 1) - ); - } - - isAttackedByQN(sq, color) { - return ( - this.isAttackedBySlideNJump( - sq, color, V.QN, V.steps[V.BISHOP].concat(V.steps[V.ROOK])) || - this.isAttackedBySlideNJump( - sq, color, V.QN, V.steps[V.KNIGHT], 1) - ); - } - - static get VALUES() { - return Object.assign( - { a: 12, e: 7, s: 5 }, - ChessRules.VALUES - ); - } - - getNotation(move) { - let notation = super.getNotation(move); - if (move.vanish[0].p != V.PAWN && move.appear[0].p != move.vanish[0].p) - // Fusion (not from a pawn: handled in ChessRules) - notation += "=" + move.appear[0].p.toUpperCase(); - return notation; + super.postProcessPotentialMoves(moves); } }; diff --git a/variants/Absorption/pieces.css b/variants/Absorption/pieces.css deleted file mode 100644 index afafb05..0000000 --- a/variants/Absorption/pieces.css +++ /dev/null @@ -1,23 +0,0 @@ -piece.black.amazon { - background-image: url(''); -} - -piece.black.empress { - background-image: url(''); -} - -piece.black.princess { - background-image: url(''); -} - -piece.white.amazon { - background-image: url(''); -} - -piece.white.empress { - background-image: url(''); -} - -piece.white.princess { - background-image: url(''); -} diff --git a/variants/Absorption/rules.html b/variants/Absorption/rules.html new file mode 100644 index 0000000..53d5e45 --- /dev/null +++ b/variants/Absorption/rules.html @@ -0,0 +1 @@ +

The capturer absorbs abilities of the captured piece (except for pawns and king).

diff --git a/variants/Absorption/style.css b/variants/Absorption/style.css index 2964ee1..14714ab 100644 --- a/variants/Absorption/style.css +++ b/variants/Absorption/style.css @@ -1,2 +1,25 @@ -@import "../../base_pieces.css" -@import "pieces.css" +@import url("/base_pieces.css"); + +piece.black.amazon { + background-image: url('/pieces/black_amazon.svg'); +} + +piece.black.empress { + background-image: url('/pieces/black_empress.svg'); +} + +piece.black.princess { + background-image: url('/pieces/black_princess.svg'); +} + +piece.white.amazon { + background-image: url('/pieces/white_amazon.svg'); +} + +piece.white.empress { + background-image: url('/pieces/white_empress.svg'); +} + +piece.white.princess { + background-image: url('/pieces/white_princess.svg'); +} diff --git a/variants/Atomic/style.css b/variants/Atomic/style.css index 9378e29..a3550bc 100644 --- a/variants/Atomic/style.css +++ b/variants/Atomic/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Balanced/style.css b/variants/Balanced/style.css index 9378e29..a3550bc 100644 --- a/variants/Balanced/style.css +++ b/variants/Balanced/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Benedict/style.css b/variants/Benedict/style.css index 9378e29..a3550bc 100644 --- a/variants/Benedict/style.css +++ b/variants/Benedict/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Cannibal/style.css b/variants/Cannibal/style.css index 9378e29..a3550bc 100644 --- a/variants/Cannibal/style.css +++ b/variants/Cannibal/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Capture/style.css b/variants/Capture/style.css index 9378e29..290a6f4 100644 --- a/variants/Capture/style.css +++ b/variants/Capture/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css") diff --git a/variants/Chakart/style.css b/variants/Chakart/style.css index 620a9e8..6cfd883 100644 --- a/variants/Chakart/style.css +++ b/variants/Chakart/style.css @@ -1,4 +1,4 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); /* TODO: images des bananes/bombes/oeufs/champis + Chomp etc pour animations * Toadette/Waluigi/Luigi/King boo/Koopa/Daisy/Bowser avec petit texte */ diff --git a/variants/Chess960/style.css b/variants/Chess960/style.css index 9378e29..a3550bc 100644 --- a/variants/Chess960/style.css +++ b/variants/Chess960/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Crazyhouse/style.css b/variants/Crazyhouse/style.css index 9378e29..a3550bc 100644 --- a/variants/Crazyhouse/style.css +++ b/variants/Crazyhouse/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Cylinder/style.css b/variants/Cylinder/style.css index 9378e29..a3550bc 100644 --- a/variants/Cylinder/style.css +++ b/variants/Cylinder/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Dark/style.css b/variants/Dark/style.css index 9378e29..a3550bc 100644 --- a/variants/Dark/style.css +++ b/variants/Dark/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Doublemove/style.css b/variants/Doublemove/style.css index 9378e29..a3550bc 100644 --- a/variants/Doublemove/style.css +++ b/variants/Doublemove/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css"); diff --git a/variants/Madrasi/class.js b/variants/Madrasi/class.js new file mode 100644 index 0000000..eaabf07 --- /dev/null +++ b/variants/Madrasi/class.js @@ -0,0 +1,38 @@ +import ChessRules from "/base_rules.js"; + +export default class MadrasiRules extends ChessRules { + + static get Options() { + return { + select: C.Options.select, + check: [ + { + label: "Rex Incl.", + defaut: false, + variable: "rexincl" + } + ].concat(C.Options.check), + styles: C.Options.styles.filter(s => s != "madrasi") + }; + } + + constructor(o) { + o.options["madrasi"] = true; + super(o); + } + + underCheck([x, y], color) { + if (this.options["rexincl"]) { + // If Rex Inclusive, kings do not check each other: + // we just replace it very temporarily. + const [ox, oy] = this.searchKingPos(color); + const saveOppKing = this.board[ox][oy]; + this.board[ox][oy] = C.GetOppCol(color) + "q"; //arbitrary + const res = super.underCheck([x, y], color); + this.board[ox][oy] = saveOppKing; + return res; + } + return super.underCheck([x, y], color); + } + +}; diff --git a/variants/Madrasi/rules.html b/variants/Madrasi/rules.html new file mode 100644 index 0000000..22b6681 --- /dev/null +++ b/variants/Madrasi/rules.html @@ -0,0 +1 @@ +

Pieces of same nature attacking each other are immobilized.

diff --git a/variants/Madrasi/style.css b/variants/Madrasi/style.css new file mode 100644 index 0000000..a3550bc --- /dev/null +++ b/variants/Madrasi/style.css @@ -0,0 +1 @@ +@import url("/base_pieces.css"); diff --git a/variants/Zen/style.css b/variants/Zen/style.css index 9378e29..a3550bc 100644 --- a/variants/Zen/style.css +++ b/variants/Zen/style.css @@ -1 +1 @@ -@import "../../base_pieces.css" +@import url("/base_pieces.css");