From 012caef5060ee876e5dfd6d4defd1795fce30d8f Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Lucien=20Asti=C3=A9?= <lucien@MacBook-Air-de-Lucien.local>
Date: Fri, 9 Aug 2024 14:38:16 +0200
Subject: [PATCH] Add avatar image + generateur de pseudo

---
 app/[id]/page.tsx       | 36 +++++++++-------
 app/avatarImage.ts      |  1 +
 app/page.tsx            | 94 ++++++++++++++++++++++++++++++++++++++---
 app/usernameGenerate.ts | 25 +++++++++++
 package-lock.json       | 11 +++++
 package.json            |  1 +
 server.mjs              | 21 ++++-----
 7 files changed, 159 insertions(+), 30 deletions(-)
 create mode 100644 app/avatarImage.ts
 create mode 100644 app/usernameGenerate.ts

diff --git a/app/[id]/page.tsx b/app/[id]/page.tsx
index dca3fa4..7c88966 100644
--- a/app/[id]/page.tsx
+++ b/app/[id]/page.tsx
@@ -4,6 +4,7 @@ import { io } from "socket.io-client"
 import { getRandomQuestion } from "./questions"
 import { useForceUpdate } from "./forceUpdate"
 import { IconCrown } from "@tabler/icons-react"
+import { defaultAvatarImage } from '../avatarImage'
 
 interface roomProps {
   params: {
@@ -17,6 +18,7 @@ export default function Home({ params }: roomProps) {
 
   const [role, setRole] = useState("")
   const [name, setName] = useState("")
+  const [avatar, setAvatar] = useState(defaultAvatarImage)
 
   const [gameStarted, setGameStarted] = useState(false)
   const [gameEnded, setGameEnded] = useState(false)
@@ -37,14 +39,16 @@ export default function Home({ params }: roomProps) {
   const roomNameDisplay = id.substring(0,3) + " " + id.substring(3,6)
 
   useEffect(() => {
-        const username = localStorage.getItem('name')
-        setName(username)
+        const localName = localStorage.getItem('name')
+        setName(localName)
+        const localAvatar = localStorage.getItem('avatar')
+        setAvatar(localAvatar)
         // Listen for incoming setMessages
         socketRef.current = io("ws://localhost:3000");
 
         socketRef.current.on("connect", () => {
           setIsConnected(true)
-          socketRef.current.emit('room_connect', {"id": id, "name": username})
+          socketRef.current.emit('room_connect', {id: id, name: localName, avatar: localAvatar})
         });
 
         socketRef.current.on("new_player", (params) => {
@@ -195,19 +199,21 @@ export default function Home({ params }: roomProps) {
             }
             { (isConnected && players.length > 0) &&
               <div className="flex flex-col space-y-16">
-                <div>
+                <div className="grid grid-cols-3 gap-4">
                   { players.map((player) => {
-                    console.log(player)
-                    if(player.role == "player") {
-                      return (
-                        <p>{player.name}</p>
-                      )
-                    }
-                    if(player.role == "owner") {
-                      return (
-                        <p className="flex flex-row">{player.name}<IconCrown /></p>
-                      )
-                    }
+                    return (
+                      <>
+                        <div className="flex flex-col items-center">
+                          <div className="avatar indicator">
+                            {player.role == "owner" && <IconCrown className="indicator-item" />}
+                            <div className="w-16 rounded">
+                              <img src={player.avatar} />
+                            </div>
+                          </div>
+                          <p className="flex flex-row">{player.name}</p>
+                        </div>
+                      </>
+                    )
                   })}
                 </div>
                 { role == "owner" && 
diff --git a/app/avatarImage.ts b/app/avatarImage.ts
new file mode 100644
index 0000000..9a72002
--- /dev/null
+++ b/app/avatarImage.ts
@@ -0,0 +1 @@
+export const defaultAvatarImage = ""
diff --git a/app/page.tsx b/app/page.tsx
index 5672327..b3634c3 100644
--- a/app/page.tsx
+++ b/app/page.tsx
@@ -2,10 +2,16 @@
 
 import { navigate } from './action'
 import { useState, useEffect, useRef } from 'react'
-import { IconPencilMinus } from "@tabler/icons-react"
+import { IconPencilMinus, IconCamera, IconDice6 } from "@tabler/icons-react"
+import { defaultAvatarImage } from './avatarImage'
+import { getUsername } from './usernameGenerate'
+import Webcam from 'react-webcam'
 
 export default function Home() {
   const [name, setName] = useState("") 
+  const [avatar, setAvatar] = useState(defaultAvatarImage)
+  const [showWebcam, setShowWebcam] = useState(false)
+  const webcamRef = useRef()
   const modal = useRef()
   const inputName = useRef()
 
@@ -14,35 +20,113 @@ export default function Home() {
     localStorage.setItem("name", inputName.current.value)
   }
 
+  function setAndStoreAvatar(data){
+    resizeBase64Image(data).then((data) => {
+      localStorage.setItem("avatar", data)
+      setAvatar(data)
+    })
+  }
+
+  function takePictureAvatar() {
+    const imageSrc = webcamRef.current.getScreenshot()
+    setAndStoreAvatar(imageSrc)
+    setShowWebcam(false)
+  }
+
+  function getBase64OfImage(file, cb) {
+    let reader = new FileReader();
+    reader.readAsDataURL(file);
+    reader.onload = function () {
+        cb(reader.result)
+    };
+    reader.onerror = function (error) {
+        console.log('Error: ', error);
+    };
+  }
+
+  function resizeBase64Image(base64Image) {
+    return new Promise((resolve, reject) => {
+      const maxSizeInKB = 500;
+      const maxSizeInBytes = maxSizeInKB * 1024;
+      const img = new Image();
+      img.src = base64Image;
+      img.onload = function () {
+        const canvas = document.createElement("canvas");
+        const ctx = canvas.getContext('2d');
+        const width = img.width;
+        const height = img.height;
+        const aspectRatio = width / height;
+        const newWidth = Math.sqrt(maxSizeInBytes * aspectRatio);
+        const newHeight = Math.sqrt(maxSizeInBytes / aspectRatio);
+        canvas.width = newWidth;
+        canvas.height = newHeight;
+        ctx.drawImage(img, 0, 0, newWidth, newHeight);
+        let quality = 0.8;
+        let dataURL = canvas.toDataURL('image/jpeg', quality);
+        resolve(dataURL);
+      };
+    });
+  }
+
   useEffect(() => {
     let localName = localStorage.getItem("name")
+    let localAvatar = localStorage.getItem("avatar")
     if(localName != null){
       setName(localName)
     } else {
       modal.current.showModal()
     }
+
+    if(localAvatar != null) {
+      setAvatar(localAvatar)
+    }
   }, [])
 
   return (
     <main data-theme="light" className="flex min-h-screen flex-col items-center space-y-16 p-24">
       <dialog ref={modal} className="modal">
-        <div className="modal-box">
-          <label className="form-control w-full max-w-xs">
+        <div className="modal-box flex flex-col space-y-4">
+          <div className="avatar flex flex-col items-center space-y-4">
+            <div className="w-24 rounded">
+              <img src={avatar} />
+            </div>
+            { showWebcam &&
+              <>
+                <Webcam ref={webcamRef} />
+                <button className="btn btn-primary" onClick={takePictureAvatar}><IconCamera /></button>
+              </>
+            }
+            { !showWebcam &&
+              <>
+                <button className="btn btn-primary" onClick={() => {setShowWebcam(true)}}>Use Webcam</button>
+                <input type="file" onChange={(e) => {getBase64OfImage(e.target.files[0], (data) => setAndStoreAvatar(data))}} className="file-input w-full max-w-xs"/>
+              </>
+            }
+          </div>
+          <label className="form-control w-full">
             <div className="label">
               <span className="label-text">What is your name?</span>
             </div>
-            <input type="text" ref={inputName} defaultValue={name} placeholder="Name" className="input input-bordered w-full max-w-xs" />
+            <div className="flex flex-row w-full space-x-4">
+              <input type="text" ref={inputName} defaultValue={name} placeholder="Name" className="input input-bordered w-full" />
+              <button onClick={() => {inputName.current.value = getUsername()}} ><IconDice6 /></button>
+            </div>
           </label>
           <div className="modal-action">
             <form method="dialog">
               {/* if there is a button in form, it will close the modal */}
-              <button onClick={setUsername} className="btn">Close</button>
+              <button onClick={setUsername} className="btn">Save</button>
             </form>
           </div>
         </div>
       </dialog>
       <h1 className="text-3xl">Bazaar</h1>
       <div className="flex flex-col space-y-4 items-center">
+        <div className="avatar">
+          <div className="w-24 rounded">
+            <img src={avatar} />
+          </div>
+        </div>
         <div className="flex flex-row space-x-4">
           <span className="text-xl">{name}</span>
           <button onClick={() => { modal.current.showModal()}}>
diff --git a/app/usernameGenerate.ts b/app/usernameGenerate.ts
new file mode 100644
index 0000000..80e0d96
--- /dev/null
+++ b/app/usernameGenerate.ts
@@ -0,0 +1,25 @@
+const prenom = [
+  "Stéphane",
+  "Hubert",
+  "Alphonse",
+  "Marjolaine",
+  "Cathy",
+  "Patrick",
+  "Agathe",
+  "Erneste",
+]
+
+const nom = [
+  "Falzar",
+  "Globule",
+  "Bermuda",
+  "Zigounette",
+  "Pissenlit",
+  "Détritus",
+]
+
+export function getUsername(){
+  const prenomIndex = Math.floor(Math.random() * prenom.length)
+  const nomIndex = Math.floor(Math.random() * nom.length)
+  return prenom[prenomIndex] + " " + nom[nomIndex]
+}
diff --git a/package-lock.json b/package-lock.json
index 075fc7f..ab0c254 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -12,6 +12,7 @@
         "next": "14.2.5",
         "react": "^18",
         "react-dom": "^18",
+        "react-webcam": "^7.2.0",
         "socket.io": "^4.7.5",
         "socket.io-client": "^4.7.5"
       },
@@ -4272,6 +4273,16 @@
       "dev": true,
       "license": "MIT"
     },
+    "node_modules/react-webcam": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/react-webcam/-/react-webcam-7.2.0.tgz",
+      "integrity": "sha512-xkrzYPqa1ag2DP+2Q/kLKBmCIfEx49bVdgCCCcZf88oF+0NPEbkwYk3/s/C7Zy0mhM8k+hpdNkBLzxg8H0aWcg==",
+      "license": "MIT",
+      "peerDependencies": {
+        "react": ">=16.2.0",
+        "react-dom": ">=16.2.0"
+      }
+    },
     "node_modules/read-cache": {
       "version": "1.0.0",
       "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
diff --git a/package.json b/package.json
index e9658ec..45e0f6a 100644
--- a/package.json
+++ b/package.json
@@ -14,6 +14,7 @@
     "next": "14.2.5",
     "react": "^18",
     "react-dom": "^18",
+    "react-webcam": "^7.2.0",
     "socket.io": "^4.7.5",
     "socket.io-client": "^4.7.5"
   },
diff --git a/server.mjs b/server.mjs
index 1cfa3a5..d98f06a 100644
--- a/server.mjs
+++ b/server.mjs
@@ -18,18 +18,19 @@ app.prepare().then(() => {
 
   io.on("connection", (socket) => {
     console.log("User connected " + socket.id)
-    socket.on('room_connect', (room) => {
-      if(!Object.keys(active_rooms).includes(room.id)){
-        console.log("First person joined " + room.id + " ! " + room.name + " is owner.")
-        active_rooms[room.id] = [{id: socket.id, name: room.name, role: "owner", vote: ""}]
-        socket.emit("room_joined", {"room_users": active_rooms[room.id], "role": "owner"})
+
+    socket.on('room_connect', (params) => {
+      if(!Object.keys(active_rooms).includes(params.id)){
+        console.log("First person joined " + params.id + " ! " + params.name + " is owner.")
+        active_rooms[params.id] = [{id: socket.id, name: params.name, avatar: params.avatar, role: "owner", vote: ""}]
+        socket.emit("room_joined", {"room_users": active_rooms[params.id], "role": "owner"})
       } else {
-        socket.to(room.id).emit("new_player",{"id": socket.id, "name": room.name, role: "player"})
-        active_rooms[room.id].push({id: socket.id, name: room.name, role: "player", vote: ""})
-        socket.emit("room_joined", {"room_users": active_rooms[room.id], role: "player"})
-        console.log("New person joined " + room.id + " ! " + room.name + " is player.")
+        socket.to(params.id).emit("new_player",{"id": socket.id, "name": params.name, avatar: params.avatar, role: "player"})
+        active_rooms[params.id].push({id: socket.id, name: params.name, avatar: params.avatar, role: "player", vote: ""})
+        socket.emit("room_joined", {"room_users": active_rooms[params.id], role: "player"})
+        console.log("New person joined " + params.id + " ! " + params.name + " is player.")
       }
-      socket.join(room.id)
+      socket.join(params.id)
     })
 
     socket.on("start_game", (params) => {