From: Benjamin Auder Date: Sun, 24 Apr 2022 20:16:29 +0000 (+0200) Subject: Start Absoption X-Git-Url: https://git.auder.net/variants/Cwda/current/img/cross.svg?a=commitdiff_plain;h=b99ce1fb4539b6ac0afd686acc39e2776e7961b4;p=xogo.git Start Absoption --- diff --git a/base_rules.js b/base_rules.js index e987492..53393df 100644 --- a/base_rules.js +++ b/base_rules.js @@ -1102,7 +1102,8 @@ export default class ChessRules { // Is (x,y) on the chessboard? onBoard(x, y) { - return x >= 0 && x < this.size.x && y >= 0 && y < this.size.y; + return (x >= 0 && x < this.size.x && + y >= 0 && y < this.size.y); } // Used in interface: 'side' arg == player color @@ -1363,23 +1364,24 @@ export default class ChessRules { return moves; } + // NOTE: using special symbols to not interfere with variants' pieces codes static get CannibalKings() { return { - "s": "p", - "u": "r", - "o": "n", - "c": "b", - "t": "q" + "!": "p", + "#": "r", + "$": "n", + "%": "b", + "*": "q" }; } static get CannibalKingCode() { return { - "p": "s", - "r": "u", - "n": "o", - "b": "c", - "q": "t", + "p": "!", + "r": "#", + "n": "$", + "b": "%", + "q": "*", "k": "k" }; } diff --git a/variants.js b/variants.js index 879b139..a5d62ce 100644 --- a/variants.js +++ b/variants.js @@ -1,6 +1,7 @@ const variants = [ -// { name: 'Absorption', desc: 'Absorb powers' }, -// { name: 'Alapo', desc: 'Geometric Chess' }, + // TODO: https://mancala.fandom.com/wiki/William_Daniel_Troyka Cleopatra chess + { name: 'Absorption', desc: 'Absorb powers' }, +// { name: 'Alapo', desc: 'Geometric Chess' }, //TODO // { name: 'Alice', desc: 'Both sides of the mirror' }, // { name: 'Align4', desc: 'Align four pawns' }, // { name: 'Allmate', desc: 'Mate any piece' }, @@ -21,7 +22,6 @@ const variants = [ { name: "Benedict", desc: "Change colors" }, // { name: 'Berolina', desc: 'Pawns move diagonally' }, // { name: 'Bicolour', desc: 'Harassed kings' }, -// { name: 'Bishopawns', desc: 'Bishop versus pawns', disp: 'Bishop-Pawns' }, // { name: 'Brotherhood', desc: 'Friendly pieces' }, { name: 'Cannibal', desc: 'Capture powers' }, // { name: 'Capablanca', desc: 'Capablanca Chess', disp: 'Capablanca Chess' }, @@ -81,7 +81,6 @@ const variants = [ // { name: 'Kinglet', desc: 'Protect your pawns' }, // { name: 'Kingsmaker', desc: 'Promote into kings' }, // { name: 'Knightmate', desc: 'Mate the knight' }, -// { name: 'Knightpawns', desc: 'Knight versus pawns', disp: 'Knight-Pawns' }, // { name: 'Knightrelay', desc: 'Move like a knight' }, // { name: 'Konane', desc: 'Hawaiian Checkers' }, // { name: 'Koopa', desc: 'Stun & kick pieces' }, @@ -115,7 +114,6 @@ const variants = [ // { name: 'Perfect', desc: 'Powerful pieces' }, // { name: 'Pocketknight', desc: 'Knight in pocket', disp: 'Pocket Knight' }, // { name: 'Progressive', desc: 'Play more and more moves' }, -// { name: 'Queenpawns', desc: 'Queen versus pawns', disp: 'Queen-Pawns' }, // { 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' }, @@ -124,10 +122,9 @@ const variants = [ // { name: 'Refusal', desc: 'Do not play that!' }, // { name: 'Rollerball', desc: 'As in the movie' }, // { name: 'Rococo', desc: 'Capture on the edge' }, -// { name: 'Rookpawns', desc: 'Rook versus pawns', disp: 'Rook-Pawns' }, // { name: 'Royalrace', desc: 'Kings cross the 11x11 board', disp: 'Royal Race' }, // { name: 'Rugby', desc: 'Transform an essay' }, -// { name: 'Schess', desc: 'Seirawan-Harper Chess', disp: 'Seirawan-Harper Chess' }, +// { name: 'Schess', desc: 'Seirawan-Harper Chess', disp: 'S-Chess' }, // { name: 'Screen', desc: 'Free initial setup' }, // { name: 'Selfabsorb', desc: 'Fusion pieces (v2)', disp: 'Self-Absorption' }, // { name: 'Shako', desc: 'Non-conformism and utopia' }, diff --git a/variants/Absorption/class.js b/variants/Absorption/class.js new file mode 100644 index 0000000..ac3445b --- /dev/null +++ b/variants/Absorption/class.js @@ -0,0 +1,195 @@ +import ChessRules from "/base_rules.js"; + +export default class AbsorptionRules extends ChessRules { + + static get Options() { + return { + select: C.Options.select, + check: [], + styles: [ + "balance", + "capture", + "cylinder", + "dark", + "doublemove", + "progressive", + "recycle", + "rifle", //TODO? absorb powers from afar? + "teleport", + "zen" + ] + }; + } + + pieces(color) { + const fusions = {{ + // amazon + 'a': { + "class": "amazon", + steps: [ + [0, 1], [0, -1], [1, 0], [-1, 0], + [1, 1], [1, -1], [-1, 1], [-1, -1] + ] + +//TODO: steps object avec range + steps... "moving"? + + steps: [ + [1, 2], [1, -2], [-1, 2], [-1, -2], + [2, 1], [-2, 1], [2, -1], [-2, -1] + ], + steps: [[0, 1], [0, -1], [1, 0], [-1, 0]] + }, + // empress + 'e': { + "class": "empress", + steps: [ + [1, 2], [1, -2], [-1, 2], [-1, -2], + [2, 1], [-2, 1], [2, -1], [-2, -1] + ], + }, + // princess + 'b': { + "class": "bishop", + steps: [[1, 1], [1, -1], [-1, 1], [-1, -1]] + }, + // queen + 'q': { + "class": "queen", + }, + + }, + return ( + Object.assign( + super.pieces(color) + ) + ); + } + + static get MergeComposed() { + return { + "be": "a", + "bs": "s", + "er": "e", + "rs": "a", + "eq": "a", + "qs": "a", + "ee": "e", + "es": "a", + "ss": "s" + }; + } + + 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; + // 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("")]; + } + // bishop + rook, or queen + [bishop or rook] + return V.QUEEN; + } + + 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); + } + // 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) + ); + }); + moves.forEach(m => { + if ( + m.vanish.length == 2 && + m.appear.length == 1 && + piece != m.vanish[1].p + ) { + // Augment pieces abilities in case of captures + m.appear[0].p = V.Fusion(piece, 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; + } + +}; diff --git a/variants/Absorption/pieces.css b/variants/Absorption/pieces.css new file mode 100644 index 0000000..afafb05 --- /dev/null +++ b/variants/Absorption/pieces.css @@ -0,0 +1,23 @@ +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/style.css b/variants/Absorption/style.css new file mode 100644 index 0000000..2964ee1 --- /dev/null +++ b/variants/Absorption/style.css @@ -0,0 +1,2 @@ +@import "../../base_pieces.css" +@import "pieces.css" diff --git a/variants/Benedict/class.js b/variants/Benedict/class.js index e08a79c..12ef3c8 100644 --- a/variants/Benedict/class.js +++ b/variants/Benedict/class.js @@ -7,14 +7,14 @@ export default class BenedictRules extends ChessRules { return { select: C.Options.select, check: [], - styles: ( - C.Options.styles.filter(s => { - return ( - ["balance", "cylinder", "dark", "doublemove", "progressive", "zen"] - .includes(s) - ); - }) - ) + styles: [ + "balance", + "cylinder", + "dark", + "doublemove", + "progressive", + "zen" + ] }; }