From 2eef6db6cdce30fe785e601b88858c7fc743eee8 Mon Sep 17 00:00:00 2001
From: Benjamin Auder <benjamin.auder@somewhere>
Date: Wed, 12 Dec 2018 03:02:19 +0100
Subject: [PATCH] Fix HalfChess, more random Loser initial position, continue
 draft of Ultima

---
 public/images/pieces/Ultima/bm.svg    |  92 ++++++++++
 public/images/pieces/Ultima/wm.svg    |  97 ++++++++++
 public/javascripts/base_rules.js      |   3 +-
 public/javascripts/components/game.js |   1 -
 public/javascripts/variants/Half.js   |  72 ++++++--
 public/javascripts/variants/Loser.js  |  56 ++++++
 public/javascripts/variants/Ultima.js | 247 +++++++++++++++++++++++++-
 public/stylesheets/variant.sass       |   4 -
 views/rules/Half.pug                  |   6 +-
 views/rules/Ultima.pug                |   2 +-
 10 files changed, 551 insertions(+), 29 deletions(-)
 create mode 100644 public/images/pieces/Ultima/bm.svg
 create mode 100644 public/images/pieces/Ultima/wm.svg

diff --git a/public/images/pieces/Ultima/bm.svg b/public/images/pieces/Ultima/bm.svg
new file mode 100644
index 00000000..fdc0ee59
--- /dev/null
+++ b/public/images/pieces/Ultima/bm.svg
@@ -0,0 +1,92 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   height="100%"
+   width="100%"
+   version="1.1"
+   viewBox="0 0 2048 2048"
+   id="svg44"
+   sodipodi:docname="bu.svg"
+   inkscape:version="0.92.2 2405546, 2018-03-11">
+  <metadata
+     id="metadata50">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs48" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="960"
+     inkscape:window-height="1060"
+     id="namedview46"
+     showgrid="false"
+     inkscape:zoom="0.11523438"
+     inkscape:cx="1041.3559"
+     inkscape:cy="1024"
+     inkscape:window-x="0"
+     inkscape:window-y="20"
+     inkscape:window-maximized="0"
+     inkscape:current-layer="svg44" />
+  <path
+     style="color:#000000;display:block;fill:#000000;fill-rule:nonzero"
+     d="m 1161,1706 h 170 v 137 h 274 V 1468 L 1383,1297 V 819 L 1553,649 V 444 h 153 V 205 H 341 v 239 h 153 v 205 l 171,170 v 478 l -222,171 v 375 h 273 v -137 h 171 v 137 h 274 z M 564,460 V 358 h 920 v 102 z m 460,1092 H 512 v -46 l 73,-55 h 879 l 71,55 v 46 z m 0,-169 H 674 l 60,-47 v -57 h 580 v 57 l 60,47 z m 0,-546 H 734 v -46 l -60,-58 h 700 l -60,58 v 46 z m 0,-172 H 610 l -46,-43 v -58 h 920 v 58 l -46,43 z"
+     display="block"
+     id="path30"
+     inkscape:connector-curvature="0" />
+  <g
+     id="g42"
+     transform="matrix(1,0,0,-1,0,2048)"
+     style="fill:#ffffff;fill-rule:nonzero">
+    <path
+       style="color:#000000;display:block"
+       d="m 564,1588 v 102 h 920 v -102 z"
+       display="block"
+       id="path32"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1024,496 H 512 v 46 l 73,55 h 879 l 71,-55 v -46 z"
+       display="block"
+       id="path34"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1024,665 H 674 l 60,47 v 57 h 580 v -57 l 60,-47 z"
+       display="block"
+       id="path36"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1024,1211 H 734 v 46 l -60,58 h 700 l -60,-58 v -46 z"
+       display="block"
+       id="path38"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1024,1383 H 610 l -46,43 v 58 h 920 v -58 l -46,-43 z"
+       display="block"
+       id="path40"
+       inkscape:connector-curvature="0" />
+  </g>
+</svg>
diff --git a/public/images/pieces/Ultima/wm.svg b/public/images/pieces/Ultima/wm.svg
new file mode 100644
index 00000000..bf9f16ad
--- /dev/null
+++ b/public/images/pieces/Ultima/wm.svg
@@ -0,0 +1,97 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   height="100%"
+   width="100%"
+   version="1.1"
+   viewBox="0 0 2048 2048"
+   id="svg70"
+   sodipodi:docname="wu.svg"
+   inkscape:version="0.92.2 2405546, 2018-03-11">
+  <metadata
+     id="metadata76">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs74" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="960"
+     inkscape:window-height="1060"
+     id="namedview72"
+     showgrid="false"
+     inkscape:zoom="0.11523438"
+     inkscape:cx="1041.3559"
+     inkscape:cy="1024"
+     inkscape:window-x="0"
+     inkscape:window-y="20"
+     inkscape:window-maximized="0"
+     inkscape:current-layer="svg70" />
+  <path
+     style="color:#000000;display:block;fill:#000000;fill-rule:nonzero"
+     d="m 1161,1706 h 170 v 137 h 274 V 1468 L 1383,1297 V 819 L 1553,649 V 444 h 153 V 205 H 341 v 239 h 153 v 205 l 171,170 v 478 l -222,171 v 375 h 273 v -137 h 171 v 137 h 274 z M 1639,376 H 409 V 273 H 1639 Z M 1484,580 H 564 V 444 h 920 z m -170,717 H 734 V 819 h 580 z m 222,239 v 239 h -137 v -137 h -308 v 137 H 956 V 1638 H 649 v 137 H 512 V 1536 Z M 1459,649 1356,751 H 693 L 588,649 Z m -110,716 127,103 H 572 l 128,-103 z"
+     display="block"
+     id="path54"
+     inkscape:connector-curvature="0" />
+  <g
+     id="g64"
+     transform="matrix(1,0,0,-1,0,2048)"
+     style="fill:#ffffff;fill-rule:nonzero">
+    <path
+       style="color:#000000;display:block"
+       d="M 1639,1672 H 409 v 103 h 1230 z"
+       display="block"
+       id="path56"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1484,1468 H 564 v 136 h 920 z"
+       display="block"
+       id="path58"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1314,751 H 734 v 478 h 580 z"
+       display="block"
+       id="path60"
+       inkscape:connector-curvature="0" />
+    <path
+       style="color:#000000;display:block"
+       d="M 1536,512 V 273 H 1399 V 410 H 1091 V 273 H 956 V 410 H 649 V 273 H 512 v 239 z"
+       display="block"
+       id="path62"
+       inkscape:connector-curvature="0" />
+  </g>
+  <path
+     style="color:#000000;display:block;fill:#ffffff;fill-rule:nonzero"
+     d="M 1459,649 1356,751 H 693 L 588,649 Z"
+     display="block"
+     id="path66"
+     inkscape:connector-curvature="0" />
+  <path
+     style="color:#000000;display:block;fill:#ffffff;fill-rule:nonzero"
+     d="m 1349,1365 127,103 H 572 l 128,-103 z"
+     display="block"
+     id="path68"
+     inkscape:connector-curvature="0" />
+</svg>
diff --git a/public/javascripts/base_rules.js b/public/javascripts/base_rules.js
index 750cd2d4..8de0176c 100644
--- a/public/javascripts/base_rules.js
+++ b/public/javascripts/base_rules.js
@@ -1033,11 +1033,10 @@ class ChessRules
 			pieces[c][knight2Pos] = 'n';
 			pieces[c][rook2Pos] = 'r';
 		}
-		let fen = pieces["b"].join("") +
+		return pieces["b"].join("") +
 			"/pppppppp/8/8/8/8/PPPPPPPP/" +
 			pieces["w"].join("").toUpperCase() +
 			" 1111"; //add flags
-		return fen;
 	}
 
 	// Return current fen according to pieces+colors state
diff --git a/public/javascripts/components/game.js b/public/javascripts/components/game.js
index d76ab011..208e0482 100644
--- a/public/javascripts/components/game.js
+++ b/public/javascripts/components/game.js
@@ -25,7 +25,6 @@ Vue.component('my-game', {
 	},
 	render(h) {
 		const [sizeX,sizeY] = VariantRules.size;
-		console.log(sizeX + " " + sizeY);
 		const smallScreen = (screen.width <= 420);
 		// Precompute hints squares to facilitate rendering
 		let hintSquares = doubleArray(sizeX, sizeY, false);
diff --git a/public/javascripts/variants/Half.js b/public/javascripts/variants/Half.js
index 2f1174ed..6c6c7ebb 100644
--- a/public/javascripts/variants/Half.js
+++ b/public/javascripts/variants/Half.js
@@ -1,8 +1,35 @@
+// Standard rules on a 4x8 board with no pawns
 class HalfRules extends ChessRules
 {
-	// Standard rules on a 4x8 board with no pawns
-
-	initVariables(fen) { } //nothing to do
+	initVariables(fen)
+	{
+		this.kingPos = {'w':[-1,-1], 'b':[-1,-1]};
+		const fenParts = fen.split(" ");
+		const position = fenParts[0].split("/");
+		for (let i=0; i<position.length; i++)
+		{
+			let k = 0;
+			for (let j=0; j<position[i].length; j++)
+			{
+				switch (position[i].charAt(j))
+				{
+					case 'k':
+						this.kingPos['b'] = [i,k];
+						break;
+					case 'K':
+						this.kingPos['w'] = [i,k];
+						break;
+					default:
+						let num = parseInt(position[i].charAt(j));
+						if (!isNaN(num))
+							k += (num-1);
+				}
+				k++;
+			}
+		}
+		// No pawns so no ep., but otherwise we must redefine play()
+		this.epSquares = [];
+	}
 
 	setFlags(fen)
 	{
@@ -10,7 +37,7 @@ class HalfRules extends ChessRules
 		this.castleFlags = { "w":[false,false], "b":[false,false] };
 	}
 
-	static get size() { return [8,4]; }
+	static get size() { return [4,8]; }
 
 	getPotentialKingMoves(sq)
 	{
@@ -29,11 +56,17 @@ class HalfRules extends ChessRules
 			|| this.isAttackedByKing(sq, colors));
 	}
 
-	// Unused:
-	updateVariables(move) { }
-	unupdateVariables(move) { }
-
-	static get SEARCH_DEPTH() { return 4; }
+	updateVariables(move)
+	{
+		// Just update king position
+		const piece = this.getPiece(move.start.x,move.start.y);
+		const c = this.getColor(move.start.x,move.start.y);
+		if (piece == VariantRules.KING)
+		{
+			this.kingPos[c][0] = move.appear[0].x;
+			this.kingPos[c][1] = move.appear[0].y;
+		}
+	}
 
 	static GenRandInitFen()
 	{
@@ -71,18 +104,29 @@ class HalfRules extends ChessRules
 			let queenPos = positions[randIndex];
 			positions.splice(randIndex, 1);
 
+			// Random square for king (no castle)
+			randIndex = _.random(2);
+			let kingPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
 			// Rooks and king positions:
 			let rook1Pos = positions[0];
-			let kingPos = positions[1];
-			let rook2Pos = positions[2];
+			let rook2Pos = positions[1];
 
 			majorPieces[c][rook1Pos] = 'r';
 			majorPieces[c][rook2Pos] = 'r';
 			majorPieces[c][kingPos] = 'k';
 			majorPieces[c][queenPos] = 'q';
 		}
-		return majorPieces["b"].join("") + "/" + minorPieces["b"].join("") + "/4/4/4/4/" +
-			minorPieces["w"].join("").toUpperCase() + "/" +
-			majorPieces["w"].join("").toUpperCase() + " 0000"; //TODO: flags?!
+		let fen = "";
+		for (let i=0; i<4; i++)
+		{
+			fen += majorPieces["b"][i] + minorPieces["b"][i] + "4" +
+				minorPieces["w"][i].toUpperCase() + majorPieces["w"][i].toUpperCase();
+			if (i < 3)
+				fen += "/";
+		}
+		fen += " 0000"; //TODO: flags?!
+		return fen;
 	}
 }
diff --git a/public/javascripts/variants/Loser.js b/public/javascripts/variants/Loser.js
index da4999fe..6a322b9d 100644
--- a/public/javascripts/variants/Loser.js
+++ b/public/javascripts/variants/Loser.js
@@ -142,4 +142,60 @@ class LoserRules extends ChessRules
 	{
 		return - super.evalPosition(); //better with less material
 	}
+
+	static GenRandInitFen()
+	{
+		let pieces = { "w": new Array(8), "b": new Array(8) };
+		// Shuffle pieces on first and last rank
+		for (let c of ["w","b"])
+		{
+			let positions = _.range(8);
+
+			// Get random squares for bishops
+			let randIndex = 2 * _.random(3);
+			let bishop1Pos = positions[randIndex];
+			// The second bishop must be on a square of different color
+			let randIndex_tmp = 2 * _.random(3) + 1;
+			let bishop2Pos = positions[randIndex_tmp];
+			// Remove chosen squares
+			positions.splice(Math.max(randIndex,randIndex_tmp), 1);
+			positions.splice(Math.min(randIndex,randIndex_tmp), 1);
+
+			// Get random squares for knights
+			randIndex = _.random(5);
+			let knight1Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+			randIndex = _.random(4);
+			let knight2Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			// Get random square for queen
+			randIndex = _.random(3);
+			let queenPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			// Random square for king (no castle)
+			randIndex = _.random(2);
+			let kingPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			// Rooks positions are now fixed
+			let rook1Pos = positions[0];
+			let rook2Pos = positions[1];
+
+			// Finally put the shuffled pieces in the board array
+			pieces[c][rook1Pos] = 'r';
+			pieces[c][knight1Pos] = 'n';
+			pieces[c][bishop1Pos] = 'b';
+			pieces[c][queenPos] = 'q';
+			pieces[c][kingPos] = 'k';
+			pieces[c][bishop2Pos] = 'b';
+			pieces[c][knight2Pos] = 'n';
+			pieces[c][rook2Pos] = 'r';
+		}
+		return pieces["b"].join("") +
+			"/pppppppp/8/8/8/8/PPPPPPPP/" +
+			pieces["w"].join("").toUpperCase() +
+			" 0000"; //add flags (TODO?!)
+	}
 }
diff --git a/public/javascripts/variants/Ultima.js b/public/javascripts/variants/Ultima.js
index 812d440b..9212afba 100644
--- a/public/javascripts/variants/Ultima.js
+++ b/public/javascripts/variants/Ultima.js
@@ -1,8 +1,247 @@
 class UltimaRules extends ChessRules
 {
-	// TODO: think about move UI for "removing an immobilized  piece from the board"
-	// (extend game.js and feedback Rules.js with "there was a click, is it a move?")
+	static getPpath(b)
+	{
+		if (b[1] == "m") //'m' for Immobilizer (I is too similar to 1)
+			return "Ultima/" + b;
+		return b; //usual piece
+	}
 
-	// TODO: Keep usual pieces names here (but comment with Ultima pieces names)
-	// Just change moving + capturing modes.
+	initVariables(fen)
+	{
+		this.kingPos = {'w':[-1,-1], 'b':[-1,-1]};
+		const fenParts = fen.split(" ");
+		const position = fenParts[0].split("/");
+		for (let i=0; i<position.length; i++)
+		{
+			let k = 0;
+			for (let j=0; j<position[i].length; j++)
+			{
+				switch (position[i].charAt(j))
+				{
+					case 'k':
+						this.kingPos['b'] = [i,k];
+						break;
+					case 'K':
+						this.kingPos['w'] = [i,k];
+						break;
+					default:
+						let num = parseInt(position[i].charAt(j));
+						if (!isNaN(num))
+							k += (num-1);
+				}
+				k++;
+			}
+		}
+		this.epSquares = []; //no en-passant here
+	}
+
+	setFlags(fen)
+	{
+		// TODO: for compatibility?
+		this.castleFlags = {"w":[false,false], "b":[false,false]};
+	}
+
+	static get IMMOBILIZER() { return 'm'; }
+	// Although other pieces keep their names here for coding simplicity,
+	// keep in mind that:
+	//  - a "rook" is a coordinator, capturing by coordinating with the king
+	//  - a "knight" is a long-leaper, capturing as in draughts
+	//  - a "bishop" is a chameleon, capturing as its prey
+	//  - a "queen" is a withdrawer, capturing by moving away from pieces
+
+	getPotentialMovesFrom([x,y])
+	{
+		switch (this.getPiece(x,y))
+		{
+			case VariantRules.IMMOBILIZER:
+				return this.getPotentialImmobilizerMoves([x,y]);
+			default:
+				return super.getPotentialMovesFrom([x,y]);
+		}
+		// TODO: add potential suicides as a move "taking the immobilizer"
+		// TODO: add long-leaper captures
+		// TODO: mark matching coordinator/withdrawer/chameleon moves as captures
+		// (will be a bit tedious for chameleons)
+		// --> filter directly in functions below
+	}
+
+	getSlideNJumpMoves([x,y], steps, oneStep)
+	{
+		const color = this.getColor(x,y);
+		const piece = this.getPiece(x,y);
+		let moves = [];
+		const [sizeX,sizeY] = VariantRules.size;
+		outerLoop:
+		for (let step of steps)
+		{
+			let i = x + step[0];
+			let j = y + step[1];
+			while (i>=0 && i<sizeX && j>=0 && j<sizeY
+				&& this.board[i][j] == VariantRules.EMPTY)
+			{
+				moves.push(this.getBasicMove([x,y], [i,j]));
+				if (oneStep !== undefined)
+					continue outerLoop;
+				i += step[0];
+				j += step[1];
+			}
+			// Only king can take on occupied square:
+			if (piece==VariantRules.KING && i>=0 && i<sizeX && j>=0
+				&& j<sizeY && this.canTake([x,y], [i,j]))
+			{
+				moves.push(this.getBasicMove([x,y], [i,j]));
+			}
+		}
+		return moves;
+	}
+
+	getPotentialPawnMoves([x,y])
+	{
+		return super.getPotentialRookMoves([x,y]);
+	}
+
+	getPotentialRookMoves(sq)
+	{
+		return super.getPotentialQueenMoves(sq);
+	}
+
+	getPotentialKnightMoves(sq)
+	{
+		return super.getPotentialQueenMoves(sq);
+	}
+
+	getPotentialBishopMoves(sq)
+	{
+		return super.getPotentialQueenMoves(sq);
+	}
+
+	getPotentialQueenMoves(sq)
+	{
+		return super.getPotentialQueenMoves(sq);
+	}
+
+	getPotentialKingMoves(sq)
+	{
+		const V = VariantRules;
+		return this.getSlideNJumpMoves(sq,
+			V.steps[V.ROOK].concat(V.steps[V.BISHOP]), "oneStep");
+	}
+
+	// isAttacked() is OK because the immobilizer doesn't take
+
+	isAttackedByPawn([x,y], colors)
+	{
+		// Square (x,y) must be surrounded by two enemy pieces,
+		// and one of them at least should be a pawn
+		return false;
+	}
+
+	isAttackedByRook(sq, colors)
+	{
+		// Enemy king must be on same file and a rook on same row (or reverse)
+	}
+
+	isAttackedByKnight(sq, colors)
+	{
+		// Square (x,y) must be on same line as a knight,
+		// and there must be empty square(s) behind.
+	}
+
+	isAttackedByBishop(sq, colors)
+	{
+		// switch on piece nature on square sq: a chameleon attack as this piece
+		// ==> call the appropriate isAttackedBy... (exception of immobilizers)
+		// Other exception: a chameleon cannot attack a chameleon (seemingly...)
+	}
+
+	isAttackedByQueen(sq, colors)
+	{
+		// Square (x,y) must be adjacent to a queen, and the queen must have
+		// some free space in the opposite direction from (x,y)
+	}
+
+	updateVariables(move)
+	{
+		// Just update king position
+		const piece = this.getPiece(move.start.x,move.start.y);
+		const c = this.getColor(move.start.x,move.start.y);
+		if (piece == VariantRules.KING && move.appear.length > 0)
+		{
+			this.kingPos[c][0] = move.appear[0].x;
+			this.kingPos[c][1] = move.appear[0].y;
+		}
+	}
+
+	static get VALUES() { //TODO: totally experimental!
+		return {
+			'p': 1,
+			'r': 2,
+			'n': 5,
+			'b': 3,
+			'q': 3,
+			'm': 5,
+			'k': 1000
+		};
+	}
+
+	static get SEARCH_DEPTH() { return 2; } //TODO?
+
+	static GenRandInitFen()
+	{
+		let pieces = { "w": new Array(8), "b": new Array(8) };
+		// Shuffle pieces on first and last rank
+		for (let c of ["w","b"])
+		{
+			let positions = _.range(8);
+			// Get random squares for every piece, totally freely
+
+			let randIndex = _.random(7);
+			const bishop1Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(6);
+			const bishop2Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(5);
+			const knight1Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(4);
+			const knight2Pos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(3);
+			const queenPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(2);
+			const kingPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+
+			randIndex = _.random(1);
+			const rookPos = positions[randIndex];
+			positions.splice(randIndex, 1);
+			const immobilizerPos = positions[2];
+
+			pieces[c][bishop1Pos] = 'b';
+			pieces[c][bishop2Pos] = 'b';
+			pieces[c][knight1Pos] = 'n';
+			pieces[c][knight2Pos] = 'n';
+			pieces[c][queenPos] = 'q';
+			pieces[c][kingPos] = 'k';
+			pieces[c][rookPos] = 'r';
+			pieces[c][immobilizerPos] = 'm';
+		}
+		return pieces["b"].join("") +
+			"/pppppppp/8/8/8/8/PPPPPPPP/" +
+			pieces["w"].join("").toUpperCase() +
+			" 0000"; //TODO: flags?!
+	}
+
+	getFlagsFen()
+	{
+		return "0000"; //TODO: or "-" ?
+	}
 }
diff --git a/public/stylesheets/variant.sass b/public/stylesheets/variant.sass
index 8839c342..3eec2900 100644
--- a/public/stylesheets/variant.sass
+++ b/public/stylesheets/variant.sass
@@ -81,10 +81,6 @@ div.board
   display: inline-block
   position: relative
 
-div.board4
-  width: 25%
-  padding-bottom: 25%
-
 div.board8
   width: 12.5%
   padding-bottom: 12.5%
diff --git a/views/rules/Half.pug b/views/rules/Half.pug
index 605eda4b..91b4a4e2 100644
--- a/views/rules/Half.pug
+++ b/views/rules/Half.pug
@@ -1,15 +1,15 @@
 p.boxed
-	| 8x4 board with no pawns. Orthodox rules.
+	| 4x8 board with no pawns. Orthodox rules.
 
 figure.diagram-container
 	.diagram
-		| fen:rkqr/nbbn/4/4/4/4/NBBN/RKQR:
+		| fen:rn4NR/kb4BK/qb4BQ/rn4NR:
 	figcaption Initial position (non-random)
 
 h3 Specifications
 
 ul
-	li Chessboard: 8x4 (see diagram).
+	li Chessboard: 4x8 (see diagram).
 	li Material: no pawns.
 	li Non-capturing moves: standard.
 	li Special moves: none.
diff --git a/views/rules/Ultima.pug b/views/rules/Ultima.pug
index 77bfd093..0a3cc7f0 100644
--- a/views/rules/Ultima.pug
+++ b/views/rules/Ultima.pug
@@ -1,7 +1,7 @@
 p.boxed
 	| Pieces look the same but behave very differently.
 	| They generally move like an orthodox queen,
-	| but capturing rules are complex: you need to read on :)
+	| but capturing rules are complex.
 
 h3 Specifications
 
-- 
2.44.0