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) => {