selectedPiece: null, //moving piece (or clicked piece)
                        conn: null, //socket connection
                        score: "*", //'*' means 'unfinished'
-                       mode: "idle", //human, friend, computer or idle (when not playing)
+                       mode: "idle", //human, chat, friend, problem, computer or idle (if not playing)
+                       myid: "", //our ID, always set
                        oppid: "", //opponent ID in case of HH game
+                       myname: getCookie("username","anonymous"),
+                       oppName: "anonymous", //opponent name, revealed after a game (if provided)
+                       chats: [], //chat messages after human game
                        oppConnected: false,
                        seek: false,
                        fenStart: "",
                };
        },
        watch: {
-               problem: function(p, pp) {
+               problem: function(p) {
                        // 'problem' prop changed: update board state
                        this.newGame("problem", p.fen, V.ParseFen(p.fen).turn);
                },
                        },
                        [h('i', { 'class': { "material-icons": true } }, "accessibility")])
                );
-               if (["idle","computer"].includes(this.mode))
+               if (["idle","chat","computer"].includes(this.mode))
                {
                        actionArray.push(
                                h('button',
                                [h('i', { 'class': { "material-icons": true } }, "computer")])
                        );
                }
-               if (["idle","friend"].includes(this.mode))
+               if (["idle","chat","friend"].includes(this.mode))
                {
                        actionArray.push(
                                h('button',
                                );
                                elementArray.push(connectedIndic);
                        }
+                       else if (this.mode == "chat")
+                       {
+                               // Also show connection indication, but also nickname
+                               const nicknameOpponent = h(
+                                       'div',
+                                       {
+                                               "class": {
+                                                       "clickable": true,
+                                                       "topindicator": true,
+                                                       "indic-left": true,
+                                                       "name-connected": this.oppConnected,
+                                                       "name-disconnected": !this.oppConnected,
+                                               },
+                                               domProps: {
+                                                       innerHTML: this.oppName,
+                                               },
+                                               on: {
+                                                       "click": () => { document.getElementById("modal-chat").checked = true; },
+                                               },
+                                       }
+                               );
+                               elementArray.push(nicknameOpponent);
+                       }
                        const turnIndic = h(
                                'div',
                                {
                                        'class': {
                                                "tooltip": true,
                                                "topindicator": true,
-                                               "indic-left": true,
+                                               "indic-right": true,
                                                "settings-btn": !smallScreen,
                                                "settings-btn-small": smallScreen,
                                        },
                                        h('div',
                                                {
                                                        attrs: { id: "instructions-div" },
-                                                       "class": { "section-content": true },
+                                                       "class": {
+                                                               "clearer": true,
+                                                               "section-content": true,
+                                                       },
                                                },
                                                [
                                                        h('p',
                        // Create board element (+ reserves if needed by variant or mode)
                        const lm = this.vr.lastMove;
                        const showLight = this.hints &&
-                               (this.mode!="idle" || this.cursor==this.vr.moves.length);
+                               (!["idle","chat"].includes(this.mode) || this.cursor==this.vr.moves.length);
                        const gameDiv = h('div',
                                {
                                        'class': { 'game': true },
                                        );
                                }), choices]
                        );
-                       if (this.mode != "idle")
+                       if (!["idle","chat"].includes(this.mode))
                        {
                                actionArray.push(
                                        h('button',
                                                          { },
                                                                [
                                                                        //h('legend', { domProps: { innerHTML: "Legend title" } }),
+                                                                       h('label',
+                                                                               {
+                                                                                       attrs: { for: "nameSetter" },
+                                                                                       domProps: { innerHTML: "My name is..." },
+                                                                               },
+                                                                       ),
+                                                                       h('input',
+                                                                               {
+                                                                                       attrs: {
+                                                                                               "id": "nameSetter",
+                                                                                               type: "text",
+                                                                                               value: this.myname,
+                                                                                       },
+                                                                                       on: { "change": this.setMyname },
+                                                                               }
+                                                                       ),
+                                                               ]
+                                                       ),
+                                                       h('fieldset',
+                                                         { },
+                                                               [
                                                                        h('label',
                                                                                {
                                                                                        attrs: { for: "setHints" },
                        )
                ];
                elementArray = elementArray.concat(modalSettings);
+               let chatEltsArray =
+               [
+                       h('label',
+                               {
+                                       attrs: { "id": "close-chat", "for": "modal-chat" },
+                                       "class": { "modal-close": true },
+                               }
+                       ),
+                       h('h3',
+                               {
+                                       attrs: { "id": "titleChat" },
+                                       "class": { "section": true },
+                                       domProps: { innerHTML: "Chat with " + this.oppName },
+                               }
+                       )
+               ];
+               for (let chat of this.chats)
+               {
+                       chatEltsArray.push(
+                               h('p',
+                                       {
+                                               "class": {
+                                                       "my-chatmsg": chat.author==this.myid,
+                                                       "opp-chatmsg": chat.author==this.oppid,
+                                               },
+                                               domProps: { innerHTML: chat.msg }
+                                       }
+                               )
+                       );
+               }
+               chatEltsArray = chatEltsArray.concat([
+                       h('input',
+                               {
+                                       attrs: {
+                                               "id": "input-chat",
+                                               type: "text",
+                                               placeholder: "Type here",
+                                       },
+                                       on: { keyup: this.trySendChat }, //if key is 'enter'
+                               }
+                       ),
+                       h('button',
+                               {
+                                       on: { click: this.sendChat },
+                                       domProps: { innerHTML: "Send" },
+                               }
+                       )
+               ]);
+               const modalChat = [
+                       h('input',
+                               {
+                                       attrs: { "id": "modal-chat", type: "checkbox" },
+                                       "class": { "modal": true },
+                               }),
+                       h('div',
+                               {
+                                       attrs: { "role": "dialog", "aria-labelledby": "titleChat" },
+                               },
+                               [
+                                       h('div',
+                                               {
+                                                       "class": { "card": true, "smallpad": true },
+                                               },
+                                               chatEltsArray
+                                       )
+                               ]
+                       )
+               ];
+               elementArray = elementArray.concat(modalChat);
                const actions = h('div',
                        {
                                attrs: { "id": "actions" },
                        const data = JSON.parse(msg.data);
                        switch (data.code)
                        {
+                               case "oppname":
+                                       // Receive opponent's name
+                                       this.oppName = data.name;
+                                       break;
+                               case "newchat":
+                                       // Receive new chat
+                                       this.chats.push({msg:data.msg, author:this.oppid});
+                                       break;
                                case "duplicate":
                                        // We opened another tab on the same game
                                        this.mode = "idle";
                                // TODO: also use (dis)connect info to count online players?
                                case "connect":
                                case "disconnect":
-                                       if (this.mode == "human" && this.oppid == data.id)
+                                       if (["human","chat"].includes(this.mode) && this.oppid == data.id)
                                                this.oppConnected = (data.code == "connect");
+                                       if (this.oppConnected)
+                                       {
+                                               // Send our name to the opponent, in case of he hasn't it
+                                               this.conn.send(JSON.stringify({
+                                                       code:"myname", name:this.myname, oppid: this.oppid}));
+                                       }
                                        break;
                        }
                };
                this.conn.onclose = socketCloseListener;
                // Listen to keyboard left/right to navigate in game
                document.onkeydown = event => {
-                       if (this.mode == "idle" && !!this.vr && this.vr.moves.length > 0
-                               && [37,39].includes(event.keyCode))
+                       if (["idle","chat"].includes(this.mode) &&
+                               !!this.vr && this.vr.moves.length > 0 && [37,39].includes(event.keyCode))
                        {
                                event.preventDefault();
                                if (event.keyCode == 37) //Back
                }
        },
        methods: {
+               setMyname: function(e) {
+                       this.myname = e.target.value;
+                       setCookie("username",this.myname);
+               },
+               trySendChat: function(e) {
+                       if (e.keyCode == 13) //'enter' key
+                               this.sendChat();
+               },
+               sendChat: function() {
+                       let chatInput = document.getElementById("input-chat");
+                       const chatTxt = chatInput.value;
+                       chatInput.value = "";
+                       this.chats.push({msg:chatTxt, author:this.myid});
+                       this.conn.send(JSON.stringify({
+                               code:"newchat", oppid: this.oppid, msg: chatTxt}));
+               },
                toggleShowSolution: function() {
                        let problemSolution = document.getElementById("problem-solution");
                        problemSolution.style.display =
                        this.pgnTxt = this.vr.getPGN(this.mycolor, this.score, this.fenStart, this.mode);
                        if (["human","computer"].includes(this.mode))
                                this.clearStorage();
-                       this.mode = "idle";
+                       if (this.mode == "human" && this.oppConnected)
+                       {
+                               // Send our nickname to opponent
+                               this.conn.send(JSON.stringify({
+                                       code:"myname", name:this.myname, oppid:this.oppid}));
+                       }
+                       this.mode = (this.mode=="human" ? "chat" : "idle");
                        this.cursor = this.vr.moves.length; //to navigate in finished game
-                       this.oppid = "";
+                       if (this.mode == "idle") //keep oppid in case of chat after human game
+                               this.oppid = "";
                },
                setStorage: function() {
                        if (this.mode=="human")
                },
                clickComputerGame: function(e) {
                        this.getRidOfTooltip(e.currentTarget);
-                       if (this.mode == "human")
-                               return; //no newgame while playing
                        this.newGame("computer");
                },
                clickFriendGame: function(e) {
                                        }
                                }
                        }
-                       if (this.mode == "computer" && mode == "human")
-                       {
-                               // Save current computer game to resume it later
-                               this.setStorage();
-                       }
                        this.vr = new VariantRules(fen, moves || []);
                        this.score = "*";
                        this.pgnTxt = ""; //redundant with this.score = "*", but cleaner
                        }
                        else if (mode == "computer")
                        {
+                               this.setStorage(); //store game state
                                this.compWorker.postMessage(["init",this.vr.getFen()]);
                                this.mycolor = Math.random() < 0.5 ? 'w' : 'b';
                                if (this.mycolor == 'b')
                                this.selectedPiece.style.zIndex = 3000;
                                const startSquare = this.getSquareFromId(e.target.parentNode.id);
                                this.possibleMoves = [];
-                               if (this.mode != "idle")
+                               if (!["idle","chat"].includes(this.mode))
                                {
                                        const color = ["friend","problem"].includes(this.mode)
                                                ? this.vr.turn
                                this.conn.send(JSON.stringify({code:"newmove", move:move, oppid:this.oppid}));
                        if (this.sound == 2)
                                new Audio("/sounds/chessmove1.mp3").play().catch(err => {});
-                       if (this.mode != "idle")
+                       if (!["idle","chat"].includes(this.mode))
                        {
                                this.incheck = this.vr.getCheckSquares(move); //is opponent in check?
                                this.vr.play(move, "ingame");
                        }
                        if (["human","computer"].includes(this.mode))
                                this.updateStorage(); //after our moves and opponent moves
-                       if (this.mode != "idle")
+                       if (!["idle","chat"].includes(this.mode))
                        {
                                const eog = this.vr.checkGameOver();
                                if (eog != "*")