From e5dc87e0e8f2d53a910b2b42ed2a0a39ea6787aa Mon Sep 17 00:00:00 2001
From: Benjamin Auder <benjamin.auder@somewhere>
Date: Thu, 10 Jan 2019 23:39:44 +0100
Subject: [PATCH] Reorganize and completed board component. Now need to finish
 game component

---
 public/javascripts/components/board.js    | 470 ++++++++++++----------
 public/javascripts/components/game.js     | 119 ++----
 public/javascripts/components/problems.js |   3 +-
 3 files changed, 295 insertions(+), 297 deletions(-)

diff --git a/public/javascripts/components/board.js b/public/javascripts/components/board.js
index 2861cf7f..6110cfb8 100644
--- a/public/javascripts/components/board.js
+++ b/public/javascripts/components/board.js
@@ -1,3 +1,12 @@
+Vue.component('my-board', {
+	// Last move cannot be guessed from here, and is required to highlight squares
+	// gotoMove : juste set FEN depuis FEN stocké dans le coup (TODO)
+	// send event after each move (or undo), to notify what was played
+	// also notify end of game (which returns here later through prop...)
+	props: ["fen","moveToPlay","moveToUndo",
+		"analyze","lastMove","orientation","userColor","gameOver"],
+	data: function () {
+		return {
 			hints: (!localStorage["hints"] ? true : localStorage["hints"] === "1"),
 			bcolor: localStorage["bcolor"] || "lichess", //lichess, chesscom or chesstempo
 			possibleMoves: [], //filled after each valid click/dragstart
@@ -6,236 +15,241 @@
 			incheck: [],
 			start: {}, //pixels coordinates + id of starting square (click or drag)
 			vr: null, //object to check moves, store them, FEN..
-	orientation: "w", //useful if click on "flip board"	
-	
-
-// TODO: watch for property change "fen"
-// send event after each move, to notify what was played
-	
-	const [sizeX,sizeY] = [V.size.x,V.size.y];
+		};
+	},
+	watch: {
+		// NOTE: maybe next 3 should be encapsulated in object to be watched (?)
+		fen: function(newFen) {
+			this.vr = new VariantRules(newFen);
+		},
+		moveToPlay: function(move) {
+			this.play(move, "animate");
+		},
+		moveToUndo: function(move) {
+			this.undo(move);
+		},
+	},
+	created: function() {
+		this.vr = new VariantRules(this.fen);
+	},
+	render(h) {
+		const [sizeX,sizeY] = [V.size.x,V.size.y];
 		// Precompute hints squares to facilitate rendering
 		let hintSquares = doubleArray(sizeX, sizeY, false);
 		this.possibleMoves.forEach(m => { hintSquares[m.end.x][m.end.y] = true; });
 		// Also precompute in-check squares
 		let incheckSq = doubleArray(sizeX, sizeY, false);
 		this.incheck.forEach(sq => { incheckSq[sq[0]][sq[1]] = true; });
-			const choices = h('div',
-				{
-					attrs: { "id": "choices" },
-					'class': { 'row': true },
-					style: {
-						"display": this.choices.length>0?"block":"none",
-						"top": "-" + ((sizeY/2)*squareWidth+squareWidth/2) + "px",
-						"width": (this.choices.length * squareWidth) + "px",
-						"height": squareWidth + "px",
-					},
+		const choices = h(
+			'div',
+			{
+				attrs: { "id": "choices" },
+				'class': { 'row': true },
+				style: {
+					"display": this.choices.length>0?"block":"none",
+					"top": "-" + ((sizeY/2)*squareWidth+squareWidth/2) + "px",
+					"width": (this.choices.length * squareWidth) + "px",
+					"height": squareWidth + "px",
 				},
-				this.choices.map( m => { //a "choice" is a move
-					return h('div',
-						{
-							'class': {
-								'board': true,
-								['board'+sizeY]: true,
-							},
-							style: {
-								'width': (100/this.choices.length) + "%",
-								'padding-bottom': (100/this.choices.length) + "%",
-							},
+			},
+			this.choices.map(m => { //a "choice" is a move
+				return h('div',
+					{
+						'class': {
+							'board': true,
+							['board'+sizeY]: true,
+						},
+						style: {
+							'width': (100/this.choices.length) + "%",
+							'padding-bottom': (100/this.choices.length) + "%",
 						},
-						[h('img',
-							{
-								attrs: { "src": '/images/pieces/' +
-									VariantRules.getPpath(m.appear[0].c+m.appear[0].p) + '.svg' },
-								'class': { 'choice-piece': true },
-								on: {
-									"click": e => { this.play(m); this.choices=[]; },
-									// NOTE: add 'touchstart' event to fix a problem on smartphones
-									"touchstart": e => { this.play(m); this.choices=[]; },
-								},
-							})
-						]
-					);
-				})
-			);
-			// Create board element (+ reserves if needed by variant or mode)
-			const lm = this.vr.lastMove;
-			const showLight = this.hints && variant.name!="Dark" &&
-				(this.mode != "idle" ||
-					(this.vr.moves.length > 0 && this.cursor==this.vr.moves.length));
-			const gameDiv = h('div',
-				{
-					'class': {
-						'game': true,
-						'clearer': true,
 					},
-				},
-				[_.range(sizeX).map(i => {
-					let ci = (this.mycolor=='w' ? i : sizeX-i-1);
-					return h(
-						'div',
+					[h('img',
 						{
-							'class': {
-								'row': true,
+							attrs: { "src": '/images/pieces/' +
+								V.getPpath(m.appear[0].c+m.appear[0].p) + '.svg' },
+							'class': { 'choice-piece': true },
+							on: {
+								"click": e => { this.play(m); this.choices=[]; },
+								// NOTE: add 'touchstart' event to fix a problem on smartphones
+								"touchstart": e => { this.play(m); this.choices=[]; },
 							},
-							style: { 'opacity': this.choices.length>0?"0.5":"1" },
-						},
-						_.range(sizeY).map(j => {
-							let cj = (this.mycolor=='w' ? j : sizeY-j-1);
-							let elems = [];
-							if (this.vr.board[ci][cj] != VariantRules.EMPTY && (variant.name!="Dark"
-								|| this.score!="*" || this.vr.enlightened[this.mycolor][ci][cj]))
-							{
-								elems.push(
-									h(
-										'img',
-										{
-											'class': {
-												'piece': true,
-												'ghost': !!this.selectedPiece
-													&& this.selectedPiece.parentNode.id == "sq-"+ci+"-"+cj,
-											},
-											attrs: {
-												src: "/images/pieces/" +
-													VariantRules.getPpath(this.vr.board[ci][cj]) + ".svg",
-											},
-										}
-									)
-								);
-							}
-							if (this.hints && hintSquares[ci][cj])
-							{
-								elems.push(
-									h(
-										'img',
-										{
-											'class': {
-												'mark-square': true,
-											},
-											attrs: {
-												src: "/images/mark.svg",
-											},
-										}
-									)
-								);
-							}
-							return h(
-								'div',
-								{
-									'class': {
-										'board': true,
-										['board'+sizeY]: true,
-										'light-square': (i+j)%2==0,
-										'dark-square': (i+j)%2==1,
-										[this.bcolor]: true,
-										'in-shadow': variant.name=="Dark" && this.score=="*"
-											&& !this.vr.enlightened[this.mycolor][ci][cj],
-										'highlight': showLight && !!lm && _.isMatch(lm.end, {x:ci,y:cj}),
-										'incheck': showLight && incheckSq[ci][cj],
-									},
-									attrs: {
-										id: this.getSquareId({x:ci,y:cj}),
-									},
-								},
-								elems
-							);
 						})
-					);
-				}), choices]
-			);
-			if (!!this.vr.reserve)
+					]
+				);
+			})
+		);
+		// Create board element (+ reserves if needed by variant or mode)
+		const lm = this.lastMove;
+		const showLight = this.hints && variant.name != "Dark";
+		const gameDiv = h(
+			'div',
 			{
-				const shiftIdx = (this.mycolor=="w" ? 0 : 1);
-				let myReservePiecesArray = [];
-				for (let i=0; i<VariantRules.RESERVE_PIECES.length; i++)
-				{
-					myReservePiecesArray.push(h('div',
+				'class': {
+					'game': true,
+					'clearer': true,
+				},
+			},
+			[_.range(sizeX).map(i => {
+				let ci = (this.orientation=='w' ? i : sizeX-i-1);
+				return h(
+					'div',
 					{
-						'class': {'board':true, ['board'+sizeY]:true},
-						attrs: { id: this.getSquareId({x:sizeX+shiftIdx,y:i}) }
+						'class': {
+							'row': true,
+						},
+						style: { 'opacity': this.choices.length>0?"0.5":"1" },
 					},
-					[
-						h('img',
+					_.range(sizeY).map(j => {
+						let cj = (this.orientation=='w' ? j : sizeY-j-1);
+						let elems = [];
+						if (this.vr.board[ci][cj] != V.EMPTY && (variant.name!="Dark"
+							|| this.gameOver || this.vr.enlightened[this.userColor][ci][cj]))
 						{
-							'class': {"piece":true, "reserve":true},
-							attrs: {
-								"src": "/images/pieces/" +
-									this.vr.getReservePpath(this.mycolor,i) + ".svg",
-							}
-						}),
-						h('sup',
-							{"class": { "reserve-count": true } },
-							[ this.vr.reserve[this.mycolor][VariantRules.RESERVE_PIECES[i]] ]
-						)
-					]));
-				}
-				let oppReservePiecesArray = [];
-				const oppCol = this.vr.getOppCol(this.mycolor);
-				for (let i=0; i<VariantRules.RESERVE_PIECES.length; i++)
-				{
-					oppReservePiecesArray.push(h('div',
-					{
-						'class': {'board':true, ['board'+sizeY]:true},
-						attrs: { id: this.getSquareId({x:sizeX+(1-shiftIdx),y:i}) }
-					},
-					[
-						h('img',
+							elems.push(
+								h(
+									'img',
+									{
+										'class': {
+											'piece': true,
+											'ghost': !!this.selectedPiece
+												&& this.selectedPiece.parentNode.id == "sq-"+ci+"-"+cj,
+										},
+										attrs: {
+											src: "/images/pieces/" +
+												V.getPpath(this.vr.board[ci][cj]) + ".svg",
+										},
+									}
+								)
+							);
+						}
+						if (this.hints && hintSquares[ci][cj])
 						{
-							'class': {"piece":true, "reserve":true},
-							attrs: {
-								"src": "/images/pieces/" +
-									this.vr.getReservePpath(oppCol,i) + ".svg",
-							}
-						}),
-						h('sup',
-							{"class": { "reserve-count": true } },
-							[ this.vr.reserve[oppCol][VariantRules.RESERVE_PIECES[i]] ]
-						)
-					]));
-				}
-				let reserves = h('div',
-					{
-						'class':{
-							'game': true,
-							"reserve-div": true,
-						},
-					},
-					[
-						h('div',
+							elems.push(
+								h(
+									'img',
+									{
+										'class': {
+											'mark-square': true,
+										},
+										attrs: {
+											src: "/images/mark.svg",
+										},
+									}
+								)
+							);
+						}
+						return h(
+							'div',
 							{
 								'class': {
-									'row': true,
-									"reserve-row-1": true,
+									'board': true,
+									['board'+sizeY]: true,
+									'light-square': (i+j)%2==0,
+									'dark-square': (i+j)%2==1,
+									[this.bcolor]: true,
+									'in-shadow': variant.name=="Dark" && !this.gameOver
+										&& !this.vr.enlightened[this.userColor][ci][cj],
+									'highlight': showLight && !!lm && _.isMatch(lm.end, {x:ci,y:cj}),
+									'incheck': showLight && incheckSq[ci][cj],
+								},
+								attrs: {
+									id: this.getSquareId({x:ci,y:cj}),
 								},
 							},
-							myReservePiecesArray
-						),
-						h('div',
-							{ 'class': { 'row': true }},
-							oppReservePiecesArray
-						)
-					]
+							elems
+						);
+					})
 				);
-				elementArray.push(reserves);
+			}), choices]
+		);
+		let elementArray = [choices, gameDiv];
+		if (!!this.vr.reserve)
+		{
+			const shiftIdx = (this.userColor=="w" ? 0 : 1);
+			let myReservePiecesArray = [];
+			for (let i=0; i<V.RESERVE_PIECES.length; i++)
+			{
+				myReservePiecesArray.push(h('div',
+				{
+					'class': {'board':true, ['board'+sizeY]:true},
+					attrs: { id: this.getSquareId({x:sizeX+shiftIdx,y:i}) }
+				},
+				[
+					h('img',
+					{
+						'class': {"piece":true, "reserve":true},
+						attrs: {
+							"src": "/images/pieces/" +
+								this.vr.getReservePpath(this.userColor,i) + ".svg",
+						}
+					}),
+					h('sup',
+						{"class": { "reserve-count": true } },
+						[ this.vr.reserve[this.userColor][V.RESERVE_PIECES[i]] ]
+					)
+				]));
+			}
+			let oppReservePiecesArray = [];
+			const oppCol = this.vr.getOppCol(this.userColor);
+			for (let i=0; i<V.RESERVE_PIECES.length; i++)
+			{
+				oppReservePiecesArray.push(h('div',
+				{
+					'class': {'board':true, ['board'+sizeY]:true},
+					attrs: { id: this.getSquareId({x:sizeX+(1-shiftIdx),y:i}) }
+				},
+				[
+					h('img',
+					{
+						'class': {"piece":true, "reserve":true},
+						attrs: {
+							"src": "/images/pieces/" +
+								this.vr.getReservePpath(oppCol,i) + ".svg",
+						}
+					}),
+					h('sup',
+						{"class": { "reserve-count": true } },
+						[ this.vr.reserve[oppCol][V.RESERVE_PIECES[i]] ]
+					)
+				]));
 			}
-				// Show current FEN (just below board, lower right corner)
-// (if mode != Dark ...)
-				elementArray.push(
+			let reserves = h('div',
+				{
+					'class':{
+						'game': true,
+						"reserve-div": true,
+					},
+				},
+				[
 					h('div',
 						{
-							attrs: { id: "fen-div" },
-							"class": { "section-content": true },
+							'class': {
+								'row': true,
+								"reserve-row-1": true,
+							},
 						},
-						[
-							h('p',
-								{
-									attrs: { id: "fen-string" },
-									domProps: { innerHTML: this.vr.getBaseFen() },
-									"class": { "text-center": true },
-								}
-							)
-						]
+						myReservePiecesArray
+					),
+					h('div',
+						{ 'class': { 'row': true }},
+						oppReservePiecesArray
 					)
-				);
+				]
+			);
+			elementArray.push(reserves);
+		}
+		return h(
+			'div',
+			{
+				'class': {
+					"col-sm-12":true,
+					"col-md-10":true,
+					"col-md-offset-1":true,
+					"col-lg-8":true,
+					"col-lg-offset-2":true,
+				},
+				// NOTE: click = mousedown + mouseup
 				on: {
 					mousedown: this.mousedown,
 					mousemove: this.mousemove,
@@ -244,10 +258,12 @@
 					touchmove: this.mousemove,
 					touchend: this.mouseup,
 				},
-
-
-		// TODO: "chessground-like" component
-		// Get the identifier of a HTML table cell from its numeric coordinates o.x,o.y.
+			},
+			elementArray
+		);
+	},
+	methods: {
+		// Get the identifier of a HTML square from its numeric coordinates o.x,o.y.
 		getSquareId: function(o) {
 			// NOTE: a separator is required to allow any size of board
 			return  "sq-" + o.x + "-" + o.y;
@@ -289,18 +305,11 @@
 				this.selectedPiece.style.zIndex = 3000;
 				const startSquare = this.getSquareFromId(e.target.parentNode.id);
 				this.possibleMoves = [];
-				if (this.score == "*")
-				{
-
-// TODO: essentially adapt this (all other things do not change much)
-// if inside a real game, mycolor should be provided ? (simplest way)
-
-					const color = ["friend","problem"].includes(this.mode)
-						? this.vr.turn
-						: this.mycolor;
-					if (this.vr.canIplay(color,startSquare))
-						this.possibleMoves = this.vr.getPossibleMovesFrom(startSquare);
-				}
+				const color = this.analyze || this.gameOver
+					? this.vr.turn
+					: this.userColor;
+				if (this.vr.canIplay(color,startSquare))
+					this.possibleMoves = this.vr.getPossibleMovesFrom(startSquare);
 				// Next line add moving piece just after current image
 				// (required for Crazyhouse reserve)
 				e.target.parentNode.insertBefore(this.selectedPiece, e.target.nextSibling);
@@ -389,3 +398,26 @@
 				this.play(move);
 			}, 250);
 		},
+		play: function(move, programmatic) {
+			if (!!programmatic) //computer or human opponent
+				return this.animateMove(move);
+			// Not programmatic, or animation is over
+			this.vr.play(move);
+			if (this.sound == 2)
+				new Audio("/sounds/move.mp3").play().catch(err => {});
+			// Is opponent in check?
+			this.incheck = this.vr.getCheckSquares(this.vr.turn);
+			const eog = this.vr.getCurrentScore();
+			if (eog != "*")
+			{
+				// TODO: notify end of game (give score)
+			}
+		},
+		undo: function(move) {
+			this.vr.undo(move);
+			if (this.sound == 2)
+				new Audio("/sounds/undo.mp3").play().catch(err => {});
+			this.incheck = this.vr.getCheckSquares(this.vr.turn);
+		},
+	},
+})
diff --git a/public/javascripts/components/game.js b/public/javascripts/components/game.js
index 84088c05..2b527d93 100644
--- a/public/javascripts/components/game.js
+++ b/public/javascripts/components/game.js
@@ -54,6 +54,25 @@ Vue.component('my-game', {
 			// TODO: controls: abort, clear, resign, draw (avec confirm box)
 			// et si partie terminée : (mode analyse) just clear, back / play
 			// + flip button toujours disponible
+				// Show current FEN (just below board, lower right corner)
+// (if mode != Dark ...)
+				elementArray.push(
+					h('div',
+						{
+							attrs: { id: "fen-div" },
+							"class": { "section-content": true },
+						},
+						[
+							h('p',
+								{
+									attrs: { id: "fen-string" },
+									domProps: { innerHTML: this.vr.getBaseFen() },
+									"class": { "text-center": true },
+								}
+							)
+						]
+					)
+				);
 			
 			<div id="pgn-div" class="section-content">
 				<a id="download" href: "#"></a>
@@ -264,83 +283,6 @@ Vue.component('my-game', {
 			this.compWorker.postMessage(["askmove"]);
 		},
 		// OK, these last functions can stay here (?!)
-		play: function(move, programmatic) {
-			if (!move)
-			{
-				// Navigate after game is over
-				if (this.cursor >= this.moves.length)
-					return; //already at the end
-				move = this.moves[this.cursor++];
-			}
-			if (!!programmatic) //computer or human opponent
-				return this.animateMove(move);
-			// Not programmatic, or animation is over
-			if (this.mode == "human" && this.vr.turn == this.mycolor)
-				this.conn.send(JSON.stringify({code:"newmove", move:move, oppid:this.oppid}));
-			
-			
-			// TODO: play move, and stack it on this.moves (if a move was provided; otherwise just navigate)
-			
-			if (this.score == "*") //TODO: I don't like this if()
-			{
-				// Emergency check, if human game started "at the same time"
-				// TODO: robustify this...
-				if (this.mode == "human" && !!move.computer)
-					return;
-				this.vr.play(move, "ingame");
-				// Is opponent in check?
-				this.incheck = this.vr.getCheckSquares(this.vr.turn);
-				if (this.sound == 2)
-					new Audio("/sounds/move.mp3").play().catch(err => {});
-				if (this.mode == "computer")
-				{
-					// Send the move to web worker (TODO: including his own moves?!)
-					this.compWorker.postMessage(["newmove",move]);
-				}
-				const eog = this.vr.getCurrentScore();
-				if (eog != "*")
-				{
-					if (["human","computer"].includes(this.mode))
-						this.endGame(eog);
-					else
-					{
-						// Just show score on screen (allow undo)
-						this.score = eog;
-						this.showScoreMsg();
-					}
-				}
-			}
-//			else
-//			{
-//				VariantRules.PlayOnBoard(this.vr.board, move);
-//				this.$forceUpdate(); //TODO: ?!
-//			}
-			if (["human","computer","friend"].includes(this.mode))
-				this.updateStorage(); //after our moves and opponent moves
-			if (this.mode == "computer" && this.vr.turn != this.mycolor && this.score == "*")
-				this.playComputerMove();
-		},
-		// TODO: merge two next functions
-		undo: function() {
-			// Navigate after game is over
-			if (this.cursor == 0)
-				return; //already at the beginning
-			if (this.cursor == this.vr.moves.length)
-				this.incheck = []; //in case of...
-			const move = this.vr.moves[--this.cursor];
-			VariantRules.UndoOnBoard(this.vr.board, move);
-			this.$forceUpdate(); //TODO: ?!
-		},
-		undoInGame: function() {
-			const lm = this.vr.lastMove;
-			if (!!lm)
-			{
-				this.vr.undo(lm);
-				if (this.sound == 2)
-					new Audio("/sounds/undo.mp3").play().catch(err => {});
-				this.incheck = this.vr.getCheckSquares(this.vr.turn);
-			}
-		},
 	},
 })
 
@@ -356,3 +298,26 @@ Vue.component('my-game', {
 //TODO: confirm dialog with "opponent offers draw", avec possible bouton "prevent future offers" + bouton "proposer nulle"
 //+ bouton "abort" avec score == "?" + demander confirmation pour toutes ces actions,
 //comme sur lichess
+			
+// send move from here:
+//if (this.mode == "human" && this.vr.turn == this.mycolor)
+			//this.conn.send(JSON.stringify({code:"newmove", move:move, oppid:this.oppid}));
+			// TODO: play move, and stack it on this.moves (if a move was provided; otherwise just navigate)
+			
+//			if (["human","computer","friend"].includes(this.mode))
+//				this.updateStorage(); //after our moves and opponent moves
+//			if (this.mode == "computer" && this.vr.turn != this.mycolor && this.score == "*")
+//				this.playComputerMove();
+//			if (this.mode == "computer")
+//			{
+//				// Send the move to web worker (TODO: including his own moves?!)
+//				this.compWorker.postMessage(["newmove",move]);
+//			}
+//				if (["human","computer"].includes(this.mode))
+//					this.endGame(eog);
+//				else
+//				{
+//					// Just show score on screen (allow undo)
+//					this.score = eog;
+//					this.showScoreMsg();
+//				}
diff --git a/public/javascripts/components/problems.js b/public/javascripts/components/problems.js
index 897babbe..9a14c709 100644
--- a/public/javascripts/components/problems.js
+++ b/public/javascripts/components/problems.js
@@ -38,7 +38,8 @@ Vue.component('my-problems', {
 						{{ curProb.instructions }}
 					</p>
 				</div>
-				<my-board :fen="curProb.fen"></my-board>
+				<my-board :fen="curProb.fen" :analyze:"true" .................> //TODO: use my-game in analyze mode ?
+				</my-board>
 				<div id="solution-div" class="section-content">
 					<h3 class="clickable" @click="showSolution = !showSolution">
 						{{ translations["Show solution"] }}
-- 
2.44.0