diff --git a/.gitignore b/.gitignore index d35bbf7..1489c98 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,3 @@ node_modules/ -.idea/ \ No newline at end of file +.idea/ +.build/ diff --git a/package-lock.json b/package-lock.json index 7df5bd6..2b943ee 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13,12 +13,12 @@ "@testing-library/react": "^13.4.0", "@testing-library/user-event": "^13.5.0", "axios": "^1.6.5", - "buffer": "^6.0.3", + "date-fns": "^3.6.0", "jwt-decode": "^4.0.0", "react": "^18.2.0", "react-dom": "^18.2.0", - "react-paginate": "^8.2.0", "react-image-picker-editor": "^1.3.3", + "react-paginate": "^8.2.0", "react-redux": "^8.1.3", "react-router-dom": "^6.18.0", "react-scripts": "^5.0.1", @@ -50,9 +50,9 @@ } }, "node_modules/@adobe/css-tools": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.3.1.tgz", - "integrity": "sha512-/62yikz7NLScCGAAST5SHdnjaDJQBDq0M2muyRTpf2VQhw6StBg2ALiu73zSJQ4fMVLA+0uBhBHAle7Wg+2kSg==" + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.3.3.tgz", + "integrity": "sha512-rE0Pygv0sEZ4vBWHlAgJLGDU7Pm8xoO6p3wsEceb7GYAjScrOHpEo8KK/eVkAcnSM+slAEtXjA2JpdjLp4fJQQ==" }, "node_modules/@alloc/quick-lru": { "version": "5.2.0", @@ -6445,6 +6445,7 @@ "version": "1.5.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true, "funding": [ { "type": "github", @@ -6517,12 +6518,12 @@ "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" }, "node_modules/body-parser": { - "version": "1.20.1", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz", - "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==", + "version": "1.20.2", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.2.tgz", + "integrity": "sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==", "dependencies": { "bytes": "3.1.2", - "content-type": "~1.0.4", + "content-type": "~1.0.5", "debug": "2.6.9", "depd": "2.0.0", "destroy": "1.2.0", @@ -6530,7 +6531,7 @@ "iconv-lite": "0.4.24", "on-finished": "2.4.1", "qs": "6.11.0", - "raw-body": "2.5.1", + "raw-body": "2.5.2", "type-is": "~1.6.18", "unpipe": "1.0.0" }, @@ -6651,29 +6652,6 @@ "node-int64": "^0.4.0" } }, - "node_modules/buffer": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz", - "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "dependencies": { - "base64-js": "^1.3.1", - "ieee754": "^1.2.1" - } - }, "node_modules/buffer-crc32": { "version": "0.2.13", "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", @@ -7250,9 +7228,9 @@ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==" }, "node_modules/cookie": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", - "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==", + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.6.0.tgz", + "integrity": "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==", "engines": { "node": ">= 0.6" } @@ -8122,6 +8100,15 @@ "node": ">=10" } }, + "node_modules/date-fns": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-3.6.0.tgz", + "integrity": "sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/kossnocorp" + } + }, "node_modules/dayjs": { "version": "1.11.10", "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.10.tgz", @@ -8560,9 +8547,9 @@ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" }, "node_modules/ejs": { - "version": "3.1.9", - "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.9.tgz", - "integrity": "sha512-rC+QVNMJWv+MtPgkt0y+0rVEIdbtxVADApW9JXrUVlzHetgcyczP/E7DJmWJ4fJCZF2cPcBk0laWO9ZHMG3DmQ==", + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz", + "integrity": "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==", "dependencies": { "jake": "^10.8.5" }, @@ -9608,16 +9595,16 @@ } }, "node_modules/express": { - "version": "4.18.2", - "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz", - "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==", + "version": "4.19.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.19.2.tgz", + "integrity": "sha512-5T6nhjsT+EOMzuck8JjBHARTHfMht0POzlA60WV2pMD3gyXw2LZnZ+ueGdNxG+0calOJcWKbpFcuzLZ91YWq9Q==", "dependencies": { "accepts": "~1.3.8", "array-flatten": "1.1.1", - "body-parser": "1.20.1", + "body-parser": "1.20.2", "content-disposition": "0.5.4", "content-type": "~1.0.4", - "cookie": "0.5.0", + "cookie": "0.6.0", "cookie-signature": "1.0.6", "debug": "2.6.9", "depd": "2.0.0", @@ -9986,9 +9973,9 @@ "integrity": "sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ==" }, "node_modules/follow-redirects": { - "version": "1.15.5", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.5.tgz", - "integrity": "sha512-vSFWUON1B+yAw1VN4xMfxgn5fTUiaOzAJCKBwIIgT/+7CuGy9+r+5gITvP62j3RmaD5Ph65UaERdOSRGUzZtgw==", + "version": "1.15.6", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz", + "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==", "funding": [ { "type": "individual", @@ -11006,6 +10993,7 @@ "version": "1.2.1", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "dev": true, "funding": [ { "type": "github", @@ -16861,9 +16849,9 @@ } }, "node_modules/raw-body": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", - "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", + "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", "dependencies": { "bytes": "3.1.2", "http-errors": "2.0.0", @@ -20236,9 +20224,9 @@ } }, "node_modules/webpack-dev-middleware": { - "version": "5.3.3", - "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-5.3.3.tgz", - "integrity": "sha512-hj5CYrY0bZLB+eTO+x/j67Pkrquiy7kWepMHmUMoPsmcUaeEnQJqFzHJOyxgWlq746/wUuA64p9ta34Kyb01pA==", + "version": "5.3.4", + "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-5.3.4.tgz", + "integrity": "sha512-BVdTqhhs+0IfoeAf7EoH5WE+exCmqGerHfDM0IL096Px60Tq2Mn9MAbnaGUe6HiMa41KMCYF19gyzZmBcq/o4Q==", "dependencies": { "colorette": "^2.0.10", "memfs": "^3.4.3", diff --git a/package.json b/package.json index 6b41de6..224eea4 100644 --- a/package.json +++ b/package.json @@ -7,11 +7,12 @@ "@testing-library/react": "^13.4.0", "@testing-library/user-event": "^13.5.0", "axios": "^1.6.5", + "date-fns": "^3.6.0", "jwt-decode": "^4.0.0", "react": "^18.2.0", "react-dom": "^18.2.0", - "react-paginate": "^8.2.0", "react-image-picker-editor": "^1.3.3", + "react-paginate": "^8.2.0", "react-redux": "^8.1.3", "react-router-dom": "^6.18.0", "react-scripts": "^5.0.1", diff --git a/src/Games/GameDetails.js b/src/Games/GameDetails.js new file mode 100644 index 0000000..421d185 --- /dev/null +++ b/src/Games/GameDetails.js @@ -0,0 +1,100 @@ +import { useEffect, useState } from "react"; +import {GameService} from "../services/GameService"; +import { useParams } from 'react-router-dom'; +import { format } from 'date-fns'; +import './GameDetails.scss' + +function GameDetails() { + + const { gameId } = useParams(); + const [game, setGame] = useState(''); + const [message, setMessage] = useState(''); + const [date,setDate] = useState([]); + const [file, setFile] = useState(); + + useEffect(() => { + const fetchGameData = async () => { + let gr + try { + gr = await GameService.getGame(gameId); + setGame(gr.data.data); + } catch (e) { + setMessage('There was a problem with fetching match data.'); + } + try { + console.log(game.gameFileName) + const date = formatTimestamp(gr.data.data.lastModification); + setDate(date); + console.log() + } catch (e) { + console.log('There was a problem with fetching date.'); + } + try { + const gf = await GameService.getGameFile(gr.data.data.fileId); + setFile(gf.data.data); + } catch (e) { + console.log('There was a problem with fetching files.'); + } + }; + + fetchGameData(); + },[gameId, game.gameFileName]); + + function formatTimestamp(timestamp) { + try { + const normalizedTimestamp = timestamp.replace(/\.\d{3,}/, match => match.slice(0, 4)); + const date = new Date(normalizedTimestamp); + + if (isNaN(date.getTime())) { + throw new Error("Invalid date"); + } + + return [format(date, 'yyyy-MM-dd'), format(date, 'HH:mm:ss')]; + } catch (error) { + console.error("Error formatting timestamp:", error); + return ["", ""]; + } + } + console.log(file) + return ( + message.length === 0 ? + <> +
+

{game.gameFileName}

+
+

{game.interfaceDefinition}

+
+
+

{game.gameInstructions}

+
+
+
+

modification

+

{date[0]}

+

{date[1]}

+
+
+

players

+

{game.numbersOfPlayer}

+
+
+

available

+

{game.isAvailableForPlay ? "Yes" : "no"}

+
+
+
+ {file ? 'Å‚uuuuuuuuuuuu': 'no file'} +
+
+ :<> +
+

{message}

+
+ + ); +} + +export default GameDetails; + + + diff --git a/src/Games/GameDetails.scss b/src/Games/GameDetails.scss new file mode 100644 index 0000000..4e10d95 --- /dev/null +++ b/src/Games/GameDetails.scss @@ -0,0 +1,47 @@ +.game-info-container{ + margin:auto; + max-width: 80%; + + .game-info-table{ + display: flex; + flex-direction: row; + flex-wrap: wrap; + height: 100%; + justify-content: space-between; + } + + .article-content{ + border: 3px solid black; + padding: 1.5% 5%; + margin: 20px; + } + + .bold-text{ + font-weight: bold; + font-size: 0.9em; + font-family: Arial, Helvetica, sans-serif; + text-shadow: none; + letter-spacing: 1px; + color: rgb(57, 185, 31); + } + + .game-info-col{ + span > p{ + margin-bottom: 30px; + } + p{ + text-align: center; + } + } + +} + +.error-message{ + display:flex; + align-items: center; + flex-direction: column; + justify-content: center; + padding-top: 23% ; + margin-top:auto; + margin-bottom:auto; +} diff --git a/src/Tournaments/TournamentNav.js b/src/Tournaments/TournamentNav.js index 84b922b..bd3c836 100644 --- a/src/Tournaments/TournamentNav.js +++ b/src/Tournaments/TournamentNav.js @@ -1,6 +1,6 @@ import './TournamentNav.scss'; import React from "react"; -import { NavLink, Link } from 'react-router-dom'; +import { NavLink } from 'react-router-dom'; import { login, logout } from '../User/store'; import { connect } from 'react-redux'; diff --git a/src/User/ProfileView/ProfileView.css b/src/User/ProfileView/ProfileView.css index 4a45203..cda01b0 100644 --- a/src/User/ProfileView/ProfileView.css +++ b/src/User/ProfileView/ProfileView.css @@ -1,4 +1,3 @@ - .main-container { display: grid; grid-template-columns: 20% 80%; @@ -103,4 +102,4 @@ p { button { width: 100%; padding: 10px; -} +} \ No newline at end of file diff --git a/src/User/Settings/SettingPopup.js b/src/User/Settings/SettingPopup.js index b7b3685..3bc4e2f 100644 --- a/src/User/Settings/SettingPopup.js +++ b/src/User/Settings/SettingPopup.js @@ -29,7 +29,7 @@ export default function SettingPopup(props) { useEffect(() => { setContent(getContent(state)) - }, [state]) + }, [state, getContent]) if (typeof triggername !== 'string') { console.error('triggername is not a string'); diff --git a/src/index.js b/src/index.js index 9ce2e5e..16b9b21 100644 --- a/src/index.js +++ b/src/index.js @@ -8,6 +8,7 @@ import { RouterProvider, } from "react-router-dom"; import GamesList from "./lists/GamesList"; +import GameDetails from "./Games/GameDetails"; import AddGameForm from "./forms/AddGameForm"; import RegisterForm from "./forms/RegisterForm"; import LoginForm from "./forms/LoginForm"; @@ -47,6 +48,10 @@ const router = createBrowserRouter([ path: "/games", element: , }, + { + path: "/games/details/:gameId", + element: , + }, { path: "/games/add", element: , diff --git a/src/lists/GamesList.js b/src/lists/GamesList.js index aa82a09..c12f123 100644 --- a/src/lists/GamesList.js +++ b/src/lists/GamesList.js @@ -1,15 +1,20 @@ import './List.scss' import DeleteGameButton from './DeleteGameButton'; -import { Link } from 'react-router-dom'; +import { Link, useNavigate } from 'react-router-dom'; import { connect } from 'react-redux'; import { login, logout } from '../User/store'; import React, { useState, useEffect } from "react"; import {GameService} from "../services/GameService"; function GamesList({games, isAuthenticated, user, login, logout }) { + const navigate = useNavigate(); const [gamesList, setGamesList] = useState([]); const [message, setMessage] = useState(''); + const handleGameClick = (gameId) => { + navigate(`/games/details/${gameId}`); + }; + useEffect(() => { const fetchGameData = async () => { try { @@ -35,7 +40,7 @@ function GamesList({games, isAuthenticated, user, login, logout }) { )} {gamesList.map((game, index) => (
- + {isAuthenticated && user.role === 'Admin' && ( )} diff --git a/src/services/GameService.js b/src/services/GameService.js index 9b2b71f..a1a0624 100644 --- a/src/services/GameService.js +++ b/src/services/GameService.js @@ -40,7 +40,7 @@ export const GameService = { return await Api.req(() => {return Api.get(`GameType/getAvailable?page=${page}&pagesize=${pagesize}`)}) }, getGame: async function (id) { - return await Api.req(() => {return Api.get(`GameType/get?id=${id}`)}) + return await Api.req(() => {return Api.get(`GameType/getOne?id=${id}`)}) }, getByName: async function (name, page, pagesize) { return await Api.req(() => {return Api.get(`GameType/getByName?name=${name}&page=${page}&pagesize=${pagesize}`)}) @@ -55,4 +55,7 @@ export const GameService = { isAvaiableForPlay: isAvaiableForPlay })}) }, + getGameFile: async function (id) { + return await Api.req(() => {return Api.post(`GameType/getFile?id=${id}`)}) + }, } \ No newline at end of file