Compare commits
	
		
			2 Commits
		
	
	
		
			master
			...
			feat/DayOf
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a8dc6c317d | |||
| cfcbd7a68b | 
							
								
								
									
										23
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										23
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +1,23 @@ | ||||
| # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. | ||||
| 
 | ||||
| # dependencies | ||||
| node_modules | ||||
| types/gen | ||||
| /.pnp | ||||
| .pnp.js | ||||
| 
 | ||||
| # testing | ||||
| /coverage | ||||
| 
 | ||||
| # production | ||||
| /build | ||||
| 
 | ||||
| # misc | ||||
| .DS_Store | ||||
| .env.local | ||||
| .env.development.local | ||||
| .env.test.local | ||||
| .env.production.local | ||||
| 
 | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
|  | ||||
| @ -1,64 +0,0 @@ | ||||
| variables: | ||||
|   - &node_image "node:22-alpine" | ||||
|   - &branch "master" | ||||
| 
 | ||||
| when: | ||||
|   - event: push | ||||
|     branch: *branch | ||||
| 
 | ||||
| steps: | ||||
|   - name: Generate TypeScript types | ||||
|     image: *node_image | ||||
|     commands: | ||||
|       - cd types | ||||
|       - yarn install --frozen-lockfile | ||||
|       - yarn openapi-ts | ||||
|   - name: Install server dependencies | ||||
|     image: *node_image | ||||
|     commands: | ||||
|       - cd server | ||||
|       - yarn install --frozen-lockfile | ||||
|     depends_on: [Generate TypeScript types] | ||||
|   - name: Install client dependencies | ||||
|     image: *node_image | ||||
|     commands: | ||||
|       - cd client | ||||
|       - yarn install --frozen-lockfile | ||||
|     depends_on: [Generate TypeScript types] | ||||
|   - name: Build server | ||||
|     depends_on: [Install server dependencies] | ||||
|     image: *node_image | ||||
|     commands: | ||||
|       - cd server | ||||
|       - yarn build | ||||
|   - name: Build client | ||||
|     depends_on: [Install client dependencies] | ||||
|     image: *node_image | ||||
|     commands: | ||||
|       - cd client | ||||
|       - yarn build | ||||
|   - name: Build Docker image | ||||
|     depends_on: [Build server, Build client] | ||||
|     image: woodpeckerci/plugin-docker-buildx | ||||
|     settings: | ||||
|       dockerfile: Dockerfile-Woodpecker | ||||
|       platforms: linux/amd64 | ||||
|       registry: | ||||
|         from_secret: REPO_URL | ||||
|       username: | ||||
|         from_secret: REPO_USERNAME | ||||
|       password: | ||||
|         from_secret: REPO_PASSWORD | ||||
|       repo: | ||||
|         from_secret: REPO_NAME | ||||
|   - name: Discord notification - build | ||||
|     image: appleboy/drone-discord | ||||
|     depends_on: [Build Docker image] | ||||
|     when: | ||||
|       - status: [success, failure] | ||||
|     settings: | ||||
|       webhook_id: | ||||
|         from_secret: DISCORD_WEBHOOK_ID | ||||
|       webhook_token: | ||||
|         from_secret: DISCORD_WEBHOOK_TOKEN | ||||
|       message: "{{#success build.status}}✅ Sestavení {{build.number}} proběhlo úspěšně.{{else}}❌ Sestavení {{build.number}} selhalo.{{/success}}\n\nPipeline: {{build.link}}\nPoslední commit: {{commit.message}}Autor: {{commit.author}}" | ||||
							
								
								
									
										40
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -1,18 +1,8 @@ | ||||
| ARG NODE_VERSION="node:22-alpine" | ||||
| 
 | ||||
| # Builder | ||||
| FROM ${NODE_VERSION} AS builder | ||||
| FROM node:18-alpine3.18 AS builder | ||||
| 
 | ||||
| WORKDIR /build | ||||
| 
 | ||||
| # Zkopírování závislostí - OpenAPI generátor | ||||
| COPY types/package.json ./types/ | ||||
| COPY types/yarn.lock ./types/ | ||||
| COPY types/api.yml ./types/ | ||||
| COPY types/schemas ./types/schemas/ | ||||
| COPY types/paths ./types/paths/ | ||||
| COPY types/openapi-ts.config.ts ./types/ | ||||
| 
 | ||||
| # Zkopírování závislostí - server | ||||
| COPY server/package.json ./server/ | ||||
| COPY server/yarn.lock ./server/ | ||||
| @ -21,10 +11,6 @@ COPY server/yarn.lock ./server/ | ||||
| COPY client/package.json ./client/ | ||||
| COPY client/yarn.lock ./client/ | ||||
| 
 | ||||
| # Instalace závislostí - OpenAPI generátor | ||||
| WORKDIR /build/types | ||||
| RUN yarn install --frozen-lockfile | ||||
| 
 | ||||
| # Instalace závislostí - server | ||||
| WORKDIR /build/server | ||||
| RUN yarn install --frozen-lockfile | ||||
| @ -48,11 +34,7 @@ COPY client/src ./client/src | ||||
| COPY client/public ./client/public | ||||
| 
 | ||||
| # Zkopírování společných typů | ||||
| COPY types/index.ts ./types/ | ||||
| 
 | ||||
| # Vygenerování společných typů z OpenAPI | ||||
| WORKDIR /build/types | ||||
| RUN yarn openapi-ts | ||||
| COPY types ./types/ | ||||
| 
 | ||||
| # Sestavení serveru | ||||
| WORKDIR /build/server | ||||
| @ -63,30 +45,26 @@ WORKDIR /build/client | ||||
| RUN yarn build | ||||
| 
 | ||||
| # Runner | ||||
| FROM ${NODE_VERSION} | ||||
| 
 | ||||
| RUN apk add --no-cache tzdata | ||||
| ENV TZ=Europe/Prague \ | ||||
|     LC_ALL=cs_CZ.UTF-8 \ | ||||
|     NODE_ENV=production | ||||
| FROM node:18-alpine3.18 | ||||
| ENV LANG cs_CZ.UTF-8 | ||||
| ENV NODE_ENV production | ||||
| 
 | ||||
| WORKDIR /app | ||||
| 
 | ||||
| # Vykopírování sestaveného serveru | ||||
| COPY --from=builder /build/server/node_modules ./server/node_modules | ||||
| COPY --from=builder /build/server/dist ./ | ||||
| COPY server/resources ./server/resources | ||||
| 
 | ||||
| # Vykopírování sestaveného klienta | ||||
| COPY --from=builder /build/client/dist ./public | ||||
| 
 | ||||
| # Zkopírování produkčních .env serveru | ||||
| COPY /server/.env.production ./server | ||||
| COPY /server/.env.production ./server/src | ||||
| 
 | ||||
| # Zkopírování konfigurace easter eggů | ||||
| RUN if [ -f /server/.easter-eggs.json ]; then cp /server/.easter-eggs.json ./server/; fi | ||||
| 
 | ||||
| # Export /data/db.json do složky /data | ||||
| VOLUME ["/data"] | ||||
| # TODO tohle spadne když nebude existovat! | ||||
| COPY /server/.easter-eggs.json ./server/ | ||||
| 
 | ||||
| EXPOSE 3000 | ||||
| 
 | ||||
|  | ||||
| @ -1,26 +0,0 @@ | ||||
| ARG NODE_VERSION="node:22-alpine" | ||||
| 
 | ||||
| FROM ${NODE_VERSION} | ||||
| 
 | ||||
| RUN apk add --no-cache tzdata | ||||
| ENV TZ=Europe/Prague \ | ||||
|     LC_ALL=cs_CZ.UTF-8 \ | ||||
|     NODE_ENV=production | ||||
| 
 | ||||
| WORKDIR /app | ||||
| 
 | ||||
| # Vykopírování sestaveného serveru | ||||
| COPY ./server/node_modules ./server/node_modules | ||||
| COPY ./server/dist ./ | ||||
| # TODO tohle není dobře, má to být součástí serveru | ||||
| # COPY ./server/resources ./resources | ||||
| 
 | ||||
| # Vykopírování sestaveného klienta | ||||
| COPY ./client/dist ./public | ||||
| 
 | ||||
| # Zkopírování konfigurace easter eggů | ||||
| RUN if [ -f ./server/.easter-eggs.json ]; then cp ./server/.easter-eggs.json ./server/; fi | ||||
| 
 | ||||
| EXPOSE 3000 | ||||
| 
 | ||||
| CMD [ "node", "./server/src/index.js" ] | ||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @ -1,9 +1,7 @@ | ||||
| # Luncher | ||||
| Aplikace pro profesionální management obědů. | ||||
| 
 | ||||
| Aplikace sestává ze tří modulů. | ||||
| - types | ||||
|   - OpenAPI definice společných typů, generované přes [openapi-ts](https://github.com/hey-api/openapi-ts) | ||||
| Aplikace sestává ze dvou modulů + společných TypeScript definic (adresář `types`). | ||||
| - server | ||||
|   - backend psaný v [node.js](https://nodejs.dev) | ||||
| - client | ||||
| @ -12,27 +10,19 @@ Aplikace sestává ze tří modulů. | ||||
| ## Spuštění pro vývoj | ||||
| ### Závislosti | ||||
| #### Klient/server | ||||
| - [Node.js 22.x (>= 22.11)](https://nodejs.dev) | ||||
| - [Node.js 18.x](https://nodejs.dev) | ||||
| - [Yarn 1.22.x (Classic)](https://classic.yarnpkg.com) | ||||
| 
 | ||||
| ### Spuštění na *nix platformách | ||||
| - Nainstalovat závislosti viz předchozí bod | ||||
| - Zkopírovat `server/.env.template` do `server/.env.development` a upravit dle potřeby | ||||
| - Vygenerovat společné TypeScript typy | ||||
|   - `cd types && yarn install && yarn openapi-ts` | ||||
| - Server | ||||
|   - `cd server && yarn install && export NODE_ENV=development && yarn startReload` | ||||
| - Klient | ||||
|   - `cd client && yarn install && yarn start` | ||||
| - Spustit `./run_dev.sh`. Na jiných platformách se lze inspirovat jeho obsahem, postup by měl být víceméně stejný. | ||||
| 
 | ||||
| ## Sestavení a spuštění produkční verze v Docker | ||||
| ### Závislosti | ||||
| - [Docker](https://www.docker.com) | ||||
| - [Docker Compose](https://docs.docker.com/compose) | ||||
| 
 | ||||
| ### Spuštění | ||||
| - `docker compose up --build -d` | ||||
| 
 | ||||
| ### Spuštení s traefik | ||||
| - `docker compose -f compose-traefik.yml up --build -d` | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										1
									
								
								client/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								client/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +1,3 @@ | ||||
| build | ||||
| dist | ||||
| src/types | ||||
| @ -6,36 +6,34 @@ | ||||
|   "type": "module", | ||||
|   "homepage": ".", | ||||
|   "dependencies": { | ||||
|     "@fortawesome/fontawesome-svg-core": "^7.1.0", | ||||
|     "@fortawesome/free-regular-svg-icons": "^7.1.0", | ||||
|     "@fortawesome/free-solid-svg-icons": "^7.1.0", | ||||
|     "@fortawesome/react-fontawesome": "^3.1.0", | ||||
|     "@types/jest": "^30.0.0", | ||||
|     "@types/node": "^24.7.1", | ||||
|     "@types/react": "^19.2.2", | ||||
|     "@types/react-dom": "^19.2.1", | ||||
|     "@vitejs/plugin-react": "^5.0.4", | ||||
|     "bootstrap": "^5.3.8", | ||||
|     "react": "^19.2.0", | ||||
|     "react-bootstrap": "^2.10.10", | ||||
|     "react-dom": "^19.2.0", | ||||
|     "react-jwt": "^1.3.0", | ||||
|     "react-modal": "^3.16.3", | ||||
|     "react-router": "^7.9.4", | ||||
|     "react-router-dom": "^7.9.4", | ||||
|     "@fortawesome/fontawesome-svg-core": "^6.4.0", | ||||
|     "@fortawesome/free-regular-svg-icons": "^6.4.0", | ||||
|     "@fortawesome/free-solid-svg-icons": "^6.4.0", | ||||
|     "@fortawesome/react-fontawesome": "^0.2.0", | ||||
|     "@types/jest": "^29.5.12", | ||||
|     "@types/node": "^20.11.20", | ||||
|     "@types/react": "^19.0.0", | ||||
|     "@types/react-dom": "^19.0.0", | ||||
|     "@vitejs/plugin-react": "^4.3.4", | ||||
|     "bootstrap": "^5.2.3", | ||||
|     "react": "^19.0.0", | ||||
|     "react-bootstrap": "^2.7.2", | ||||
|     "react-dom": "^19.0.0", | ||||
|     "react-jwt": "^1.2.0", | ||||
|     "react-modal": "^3.16.1", | ||||
|     "react-select-search": "^4.1.6", | ||||
|     "react-snowfall": "^2.3.0", | ||||
|     "react-toastify": "^11.0.5", | ||||
|     "recharts": "^3.2.1", | ||||
|     "sass": "^1.93.2", | ||||
|     "react-snowfall": "^2.2.0", | ||||
|     "react-toastify": "^10.0.4", | ||||
|     "sass": "^1.80.6", | ||||
|     "socket.io-client": "^4.6.1", | ||||
|     "typescript": "^5.9.3", | ||||
|     "vite": "^7.1.9", | ||||
|     "typescript": "^5.3.3", | ||||
|     "vite": "^6.0.3", | ||||
|     "vite-tsconfig-paths": "^5.1.4" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "start": "yarn vite", | ||||
|     "build": "tsc --noEmit && yarn vite build" | ||||
|     "copy-types": "cp -r ../types ./src", | ||||
|     "start": "yarn copy-types && vite", | ||||
|     "build": "yarn copy-types && vite build" | ||||
|   }, | ||||
|   "eslintConfig": { | ||||
|     "extends": [ | ||||
| @ -56,6 +54,6 @@ | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/plugin-proposal-private-property-in-object": "^7.21.11", | ||||
|     "prettier": "^3.6.2" | ||||
|     "prettier": "^3.2.5" | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#8B4513" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#228B22" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#D2691E" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#B22222" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#FFD700" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -1,22 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||
| 	 width="100px" height="100px" viewBox="1 0 100 100" enable-background="new 1 0 100 100" xml:space="preserve"> | ||||
| <path fill="#471C37" d="M31.5,77.2c0,0,2.6-1.9,3.4-2.8c0.8-0.9,2-2.5,2-2.5s-4.7-0.7-7.9-1.6c-3.2-0.9-9.7-3.7-9.7-3.7 | ||||
| 	s1.3-0.4,2.9-1.4c1.6-1.1,2-1.7,2-1.7s-4.7-1.3-7.5-2.4c-2.8-1-7.3-2.9-7.3-2.9s1.4-0.2,2.8-1.3c1.4-1.1,1.5-2.2,1.5-2.2 | ||||
| 	S9.4,53.3,7,51.9c-2.4-1.5-5.3-3.7-5.3-3.7s5.2-4,8.7-5.5c3.5-1.5,9.1-2.6,9.1-2.6s-0.9-1.5-2.1-2.6c-1.1-1.1-4.2-2.9-4.2-2.9 | ||||
| 	s5.6-1.9,8.3-2.5c2.7-0.6,8.2,0,8.2,0s-0.1-1.1-1.3-2.2c-1.2-1.1-2.7-2-2.7-2S33,25.4,37,24.4c4-1,10.8-0.6,10.8-0.6 | ||||
| 	S46.1,22.1,45,21c-1.1-1.1-2.7-2.3-2.7-2.3s5.4-0.4,9.1-0.4c3.7,0.1,9.5,1.1,9.5,1.1s-0.7-1-1.4-2.3c-0.8-1.3-1.5-2.6-1.5-2.6 | ||||
| 	s7.1,1.7,9.5,2.8c2.4,1,7.6,3.9,7.6,3.9s-0.1-1.8-1-3.5c-0.9-1.7-1.8-3-1.8-3s27.7,17.2,28.1,33.5c0.1,1.8-0.1,3.7-0.3,5.5 | ||||
| 	c-1.5-0.7-3.1-1.3-4.7-1.8c-2.3-6.3-4.3-9.9-6.4-12.6c-2.4-2.9-6.9-8-9.6-9.1c2.3,1.9,6.8,6.2,9.7,11.8c2.3,4.6,3,5.6,3.8,9.2 | ||||
| 	c-3.6-1.1-7.3-1.9-11-2.5c-0.7-0.1-1.4-0.2-2.1-0.3c-2-4.3-4.4-8.8-7.6-11.5c-3.7-3.2-8.3-6.7-11.1-7.5c2.6,1.5,8.7,6.8,11.8,10.9 | ||||
| 	c2.2,3,2.9,5.1,4,7.7c-5-0.6-9.9-0.9-14.9-1.1c-2-3.7-4.5-7.5-7.6-9.6c-3.4-2.4-7.6-5-10-5.5c2.3,1.1,7.9,5.1,10.8,8.3 | ||||
| 	c2.3,2.6,3,4.4,4.2,6.9c-11.1,0.4-22,1.7-32.8,3.6c9.4-0.8,18.9-1,28.3-0.6c-1.1,1.7-1.8,3-3.8,4.7c-2.6,2.2-7.5,4.5-9.5,4.9 | ||||
| 	c2,0.1,5.6-1.3,8.6-2.6c2.7-1.3,5.1-4.1,7.1-6.9c1.8,0.1,3.5,0.2,5.3,0.3c2.9,0.1,5.8,0.3,8.7,0.5c-1.4,3-2,5-4.6,8.1 | ||||
| 	c-3.1,3.6-9.1,8-11.6,9.2c2.7-0.4,7.2-3.3,10.8-5.9c3.3-2.4,6-6.8,8.1-11c2.7,0.3,5.4,0.6,8.1,1.1c2,0.3,3.9,0.7,5.9,1.2 | ||||
| 	c-1.9,4.1-2,4.6-5,8c-3.4,3.9-9.1,8.9-11.6,10.4c2.9-0.7,8.5-4.7,11.1-6.7c2.4-1.8,5.8-5.5,8.2-11.1c3,0.8,5.9,1.9,8.7,3.2 | ||||
| 	c-3.1,12.9-11.9,24.1-11.9,24.1s0.2-1.7-0.5-3.9c-0.8-2.2-1-2-1-2s-1.1,2.9-5,6.8c-3.9,3.9-8.9,5-8.9,5s0.7-1.1,0.8-2.4 | ||||
| 	c0.1-1.3-0.3-2-0.3-2s-2.6,1.9-5.3,3c-2.7,1.2-8.6,2.5-8.6,2.5s1.6-1.9,1.9-3c0.3-1-0.3-1.9-0.3-1.9s-3.9,1.5-7.5,1.5 | ||||
| 	c-3.6,0-7.7-1.5-7.7-1.5s1-0.5,1.7-1.8c0.8-1.3,0.2-1.9,0.2-1.9s-4.4,0.5-7.5,0.1C36.6,79.3,31.5,77.2,31.5,77.2z"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.3 KiB | 
| @ -51,24 +51,8 @@ | ||||
|   font-size: 64px; | ||||
| } | ||||
| 
 | ||||
| /* Sticky footer layout */ | ||||
| html, | ||||
| body, | ||||
| #root { | ||||
|   height: 100%; | ||||
|   margin: 0; | ||||
| } | ||||
| 
 | ||||
| .app-container { | ||||
|   min-height: 100vh; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
| } | ||||
| 
 | ||||
| .wrapper { | ||||
|   padding: 20px; | ||||
|   flex: 1; | ||||
|   /* Zabere zbytek místa */ | ||||
| } | ||||
| 
 | ||||
| .title { | ||||
|  | ||||
| @ -1,27 +1,29 @@ | ||||
| import React, { useContext, useEffect, useMemo, useRef, useState, useCallback } from 'react'; | ||||
| import 'bootstrap/dist/css/bootstrap.min.css'; | ||||
| import { EVENT_DISCONNECT, EVENT_MESSAGE, SocketContext } from './context/socket'; | ||||
| import { addPizza, createPizzaDay, deletePizzaDay, finishDelivery, finishOrder, lockPizzaDay, removePizza, unlockPizzaDay, updatePizzaDayNote } from './api/PizzaDayApi'; | ||||
| import { useAuth } from './context/auth'; | ||||
| import Login from './Login'; | ||||
| import { Alert, Button, Col, Form, Row, Table } from 'react-bootstrap'; | ||||
| import Header from './components/Header'; | ||||
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | ||||
| import PizzaOrderList from './components/PizzaOrderList'; | ||||
| import SelectSearch, { SelectedOptionValue, SelectSearchOption } from 'react-select-search'; | ||||
| import SelectSearch, { SelectedOptionValue } from 'react-select-search'; | ||||
| import 'react-select-search/style.css'; | ||||
| import './App.scss'; | ||||
| import { SelectSearchOption } from 'react-select-search'; | ||||
| import { faCircleCheck, faNoteSticky, faTrashCan } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { useSettings } from './context/settings'; | ||||
| import { ClientData, Restaurants, Food, Order, Locations, PizzaOrder, PizzaDayState, FoodChoices, DayMenu, DepartureTime, LocationKey } from './types'; | ||||
| import Footer from './components/Footer'; | ||||
| import { faChainBroken, faChevronLeft, faChevronRight, faGear, faSatelliteDish, faSearch } from '@fortawesome/free-solid-svg-icons'; | ||||
| import Loader from './components/Loader'; | ||||
| import { getData, errorHandler, getQrUrl } from './api/Api'; | ||||
| import { addChoice, removeChoices, removeChoice, changeDepartureTime, jdemeObed, updateNote } from './api/FoodApi'; | ||||
| import { getHumanDateTime, isInTheFuture } from './Utils'; | ||||
| import NoteModal from './components/modals/NoteModal'; | ||||
| import { useEasterEgg } from './context/eggs'; | ||||
| import { ClientData, Food, PizzaOrder, DepartureTime, PizzaDayState, Restaurant, RestaurantDayMenu, RestaurantDayMenuMap, LunchChoice, UserLunchChoice, PizzaVariant, getData, getEasterEggImage, addPizza, removePizza, updatePizzaDayNote, createPizzaDay, deletePizzaDay, lockPizzaDay, unlockPizzaDay, finishOrder, finishDelivery, addChoice, jdemeObed, removeChoices, removeChoice, updateNote, changeDepartureTime } from '../../types'; | ||||
| import { getLunchChoiceName } from './enums'; | ||||
| import FallingLeaves, { LEAF_PRESETS, LEAF_COLOR_THEMES } from './FallingLeaves'; | ||||
| import './FallingLeaves.scss'; | ||||
| import { getImage } from './api/EasterEggApi'; | ||||
| 
 | ||||
| const EVENT_CONNECT = "connect" | ||||
| 
 | ||||
| @ -32,37 +34,17 @@ const EASTER_EGG_STYLE = { | ||||
|   animationTimingFunction: "ease" | ||||
| } | ||||
| 
 | ||||
| // Mapování čísel alergenů na jejich názvy
 | ||||
| const ALLERGENS: { [key: number]: string } = { | ||||
|   1: "Obiloviny obsahující lepek", | ||||
|   2: "Korýši a výrobky z nich", | ||||
|   3: "Vejce a výrobky z nich", | ||||
|   4: "Ryby a výrobky z nich", | ||||
|   5: "Arašidy a výrobky z nich", | ||||
|   6: "Sója a výrobky z nich", | ||||
|   7: "Mléko a výrobky z nich (včetně laktózy)", | ||||
|   8: "Skořápkové plody", | ||||
|   9: "Celer a výrobky z něj", | ||||
|   10: "Hořčice a výrobky z ní", | ||||
|   11: "Sezamová semena a výrobky z nich", | ||||
|   12: "Oxid siřičitý a siřičitany", | ||||
|   13: "Vlčí bob (Lupina) a výrobky z něj", | ||||
|   14: "Měkkýši a výrobky z nich" | ||||
| } | ||||
| 
 | ||||
| const LINK_ALLERGENS = 'https://www.strava.cz/Strava/Napoveda/cz/Prilohy/alergeny.pdf'; | ||||
| 
 | ||||
| // Výchozí doba trvání animace v sekundách, pokud není přetíženo v konfiguračním JSONu
 | ||||
| const EASTER_EGG_DEFAULT_DURATION = 0.75; | ||||
| 
 | ||||
| function App() { | ||||
|   const auth = useAuth(); | ||||
|   const settings = useSettings(); | ||||
|   const [easterEgg, _] = useEasterEgg(auth); | ||||
|   const [easterEgg, easterEggLoading] = useEasterEgg(auth); | ||||
|   const [isConnected, setIsConnected] = useState<boolean>(false); | ||||
|   const [data, setData] = useState<ClientData>(); | ||||
|   const [food, setFood] = useState<RestaurantDayMenuMap>(); | ||||
|   const [myOrder, setMyOrder] = useState<PizzaOrder>(); | ||||
|   const [food, setFood] = useState<{ [key in Restaurants]?: DayMenu }>(); | ||||
|   const [myOrder, setMyOrder] = useState<Order>(); | ||||
|   const [foodChoiceList, setFoodChoiceList] = useState<Food[]>(); | ||||
|   const [closed, setClosed] = useState<boolean>(false); | ||||
|   const socket = useContext(SocketContext); | ||||
| @ -82,17 +64,14 @@ function App() { | ||||
| 
 | ||||
|   // Načtení dat po přihlášení
 | ||||
|   useEffect(() => { | ||||
|     if (!auth?.login) { | ||||
|     if (!auth || !auth.login) { | ||||
|       return | ||||
|     } | ||||
|     getData().then(response => { | ||||
|       const data = response.data | ||||
|       if (data) { | ||||
|     getData().then((data: ClientData) => { | ||||
|       setData(data); | ||||
|         setDayIndex(data.dayIndex); | ||||
|         dayIndexRef.current = data.dayIndex; | ||||
|       setDayIndex(data.weekIndex); | ||||
|       dayIndexRef.current = data.weekIndex; | ||||
|       setFood(data.menus); | ||||
|       } | ||||
|     }).catch(e => { | ||||
|       setFailure(true); | ||||
|     }) | ||||
| @ -100,15 +79,12 @@ function App() { | ||||
| 
 | ||||
|   // Přenačtení pro zvolený den
 | ||||
|   useEffect(() => { | ||||
|     if (!auth?.login) { | ||||
|     if (!auth || !auth.login) { | ||||
|       return | ||||
|     } | ||||
|     getData({ query: { dayIndex: dayIndex } }).then(response => { | ||||
|       const data = response.data; | ||||
|     getData(dayIndex).then((data: ClientData) => { | ||||
|       setData(data); | ||||
|       if (data) { | ||||
|       setFood(data.menus); | ||||
|       } | ||||
|     }).catch(e => { | ||||
|       setFailure(true); | ||||
|     }) | ||||
| @ -117,15 +93,17 @@ function App() { | ||||
|   // Registrace socket eventů
 | ||||
|   useEffect(() => { | ||||
|     socket.on(EVENT_CONNECT, () => { | ||||
|       // console.log("Connected!");
 | ||||
|       setIsConnected(true); | ||||
|     }); | ||||
|     socket.on(EVENT_DISCONNECT, () => { | ||||
|       // console.log("Disconnected!");
 | ||||
|       setIsConnected(false); | ||||
|     }); | ||||
|     socket.on(EVENT_MESSAGE, (newData: ClientData) => { | ||||
|       // console.log("Přijata nová data ze socketu", newData);
 | ||||
|       // Aktualizujeme pouze, pokud jsme dostali data pro den, který máme aktuálně zobrazený
 | ||||
|       if (dayIndexRef.current == null || newData.dayIndex === dayIndexRef.current) { | ||||
|       if (dayIndexRef.current == null || newData.weekIndex === dayIndexRef.current) { | ||||
|         setData(newData); | ||||
|       } | ||||
|     }); | ||||
| @ -138,7 +116,7 @@ function App() { | ||||
|   }, [socket]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!auth?.login) { | ||||
|     if (!auth || !auth.login) { | ||||
|       return | ||||
|     } | ||||
|     // TODO tohle občas náhodně nezafunguje, nutno přepsat, viz https://medium.com/@teh_builder/ref-objects-inside-useeffect-hooks-eb7c15198780
 | ||||
| @ -163,10 +141,10 @@ function App() { | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (choiceRef?.current?.value && choiceRef.current.value !== "") { | ||||
|       const locationKey = choiceRef.current.value as LunchChoice; | ||||
|       const restaurantKey = Object.keys(Restaurant).indexOf(locationKey); | ||||
|       const locationKey = choiceRef.current.value as LocationKey; | ||||
|       const restaurantKey = Object.keys(Restaurants).indexOf(locationKey); | ||||
|       if (restaurantKey > -1 && food) { | ||||
|         const restaurant = Object.keys(Restaurant)[restaurantKey] as Restaurant; | ||||
|         const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||
|         setFoodChoiceList(food[restaurant]?.food); | ||||
|         setClosed(food[restaurant]?.closed ?? false); | ||||
|       } else { | ||||
| @ -198,9 +176,9 @@ function App() { | ||||
|   // Stažení a nastavení easter egg obrázku
 | ||||
|   useEffect(() => { | ||||
|     if (auth?.login && easterEgg?.url && !eggImage) { | ||||
|       getEasterEggImage({ path: { url: easterEgg.url } }).then(response => { | ||||
|         if (response.data) { | ||||
|           setEggImage(response.data); | ||||
|       getImage(easterEgg.url).then(data => { | ||||
|         if (data) { | ||||
|           setEggImage(data); | ||||
|           // Smazání obrázku z DOMu po animaci
 | ||||
|           setTimeout(() => { | ||||
|             if (eggRef?.current) { | ||||
| @ -212,18 +190,10 @@ function App() { | ||||
|     } | ||||
|   }, [auth?.login, easterEgg?.duration, easterEgg?.url, eggImage]); | ||||
| 
 | ||||
|   const doAddClickFoodChoice = async (location: LunchChoice, foodIndex?: number) => { | ||||
|     if (document.getSelection()?.type !== 'Range') { // pouze pokud se nejedná o výběr textu
 | ||||
|       if (auth?.login) { | ||||
|         await addChoice({ body: { locationKey: location, foodIndex, dayIndex } }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const doAddChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||
|     const locationKey = event.target.value as LunchChoice; | ||||
|     const locationKey = event.target.value as LocationKey; | ||||
|     if (auth?.login) { | ||||
|       await addChoice({ body: { locationKey, dayIndex } }); | ||||
|       await errorHandler(() => addChoice(locationKey, undefined, dayIndex)); | ||||
|       if (foodChoiceRef.current?.value) { | ||||
|         foodChoiceRef.current.value = ""; | ||||
|       } | ||||
| @ -238,16 +208,16 @@ function App() { | ||||
| 
 | ||||
|   const doAddFoodChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||
|     if (event.target.value && foodChoiceList?.length && choiceRef.current?.value) { | ||||
|       const locationKey = choiceRef.current.value as LunchChoice; | ||||
|       const locationKey = choiceRef.current.value as LocationKey; | ||||
|       if (auth?.login) { | ||||
|         await addChoice({ body: { locationKey, foodIndex: Number(event.target.value), dayIndex } }); | ||||
|         await errorHandler(() => addChoice(locationKey, Number(event.target.value), dayIndex)); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const doRemoveChoices = async (locationKey: LunchChoice) => { | ||||
|   const doRemoveChoices = async (locationKey: LocationKey) => { | ||||
|     if (auth?.login) { | ||||
|       await removeChoices({ body: { locationKey, dayIndex } }); | ||||
|       await errorHandler(() => removeChoices(locationKey, dayIndex)); | ||||
|       // Vyresetujeme výběr, aby bylo jasné pro který případně vybíráme jídlo
 | ||||
|       if (choiceRef?.current?.value) { | ||||
|         choiceRef.current.value = ""; | ||||
| @ -258,9 +228,9 @@ function App() { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const doRemoveFoodChoice = async (locationKey: LunchChoice, foodIndex: number) => { | ||||
|   const doRemoveFoodChoice = async (locationKey: LocationKey, foodIndex: number) => { | ||||
|     if (auth?.login) { | ||||
|       await removeChoice({ body: { locationKey, foodIndex, dayIndex } }); | ||||
|       await errorHandler(() => removeChoice(locationKey, foodIndex, dayIndex)); | ||||
|       if (choiceRef?.current?.value) { | ||||
|         choiceRef.current.value = ""; | ||||
|       } | ||||
| @ -272,7 +242,7 @@ function App() { | ||||
| 
 | ||||
|   const saveNote = async (note?: string) => { | ||||
|     if (auth?.login) { | ||||
|       await updateNote({ body: { note, dayIndex } }); | ||||
|       await errorHandler(() => updateNote(note, dayIndex)); | ||||
|       setNoteModalOpen(false); | ||||
|     } | ||||
|   } | ||||
| @ -296,18 +266,18 @@ function App() { | ||||
| 
 | ||||
|   const handlePizzaChange = async (value: SelectedOptionValue | SelectedOptionValue[]) => { | ||||
|     if (auth?.login && data?.pizzaList) { | ||||
|       if (typeof value !== 'string') { | ||||
|       if (!(typeof value === 'string')) { | ||||
|         throw Error('Nepodporovaný typ hodnoty'); | ||||
|       } | ||||
|       const s = value.split('|'); | ||||
|       const pizzaIndex = Number.parseInt(s[0]); | ||||
|       const pizzaSizeIndex = Number.parseInt(s[1]); | ||||
|       await addPizza({ body: { pizzaIndex, pizzaSizeIndex } }); | ||||
|       await addPizza(pizzaIndex, pizzaSizeIndex); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const handlePizzaDelete = async (pizzaOrder: PizzaVariant) => { | ||||
|     await removePizza({ body: { pizzaOrder } }); | ||||
|   const handlePizzaDelete = async (pizzaOrder: PizzaOrder) => { | ||||
|     await removePizza(pizzaOrder); | ||||
|   } | ||||
| 
 | ||||
|   const handlePizzaPoznamkaChange = async () => { | ||||
| @ -315,7 +285,7 @@ function App() { | ||||
|       alert("Poznámka může mít maximálně 70 znaků"); | ||||
|       return; | ||||
|     } | ||||
|     updatePizzaDayNote({ body: { note: pizzaPoznamkaRef.current?.value } }); | ||||
|     updatePizzaDayNote(pizzaPoznamkaRef.current?.value); | ||||
|   } | ||||
| 
 | ||||
|   // const addToCart = async () => {
 | ||||
| @ -344,7 +314,7 @@ function App() { | ||||
| 
 | ||||
|   const handleChangeDepartureTime = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||
|     if (foodChoiceList?.length && choiceRef.current?.value) { | ||||
|       await changeDepartureTime({ body: { time: event.target.value as DepartureTime, dayIndex } }); | ||||
|       await changeDepartureTime(event.target.value, dayIndex); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| @ -362,32 +332,17 @@ function App() { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const renderFoodTable = (location: Restaurant, menu: RestaurantDayMenu) => { | ||||
|   const renderFoodTable = (name: string, menu: DayMenu) => { | ||||
|     let content; | ||||
|     if (menu?.closed) { | ||||
|       content = <h3>Zavřeno</h3> | ||||
|     } else if (menu?.food?.length && menu.food.length > 0) { | ||||
|       const hideSoups = settings?.hideSoups; | ||||
|     } else if (menu?.food?.length > 0) { | ||||
|       content = <Table striped bordered hover> | ||||
|         <tbody style={{ cursor: 'pointer' }}> | ||||
|           {menu.food.map((f: Food, index: number) => | ||||
|             (!hideSoups || !f.isSoup) && | ||||
|             <tr key={f.name} onClick={() => doAddClickFoodChoice(location, index)}> | ||||
|         <tbody> | ||||
|           {menu.food.filter(f => (settings?.hideSoups ? !f.isSoup : true)).map((f: any, index: number) => | ||||
|             <tr key={index}> | ||||
|               <td>{f.amount}</td> | ||||
|               <td> | ||||
|                 {f.name} | ||||
|                 {f.allergens && f.allergens.length > 0 && ( | ||||
|                   <> ({f.allergens.map((a, idx) => ( | ||||
|                     <span key={a}> | ||||
|                       <span title={ALLERGENS[a]} style={{ cursor: 'help', textDecoration: 'underline' }} onClick={e => { | ||||
|                         e.stopPropagation(); | ||||
|                         window.open(LINK_ALLERGENS, '_blank'); | ||||
|                       }}>{a}</span> | ||||
|                       {idx < f.allergens!.length - 1 && ','} | ||||
|                     </span> | ||||
|                   ))})</> | ||||
|                 )} | ||||
|               </td> | ||||
|               <td>{f.name}</td> | ||||
|               <td>{f.price}</td> | ||||
|             </tr> | ||||
|           )} | ||||
| @ -396,14 +351,14 @@ function App() { | ||||
|     } else { | ||||
|       content = <h3>Chyba načtení dat</h3> | ||||
|     } | ||||
|     return <Col md={12} lg={3} className='mt-3'> | ||||
|       <h3 style={{ cursor: 'pointer' }} onClick={() => doAddClickFoodChoice(location)}>{getLunchChoiceName(location)}</h3> | ||||
|     return <Col md={12} lg={4} className='mt-3'> | ||||
|       <h3>{name}</h3> | ||||
|       {menu?.lastUpdate && <small>Poslední aktualizace: {getHumanDateTime(new Date(menu.lastUpdate))}</small>} | ||||
|       {content} | ||||
|     </Col> | ||||
|   } | ||||
| 
 | ||||
|   if (!auth?.login) { | ||||
|   if (!auth || !auth.login) { | ||||
|     return <Login />; | ||||
|   } | ||||
| 
 | ||||
| @ -432,63 +387,59 @@ function App() { | ||||
|   } | ||||
| 
 | ||||
|   const noOrders = data?.pizzaDay?.orders?.length === 0; | ||||
|   const canChangeChoice = dayIndex == null || data.todayDayIndex == null || dayIndex >= data.todayDayIndex; | ||||
|   const canChangeChoice = dayIndex == null || data.todayWeekIndex == null || dayIndex >= data.todayWeekIndex; | ||||
| 
 | ||||
|   const { path, url, startOffset, endOffset, duration, ...style } = easterEgg || {}; | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="app-container"> | ||||
|     <> | ||||
|       {easterEgg && eggImage && <img ref={eggRef} alt='' src={URL.createObjectURL(eggImage)} style={{ position: 'absolute', ...EASTER_EGG_STYLE, ...style, animationDuration: `${duration ?? EASTER_EGG_DEFAULT_DURATION}s` }} />} | ||||
|       <Header /> | ||||
|       <div className='wrapper'> | ||||
|         {data.isWeekend ? <h4>Užívejte víkend :)</h4> : <> | ||||
|           <Alert variant={'primary'}> | ||||
|             {/* <img alt="" src='hat.png' style={{ position: "absolute", width: "70px", rotate: "-45deg", left: -40, top: -58 }} /> | ||||
|             <img alt="" src='snowman.png' style={{ position: "absolute", height: "110px", right: 10, top: 5 }} /> */} | ||||
|             <img src='hat.png' style={{ position: "absolute", width: "70px", rotate: "-45deg", left: -40, top: -58 }} /> | ||||
|             <img src='snowman.png' style={{ position: "absolute", height: "110px", right: 10, top: 5 }} /> | ||||
|             Poslední změny: | ||||
|             <ul> | ||||
|               <li>Zobrazení alergenu při najetí myší a proklik na seznam alergenů</li> | ||||
|               <li>Přesun přenačtení menu do samostatného dialogu</li> | ||||
|               <li>Podzimní atmosféra</li> | ||||
|               <li>Zimní atmosféra</li> | ||||
|               <li>Odstranění podniku U Motlíků</li> | ||||
|             </ul> | ||||
|           </Alert> | ||||
|           {dayIndex != null && | ||||
|             <div className='day-navigator'> | ||||
|               <span title='Předchozí den'> | ||||
|                 <FontAwesomeIcon icon={faChevronLeft} style={{ cursor: "pointer", visibility: dayIndex > 0 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex - 1)} /> | ||||
|               </span> | ||||
|               <h1 className='title' style={{ color: dayIndex === data.todayDayIndex ? 'black' : 'gray' }}>{data.date}</h1> | ||||
|               <span title="Následující den"> | ||||
|                 <FontAwesomeIcon icon={faChevronRight} style={{ cursor: "pointer", visibility: dayIndex < 4 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex + 1)} /> | ||||
|               </span> | ||||
|               <FontAwesomeIcon title="Předchozí den" icon={faChevronLeft} style={{ cursor: "pointer", visibility: dayIndex > 0 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex - 1)} /> | ||||
|               <h1 className='title' style={{ color: dayIndex === data.todayWeekIndex ? 'black' : 'gray' }}>{data.date}</h1> | ||||
|               <FontAwesomeIcon title="Následující den" icon={faChevronRight} style={{ cursor: "pointer", visibility: dayIndex < 4 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex + 1)} /> | ||||
|             </div> | ||||
|           } | ||||
|           <Row className='food-tables'> | ||||
|             {/* TODO zjednodušit, stačí iterovat klíče typu Restaurant */} | ||||
|             {food['SLADOVNICKA'] && renderFoodTable('SLADOVNICKA', food['SLADOVNICKA'])} | ||||
|             {food['TECHTOWER'] && renderFoodTable('TECHTOWER', food['TECHTOWER'])} | ||||
|             {food['ZASTAVKAUMICHALA'] && renderFoodTable('ZASTAVKAUMICHALA', food['ZASTAVKAUMICHALA'])} | ||||
|             {food['SENKSERIKOVA'] && renderFoodTable('SENKSERIKOVA', food['SENKSERIKOVA'])} | ||||
|             {food[Restaurants.SLADOVNICKA] && renderFoodTable('Sladovnická', food[Restaurants.SLADOVNICKA])} | ||||
|             {/* {food[Restaurants.UMOTLIKU] && renderFoodTable('U Motlíků', food[Restaurants.UMOTLIKU])} */} | ||||
|             {food[Restaurants.TECHTOWER] && renderFoodTable('TechTower', food[Restaurants.TECHTOWER])} | ||||
|             {food[Restaurants.ZASTAVKAUMICHALA] && renderFoodTable('Zastávka u Michala', food[Restaurants.ZASTAVKAUMICHALA])} | ||||
|           </Row> | ||||
|           <div className='content-wrapper'> | ||||
|             <div className='content'> | ||||
|               {canChangeChoice && <> | ||||
|                 <p>{`Jak to ${dayIndex == null || dayIndex === data.todayDayIndex ? 'dnes' : 'tento den'} vidíš s obědem?`}</p> | ||||
|                 <p>{`Jak to ${dayIndex == null || dayIndex === data.todayWeekIndex ? 'dnes' : 'tento den'} vidíš s obědem?`}</p> | ||||
|                 <Form.Select ref={choiceRef} onChange={doAddChoice}> | ||||
|                   <option></option> | ||||
|                   {Object.entries(LunchChoice) | ||||
|                   {Object.entries(Locations) | ||||
|                     .filter(entry => { | ||||
|                       const locationKey = entry[0] as Restaurant; | ||||
|                       return !food[locationKey]?.closed; | ||||
|                       const locationKey = entry[0] as LocationKey; | ||||
|                       const restaurantKey = Object.keys(Restaurants).indexOf(locationKey); | ||||
|                       const v = Object.values(Restaurants)[restaurantKey]; | ||||
|                       return v == null || !food[v]?.closed; | ||||
|                     }) | ||||
|                     .map(entry => <option key={entry[0]} value={entry[0]}>{getLunchChoiceName(entry[1])}</option>)} | ||||
|                     .map(entry => <option key={entry[0]} value={entry[0]}>{entry[1]}</option>)} | ||||
|                 </Form.Select> | ||||
|                 <small>Je možné vybrat jen jednu možnost. Výběr jiné odstraní předchozí.</small> | ||||
|                 {foodChoiceList && !closed && <> | ||||
|                   <p style={{ marginTop: "10px" }}>Na co dobrého? <small>(nepovinné)</small></p> | ||||
|                   <Form.Select ref={foodChoiceRef} onChange={doAddFoodChoice}> | ||||
|                     <option></option> | ||||
|                     {foodChoiceList.map((food, index) => <option key={food.name} value={index}>{food.name}</option>)} | ||||
|                     {foodChoiceList.map((food, index) => <option key={index} value={index}>{food.name}</option>)} | ||||
|                   </Form.Select> | ||||
|                 </>} | ||||
|                 {foodChoiceList && !closed && <> | ||||
| @ -505,60 +456,51 @@ function App() { | ||||
|                 <Table bordered className='mt-5'> | ||||
|                   <tbody> | ||||
|                     {Object.keys(data.choices).map(key => { | ||||
|                       const locationKey = key as LunchChoice; | ||||
|                       const locationName = getLunchChoiceName(locationKey); | ||||
|                       const locationKey = key as LocationKey; | ||||
|                       const locationName = Locations[locationKey]; | ||||
|                       const loginObject = data.choices[locationKey]; | ||||
|                       if (!loginObject) { | ||||
|                         return; | ||||
|                       } | ||||
|                       const locationLoginList = Object.entries(loginObject); | ||||
|                       const locationPickCount = locationLoginList.length | ||||
|                       return ( | ||||
|                         <tr key={key}> | ||||
|                           {(locationPickCount ?? 0) > 1 ? ( | ||||
|                             <td>{locationName} ({locationPickCount})</td> | ||||
|                           ) : ( | ||||
|                             <td>{locationName}</td>)} | ||||
|                           <td>{locationName}</td> | ||||
|                           <td className='p-0'> | ||||
|                             <Table> | ||||
|                               <tbody> | ||||
|                                 {locationLoginList.map((entry: [string, UserLunchChoice], index) => { | ||||
|                                 {locationLoginList.map((entry: [string, FoodChoices], index) => { | ||||
|                                   const login = entry[0]; | ||||
|                                   const userPayload = entry[1]; | ||||
|                                   const userChoices = userPayload?.selectedFoods; | ||||
|                                   const userChoices = userPayload?.options; | ||||
|                                   const trusted = userPayload?.trusted || false; | ||||
|                                   return <tr key={entry[0]}> | ||||
|                                   return <tr key={index}> | ||||
|                                     <td> | ||||
|                                       {trusted && <span className='trusted-icon' title='Uživatel ověřený doménovým přihlášením'> | ||||
|                                         <FontAwesomeIcon icon={faCircleCheck} style={{ cursor: "help" }} /> | ||||
|                                       {trusted && <span className='trusted-icon'> | ||||
|                                         <FontAwesomeIcon title='Uživatel ověřený doménovým přihlášením' icon={faCircleCheck} style={{ cursor: "help" }} /> | ||||
|                                       </span>} | ||||
|                                       {login} | ||||
|                                       {userPayload.departureTime && <small> ({userPayload.departureTime})</small>} | ||||
|                                       {userPayload.note && <small style={{ overflowWrap: 'anywhere' }}> ({userPayload.note})</small>} | ||||
|                                       {login === auth.login && canChangeChoice && <span title='Upravit poznámku'> | ||||
|                                         <FontAwesomeIcon onClick={() => { | ||||
|                                       {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||
|                                         setNoteModalOpen(true); | ||||
|                                       }} className='action-icon' icon={faNoteSticky} /> | ||||
|                                       </span>} | ||||
|                                       {login === auth.login && canChangeChoice && <span title={`Odstranit volbu ${locationName}, včetně případných zvolených jídel`}> | ||||
|                                         <FontAwesomeIcon onClick={() => { | ||||
|                                         doRemoveChoices(key as LunchChoice); | ||||
|                                       }} className='action-icon' icon={faTrashCan} /> | ||||
|                                       </span>} | ||||
|                                       }} title='Upravit poznámku' className='action-icon' icon={faNoteSticky} />} | ||||
|                                       {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||
|                                         doRemoveChoices(key as LocationKey); | ||||
|                                       }} title={`Odstranit volbu ${locationName}, včetně případných zvolených jídel`} className='action-icon' icon={faTrashCan} />} | ||||
|                                     </td> | ||||
|                                     {userChoices?.length && food ? <td> | ||||
|                                       <ul> | ||||
|                                         {userChoices?.map(foodIndex => { | ||||
|                                           const restaurantKey = key as Restaurant; | ||||
|                                           const foodName = food[restaurantKey]?.food?.[foodIndex].name; | ||||
|                                           // TODO narovnat, tohle je zbytečně složité
 | ||||
|                                           const restaurantKey = Object.keys(Restaurants).indexOf(key); | ||||
|                                           const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||
|                                           const foodName = food[restaurant]?.food[foodIndex].name; | ||||
|                                           return <li key={foodIndex}> | ||||
|                                             {foodName} | ||||
|                                             {login === auth.login && canChangeChoice && | ||||
|                                             <span title={`Odstranit ${foodName}`}> | ||||
|                                               <FontAwesomeIcon onClick={() => { | ||||
|                                               doRemoveFoodChoice(restaurantKey, foodIndex); | ||||
|                                             }} className='action-icon' icon={faTrashCan} /> | ||||
|                                             </span>} | ||||
|                                             {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||
|                                               doRemoveFoodChoice(key as LocationKey, foodIndex); | ||||
|                                             }} title={`Odstranit ${foodName}`} className='action-icon' icon={faTrashCan} />} | ||||
|                                           </li> | ||||
|                                         })} | ||||
|                                       </ul> | ||||
| @ -577,7 +519,7 @@ function App() { | ||||
|                 : <div className='mt-5'><i>Zatím nikdo nehlasoval...</i></div> | ||||
|               } | ||||
|             </div> | ||||
|             {dayIndex === data.todayDayIndex && | ||||
|             {dayIndex === data.todayWeekIndex && | ||||
|               <div className='mt-5'> | ||||
|                 {!data.pizzaDay && | ||||
|                   <div style={{ textAlign: 'center' }}> | ||||
| @ -630,6 +572,9 @@ function App() { | ||||
|                               <Button className='danger mb-3' title="Umožní znovu editovat objednávky." onClick={async () => { | ||||
|                                 await unlockPizzaDay(); | ||||
|                               }}>Odemknout</Button> | ||||
|                               {/* <Button className='danger mb-3' style={{ marginLeft: '20px' }} onClick={async () => { | ||||
|                           await addToCart(); | ||||
|                         }}>Přidat vše do košíku</Button> */} | ||||
|                               <Button className='danger mb-3' style={{ marginLeft: '20px' }} title={noOrders ? "Nelze objednat - neexistuje žádná objednávka" : "Použij po objednání. Objednávky zůstanou zamčeny."} disabled={noOrders} onClick={async () => { | ||||
|                                 await finishOrder(); | ||||
|                               }}>Objednáno</Button> | ||||
| @ -647,7 +592,7 @@ function App() { | ||||
|                                 await lockPizzaDay(); | ||||
|                               }}>Vrátit do "uzamčeno"</Button> | ||||
|                               <Button className='danger mb-3' style={{ marginLeft: '20px' }} title="Nastaví stav na 'Doručeno' - koncový stav." onClick={async () => { | ||||
|                                 await finishDelivery({ body: { bankAccount: settings?.bankAccount, bankAccountHolder: settings?.holderName } }); | ||||
|                                 await finishDelivery(settings?.bankAccount, settings?.holderName); | ||||
|                               }}>Doručeno</Button> | ||||
|                             </div> | ||||
|                           } | ||||
| @ -684,28 +629,24 @@ function App() { | ||||
|                         </Button> | ||||
|                       </div> | ||||
|                     } | ||||
|                     <PizzaOrderList state={data.pizzaDay.state!} orders={data.pizzaDay.orders!} onDelete={handlePizzaDelete} creator={data.pizzaDay.creator!} /> | ||||
|                     <PizzaOrderList state={data.pizzaDay.state} orders={data.pizzaDay.orders} onDelete={handlePizzaDelete} creator={data.pizzaDay.creator} /> | ||||
|                     { | ||||
|                       data.pizzaDay.state === PizzaDayState.DELIVERED && myOrder?.hasQr ? | ||||
|                       data.pizzaDay.state === PizzaDayState.DELIVERED && myOrder?.hasQr && | ||||
|                       <div className='qr-code'> | ||||
|                         <h3>QR platba</h3> | ||||
|                           <img src={`/api/qr?login=${auth.login}`} alt='QR kód' /> | ||||
|                         </div> : null | ||||
|                         <img src={getQrUrl(auth.login)} alt='QR kód' /> | ||||
|                       </div> | ||||
|                     } | ||||
|                   </div> | ||||
|                 } | ||||
|               </div> | ||||
|             } | ||||
|           </div> | ||||
|         </> || "Jejda, něco se nám nepovedlo :("} | ||||
|         </>} | ||||
|       </div> | ||||
|       <FallingLeaves | ||||
|         numLeaves={LEAF_PRESETS.NORMAL} | ||||
|         leafVariants={LEAF_COLOR_THEMES.AUTUMN} | ||||
|       /> | ||||
|       <Footer /> | ||||
|       <NoteModal isOpen={noteModalOpen} onClose={() => setNoteModalOpen(false)} onSave={saveNote} /> | ||||
|     </div> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -1,33 +0,0 @@ | ||||
| import { Routes, Route } from "react-router-dom"; | ||||
| import { ProvideSettings } from "./context/settings"; | ||||
| // import Snowfall from "react-snowfall";
 | ||||
| import { ToastContainer } from "react-toastify"; | ||||
| import { SocketContext, socket } from "./context/socket"; | ||||
| import StatsPage from "./pages/StatsPage"; | ||||
| import App from "./App"; | ||||
| 
 | ||||
| export const STATS_URL = '/stats'; | ||||
| 
 | ||||
| export default function AppRoutes() { | ||||
|     return ( | ||||
|         <Routes> | ||||
|             <Route path={STATS_URL} element={<StatsPage />} /> | ||||
|             <Route path="/" element={ | ||||
|                 <ProvideSettings> | ||||
|                     <SocketContext.Provider value={socket}> | ||||
|                         <> | ||||
|                             {/* <Snowfall style={{ | ||||
|                                 zIndex: 2, | ||||
|                                 position: 'fixed', | ||||
|                                 width: '100vw', | ||||
|                                 height: '100vh' | ||||
|                             }} /> */} | ||||
|                             <App /> | ||||
|                         </> | ||||
|                         <ToastContainer /> | ||||
|                     </SocketContext.Provider> | ||||
|                 </ProvideSettings> | ||||
|             } /> | ||||
|         </Routes> | ||||
|     ); | ||||
| } | ||||
| @ -1,31 +0,0 @@ | ||||
| .falling-leaves { | ||||
|   position: fixed; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|   width: 100vw; | ||||
|   height: 100vh; | ||||
|   pointer-events: none; | ||||
|   z-index: 0; | ||||
|   overflow: hidden; | ||||
| } | ||||
| 
 | ||||
| .leaf-scene { | ||||
|   position: absolute; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|   bottom: 0; | ||||
|   width: 100%; | ||||
|   transform-style: preserve-3d; | ||||
| 
 | ||||
|   div { | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     width: 20px; | ||||
|     height: 20px; | ||||
|     background-repeat: no-repeat; | ||||
|     background-size: 100%; | ||||
|     transform-style: preserve-3d; | ||||
|     backface-visibility: visible; | ||||
|   } | ||||
| } | ||||
| @ -1,317 +0,0 @@ | ||||
| import React, { useEffect, useRef, useCallback } from 'react'; | ||||
| 
 | ||||
| // Různé barevné varianty listů
 | ||||
| const LEAF_VARIANTS = [ | ||||
|     'leaf.svg',        // Původní tmavě hnědá
 | ||||
|     'leaf-orange.svg', // Oranžová
 | ||||
|     'leaf-yellow.svg', // Žlutá
 | ||||
|     'leaf-red.svg',    // Červená
 | ||||
|     'leaf-brown.svg',  // Světle hnědá
 | ||||
|     'leaf-green.svg',  // Zelená
 | ||||
| ] as const; | ||||
| 
 | ||||
| interface LeafData { | ||||
|     el: HTMLDivElement; | ||||
|     x: number; | ||||
|     y: number; | ||||
|     z: number; | ||||
|     rotation: { | ||||
|         axis: 'X' | 'Y' | 'Z'; | ||||
|         value: number; | ||||
|         speed: number; | ||||
|         x: number; | ||||
|     }; | ||||
|     xSpeedVariation: number; | ||||
|     ySpeed: number; | ||||
|     path: { | ||||
|         type: number; | ||||
|         start: number; | ||||
|     }; | ||||
|     image: number; | ||||
| } | ||||
| 
 | ||||
| interface WindOptions { | ||||
|     magnitude: number; | ||||
|     maxSpeed: number; | ||||
|     duration: number; | ||||
|     start: number; | ||||
|     speed: (t: number, y: number) => number; | ||||
| } | ||||
| 
 | ||||
| interface LeafSceneOptions { | ||||
|     numLeaves: number; | ||||
|     wind: WindOptions; | ||||
| } | ||||
| 
 | ||||
| interface FallingLeavesProps { | ||||
|     /** Počet padających listů (výchozí: 20) */ | ||||
|     numLeaves?: number; | ||||
|     /** CSS třída pro kontejner (výchozí: 'falling-leaves') */ | ||||
|     className?: string; | ||||
|     /** Barevné varianty listů k použití (výchozí: všechny) */ | ||||
|     leafVariants?: readonly string[]; | ||||
| } | ||||
| 
 | ||||
| class LeafScene { | ||||
|     private viewport: HTMLElement; | ||||
|     private world: HTMLDivElement; | ||||
|     private leaves: LeafData[] = []; | ||||
|     private options: LeafSceneOptions; | ||||
|     private width: number; | ||||
|     private height: number; | ||||
|     private timer: number = 0; | ||||
|     private animationId: number | null = null; | ||||
|     private leafVariants: readonly string[]; | ||||
| 
 | ||||
|     constructor(el: HTMLElement, numLeaves: number = 20, leafVariants: readonly string[] = LEAF_VARIANTS) { | ||||
|         this.viewport = el; | ||||
|         this.world = document.createElement('div'); | ||||
|         this.leafVariants = leafVariants; | ||||
| 
 | ||||
|         this.options = { | ||||
|             numLeaves, | ||||
|             wind: { | ||||
|                 magnitude: 1.2, | ||||
|                 maxSpeed: 12, | ||||
|                 duration: 300, | ||||
|                 start: 0, | ||||
|                 speed: () => 0 | ||||
|             }, | ||||
|         }; | ||||
| 
 | ||||
|         this.width = this.viewport.offsetWidth; | ||||
|         this.height = this.viewport.offsetHeight; | ||||
|     } | ||||
| 
 | ||||
|     private resetLeaf = (leaf: LeafData): LeafData => { | ||||
|         // place leaf towards the top left
 | ||||
|         leaf.x = this.width * 2 - Math.random() * this.width * 1.75; | ||||
|         leaf.y = -10; | ||||
|         leaf.z = Math.random() * 200; | ||||
| 
 | ||||
|         if (leaf.x > this.width) { | ||||
|             leaf.x = this.width + 10; | ||||
|             leaf.y = Math.random() * this.height / 2; | ||||
|         } | ||||
| 
 | ||||
|         // at the start, the leaf can be anywhere
 | ||||
|         if (this.timer === 0) { | ||||
|             leaf.y = Math.random() * this.height; | ||||
|         } | ||||
| 
 | ||||
|         // Choose axis of rotation.
 | ||||
|         // If axis is not X, chose a random static x-rotation for greater variability
 | ||||
|         leaf.rotation.speed = Math.random() * 10; | ||||
|         const randomAxis = Math.random(); | ||||
| 
 | ||||
|         if (randomAxis > 0.5) { | ||||
|             leaf.rotation.axis = 'X'; | ||||
|         } else if (randomAxis > 0.25) { | ||||
|             leaf.rotation.axis = 'Y'; | ||||
|             leaf.rotation.x = Math.random() * 180 + 90; | ||||
|         } else { | ||||
|             leaf.rotation.axis = 'Z'; | ||||
|             leaf.rotation.x = Math.random() * 360 - 180; | ||||
|             // looks weird if the rotation is too fast around this axis
 | ||||
|             leaf.rotation.speed = Math.random() * 3; | ||||
|         } | ||||
| 
 | ||||
|         // random speed
 | ||||
|         leaf.xSpeedVariation = Math.random() * 0.8 - 0.4; | ||||
|         leaf.ySpeed = Math.random() + 1.5; | ||||
| 
 | ||||
|         // randomly select leaf color variant
 | ||||
|         const randomVariantIndex = Math.floor(Math.random() * this.leafVariants.length); | ||||
|         leaf.image = randomVariantIndex; | ||||
| 
 | ||||
|         // apply the background image to the leaf element
 | ||||
|         const leafVariant = this.leafVariants[randomVariantIndex]; | ||||
|         leaf.el.style.backgroundImage = `url(${leafVariant})`; | ||||
| 
 | ||||
|         return leaf; | ||||
|     }; | ||||
| 
 | ||||
|     private updateLeaf = (leaf: LeafData): void => { | ||||
|         const leafWindSpeed = this.options.wind.speed(this.timer - this.options.wind.start, leaf.y); | ||||
| 
 | ||||
|         const xSpeed = leafWindSpeed + leaf.xSpeedVariation; | ||||
|         leaf.x -= xSpeed; | ||||
|         leaf.y += leaf.ySpeed; | ||||
|         leaf.rotation.value += leaf.rotation.speed; | ||||
| 
 | ||||
|         const transform = `translateX(${leaf.x}px) translateY(${leaf.y}px) translateZ(${leaf.z}px) rotate${leaf.rotation.axis}(${leaf.rotation.value}deg)${leaf.rotation.axis !== 'X' ? ` rotateX(${leaf.rotation.x}deg)` : '' | ||||
|             }`;
 | ||||
| 
 | ||||
|         leaf.el.style.transform = transform; | ||||
| 
 | ||||
|         // reset if out of view
 | ||||
|         if (leaf.x < -10 || leaf.y > this.height + 10) { | ||||
|             this.resetLeaf(leaf); | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     private updateWind = (): void => { | ||||
|         // wind follows a sine curve: asin(b*time + c) + a
 | ||||
|         // where a = wind magnitude as a function of leaf position, b = wind.duration, c = offset
 | ||||
|         // wind duration should be related to wind magnitude, e.g. higher windspeed means longer gust duration
 | ||||
| 
 | ||||
|         if (this.timer === 0 || this.timer > (this.options.wind.start + this.options.wind.duration)) { | ||||
|             this.options.wind.magnitude = Math.random() * this.options.wind.maxSpeed; | ||||
|             this.options.wind.duration = this.options.wind.magnitude * 50 + (Math.random() * 20 - 10); | ||||
|             this.options.wind.start = this.timer; | ||||
| 
 | ||||
|             const screenHeight = this.height; | ||||
| 
 | ||||
|             this.options.wind.speed = function (t: number, y: number) { | ||||
|                 // should go from full wind speed at the top, to 1/2 speed at the bottom, using leaf Y
 | ||||
|                 const a = this.magnitude / 2 * (screenHeight - 2 * y / 3) / screenHeight; | ||||
|                 return a * Math.sin(2 * Math.PI / this.duration * t + (3 * Math.PI / 2)) + a; | ||||
|             }; | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     public init = (): void => { | ||||
|         // Clear existing leaves
 | ||||
|         this.leaves = []; | ||||
|         this.world.innerHTML = ''; | ||||
| 
 | ||||
|         for (let i = 0; i < this.options.numLeaves; i++) { | ||||
|             const leaf: LeafData = { | ||||
|                 el: document.createElement('div'), | ||||
|                 x: 0, | ||||
|                 y: 0, | ||||
|                 z: 0, | ||||
|                 rotation: { | ||||
|                     axis: 'X', | ||||
|                     value: 0, | ||||
|                     speed: 0, | ||||
|                     x: 0 | ||||
|                 }, | ||||
|                 xSpeedVariation: 0, | ||||
|                 ySpeed: 0, | ||||
|                 path: { | ||||
|                     type: 1, | ||||
|                     start: 0, | ||||
|                 }, | ||||
|                 image: 1 | ||||
|             }; | ||||
| 
 | ||||
|             this.resetLeaf(leaf); | ||||
|             this.leaves.push(leaf); | ||||
|             this.world.appendChild(leaf.el); | ||||
|         } | ||||
| 
 | ||||
|         this.world.className = 'leaf-scene'; | ||||
|         this.viewport.appendChild(this.world); | ||||
| 
 | ||||
|         // set perspective
 | ||||
|         this.world.style.perspective = "400px"; | ||||
| 
 | ||||
|         // reset window height/width on resize
 | ||||
|         const handleResize = (): void => { | ||||
|             this.width = this.viewport.offsetWidth; | ||||
|             this.height = this.viewport.offsetHeight; | ||||
|         }; | ||||
| 
 | ||||
|         window.addEventListener('resize', handleResize); | ||||
|     }; | ||||
| 
 | ||||
|     public render = (): void => { | ||||
|         this.updateWind(); | ||||
| 
 | ||||
|         for (let i = 0; i < this.leaves.length; i++) { | ||||
|             this.updateLeaf(this.leaves[i]); | ||||
|         } | ||||
| 
 | ||||
|         this.timer++; | ||||
|         this.animationId = requestAnimationFrame(this.render); | ||||
|     }; | ||||
| 
 | ||||
|     public destroy = (): void => { | ||||
|         if (this.animationId) { | ||||
|             cancelAnimationFrame(this.animationId); | ||||
|             this.animationId = null; | ||||
|         } | ||||
| 
 | ||||
|         if (this.world && this.world.parentNode) { | ||||
|             this.world.parentNode.removeChild(this.world); | ||||
|         } | ||||
| 
 | ||||
|         window.removeEventListener('resize', () => { | ||||
|             this.width = this.viewport.offsetWidth; | ||||
|             this.height = this.viewport.offsetHeight; | ||||
|         }); | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Komponenta pro zobrazení padajících listů na pozadí stránky | ||||
|  *  | ||||
|  * @param numLeaves - Počet padajících listů (výchozí: 20) | ||||
|  * @param className - CSS třída pro kontejner (výchozí: 'falling-leaves') | ||||
|  * @param leafVariants - Barevné varianty listů k použití (výchozí: všechny) | ||||
|  *  | ||||
|  * @example | ||||
|  * // Základní použití s výchozím počtem listů
 | ||||
|  * <FallingLeaves /> | ||||
|  *  | ||||
|  * @example   | ||||
|  * // Použití s vlastním počtem listů
 | ||||
|  * <FallingLeaves numLeaves={50} /> | ||||
|  *  | ||||
|  * @example | ||||
|  * // Použití s vlastní CSS třídou a pouze podzimními barvami
 | ||||
|  * <FallingLeaves  | ||||
|  *   numLeaves={15}  | ||||
|  *   className="autumn-leaves" | ||||
|  *   leafVariants={['leaf-orange.svg', 'leaf-red.svg', 'leaf-yellow.svg']} | ||||
|  * /> | ||||
|  */ | ||||
| const FallingLeaves: React.FC<FallingLeavesProps> = ({ | ||||
|     numLeaves = 20, | ||||
|     className = 'falling-leaves', | ||||
|     leafVariants = LEAF_VARIANTS | ||||
| }) => { | ||||
|     const containerRef = useRef<HTMLDivElement>(null); | ||||
|     const leafSceneRef = useRef<LeafScene | null>(null); | ||||
| 
 | ||||
|     const initializeLeafScene = useCallback(() => { | ||||
|         if (containerRef.current) { | ||||
|             leafSceneRef.current = new LeafScene(containerRef.current, numLeaves, leafVariants); | ||||
|             leafSceneRef.current.init(); | ||||
|             leafSceneRef.current.render(); | ||||
|         } | ||||
|     }, [numLeaves, leafVariants]); | ||||
| 
 | ||||
|     useEffect(() => { | ||||
|         initializeLeafScene(); | ||||
| 
 | ||||
|         return () => { | ||||
|             if (leafSceneRef.current) { | ||||
|                 leafSceneRef.current.destroy(); | ||||
|                 leafSceneRef.current = null; | ||||
|             } | ||||
|         }; | ||||
|     }, [initializeLeafScene]); | ||||
| 
 | ||||
|     return <div ref={containerRef} className={className} />; | ||||
| }; | ||||
| 
 | ||||
| // Přednastavení pro různé účely
 | ||||
| export const LEAF_PRESETS = { | ||||
|     LIGHT: 10,      // Lehký podzimní efekt
 | ||||
|     NORMAL: 20,     // Standardní množství  
 | ||||
|     HEAVY: 40,      // Silný podzimní vítr
 | ||||
|     BLIZZARD: 80    // Hustý pád listí
 | ||||
| } as const; | ||||
| 
 | ||||
| // Přednastavené barevné kombinace
 | ||||
| export const LEAF_COLOR_THEMES = { | ||||
|     ALL: LEAF_VARIANTS,                                          // Všechny barvy
 | ||||
|     AUTUMN: ['leaf-orange.svg', 'leaf-red.svg', 'leaf-yellow.svg', 'leaf-brown.svg'] as const, // Podzimní barvy
 | ||||
|     WARM: ['leaf-orange.svg', 'leaf-red.svg', 'leaf-brown.svg'] as const,           // Teplé barvy
 | ||||
|     CLASSIC: ['leaf.svg', 'leaf-brown.svg'] as const,                               // Klasické hnědé odstíny
 | ||||
|     BRIGHT: ['leaf-yellow.svg', 'leaf-orange.svg'] as const,                        // Světlé barvy
 | ||||
| } as const; | ||||
| 
 | ||||
| export default FallingLeaves; | ||||
| @ -1,7 +1,7 @@ | ||||
| import React, { useCallback, useEffect, useRef } from 'react'; | ||||
| import { Button } from 'react-bootstrap'; | ||||
| import { useAuth } from './context/auth'; | ||||
| import { login } from '../../types'; | ||||
| import { login } from './api/Api'; | ||||
| import './Login.css'; | ||||
| 
 | ||||
| /** | ||||
| @ -14,10 +14,9 @@ export default function Login() { | ||||
|   useEffect(() => { | ||||
|     if (auth && !auth.login) { | ||||
|       // Vyzkoušíme přihlášení "naprázdno", pokud projde, přihlásili nás trusted headers
 | ||||
|       login().then(response => { | ||||
|         const token = response.data; | ||||
|       login().then(token => { | ||||
|         if (token) { | ||||
|           auth?.setToken(token as unknown as string); // TODO vyřešit, API definice je špatně, je to skutečně string
 | ||||
|           auth?.setToken(token); | ||||
|         } | ||||
|       }).catch(error => { | ||||
|         // nezajímá nás
 | ||||
| @ -26,16 +25,17 @@ export default function Login() { | ||||
|   }, [auth]); | ||||
| 
 | ||||
|   const doLogin = useCallback(async () => { | ||||
|     const length = loginRef?.current?.value.length && loginRef.current.value.replace(/\s/g, '').length | ||||
|     const length = loginRef?.current?.value && loginRef?.current?.value.length && loginRef.current.value.replace(/\s/g, '').length | ||||
|     if (length) { | ||||
|       const response = await login({ body: { login: loginRef.current?.value } }); | ||||
|       if (response.data) { | ||||
|         auth?.setToken(response.data as unknown as string); // TODO vyřešit
 | ||||
|       // TODO odchytávat cokoliv mimo 200
 | ||||
|       const token = await login(loginRef.current.value); | ||||
|       if (token) { | ||||
|         auth?.setToken(token); | ||||
|       } | ||||
|     } | ||||
|   }, [auth]); | ||||
| 
 | ||||
|   if (!auth?.login) { | ||||
|   if (!auth || !auth.login) { | ||||
|     return <div className='login'> | ||||
|       <h1>Luncher</h1> | ||||
|       <h4 style={{ marginBottom: "50px" }}>Aplikace pro profesionální management obědů</h4> | ||||
|  | ||||
| @ -16,8 +16,8 @@ export const storeToken = (token: string) => { | ||||
|  *  | ||||
|  * @returns token nebo null | ||||
|  */ | ||||
| export const getToken = (): string | undefined => { | ||||
|     return localStorage.getItem(TOKEN_KEY) ?? undefined; | ||||
| export const getToken = (): string | null => { | ||||
|     return localStorage.getItem(TOKEN_KEY); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
| @ -53,54 +53,7 @@ export function isInTheFuture(time: DepartureTime) { | ||||
|     const now = new Date(); | ||||
|     const currentHours = now.getHours(); | ||||
|     const currentMinutes = now.getMinutes(); | ||||
|     const currentDate = now.toDateString(); | ||||
|     const [hours, minutes] = time.split(':').map(Number); | ||||
| 
 | ||||
|     if (currentDate === now.toDateString()) { | ||||
|     return hours > currentHours || (hours === currentHours && minutes > currentMinutes); | ||||
| } | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí index dne v týdnu, kde pondělí=0, neděle=6 | ||||
|  *  | ||||
|  * @param date datum | ||||
|  * @returns index dne v týdnu | ||||
|  */ | ||||
| export const getDayOfWeekIndex = (date: Date) => { | ||||
|     // https://stackoverflow.com/a/4467559
 | ||||
|     return (((date.getDay() - 1) % 7) + 7) % 7; | ||||
| } | ||||
| 
 | ||||
| /** Vrátí první pracovní den v týdnu předaného data. */ | ||||
| export function getFirstWorkDayOfWeek(date: Date) { | ||||
|     const firstDay = new Date(date.getTime()); | ||||
|     firstDay.setDate(date.getDate() - getDayOfWeekIndex(date)); | ||||
|     return firstDay; | ||||
| } | ||||
| 
 | ||||
| /** Vrátí poslední pracovní den v týdnu předaného data. */ | ||||
| export function getLastWorkDayOfWeek(date: Date) { | ||||
|     const lastDay = new Date(date.getTime()); | ||||
|     lastDay.setDate(date.getDate() + (4 - getDayOfWeekIndex(date))); | ||||
|     return lastDay; | ||||
| } | ||||
| 
 | ||||
| /** Vrátí datum v ISO formátu. */ | ||||
| export function formatDate(date: Date, format?: string) { | ||||
|     let day = String(date.getDate()).padStart(2, '0'); | ||||
|     let month = String(date.getMonth() + 1).padStart(2, "0"); | ||||
|     let year = String(date.getFullYear()); | ||||
| 
 | ||||
|     const f = format ?? 'YYYY-MM-DD'; | ||||
|     return f.replace('DD', day).replace('MM', month).replace('YYYY', year); | ||||
| } | ||||
| 
 | ||||
| /** Vrátí human-readable reprezentaci předaného data pro zobrazení. */ | ||||
| export function getHumanDate(date: Date) { | ||||
|     let currentDay = String(date.getDate()).padStart(2, '0'); | ||||
|     let currentMonth = String(date.getMonth() + 1).padStart(2, "0"); | ||||
|     let currentYear = date.getFullYear(); | ||||
|     return `${currentDay}.${currentMonth}.${currentYear}`; | ||||
| } | ||||
							
								
								
									
										83
									
								
								client/src/api/Api.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								client/src/api/Api.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | ||||
| import { toast } from "react-toastify"; | ||||
| import { getToken } from "../Utils"; | ||||
| 
 | ||||
| /** | ||||
|  * Wrapper pro volání API, u kterých chceme automaticky zobrazit toaster s chybou ze serveru. | ||||
|  *  | ||||
|  * @param apiFunction volaná API funkce | ||||
|  */ | ||||
| export function errorHandler<T>(apiFunction: () => Promise<T>): Promise<T> { | ||||
|     return new Promise<T>((resolve, reject) => { | ||||
|         apiFunction().then((result) => { | ||||
|             resolve(result); | ||||
|         }).catch(e => { | ||||
|             toast.error(e.message, { theme: "colored" }); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
| 
 | ||||
| async function request<TResponse>( | ||||
|     url: string, | ||||
|     config: RequestInit = {} | ||||
| ): Promise<TResponse> { | ||||
|     config.headers = config?.headers ? new Headers(config.headers) : new Headers(); | ||||
|     config.headers.set("Authorization", `Bearer ${getToken()}`); | ||||
|     try { | ||||
|         const response = await fetch(url, config); | ||||
|         if (!response.ok) { | ||||
|             // TODO tohle je blbě, jelikož automaticky očekáváme, že v případě chyby přijde vždy JSON, což není pravda
 | ||||
|             const json = await response.json(); | ||||
|             // Vyhodíme samotnou hlášku z odpovědi, odchytí si jí errorHandler
 | ||||
|             throw new Error(json.error); | ||||
|         } | ||||
|         const contentType = response.headers.get("content-type"); | ||||
|         if (contentType && contentType.indexOf("application/json") !== -1) { | ||||
|             return response.json() as TResponse; | ||||
|         } else { | ||||
|             return response.text() as TResponse; | ||||
|         } | ||||
|     } catch (e) { | ||||
|         return Promise.reject(e); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| async function blobRequest( | ||||
|     url: string, | ||||
|     config: RequestInit = {} | ||||
| ): Promise<Blob> { | ||||
|     config.headers = config?.headers ? new Headers(config.headers) : new Headers(); | ||||
|     config.headers.set("Authorization", `Bearer ${getToken()}`); | ||||
|     try { | ||||
|         const response = await fetch(url, config); | ||||
|         if (!response.ok) { | ||||
|             const json = await response.json(); | ||||
|             // Vyhodíme samotnou hlášku z odpovědi, odchytí si jí errorHandler
 | ||||
|             throw new Error(json.error); | ||||
|         } | ||||
|         return response.blob() | ||||
|     } catch (e) { | ||||
|         return Promise.reject(e); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| export const api = { | ||||
|     get: <TResponse>(url: string) => request<TResponse>(url), | ||||
|     blobGet: (url: string) => blobRequest(url), | ||||
|     post: <TBody, TResponse>(url: string, body?: TBody) => request<TResponse>(url, { method: 'POST', body: JSON.stringify(body), headers: { 'Content-Type': 'application/json' } }), | ||||
| } | ||||
| 
 | ||||
| export const getQrUrl = (login: string) => { | ||||
|     return `/api/qr?login=${login}`; | ||||
| } | ||||
| 
 | ||||
| export const getData = async (dayIndex?: number) => { | ||||
|     let url = '/api/data'; | ||||
|     if (dayIndex != null) { | ||||
|         url += '?dayIndex=' + dayIndex; | ||||
|     } | ||||
|     return await api.get<any>(url); | ||||
| } | ||||
| 
 | ||||
| export const login = async (login?: string) => { | ||||
|     return await api.post<any, any>('/api/login', { login }); | ||||
| } | ||||
							
								
								
									
										12
									
								
								client/src/api/EasterEggApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								client/src/api/EasterEggApi.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | ||||
| import { EasterEgg } from "../types"; | ||||
| import { api } from "./Api"; | ||||
| 
 | ||||
| const EASTER_EGGS_API_PREFIX = '/api/easterEggs'; | ||||
| 
 | ||||
| export const getEasterEgg = async (): Promise<EasterEgg | undefined> => { | ||||
|     return await api.get<EasterEgg>(`${EASTER_EGGS_API_PREFIX}`); | ||||
| } | ||||
| 
 | ||||
| export const getImage = async (url: string) => { | ||||
|     return await api.blobGet(`${EASTER_EGGS_API_PREFIX}/${url}`); | ||||
| } | ||||
							
								
								
									
										28
									
								
								client/src/api/FoodApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								client/src/api/FoodApi.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | ||||
| import { AddChoiceRequest, ChangeDepartureTimeRequest, LocationKey, RemoveChoiceRequest, RemoveChoicesRequest, UpdateNoteRequest } from "../types"; | ||||
| import { api } from "./Api"; | ||||
| 
 | ||||
| const FOOD_API_PREFIX = '/api/food'; | ||||
| 
 | ||||
| export const addChoice = async (locationKey: LocationKey, foodIndex?: number, dayIndex?: number) => { | ||||
|     return await api.post<AddChoiceRequest, void>(`${FOOD_API_PREFIX}/addChoice`, { locationKey, foodIndex, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const removeChoices = async (locationKey: LocationKey, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoicesRequest, void>(`${FOOD_API_PREFIX}/removeChoices`, { locationKey, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const removeChoice = async (locationKey: LocationKey, foodIndex: number, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoiceRequest, void>(`${FOOD_API_PREFIX}/removeChoice`, { locationKey, foodIndex, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const updateNote = async (note?: string, dayIndex?: number) => { | ||||
|     return await api.post<UpdateNoteRequest, void>(`${FOOD_API_PREFIX}/updateNote`, { note, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const changeDepartureTime = async (time: string, dayIndex?: number) => { | ||||
|     return await api.post<ChangeDepartureTimeRequest, void>(`${FOOD_API_PREFIX}/changeDepartureTime`, { time, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const jdemeObed = async () => { | ||||
|     return await api.post<undefined, void>(`${FOOD_API_PREFIX}/jdemeObed`); | ||||
| } | ||||
							
								
								
									
										44
									
								
								client/src/api/PizzaDayApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								client/src/api/PizzaDayApi.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| import { AddPizzaRequest, FinishDeliveryRequest, PizzaOrder, RemovePizzaRequest, UpdatePizzaDayNoteRequest, UpdatePizzaFeeRequest } from "../types"; | ||||
| import { api } from "./Api"; | ||||
| 
 | ||||
| const PIZZADAY_API_PREFIX = '/api/pizzaDay'; | ||||
| 
 | ||||
| export const createPizzaDay = async () => { | ||||
|     return await api.post<undefined, void>(`${PIZZADAY_API_PREFIX}/create`); | ||||
| } | ||||
| 
 | ||||
| export const deletePizzaDay = async () => { | ||||
|     return await api.post<undefined, void>(`${PIZZADAY_API_PREFIX}/delete`); | ||||
| } | ||||
| 
 | ||||
| export const lockPizzaDay = async () => { | ||||
|     return await api.post<undefined, void>(`${PIZZADAY_API_PREFIX}/lock`); | ||||
| } | ||||
| 
 | ||||
| export const unlockPizzaDay = async () => { | ||||
|     return await api.post<undefined, void>(`${PIZZADAY_API_PREFIX}/unlock`); | ||||
| } | ||||
| 
 | ||||
| export const finishOrder = async () => { | ||||
|     return await api.post<undefined, void>(`${PIZZADAY_API_PREFIX}/finishOrder`); | ||||
| } | ||||
| 
 | ||||
| export const finishDelivery = async (bankAccount?: string, bankAccountHolder?: string) => { | ||||
|     return await api.post<FinishDeliveryRequest, void>(`${PIZZADAY_API_PREFIX}/finishDelivery`, { bankAccount, bankAccountHolder }); | ||||
| } | ||||
| 
 | ||||
| export const addPizza = async (pizzaIndex: number, pizzaSizeIndex: number) => { | ||||
|     return await api.post<AddPizzaRequest, void>(`${PIZZADAY_API_PREFIX}/add`, { pizzaIndex, pizzaSizeIndex }); | ||||
| } | ||||
| 
 | ||||
| export const removePizza = async (pizzaOrder: PizzaOrder) => { | ||||
|     return await api.post<RemovePizzaRequest, void>(`${PIZZADAY_API_PREFIX}/remove`, { pizzaOrder }); | ||||
| } | ||||
| 
 | ||||
| export const updatePizzaDayNote = async (note?: string) => { | ||||
|     return await api.post<UpdatePizzaDayNoteRequest, void>(`${PIZZADAY_API_PREFIX}/updatePizzaDayNote`, { note }); | ||||
| } | ||||
| 
 | ||||
| export const updatePizzaFee = async (login: string, text?: string, price?: number) => { | ||||
|     return await api.post<UpdatePizzaFeeRequest, void>(`${PIZZADAY_API_PREFIX}/updatePizzaFee`, { login, text, price }); | ||||
| } | ||||
							
								
								
									
										12
									
								
								client/src/api/VotingApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								client/src/api/VotingApi.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | ||||
| import { FeatureRequest, UpdateFeatureVoteRequest } from "../types"; | ||||
| import { api } from "./Api"; | ||||
| 
 | ||||
| const VOTING_API_PREFIX = '/api/voting'; | ||||
| 
 | ||||
| export const getFeatureVotes = async () => { | ||||
|     return await api.get<FeatureRequest[]>(`${VOTING_API_PREFIX}/getVotes`); | ||||
| } | ||||
| 
 | ||||
| export const updateFeatureVote = async (option: FeatureRequest, active: boolean) => { | ||||
|     return await api.post<UpdateFeatureVoteRequest, void>(`${VOTING_API_PREFIX}/updateVote`, { option, active }); | ||||
| } | ||||
| @ -1,12 +1,7 @@ | ||||
| import { Navbar } from "react-bootstrap"; | ||||
| 
 | ||||
| export default function Footer() { | ||||
|     return <Navbar className="text-light" variant='dark' expand="lg" style={{ | ||||
|         display: "flex", | ||||
|         justifyContent: "center", | ||||
|         marginTop: "auto", // Pushne footer na spodek
 | ||||
|         flexShrink: 0 // Zabrání zmenšování při malém obsahu
 | ||||
|     }}> | ||||
|     return <Navbar className="text-light" variant='dark' expand="lg" style={{ display: "flex", justifyContent: "center" }}> | ||||
|         <span>🄯 Žádná práva nevyhrazena. TODO a zdrojové kódy dostupné <a href="https://gitea.melancholik.eu/mates/Luncher">zde</a>.</span> | ||||
|     </Navbar > | ||||
| } | ||||
| @ -4,26 +4,24 @@ import { useAuth } from "../context/auth"; | ||||
| import SettingsModal from "./modals/SettingsModal"; | ||||
| import { useSettings } from "../context/settings"; | ||||
| import FeaturesVotingModal from "./modals/FeaturesVotingModal"; | ||||
| import { FeatureRequest } from "../types"; | ||||
| import { errorHandler } from "../api/Api"; | ||||
| import { getFeatureVotes, updateFeatureVote } from "../api/VotingApi"; | ||||
| import PizzaCalculatorModal from "./modals/PizzaCalculatorModal"; | ||||
| import RefreshMenuModal from "./modals/RefreshMenuModal"; | ||||
| import { useNavigate } from "react-router"; | ||||
| import { STATS_URL } from "../AppRoutes"; | ||||
| import { FeatureRequest, getVotes, updateVote } from "../../../types"; | ||||
| 
 | ||||
| 
 | ||||
| export default function Header() { | ||||
|     const auth = useAuth(); | ||||
|     const settings = useSettings(); | ||||
|     const navigate = useNavigate(); | ||||
|     const [settingsModalOpen, setSettingsModalOpen] = useState<boolean>(false); | ||||
|     const [votingModalOpen, setVotingModalOpen] = useState<boolean>(false); | ||||
|     const [pizzaModalOpen, setPizzaModalOpen] = useState<boolean>(false); | ||||
|     const [refreshMenuModalOpen, setRefreshMenuModalOpen] = useState<boolean>(false); | ||||
|     const [featureVotes, setFeatureVotes] = useState<FeatureRequest[] | undefined>([]); | ||||
|     const [featureVotes, setFeatureVotes] = useState<FeatureRequest[]>([]); | ||||
| 
 | ||||
|     useEffect(() => { | ||||
|         if (auth?.login) { | ||||
|             getVotes().then(response => { | ||||
|                 setFeatureVotes(response.data); | ||||
|             getFeatureVotes().then(votes => { | ||||
|                 setFeatureVotes(votes); | ||||
|             }) | ||||
|         } | ||||
|     }, [auth?.login]); | ||||
| @ -40,10 +38,6 @@ export default function Header() { | ||||
|         setPizzaModalOpen(false); | ||||
|     } | ||||
| 
 | ||||
|     const closeRefreshMenuModal = () => { | ||||
|         setRefreshMenuModalOpen(false); | ||||
|     } | ||||
| 
 | ||||
|     const isValidInteger = (str: string) => { | ||||
|         str = str.trim(); | ||||
|         if (!str) { | ||||
| @ -82,7 +76,7 @@ export default function Header() { | ||||
|                     cislo = cislo.padStart(16, '0'); | ||||
|                 } | ||||
|                 let sum = 0; | ||||
|                 for (let i = 0; i < cislo.length; i++) { | ||||
|                 for (var i = 0; i < cislo.length; i++) { | ||||
|                     const char = cislo.charAt(i); | ||||
|                     const order = (cislo.length - 1) - i; | ||||
|                     const weight = (2 ** order) % 11; | ||||
| @ -103,8 +97,8 @@ export default function Header() { | ||||
|     } | ||||
| 
 | ||||
|     const saveFeatureVote = async (option: FeatureRequest, active: boolean) => { | ||||
|         await updateVote({ body: { option, active } }); | ||||
|         const votes = [...featureVotes || []]; | ||||
|         await errorHandler(() => updateFeatureVote(option, active)); | ||||
|         const votes = [...featureVotes]; | ||||
|         if (active) { | ||||
|             votes.push(option); | ||||
|         } else { | ||||
| @ -114,23 +108,20 @@ export default function Header() { | ||||
|     } | ||||
| 
 | ||||
|     return <Navbar variant='dark' expand="lg"> | ||||
|         <Navbar.Brand href="/">Luncher</Navbar.Brand> | ||||
|         <Navbar.Brand>Luncher</Navbar.Brand> | ||||
|         <Navbar.Toggle aria-controls="basic-navbar-nav" /> | ||||
|         <Navbar.Collapse id="basic-navbar-nav"> | ||||
|             <Nav className="nav"> | ||||
|                 <NavDropdown align="end" title={auth?.login} id="basic-nav-dropdown"> | ||||
|                     <NavDropdown.Item onClick={() => setSettingsModalOpen(true)}>Nastavení</NavDropdown.Item> | ||||
|                     <NavDropdown.Item onClick={() => setRefreshMenuModalOpen(true)}>Přenačtení menu</NavDropdown.Item> | ||||
|                     <NavDropdown.Item onClick={() => setVotingModalOpen(true)}>Hlasovat o nových funkcích</NavDropdown.Item> | ||||
|                     <NavDropdown.Item onClick={() => setPizzaModalOpen(true)}>Pizza kalkulačka</NavDropdown.Item> | ||||
|                     <NavDropdown.Item onClick={() => navigate(STATS_URL)}>Statistiky</NavDropdown.Item> | ||||
|                     <NavDropdown.Divider /> | ||||
|                     <NavDropdown.Item onClick={auth?.logout}>Odhlásit se</NavDropdown.Item> | ||||
|                 </NavDropdown> | ||||
|             </Nav> | ||||
|         </Navbar.Collapse> | ||||
|         <SettingsModal isOpen={settingsModalOpen} onClose={closeSettingsModal} onSave={saveSettings} /> | ||||
|         <RefreshMenuModal isOpen={refreshMenuModalOpen} onClose={closeRefreshMenuModal} /> | ||||
|         <FeaturesVotingModal isOpen={votingModalOpen} onClose={closeVotingModal} onChange={saveFeatureVote} initialValues={featureVotes} /> | ||||
|         <PizzaCalculatorModal isOpen={pizzaModalOpen} onClose={closePizzaModal} /> | ||||
|     </Navbar> | ||||
|  | ||||
| @ -2,15 +2,15 @@ import { IconDefinition } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | ||||
| 
 | ||||
| type Props = { | ||||
|   title?: string, | ||||
|   title?: String, | ||||
|   icon: IconDefinition, | ||||
|   description: string, | ||||
|   animation?: string, | ||||
|   description: String, | ||||
|   animation?: String, | ||||
| } | ||||
| 
 | ||||
| function Loader(props: Readonly<Props>) { | ||||
| function Loader(props: Props) { | ||||
|   return <div className='loader'> | ||||
|     <h1>{props.title ?? 'Prosím čekejte...'}</h1> | ||||
|     <h1>{props.title || 'Prosím čekejte...'}</h1> | ||||
|     <FontAwesomeIcon icon={props.icon} className={`loader-icon mb-3 ` + (props.animation ?? '')} /> | ||||
|     <p>{props.description}</p> | ||||
|   </div> | ||||
|  | ||||
| @ -1,17 +1,18 @@ | ||||
| import { Table } from "react-bootstrap"; | ||||
| import { Order, PizzaDayState, PizzaOrder } from "../types"; | ||||
| import PizzaOrderRow from "./PizzaOrderRow"; | ||||
| import { PizzaDayState, PizzaOrder, PizzaVariant, updatePizzaFee } from "../../../types"; | ||||
| import { updatePizzaFee } from "../api/PizzaDayApi"; | ||||
| 
 | ||||
| type Props = { | ||||
|     state: PizzaDayState, | ||||
|     orders: PizzaOrder[], | ||||
|     onDelete: (pizzaOrder: PizzaVariant) => void, | ||||
|     orders: Order[], | ||||
|     onDelete: (pizzaOrder: PizzaOrder) => void, | ||||
|     creator: string, | ||||
| } | ||||
| 
 | ||||
| export default function PizzaOrderList({ state, orders, onDelete, creator }: Readonly<Props>) { | ||||
| export default function PizzaOrderList({ state, orders, onDelete, creator }: Props) { | ||||
|     const saveFees = async (customer: string, text?: string, price?: number) => { | ||||
|         await updatePizzaFee({ body: { login: customer, text, price } }); | ||||
|         await updatePizzaFee(customer, text, price); | ||||
|     } | ||||
| 
 | ||||
|     if (!orders?.length) { | ||||
| @ -20,7 +21,8 @@ export default function PizzaOrderList({ state, orders, onDelete, creator }: Rea | ||||
| 
 | ||||
|     const total = orders.reduce((total, order) => total + order.totalPrice, 0); | ||||
| 
 | ||||
|     return <Table className="mt-3" striped bordered hover> | ||||
|     return <> | ||||
|         <Table className="mt-3" striped bordered hover> | ||||
|             <thead> | ||||
|                 <tr> | ||||
|                     <th>Jméno</th> | ||||
| @ -40,4 +42,5 @@ export default function PizzaOrderList({ state, orders, onDelete, creator }: Rea | ||||
|                 </tr> | ||||
|             </tbody> | ||||
|         </Table> | ||||
|     </> | ||||
| } | ||||
| @ -2,46 +2,44 @@ import React, { useState } from "react"; | ||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||
| import { faMoneyBill1, faTrashCan } from "@fortawesome/free-regular-svg-icons"; | ||||
| import { useAuth } from "../context/auth"; | ||||
| import { Order, PizzaDayState, PizzaOrder } from "../types"; | ||||
| import PizzaAdditionalFeeModal from "./modals/PizzaAdditionalFeeModal"; | ||||
| import { PizzaDayState, PizzaOrder, PizzaVariant } from "../../../types"; | ||||
| 
 | ||||
| type Props = { | ||||
|     creator: string, | ||||
|     order: PizzaOrder, | ||||
|     order: Order, | ||||
|     state: PizzaDayState, | ||||
|     onDelete: (order: PizzaVariant) => void, | ||||
|     onDelete: (order: PizzaOrder) => void, | ||||
|     onFeeModalSave: (customer: string, name?: string, price?: number) => void, | ||||
| } | ||||
| 
 | ||||
| export default function PizzaOrderRow({ creator, order, state, onDelete, onFeeModalSave }: Readonly<Props>) { | ||||
| export default function PizzaOrderRow({ creator, order, state, onDelete, onFeeModalSave }: Props) { | ||||
|     const auth = useAuth(); | ||||
|     const [isFeeModalOpen, setIsFeeModalOpen] = useState<boolean>(false); | ||||
|     const [isFeeModalOpen, setFeeModalOpen] = useState<boolean>(false); | ||||
| 
 | ||||
|     const saveFees = (customer: string, text?: string, price?: number) => { | ||||
|         onFeeModalSave(customer, text, price); | ||||
|         setIsFeeModalOpen(false); | ||||
|         setFeeModalOpen(false); | ||||
|     } | ||||
| 
 | ||||
|     return <> | ||||
|         <td>{order.customer}</td> | ||||
|         <td>{order.pizzaList!.map<React.ReactNode>(pizzaOrder => | ||||
|             <span key={pizzaOrder.name}> | ||||
|         <td>{order.pizzaList.map<React.ReactNode>((pizzaOrder, index) => | ||||
|             <span key={index}> | ||||
|                 {`${pizzaOrder.name}, ${pizzaOrder.size} (${pizzaOrder.price} Kč)`} | ||||
|                 {auth?.login === order.customer && state === PizzaDayState.CREATED && | ||||
|                     <span title='Odstranit'> | ||||
|                     <FontAwesomeIcon onClick={() => { | ||||
|                         onDelete(pizzaOrder); | ||||
|                         }} className='action-icon' icon={faTrashCan} /> | ||||
|                     </span> | ||||
|                     }} title='Odstranit' className='action-icon' icon={faTrashCan} /> | ||||
|                 } | ||||
|             </span>) | ||||
|             .reduce((prev, curr, index) => [prev, <br key={`br-${index}`} />, curr])} | ||||
|         </td> | ||||
|         <td style={{ maxWidth: "200px" }}>{order.note ?? '-'}</td> | ||||
|         <td style={{ maxWidth: "200px" }}>{order.note || '-'}</td> | ||||
|         <td style={{ maxWidth: "200px" }}>{order.fee?.price ? `${order.fee.price} Kč${order.fee.text ? ` (${order.fee.text})` : ''}` : '-'}</td> | ||||
|         <td> | ||||
|             {order.totalPrice} Kč{auth?.login === creator && state === PizzaDayState.CREATED && <span title='Nastavit příplatek'><FontAwesomeIcon onClick={() => { setIsFeeModalOpen(true) }} className='action-icon' icon={faMoneyBill1} /></span>} | ||||
|             {order.totalPrice} Kč{auth?.login === creator && state === PizzaDayState.CREATED && <FontAwesomeIcon onClick={() => { setFeeModalOpen(true) }} title='Nastavit příplatek' className='action-icon' icon={faMoneyBill1} />} | ||||
|         </td> | ||||
|         <PizzaAdditionalFeeModal customerName={order.customer} isOpen={isFeeModalOpen} onClose={() => setIsFeeModalOpen(false)} onSave={saveFees} initialValues={{ text: order.fee?.text, price: order.fee?.price?.toString() }} /> | ||||
|         <PizzaAdditionalFeeModal customerName={order.customer} isOpen={isFeeModalOpen} onClose={() => setFeeModalOpen(false)} onSave={saveFees} initialValues={{ text: order.fee?.text, price: order.fee?.price?.toString() }} /> | ||||
|     </> | ||||
| } | ||||
| @ -1,5 +1,5 @@ | ||||
| import { Modal, Button, Form } from "react-bootstrap" | ||||
| import { FeatureRequest } from "../../../../types"; | ||||
| import { FeatureRequest } from "../../types"; | ||||
| 
 | ||||
| type Props = { | ||||
|     isOpen: boolean, | ||||
| @ -9,7 +9,7 @@ type Props = { | ||||
| } | ||||
| 
 | ||||
| /** Modální dialog pro hlasování o nových funkcích. */ | ||||
| export default function FeaturesVotingModal({ isOpen, onClose, onChange, initialValues }: Readonly<Props>) { | ||||
| export default function FeaturesVotingModal({ isOpen, onClose, onChange, initialValues }: Props) { | ||||
| 
 | ||||
|     const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { | ||||
|         onChange(e.currentTarget.value as FeatureRequest, e.currentTarget.checked); | ||||
| @ -31,7 +31,7 @@ export default function FeaturesVotingModal({ isOpen, onClose, onChange, initial | ||||
|                     label={FeatureRequest[key]} | ||||
|                     onChange={handleChange} | ||||
|                     value={key} | ||||
|                     defaultChecked={initialValues?.includes(key as FeatureRequest)} | ||||
|                     defaultChecked={initialValues && initialValues.includes(key as FeatureRequest)} | ||||
|                 /> | ||||
|             })} | ||||
|             <p className="mt-3" style={{ fontSize: '12px' }}>Něco jiného? Dejte vědět.</p> | ||||
|  | ||||
| @ -8,7 +8,7 @@ type Props = { | ||||
| } | ||||
| 
 | ||||
| /** Modální dialog pro úpravu obecné poznámky. */ | ||||
| export default function NoteModal({ isOpen, onClose, onSave }: Readonly<Props>) { | ||||
| export default function NoteModal({ isOpen, onClose, onSave }: Props) { | ||||
|     const note = useRef<HTMLInputElement>(null); | ||||
| 
 | ||||
|     const save = () => { | ||||
|  | ||||
| @ -10,17 +10,17 @@ type Props = { | ||||
| } | ||||
| 
 | ||||
| /** Modální dialog pro nastavení příplatků za pizzu. */ | ||||
| export default function PizzaAdditionalFeeModal({ customerName, isOpen, onClose, onSave, initialValues }: Readonly<Props>) { | ||||
| export default function PizzaAdditionalFeeModal({ customerName, isOpen, onClose, onSave, initialValues }: Props) { | ||||
|     const textRef = useRef<HTMLInputElement>(null); | ||||
|     const priceRef = useRef<HTMLInputElement>(null); | ||||
| 
 | ||||
|     const doSubmit = () => { | ||||
|         onSave(customerName, textRef.current?.value, parseInt(priceRef.current?.value ?? "0")); | ||||
|         onSave(customerName, textRef.current?.value, parseInt(priceRef.current?.value || "0")); | ||||
|     } | ||||
| 
 | ||||
|     const handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => { | ||||
|         if (e.key === 'Enter') { | ||||
|             onSave(customerName, textRef.current?.value, parseInt(priceRef.current?.value ?? "0")); | ||||
|             onSave(customerName, textRef.current?.value, parseInt(priceRef.current?.value || "0")); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -23,7 +23,7 @@ type Result = { | ||||
| } | ||||
| 
 | ||||
| /** Modální dialog pro výpočet výhodnosti pizzy. */ | ||||
| export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props>) { | ||||
| export default function PizzaCalculatorModal({ isOpen, onClose }: Props) { | ||||
|     const diameter1Ref = useRef<HTMLInputElement>(null); | ||||
|     const price1Ref = useRef<HTMLInputElement>(null); | ||||
|     const diameter2Ref = useRef<HTMLInputElement>(null); | ||||
| @ -37,7 +37,9 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props | ||||
|         // 1. pizza
 | ||||
|         if (diameter1Ref.current?.value) { | ||||
|             const diameter1 = parseInt(diameter1Ref.current?.value); | ||||
|             r.pizza1 ??= {}; | ||||
|             if (!r.pizza1) { | ||||
|                 r.pizza1 = {}; | ||||
|             } | ||||
|             if (diameter1 && diameter1 > 0) { | ||||
|                 r.pizza1.diameter = diameter1; | ||||
|                 r.pizza1.area = Math.PI * Math.pow(diameter1 / 2, 2); | ||||
| @ -57,7 +59,9 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props | ||||
|         // 2. pizza
 | ||||
|         if (diameter2Ref.current?.value) { | ||||
|             const diameter2 = parseInt(diameter2Ref.current?.value); | ||||
|             r.pizza2 ??= {}; | ||||
|             if (!r.pizza2) { | ||||
|                 r.pizza2 = {}; | ||||
|             } | ||||
|             if (diameter2 && diameter2 > 0) { | ||||
|                 r.pizza2.diameter = diameter2; | ||||
|                 r.pizza2.area = Math.PI * Math.pow(diameter2 / 2, 2); | ||||
|  | ||||
| @ -1,105 +0,0 @@ | ||||
| import { useRef, useState } from "react"; | ||||
| import { Modal, Button, Alert } from "react-bootstrap"; | ||||
| 
 | ||||
| type Props = { | ||||
|     isOpen: boolean; | ||||
|     onClose: () => void; | ||||
| }; | ||||
| 
 | ||||
| /** Modální dialog pro přenačtení menu z restaurací. */ | ||||
| export default function RefreshMenuModal({ isOpen, onClose }: Readonly<Props>) { | ||||
|     const refreshPassRef = useRef<HTMLInputElement>(null); | ||||
|     const refreshTypeRef = useRef<HTMLSelectElement>(null); | ||||
|     const [refreshLoading, setRefreshLoading] = useState(false); | ||||
|     const [refreshMessage, setRefreshMessage] = useState<{ type: 'success' | 'error', text: string } | null>(null); | ||||
| 
 | ||||
|     const handleRefresh = async () => { | ||||
|         const password = refreshPassRef.current?.value; | ||||
|         const type = refreshTypeRef.current?.value; | ||||
|         if (!password || !type) { | ||||
|             setRefreshMessage({ type: 'error', text: 'Zadejte heslo a typ refresh.' }); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         setRefreshLoading(true); | ||||
|         setRefreshMessage(null); | ||||
| 
 | ||||
|         try { | ||||
|             const res = await fetch(`/api/food/refresh?type=${type}&heslo=${encodeURIComponent(password)}`); | ||||
|             const data = await res.json(); | ||||
|             if (res.ok) { | ||||
|                 setRefreshMessage({ type: 'success', text: 'Uspesny fetch' }); | ||||
|                 if (refreshPassRef.current) { | ||||
|                     // Clean hesla xd
 | ||||
|                     refreshPassRef.current.value = ''; | ||||
|                 } | ||||
|             } else { | ||||
|                 setRefreshMessage({ type: 'error', text: data.error || 'Chyba při obnovování jídelníčku.' }); | ||||
|             } | ||||
|         } catch (error) { | ||||
|             console.error('Error refreshing menu:', error); | ||||
|             setRefreshMessage({ type: 'error', text: 'Chyba při obnovování jídelníčku.' }); | ||||
|         } finally { | ||||
|             setRefreshLoading(false); | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     const handleClose = () => { | ||||
|         setRefreshMessage(null); | ||||
|         onClose(); | ||||
|     }; | ||||
| 
 | ||||
|     return ( | ||||
|         <Modal show={isOpen} onHide={handleClose} size="lg"> | ||||
|             <Modal.Header closeButton> | ||||
|                 <Modal.Title><h2>Přenačtení menu</h2></Modal.Title> | ||||
|             </Modal.Header> | ||||
|             <Modal.Body> | ||||
|                 <p>Ruční refresh dat z restaurací.</p> | ||||
| 
 | ||||
|                 {refreshMessage && ( | ||||
|                     <Alert variant={refreshMessage.type === 'success' ? 'success' : 'danger'}> | ||||
|                         {refreshMessage.text} | ||||
|                     </Alert> | ||||
|                 )} | ||||
| 
 | ||||
|                 <div className="mb-3"> | ||||
|                     Heslo: <input | ||||
|                         ref={refreshPassRef} | ||||
|                         type="password" | ||||
|                         placeholder="Zadejte heslo" | ||||
|                         className="form-control d-inline-block" | ||||
|                         style={{ width: 'auto', marginLeft: '10px' }} | ||||
|                         onKeyDown={e => e.stopPropagation()} | ||||
|                     /> | ||||
|                 </div> | ||||
| 
 | ||||
|                 <div className="mb-3"> | ||||
|                     Typ refreshe: <select | ||||
|                         ref={refreshTypeRef} | ||||
|                         className="form-select d-inline-block" | ||||
|                         style={{ width: 'auto', marginLeft: '10px' }} | ||||
|                         defaultValue="week" | ||||
|                     > | ||||
|                         <option value="week">Týden</option> | ||||
|                         <option value="day">Den</option> | ||||
|                     </select> | ||||
|                 </div> | ||||
| 
 | ||||
|                 <Button | ||||
|                     variant="info" | ||||
|                     onClick={handleRefresh} | ||||
|                     disabled={refreshLoading} | ||||
|                     className="mb-3" | ||||
|                 > | ||||
|                     {refreshLoading ? 'Refreshing...' : 'Refresh'} | ||||
|                 </Button> | ||||
|             </Modal.Body> | ||||
|             <Modal.Footer> | ||||
|                 <Button variant="secondary" onClick={handleClose}> | ||||
|                     Zavřít | ||||
|                 </Button> | ||||
|             </Modal.Footer> | ||||
|         </Modal> | ||||
|     ); | ||||
| } | ||||
| @ -9,7 +9,7 @@ type Props = { | ||||
| } | ||||
| 
 | ||||
| /** Modální dialog pro uživatelská nastavení. */ | ||||
| export default function SettingsModal({ isOpen, onClose, onSave }: Readonly<Props>) { | ||||
| export default function SettingsModal({ isOpen, onClose, onSave }: Props) { | ||||
|     const settings = useSettings(); | ||||
|     const bankAccountRef = useRef<HTMLInputElement>(null); | ||||
|     const nameRef = useRef<HTMLInputElement>(null); | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import React, { ReactNode, useContext, useEffect, useState } from "react" | ||||
| import React, { ReactNode, useContext, useState } from "react" | ||||
| import { useEffect } from "react" | ||||
| import { useJwt } from "react-jwt"; | ||||
| import { deleteToken, getToken, storeToken } from "../Utils"; | ||||
| 
 | ||||
| @ -15,7 +16,7 @@ type ContextProps = { | ||||
| 
 | ||||
| const authContext = React.createContext<AuthContextProps | null>(null); | ||||
| 
 | ||||
| export function ProvideAuth(props: Readonly<ContextProps>) { | ||||
| export function ProvideAuth(props: ContextProps) { | ||||
|   const auth = useProvideAuth(); | ||||
|   return <authContext.Provider value={auth}>{props.children}</authContext.Provider> | ||||
| } | ||||
| @ -27,8 +28,8 @@ export const useAuth = () => { | ||||
| function useProvideAuth(): AuthContextProps { | ||||
|   const [loginName, setLoginName] = useState<string | undefined>(); | ||||
|   const [trusted, setTrusted] = useState<boolean | undefined>(); | ||||
|   const [token, setToken] = useState<string | undefined>(getToken()); | ||||
|   const { decodedToken } = useJwt(token ?? ''); | ||||
|   const [token, setToken] = useState<string | null>(getToken()); | ||||
|   const { decodedToken } = useJwt(token || ''); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (token && token.length > 0) { | ||||
| @ -51,7 +52,7 @@ function useProvideAuth(): AuthContextProps { | ||||
|   function logout() { | ||||
|     const trusted = (decodedToken as any).trusted; | ||||
|     const logoutUrl = (decodedToken as any).logoutUrl; | ||||
|     setToken(undefined); | ||||
|     setToken(null); | ||||
|     setLoginName(undefined); | ||||
|     setTrusted(undefined); | ||||
|     if (trusted && logoutUrl?.length) { | ||||
|  | ||||
| @ -1,6 +1,7 @@ | ||||
| import { useEffect, useState } from "react"; | ||||
| import { getEasterEgg } from "../api/EasterEggApi"; | ||||
| import { AuthContextProps } from "./auth"; | ||||
| import { EasterEgg, getEasterEgg } from "../../../types"; | ||||
| import { EasterEgg } from "../types"; | ||||
| 
 | ||||
| export const useEasterEgg = (auth?: AuthContextProps | null): [EasterEgg | undefined, boolean] => { | ||||
|   const [result, setResult] = useState<EasterEgg | undefined>(); | ||||
| @ -10,7 +11,7 @@ export const useEasterEgg = (auth?: AuthContextProps | null): [EasterEgg | undef | ||||
|     async function fetchEasterEgg() { | ||||
|       if (auth?.login) { | ||||
|         setLoading(true); | ||||
|         const egg = (await getEasterEgg())?.data; | ||||
|         const egg = await getEasterEgg(); | ||||
|         setResult(egg); | ||||
|         setLoading(false); | ||||
|       } | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import React, { ReactNode, useContext, useEffect, useState } from "react" | ||||
| import React, { ReactNode, useContext, useState } from "react" | ||||
| import { useEffect } from "react" | ||||
| 
 | ||||
| const BANK_ACCOUNT_NUMBER_KEY = 'bank_account_number'; | ||||
| const BANK_ACCOUNT_HOLDER_KEY = 'bank_account_holder_name'; | ||||
| @ -19,7 +20,7 @@ type ContextProps = { | ||||
| 
 | ||||
| const settingsContext = React.createContext<SettingsContextProps | null>(null); | ||||
| 
 | ||||
| export function ProvideSettings(props: Readonly<ContextProps>) { | ||||
| export function ProvideSettings(props: ContextProps) { | ||||
|   const settings = useProvideSettings(); | ||||
|   return <settingsContext.Provider value={settings}>{props.children}</settingsContext.Provider> | ||||
| } | ||||
| @ -44,7 +45,7 @@ function useProvideSettings(): SettingsContextProps { | ||||
|     } | ||||
|     const hideSoups = localStorage.getItem(HIDE_SOUPS_KEY); | ||||
|     if (hideSoups !== null) { | ||||
|       setHideSoups(hideSoups === 'true'); | ||||
|       setHideSoups(hideSoups === 'true' ? true : false); | ||||
|     } | ||||
|   }, []) | ||||
| 
 | ||||
|  | ||||
| @ -18,3 +18,8 @@ export const SocketContext = React.createContext(); | ||||
| export const EVENT_CONNECT = 'connect'; | ||||
| export const EVENT_DISCONNECT = 'disconnect'; | ||||
| export const EVENT_MESSAGE = 'message'; | ||||
| // export const EVENT_CONFIG = 'config';
 | ||||
| // export const EVENT_TOASTER = 'toaster';
 | ||||
| // export const EVENT_VOTING = 'voting';
 | ||||
| // export const EVENT_VOTE_CONFIG = 'voteSettings';
 | ||||
| // export const EVENT_ADMIN = 'admin';
 | ||||
|  | ||||
| @ -1,41 +0,0 @@ | ||||
| import { LunchChoice, Restaurant } from "../../types"; | ||||
| 
 | ||||
| export function getRestaurantName(restaurant: Restaurant) { | ||||
|     switch (restaurant) { | ||||
|         case Restaurant.SLADOVNICKA: | ||||
|             return "Sladovnická"; | ||||
|         case Restaurant.TECHTOWER: | ||||
|             return "TechTower"; | ||||
|         case Restaurant.ZASTAVKAUMICHALA: | ||||
|             return "Zastávka u Michala"; | ||||
|         case Restaurant.SENKSERIKOVA: | ||||
|             return "Šenk Šeříková"; | ||||
|         default: | ||||
|             return restaurant; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| export function getLunchChoiceName(location: LunchChoice) { | ||||
|     switch (location) { | ||||
|         case Restaurant.SLADOVNICKA: | ||||
|             return "Sladovnická"; | ||||
|         case Restaurant.TECHTOWER: | ||||
|             return "TechTower"; | ||||
|         case Restaurant.ZASTAVKAUMICHALA: | ||||
|             return "Zastávka u Michala"; | ||||
|         case Restaurant.SENKSERIKOVA: | ||||
|             return "Šenk Šeříková"; | ||||
|         case LunchChoice.SPSE: | ||||
|             return "SPŠE"; | ||||
|         case LunchChoice.PIZZA: | ||||
|             return "Pizza day"; | ||||
|         case LunchChoice.OBJEDNAVAM: | ||||
|             return "Budu objednávat"; | ||||
|         case LunchChoice.NEOBEDVAM: | ||||
|             return "Mám vlastní/neobědvám"; | ||||
|         case LunchChoice.ROZHODUJI: | ||||
|             return "Rozhoduji se"; | ||||
|         default: | ||||
|             return location; | ||||
|     } | ||||
| } | ||||
| @ -1,38 +1,33 @@ | ||||
| import React from 'react'; | ||||
| import ReactDOM from 'react-dom/client'; | ||||
| import App from './App'; | ||||
| import { SocketContext, socket } from './context/socket'; | ||||
| import { ProvideAuth } from './context/auth'; | ||||
| import { ToastContainer } from 'react-toastify'; | ||||
| import { ProvideSettings } from './context/settings'; | ||||
| import 'react-toastify/dist/ReactToastify.css'; | ||||
| import './index.css'; | ||||
| import AppRoutes from './AppRoutes'; | ||||
| import { BrowserRouter } from 'react-router'; | ||||
| import { client } from '../../types/gen/client.gen'; | ||||
| import { getToken } from './Utils'; | ||||
| import { toast } from 'react-toastify'; | ||||
| 
 | ||||
| client.setConfig({ | ||||
|   auth: () => getToken(), | ||||
|   baseUrl: '/api', // openapi-ts si to z nějakého důvodu neumí převzít z api.yml
 | ||||
| }); | ||||
| 
 | ||||
| // Interceptor na vyhození toasteru při chybě
 | ||||
| client.interceptors.response.use(async response => { | ||||
|   // TODO opravit - login je zatím výjimka, voláme ho "naprázdno" abychom zjistili, zda nás nepřihlásily trusted headers
 | ||||
|   if (!response.ok && response.url.indexOf("/login") == -1) { | ||||
|     const json = await response.json(); | ||||
|     toast.error(json.error, { theme: "colored" }); | ||||
|   } | ||||
|   return response; | ||||
| }); | ||||
| import Snowfall from 'react-snowfall'; | ||||
| 
 | ||||
| const root = ReactDOM.createRoot( | ||||
|   document.getElementById('root') as HTMLElement | ||||
| ); | ||||
| root.render( | ||||
|   <React.StrictMode> | ||||
|     <BrowserRouter> | ||||
|     <ProvideAuth> | ||||
|         <AppRoutes /> | ||||
|       <ProvideSettings> | ||||
|         <SocketContext.Provider value={socket}> | ||||
|           <> | ||||
|             <Snowfall style={{ | ||||
|               zIndex: 2, | ||||
|               position: 'fixed', | ||||
|               width: '100vw', | ||||
|               height: '100vh'}} /> | ||||
|             <App /> | ||||
|           </> | ||||
|           <ToastContainer /> | ||||
|         </SocketContext.Provider> | ||||
|       </ProvideSettings> | ||||
|     </ProvideAuth> | ||||
|     </BrowserRouter> | ||||
|   </React.StrictMode> | ||||
| ); | ||||
|  | ||||
| @ -1,16 +0,0 @@ | ||||
| .stats-page { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   align-items: center; | ||||
|   padding: 20px; | ||||
| 
 | ||||
|   .week-navigator { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     font-size: xx-large; | ||||
| 
 | ||||
|     .date-range { | ||||
|       margin: 5px 20px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -1,129 +0,0 @@ | ||||
| import { useCallback, useEffect, useState } from "react"; | ||||
| import Footer from "../components/Footer"; | ||||
| import Header from "../components/Header"; | ||||
| import { useAuth } from "../context/auth"; | ||||
| import Login from "../Login"; | ||||
| import { formatDate, getFirstWorkDayOfWeek, getHumanDate, getLastWorkDayOfWeek } from "../Utils"; | ||||
| import { WeeklyStats, LunchChoice, getStats } from "../../../types"; | ||||
| import Loader from "../components/Loader"; | ||||
| import { faChevronLeft, faChevronRight, faGear } from "@fortawesome/free-solid-svg-icons"; | ||||
| import { Legend, Line, LineChart, Tooltip, XAxis, YAxis } from "recharts"; | ||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||
| import { getLunchChoiceName } from "../enums"; | ||||
| import './StatsPage.scss'; | ||||
| 
 | ||||
| const CHART_WIDTH = 1400; | ||||
| const CHART_HEIGHT = 700; | ||||
| const STROKE_WIDTH = 2.5; | ||||
| 
 | ||||
| const COLORS = [ | ||||
|   // Komentáře jsou kvůli vizualizaci barev ve VS Code
 | ||||
|   '#ff1493', // #ff1493
 | ||||
|   '#1e90ff', // #1e90ff
 | ||||
|   '#c5a700', // #c5a700
 | ||||
|   '#006400', // #006400
 | ||||
|   '#b300ff', // #b300ff
 | ||||
|   '#ff4500', // #ff4500
 | ||||
|   '#bc8f8f', // #bc8f8f
 | ||||
|   '#00ff00', // #00ff00
 | ||||
|   '#7c7c7c', // #7c7c7c
 | ||||
| ] | ||||
| 
 | ||||
| export default function StatsPage() { | ||||
|   const auth = useAuth(); | ||||
|   const [dateRange, setDateRange] = useState<Date[]>(); | ||||
|   const [data, setData] = useState<WeeklyStats>(); | ||||
| 
 | ||||
|   // Prvotní nastavení aktuálního týdne
 | ||||
|   useEffect(() => { | ||||
|     const today = new Date(); | ||||
|     setDateRange([getFirstWorkDayOfWeek(today), getLastWorkDayOfWeek(today)]); | ||||
|   }, []); | ||||
| 
 | ||||
|   // Přenačtení pro zvolený týden
 | ||||
|   useEffect(() => { | ||||
|     if (dateRange) { | ||||
|       getStats({ query: { startDate: formatDate(dateRange[0]), endDate: formatDate(dateRange[1]) } }).then(response => { | ||||
|         setData(response.data); | ||||
|       }); | ||||
|     } | ||||
|   }, [dateRange]); | ||||
| 
 | ||||
|   const renderLine = (location: LunchChoice) => { | ||||
|     const index = Object.values(LunchChoice).indexOf(location); | ||||
|     return <Line key={location} name={getLunchChoiceName(location)} type="monotone" dataKey={data => data.locations[location] ?? 0} stroke={COLORS[index]} strokeWidth={STROKE_WIDTH} /> | ||||
|   } | ||||
| 
 | ||||
|   const handlePreviousWeek = () => { | ||||
|     if (dateRange) { | ||||
|       const previousStartDate = new Date(dateRange[0]); | ||||
|       previousStartDate.setDate(previousStartDate.getDate() - 7); | ||||
|       const previousEndDate = new Date(previousStartDate); | ||||
|       previousEndDate.setDate(previousEndDate.getDate() + 4); | ||||
|       setDateRange([previousStartDate, previousEndDate]); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const handleNextWeek = () => { | ||||
|     if (dateRange) { | ||||
|       const nextStartDate = new Date(dateRange[0]); | ||||
|       nextStartDate.setDate(nextStartDate.getDate() + 7); | ||||
|       const nextEndDate = new Date(nextStartDate); | ||||
|       nextEndDate.setDate(nextEndDate.getDate() + 4); | ||||
|       setDateRange([nextStartDate, nextEndDate]); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const handleKeyDown = useCallback((e: any) => { | ||||
|     if (e.keyCode === 37) { | ||||
|       handlePreviousWeek(); | ||||
|     } else if (e.keyCode === 39) { | ||||
|       handleNextWeek() | ||||
|     } | ||||
|   }, [dateRange]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     } | ||||
|   }, [handleKeyDown]); | ||||
| 
 | ||||
|   if (!auth?.login) { | ||||
|     return <Login />; | ||||
|   } | ||||
| 
 | ||||
|   if (!dateRange) { | ||||
|     return <Loader | ||||
|       icon={faGear} | ||||
|       description={'Načítám data...'} | ||||
|       animation={'fa-bounce'} | ||||
|     /> | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <> | ||||
|       <Header /> | ||||
|       <div className="stats-page"> | ||||
|         <h1>Statistiky</h1> | ||||
|         <div className="week-navigator"> | ||||
|           <span title="Předchozí týden"> | ||||
|             <FontAwesomeIcon icon={faChevronLeft} style={{ cursor: "pointer" }} onClick={handlePreviousWeek} /> | ||||
|           </span> | ||||
|           <h2 className="date-range">{getHumanDate(dateRange[0])} - {getHumanDate(dateRange[1])}</h2> | ||||
|           <span title="Následující týden"> | ||||
|             <FontAwesomeIcon icon={faChevronRight} style={{ cursor: "pointer" }} onClick={handleNextWeek} /> | ||||
|           </span> | ||||
|         </div> | ||||
|         <LineChart width={CHART_WIDTH} height={CHART_HEIGHT} data={data}> | ||||
|           {Object.values(LunchChoice).map(location => renderLine(location))} | ||||
|           <XAxis dataKey="date" /> | ||||
|           <YAxis /> | ||||
|           <Tooltip /> | ||||
|           <Legend /> | ||||
|         </LineChart> | ||||
|       </div> | ||||
|       <Footer /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| @ -1,5 +1,6 @@ | ||||
| { | ||||
|   "compilerOptions": { | ||||
|     "target": "ESNext", | ||||
|     "lib": [ | ||||
|       "dom", | ||||
|       "dom.iterable", | ||||
| @ -15,12 +16,10 @@ | ||||
|     "strict": true, | ||||
|     "forceConsistentCasingInFileNames": true, | ||||
|     "noFallthroughCasesInSwitch": true, | ||||
|     "moduleResolution": "bundler", | ||||
|     "module": "ESNext", | ||||
|     "target": "ESNext", | ||||
|     "module": "esnext", | ||||
|     "moduleResolution": "node", | ||||
|     "resolveJsonModule": true, | ||||
|     "isolatedModules": true, | ||||
|     "allowImportingTsExtensions": true, | ||||
|     "noEmit": true, | ||||
|     "jsx": "react-jsx" | ||||
|   } | ||||
|  | ||||
							
								
								
									
										1571
									
								
								client/yarn.lock
									
									
									
									
									
								
							
							
						
						
									
										1571
									
								
								client/yarn.lock
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										23
									
								
								run_dev.sh
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								run_dev.sh
									
									
									
									
									
								
							| @ -1,18 +1,5 @@ | ||||
| #!/bin/bash | ||||
| # Spustí server a klienta v samostatných panelech uvnitř stejného tmux okna. | ||||
| # Pokud už daná tmux session existuje, pouze se k ní připojí. | ||||
| 
 | ||||
| SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) | ||||
| 
 | ||||
| SESSION="luncher" | ||||
| 
 | ||||
| if ! tmux has-session -t $SESSION 2>/dev/null; then | ||||
|     cd types && yarn openapi-ts && cd .. | ||||
|     tmux new-session -d -s $SESSION | ||||
|     tmux send-keys -t $SESSION:0 "cd $SCRIPT_DIR" Enter | ||||
|     tmux split-window -v | ||||
|     tmux send-keys -t $SESSION:0.0 "cd server && export NODE_ENV=development && yarn startReload" Enter | ||||
|     tmux send-keys -t $SESSION:0.1 "cd client && yarn start" Enter | ||||
| fi | ||||
| 
 | ||||
| tmux attach-session -t $SESSION | ||||
| export NODE_ENV=development | ||||
| yarn install | ||||
| cd server && yarn start & | ||||
| cd client && yarn start & | ||||
| wait | ||||
							
								
								
									
										6
									
								
								server/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								server/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,7 +1,7 @@ | ||||
| /data | ||||
| /node_modules | ||||
| /dist | ||||
| /resources/easterEggs | ||||
| /src/gen | ||||
| data.json | ||||
| .env.production | ||||
| .env.development | ||||
| .easter-eggs.json | ||||
| /resources/easterEggs | ||||
| @ -11,28 +11,28 @@ | ||||
|     "test": "jest" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/core": "^7.28.4", | ||||
|     "@babel/preset-env": "^7.28.3", | ||||
|     "@babel/preset-typescript": "^7.27.1", | ||||
|     "@types/express": "^5.0.3", | ||||
|     "@types/jest": "^30.0.0", | ||||
|     "@types/jsonwebtoken": "^9.0.10", | ||||
|     "@types/node": "^24.7.1", | ||||
|     "@babel/core": "^7.23.0", | ||||
|     "@babel/preset-env": "^7.22.20", | ||||
|     "@babel/preset-typescript": "^7.23.0", | ||||
|     "@types/express": "^4.17.17", | ||||
|     "@types/jest": "^29.5.14", | ||||
|     "@types/jsonwebtoken": "^9.0.6", | ||||
|     "@types/node": "^20.11.20", | ||||
|     "@types/request-promise": "^4.1.48", | ||||
|     "babel-jest": "^30.2.0", | ||||
|     "jest": "^30.2.0", | ||||
|     "nodemon": "^3.1.10", | ||||
|     "babel-jest": "^29.7.0", | ||||
|     "jest": "^29.7.0", | ||||
|     "nodemon": "^3.1.0", | ||||
|     "ts-node": "^10.9.1", | ||||
|     "typescript": "^5.9.3" | ||||
|     "typescript": "^5.0.2" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "axios": "^1.12.2", | ||||
|     "cheerio": "^1.1.2", | ||||
|     "axios": "^1.4.0", | ||||
|     "cheerio": "^1.0.0-rc.12", | ||||
|     "cors": "^2.8.5", | ||||
|     "dotenv": "^17.2.3", | ||||
|     "express": "^5.1.0", | ||||
|     "dotenv": "^16.4.5", | ||||
|     "express": "^4.18.2", | ||||
|     "jsonwebtoken": "^9.0.0", | ||||
|     "redis": "^5.8.3", | ||||
|     "redis": "^4.6.7", | ||||
|     "simple-json-db": "^2.0.0", | ||||
|     "socket.io": "^4.6.1" | ||||
|   } | ||||
|  | ||||
| @ -49,16 +49,16 @@ export async function downloadPizzy(mock: boolean): Promise<Pizza[]> { | ||||
|     const $ = load(html); | ||||
|     const links = $('.vypisproduktu > div > h4 > a'); | ||||
|     const urls = []; | ||||
|     for (const element of links) { | ||||
|         if (element.name === 'a' && element.attribs?.href) { | ||||
|             const pizzaUrl = element.attribs?.href; | ||||
|     for (let i = 0; i < links.length; i++) { | ||||
|         if (links[i].name === 'a' && links[i].attribs?.href) { | ||||
|             const pizzaUrl = links[i].attribs?.href; | ||||
|             urls.push(buildPizzaUrl(pizzaUrl)); | ||||
|         } | ||||
|     } | ||||
|     // Scrapneme jednotlivé pizzy
 | ||||
|     const result: Pizza[] = []; | ||||
|     for (const element of urls) { | ||||
|         const pizzaUrl = element; | ||||
|     for (let i = 0; i < urls.length; i++) { | ||||
|         const pizzaUrl = urls[i]; | ||||
|         const pizzaHtml = await axios.get(pizzaUrl).then(res => res.data); | ||||
|         // Název
 | ||||
|         const name = $('.produkt > h2', pizzaHtml).first().text() | ||||
|  | ||||
| @ -9,13 +9,12 @@ import { generateToken, verify } from "./auth"; | ||||
| import { InsufficientPermissions } from "./utils"; | ||||
| import { initWebsocket } from "./websocket"; | ||||
| import pizzaDayRoutes from "./routes/pizzaDayRoutes"; | ||||
| import foodRoutes, { refreshMetoda } from "./routes/foodRoutes"; | ||||
| import foodRoutes from "./routes/foodRoutes"; | ||||
| import votingRoutes from "./routes/votingRoutes"; | ||||
| import easterEggRoutes from "./routes/easterEggRoutes"; | ||||
| import statsRoutes from "./routes/statsRoutes"; | ||||
| 
 | ||||
| const ENVIRONMENT = process.env.NODE_ENV ?? 'production'; | ||||
| dotenv.config({ path: path.resolve(__dirname, `../.env.${ENVIRONMENT}`) }); | ||||
| const ENVIRONMENT = process.env.NODE_ENV || 'production'; | ||||
| dotenv.config({ path: path.resolve(__dirname, `./.env.${ENVIRONMENT}`) }); | ||||
| 
 | ||||
| // Validace nastavení JWT tokenu - nemá bez něj smysl vůbec povolit server spustit
 | ||||
| if (!process.env.JWT_SECRET) { | ||||
| @ -35,7 +34,7 @@ app.use(cors({ | ||||
| 
 | ||||
| // Zapínatelný login přes hlavičky - pokud je zapnutý nepovolí "basicauth"
 | ||||
| const HTTP_REMOTE_USER_ENABLED = process.env.HTTP_REMOTE_USER_ENABLED === 'true' || false; | ||||
| const HTTP_REMOTE_USER_HEADER_NAME = process.env.HTTP_REMOTE_USER_HEADER_NAME ?? 'remote-user'; | ||||
| const HTTP_REMOTE_USER_HEADER_NAME = process.env.HTTP_REMOTE_USER_HEADER_NAME || 'remote-user'; | ||||
| if (HTTP_REMOTE_USER_ENABLED) { | ||||
|     if (!process.env.HTTP_REMOTE_TRUSTED_IPS) { | ||||
|         throw new Error('Je zapnutý login z hlaviček, ale není nastaven rozsah adres ze kterých hlavička může přijít.'); | ||||
| @ -54,10 +53,6 @@ app.get("/api/whoami", (req, res) => { | ||||
|     if (!HTTP_REMOTE_USER_ENABLED) { | ||||
|         res.status(403).json({ error: 'Není zapnuté přihlášení z hlaviček' }); | ||||
|     } | ||||
|     if(process.env.ENABLE_HEADERS_LOGGING === 'yes'){ | ||||
|         delete req.headers["cookie"] | ||||
|         console.log(req.headers) | ||||
|     } | ||||
|     res.send(req.header(HTTP_REMOTE_USER_HEADER_NAME)); | ||||
| }) | ||||
| 
 | ||||
| @ -65,11 +60,11 @@ app.post("/api/login", (req, res) => { | ||||
|     if (HTTP_REMOTE_USER_ENABLED) { // je rovno app.enabled('trust proxy')
 | ||||
|         // Autentizace pomocí trusted headers
 | ||||
|         const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME); | ||||
|         //const remoteName = req.header('remote-name');
 | ||||
|         if (remoteUser && remoteUser.length > 0 ) { | ||||
|             res.status(200).json(generateToken(Buffer.from(remoteUser, 'latin1').toString(), true)); | ||||
|         const remoteName = req.header('remote-name'); | ||||
|         if (remoteUser && remoteUser.length > 0 && remoteName && remoteName.length > 0) { | ||||
|             res.status(200).json(generateToken(Buffer.from(remoteName, 'latin1').toString(), true)); | ||||
|         } else { | ||||
|             throw Error("Je zapnuto přihlášení přes hlavičky, ale nepřišla hlavička nebo ??"); | ||||
|             throw Error("Tohle nema nastat nekdo neco dela spatne."); | ||||
|         } | ||||
|     } else { | ||||
|         // Klasická autentizace loginem
 | ||||
| @ -83,6 +78,7 @@ app.post("/api/login", (req, res) => { | ||||
| 
 | ||||
| // TODO dočasné řešení - QR se zobrazuje přes <img>, nemáme sem jak dostat token
 | ||||
| app.get("/api/qr", (req, res) => { | ||||
|     // const login = getLogin(parseToken(req));
 | ||||
|     if (!req.query?.login) { | ||||
|         throw Error("Nebyl předán login"); | ||||
|     } | ||||
| @ -96,23 +92,15 @@ app.get("/api/qr", (req, res) => { | ||||
| 
 | ||||
| // ----------------------------------------------------
 | ||||
| 
 | ||||
| // Přeskočení auth pro refresh dat xd
 | ||||
| app.use("/api/food/refresh", refreshMetoda); | ||||
| 
 | ||||
| /** Middleware ověřující JWT token */ | ||||
| app.use("/api/", (req, res, next) => { | ||||
|     if (HTTP_REMOTE_USER_ENABLED) { | ||||
|         // Autentizace pomocí trusted headers
 | ||||
|         const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME); | ||||
|         if(process.env.ENABLE_HEADERS_LOGGING === 'yes'){ | ||||
|             delete req.headers["cookie"] | ||||
|             console.log(req.headers) | ||||
|         } | ||||
|         if (remoteUser && remoteUser.length > 0) { | ||||
|             const remoteName = Buffer.from(remoteUser, 'latin1').toString(); | ||||
|             if (ENVIRONMENT !== "production") { | ||||
|                 console.log("Tvuj username: %s.", remoteName); | ||||
|             } | ||||
|         const userHeader = req.header(HTTP_REMOTE_USER_HEADER_NAME); | ||||
|         const nameHeader = req.header('remote-name'); | ||||
|         const emailHeader = req.header('remote-email'); | ||||
|         if (userHeader !== undefined && nameHeader !== undefined) { | ||||
|             const remoteName = Buffer.from(nameHeader, 'latin1').toString(); | ||||
|             console.log("Tvuj username, name a email: %s, %s, %s.", userHeader, remoteName, emailHeader); | ||||
|         } | ||||
|     } | ||||
|     if (!req.headers.authorization) { | ||||
| @ -142,10 +130,7 @@ app.use("/api/pizzaDay", pizzaDayRoutes); | ||||
| app.use("/api/food", foodRoutes); | ||||
| app.use("/api/voting", votingRoutes); | ||||
| app.use("/api/easterEggs", easterEggRoutes); | ||||
| app.use("/api/stats", statsRoutes); | ||||
| 
 | ||||
| app.use('/stats', express.static('public')); | ||||
| app.use(express.static('public')); | ||||
| app.use(express.static('public')) | ||||
| 
 | ||||
| // Middleware pro zpracování chyb
 | ||||
| app.use((err: any, req: any, res: any, next: any) => { | ||||
| @ -157,8 +142,8 @@ app.use((err: any, req: any, res: any, next: any) => { | ||||
|     next(); | ||||
| }); | ||||
| 
 | ||||
| const PORT = process.env.PORT ?? 3001; | ||||
| const HOST = process.env.HOST ?? '0.0.0.0'; | ||||
| const PORT = process.env.PORT || 3001; | ||||
| const HOST = process.env.HOST || '0.0.0.0'; | ||||
| 
 | ||||
| server.listen(PORT, () => { | ||||
|     console.log(`Server listening on ${HOST}, port ${PORT}`); | ||||
|  | ||||
| @ -1,159 +1,137 @@ | ||||
| import { WeeklyStats, LunchChoice } from "../../types/gen/types.gen"; | ||||
| 
 | ||||
| // Mockovací data pro podporované podniky, na jeden týden
 | ||||
| const MOCK_DATA = { | ||||
|     'sladovnicka': [ | ||||
|         [ | ||||
|     'sladovnicka': { | ||||
|         'MONDAY': [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Česnečka s uzeným masem a krutony", | ||||
|                 name: "Kulajda", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 3, 7, 9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "250g", | ||||
|                 name: "Přírodní roštěná s jasmínovou rýží", | ||||
|                 name: "Kuřecí křidélka s vařeným bramborem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 9, 10] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Noky s kuřecím masem a sýrovou omáčkou", | ||||
|                 name: "Hovězí hamburger s BBQ omáčkou a hranolky", | ||||
|                 price: "145\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Kuřecí stehno pečené na Moravance s feferony, bramborový knedlík", | ||||
|                 name: "Frankfurtská hovězí pečeně s jasmínovou rýží", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7, 9] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|         'TUESDAY': [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Hovězí vývar s kapáním", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "200g", | ||||
|                 name: "Smažený karbanátek s bramborovou kaší", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Vepřová plec na smetaně s kynutým knedlíkem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Trhané kachní maso se zeleninovým kuskusem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         'WEDNESDAY': [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Zelná polévka s klobásou", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Hovězí na česneku s bramborovým knedlíkem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "250g", | ||||
|                 name: "Přírodní holandský řízek s bramborovou kaší, rajčatový salát", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "350g", | ||||
|                 name: "Bagel s vinnou klobásou, cibulový konfit, kysané zelí, slanina a hořčicová mayo, hranolky, curry omáčka", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         'THURSDAY': [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Kuřecí vývar s nudlemi", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 3, 7, 9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "200g", | ||||
|                 name: "Hovězí maso v rajské omáčce s kynutým knedlíkem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Krůtí roláda se sušenými rajčaty , mozzarellou a bramborovou kaší", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Telecí játra na grilu, restované brambory, tatarská omáčka , polníčkový salát", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [3, 7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Zeleninová polévka", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [3, 9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Smažené rybí filé s vařeným bramborem, tatarka", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "250g", | ||||
|                 name: "Vepřové výpečky se špenátem, bramborový knedlík 1,3,7", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "350g", | ||||
|                 name: "Kuřecí řízek \" Ondráš \" , kysané zelí", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [3, 7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Hovězí vývar s játrovými knedlíčky", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 3] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Merguez klobáska, bílé fazole na kyselo, sázené vejce a vídeňská cibulka", | ||||
|                 name: "Kovbojské fazole s klobásou a chlebem", | ||||
|                 price: "125\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Kuřecí steak s liškovou omáčkou a opečený brambor", | ||||
|                 name: "Kuřecí rarášci s vařeným bramborem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Kaťák – vepřové kostky s feferonou, cibulí, kečupem ,česnekem, smažené krokety", | ||||
|                 name: "Hovězí pečeně na slanině s jasmínovou rýží", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [3, 7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|         'FRIDAY': [ | ||||
|             { | ||||
|                 amount: "0,25l", | ||||
|                 name: "Čočková polévka", | ||||
|                 name: "Dršťková polévka", | ||||
|                 price: "35\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [9, 12] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Ovocné knedlíky s tvarohem", | ||||
|                 name: "Tortilla s kuřecím masem, čedarem, zeleninou a papričkami jalapeňos", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Smažený vepřový řízek s bramborovým salátem", | ||||
|                 name: "Segedínský guláš s kynutým knedlíkem", | ||||
|                 price: "135\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7, 9, 10] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "150g", | ||||
|                 name: "Znojemský hovězí guláš s jasmínovou rýží", | ||||
|                 name: "Filet z krůtích prsou, omáčka z modrého sýra, pečené brambory", | ||||
|                 price: "145\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 9] | ||||
|             } | ||||
|         ] | ||||
|     ], | ||||
|     }, | ||||
|     'uMotliku': [ | ||||
|         [ | ||||
|             { | ||||
| @ -290,185 +268,161 @@ const MOCK_DATA = { | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Batátový krém s chilli a kokosovým mlékem", | ||||
|                 name: "Uzený vývar s kapustou", | ||||
|                 price: "40\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí stehno na paprice, knedlík", | ||||
|                 name: "Čočka na kyselo, opečená klobása, okurka, chléb", | ||||
|                 price: "130\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Těstoviny se sušenými rajčaty a cuketou, parmezán", | ||||
|                 name: "Smažená brokolice, brambory, tatarská omáčka", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Quesadilla s trhaným vepřovým masem, salát coleslaw, hranolky", | ||||
|                 name: "Uzený vepřový bůček, bramborové pyré", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Smažený kuřecí řízek v sezamové strouhance, vařené brambory, wasabi majonéza", | ||||
|                 name: "Kuřecí medailonky v sýrové omáčce, hranolky", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Ovarová", | ||||
|                 name: "Slepičí s nudlemi", | ||||
|                 price: "40\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Zapečené těstoviny s uzeným masem, okurka", | ||||
|                 name: "Zvěřinový guláš, knedlík", | ||||
|                 price: "130\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Cheddarové kuličky s jalapeños, máslové brambory, tatarská omáčka", | ||||
|                 name: "Čínské nudle se zeleninou a vejcem", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Steak z krkovice s miso omáčkou, jasmínová rýže", | ||||
|                 name: "Jitrnice/jelito, brambory, zelný salát s křenem, hořčice", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 6, 11] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí supreme s bramborovo-mrkvovým pyré, restovaná cuketa", | ||||
|                 name: "Vídeňská roštěná se smaženou cibulkou, jasmínová rýže", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Hovězí s hráškem a rýží", | ||||
|                 name: "Dýňový krém se smetanou", | ||||
|                 price: "40\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Rizoto s kuřecím masem a zeleninou, okurka, sýr", | ||||
|                 name: "Kuřecí směs se zeleninou, rýže", | ||||
|                 price: "130\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [7, 9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Smažené rýžové nudle Pad thai s arašídy, zeleninou a vejcem", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 6, 8, 11] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Vykoštěné vepřové koleno s křenem a hořčicí, chléb", | ||||
|                 name: "Tvarohové knedlíky s meruňkami, strouhaný tvaroh, máslo, cukr", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Gordon bleu, hranolky, pikantní dip", | ||||
|                 name: "Ovar, křen, hořčice, pečivo", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Telecí holandský řízek s uzeným sýrem, bramborové pyré", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Dýňová", | ||||
|                 name: "Zeleninová s jáhly", | ||||
|                 price: "40\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Uzená plec, křenová omáčka, knedlík", | ||||
|                 name: "Rizoto s vepřovým masem, okurka", | ||||
|                 price: "130\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Palačinky s marmeládou přelité čokoládou, sypané cukrem", | ||||
|                 name: "Tortellini s parmezánovou omáčkou", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Smažený holandský řízek s bramborovou kaší a nakládanou zeleninou", | ||||
|                 name: "Pečený prejt, brambory, zelný salát", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí jatýrka na smetaně s čerstvou majoránkou, jasmínová rýže", | ||||
|                 name: "Chobotnice na grilu, grilovaná zelenina, bylinková bageta", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [7] | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Hovězí vývar s játrovými knedlíčky", | ||||
|                 name: "Fazolová s uzeninou", | ||||
|                 price: "40\xA0Kč", | ||||
|                 isSoup: true, | ||||
|                 allergens: [1, 3, 7, 9] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí Kung-pao, jasmínová rýže", | ||||
|                 name: "Krůtí perkelt, těstoviny", | ||||
|                 price: "130\xA0Kč", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 5, 6] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Sýrové tortelliny s pažitkovou omáčkou", | ||||
|                 name: "Grilovaný hermelín, bulgurový salát se zeleninou", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Teriyaki losos burger s frisée salátem a citrusovou majonézou, bramborové lupínky", | ||||
|                 name: "Zabijačkový guláš, karlovarský knedlík", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 6, 7, 11] | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Vepřové výpečky s červeným zelím, bramborové knedlíky se smaženou cibulkou", | ||||
|                 name: "Vepřový plátek na žampionech, jasmínová rýže", | ||||
|                 price: "na\xA0váhu", | ||||
|                 isSoup: false, | ||||
|                 allergens: [1, 3, 7] | ||||
|             } | ||||
|         ] | ||||
|     ], | ||||
| @ -561,91 +515,7 @@ const MOCK_DATA = { | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|     ], | ||||
|     'senkSerikova': [ | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Drůbeží vývar s masem a nudlemi", | ||||
|                 price: "45\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Vepřová pečeně se zelím a houskovým knedlíkem", | ||||
|                 price: "155\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Špagety s kuřecím masem, špenátem a smetanou", | ||||
|                 price: "145\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Medailonky z vepřové panenky s fazolkami se slaninou, šťouchané brambory", | ||||
|                 price: "185\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Mrkvová polévka se zázvorem a kokosovým mlékem", | ||||
|                 price: "45\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Hovězí po Burgundsku, bramborová kaše", | ||||
|                 price: "155\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Hovězí vývar s játrovými knedlíčky", | ||||
|                 price: "45\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí plátky na sušených rajčatech, bylinkách a česneku, bramborová kaše", | ||||
|                 price: "155\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Kuřecí vývar s rýží", | ||||
|                 price: "45\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Rajská s plněnou paprikou, knedlík", | ||||
|                 price: "170\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|         [ | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Mexická fazolová polévka", | ||||
|                 price: "45\xA0Kč", | ||||
|                 isSoup: true, | ||||
|             }, | ||||
|             { | ||||
|                 amount: "-", | ||||
|                 name: "Ragú z trhané kachny, onsen vejce, soté ze špenátu a ředkvičky, bramborové pyré, lanýžová sůl, zelený olej", | ||||
|                 price: "189\xA0Kč", | ||||
|                 isSoup: false, | ||||
|             } | ||||
|         ], | ||||
|     ], | ||||
|     ] | ||||
| } | ||||
| 
 | ||||
| // Mockovací data pro Pizza day
 | ||||
| @ -1402,7 +1272,7 @@ const MOCK_PIZZA_LIST = [ | ||||
|  * Funkce vrací mock datu ve formátu YYYY-MM-DD | ||||
|  */ | ||||
| export const getTodayMock = (): Date => { | ||||
|     return new Date('2025-01-10'); // pátek
 | ||||
|     return new Date('2025-01-08'); // pátek
 | ||||
| } | ||||
| 
 | ||||
| export const getMenuSladovnickaMock = () => { | ||||
| @ -1421,35 +1291,6 @@ export const getMenuZastavkaUmichalaMock = () => { | ||||
|     return MOCK_DATA['zastavkaUmichala']; | ||||
| } | ||||
| 
 | ||||
| export const getMenuSenkSerikovaMock = () => { | ||||
|     return MOCK_DATA['senkSerikova']; | ||||
| } | ||||
| 
 | ||||
| export const getPizzaListMock = () => { | ||||
|     return MOCK_PIZZA_LIST; | ||||
| } | ||||
| 
 | ||||
| export const getStatsMock = (): WeeklyStats => { | ||||
|     return [ | ||||
|         { | ||||
|             date: '24.02.', | ||||
|             locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) } | ||||
|         }, | ||||
|         { | ||||
|             date: '25.02.', | ||||
|             locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) } | ||||
|         }, | ||||
|         { | ||||
|             date: '26.02.', | ||||
|             locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) } | ||||
|         }, | ||||
|         { | ||||
|             date: '27.02.', | ||||
|             locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) } | ||||
|         }, | ||||
|         { | ||||
|             date: '28.02.', | ||||
|             locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) } | ||||
|         } | ||||
|     ]; | ||||
| } | ||||
| @ -1,14 +1,60 @@ | ||||
| import axios from 'axios'; | ||||
| /** Notifikace pro gotify*/ | ||||
| import { ClientData, GotifyServer, NotififaceInput, NotifikaceData } from '../../types'; | ||||
| import axios, { AxiosError } from 'axios'; | ||||
| import dotenv from 'dotenv'; | ||||
| import path from 'path'; | ||||
| import { getClientData, getToday } from "./service"; | ||||
| import { getUsersByLocation, getHumanTime } from "./utils"; | ||||
| import { NotifikaceData, NotifikaceInput } from '../../types'; | ||||
| import { getToday } from "./service"; | ||||
| import { formatDate, getUsersByLocation } from "./utils"; | ||||
| import getStorage from "./storage"; | ||||
| 
 | ||||
| const ENVIRONMENT = process.env.NODE_ENV ?? 'production'; | ||||
| const storage = getStorage(); | ||||
| const ENVIRONMENT = process.env.NODE_ENV || 'production' | ||||
| dotenv.config({ path: path.resolve(__dirname, `../.env.${ENVIRONMENT}`) }); | ||||
| 
 | ||||
| export const ntfyCall = async (data: NotifikaceInput) => { | ||||
| const gotifyDataRaw = process.env.GOTIFY_SERVERS_AND_KEYS || "{}"; | ||||
| const gotifyData: GotifyServer[] = JSON.parse(gotifyDataRaw); | ||||
| export const gotifyCall = async (data: NotififaceInput, gotifyServers?: GotifyServer[]): Promise<any[]> => { | ||||
|     if (!Array.isArray(gotifyServers)) { | ||||
|         return [] | ||||
|     } | ||||
|     const urls = gotifyServers.flatMap(gotifyServer => | ||||
|         gotifyServer.api_keys.map(apiKey => `${gotifyServer.server}/message?token=${apiKey}`)); | ||||
| 
 | ||||
|     const dataPayload = { | ||||
|         title: "Luncher", | ||||
|         message: `${data.udalost} - spustil:${data.user}`, | ||||
|         priority: 7, | ||||
|     }; | ||||
| 
 | ||||
|     const headers = { "Content-Type": "application/json" }; | ||||
| 
 | ||||
|     const promises = urls.map(url => | ||||
|         axios.post(url, dataPayload, { headers }).then(response => { | ||||
|             response.data = { | ||||
|                 success: true, | ||||
|                 message: "Notifikace doručena", | ||||
|             }; | ||||
|             return response; | ||||
|         }).catch(error => { | ||||
|             if (axios.isAxiosError(error)) { | ||||
|                 const axiosError = error as AxiosError; | ||||
|                 if (axiosError.response) { | ||||
|                     axiosError.response.data = { | ||||
|                         success: false, | ||||
|                         message: "fail", | ||||
|                     }; | ||||
|                     console.log(error) | ||||
|                     return axiosError.response; | ||||
|                 } | ||||
|             } | ||||
|             // Handle unknown error without a response
 | ||||
|             console.log(error, "unknown error"); | ||||
|         }) | ||||
|     ); | ||||
|     return promises; | ||||
| }; | ||||
| 
 | ||||
| export const ntfyCall = async (data: NotififaceInput) => { | ||||
|     const url = process.env.NTFY_HOST | ||||
|     const username = process.env.NTFY_USERNAME; | ||||
|     const password = process.env.NTFY_PASSWD; | ||||
| @ -24,7 +70,8 @@ export const ntfyCall = async (data: NotifikaceInput) => { | ||||
|         console.log("NTFY_PASSWD není definován v env") | ||||
|         return | ||||
|     } | ||||
|     let clientData = await getClientData(getToday()); | ||||
|     const today = formatDate(getToday()); | ||||
|     let clientData: ClientData = await storage.getData(today); | ||||
|     const userByCLocation = getUsersByLocation(clientData.choices, data.user) | ||||
| 
 | ||||
|     const token = Buffer.from(`${username}:${password}`, 'utf8').toString('base64'); | ||||
| @ -50,64 +97,27 @@ export const ntfyCall = async (data: NotifikaceInput) => { | ||||
|     return promises; | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| export const teamsCall = async (data: NotifikaceInput) => { | ||||
|     const url = process.env.TEAMS_WEBHOOK_URL; | ||||
|     const title = data.udalost; | ||||
|     let time = new Date(); | ||||
|     time.setTime(time.getTime() + 1000 * 60); | ||||
|     const message = 'Odcházíme v ' + getHumanTime(time) + ', ' + data.user; | ||||
|     const card = { | ||||
|         '@type': 'MessageCard', | ||||
|         '@context': 'http://schema.org/extensions', | ||||
|         'themeColor': "0072C6", // light blue
 | ||||
|         summary: 'Summary description', | ||||
|         sections: [ | ||||
|             { | ||||
|                 activityTitle: title, | ||||
|                 text: message, | ||||
|             }, | ||||
|         ], | ||||
|     }; | ||||
| 
 | ||||
|     if (!url) { | ||||
|         console.log("TEAMS_WEBHOOK_URL není definován v env") | ||||
|         return | ||||
|     } | ||||
| 
 | ||||
|     try { | ||||
|         const response = await axios.post(url, card, { | ||||
|             headers: { | ||||
|                 'content-type': 'application/vnd.microsoft.teams.card.o365connector' | ||||
|             }, | ||||
|         }); | ||||
|         return `${response.status} - ${response.statusText}`; | ||||
|     } catch (err) { | ||||
|         return err; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** Zavolá notifikace na všechny konfigurované způsoby notifikace, přetížení proměných na false pro jednotlivé způsoby je vypne*/ | ||||
| export const callNotifikace = async ({ input, teams = true, gotify = false, ntfy = true }: NotifikaceData) => { | ||||
|     const notifications = []; | ||||
| 
 | ||||
|     if (ntfy) { | ||||
|         const ntfyPromises = await ntfyCall(input); | ||||
|         if (ntfyPromises) { | ||||
|             notifications.push(...ntfyPromises); | ||||
|         } | ||||
|     } | ||||
|     /* Zatím není | ||||
|     if (teams) { | ||||
|         const teamsPromises = await teamsCall(input); | ||||
|         if (teamsPromises) { | ||||
|             notifications.push(teamsPromises); | ||||
|         } | ||||
|     } | ||||
|          notifications.push(teamsCall(input)); | ||||
|      }*/ | ||||
| 
 | ||||
|     // Add more notifications as necessary
 | ||||
| 
 | ||||
|     //gotify bych řekl, že už je deprecated
 | ||||
|     // if (gotify) {
 | ||||
|     //     const gotifyPromises = await gotifyCall(input, gotifyData);
 | ||||
|     //     notifications.push(...gotifyPromises);
 | ||||
|     // }
 | ||||
|     if (gotify) { | ||||
|         const gotifyPromises = await gotifyCall(input, gotifyData); | ||||
|         notifications.push(...gotifyPromises); | ||||
|     } | ||||
| 
 | ||||
|     try { | ||||
|         const results = await Promise.all(notifications); | ||||
|  | ||||
| @ -1,10 +1,10 @@ | ||||
| import { formatDate } from "./utils"; | ||||
| import { callNotifikace } from "./notifikace"; | ||||
| import { generateQr } from "./qr"; | ||||
| import { ClientData, PizzaDayState, UdalostEnum, Pizza, PizzaSize, Order, PizzaOrder, DayData } from "../../types"; | ||||
| import getStorage from "./storage"; | ||||
| import { downloadPizzy } from "./chefie"; | ||||
| import { getClientData, getToday, initIfNeeded } from "./service"; | ||||
| import { Pizza, ClientData, PizzaDayState, PizzaSize, PizzaOrder, PizzaVariant, UdalostEnum } from "../../types/gen/types.gen"; | ||||
| import { getToday, initIfNeeded } from "./service"; | ||||
| 
 | ||||
| const storage = getStorage(); | ||||
| 
 | ||||
| @ -14,7 +14,8 @@ const storage = getStorage(); | ||||
|  */ | ||||
| export async function getPizzaList(): Promise<Pizza[] | undefined> { | ||||
|     await initIfNeeded(); | ||||
|     let clientData = await getClientData(getToday()); | ||||
|     const today = formatDate(getToday()); | ||||
|     let clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaList) { | ||||
|         const mock = process.env.MOCK_DATA === 'true'; | ||||
|         clientData = await savePizzaList(await downloadPizzy(mock)); | ||||
| @ -30,9 +31,9 @@ export async function getPizzaList(): Promise<Pizza[] | undefined> { | ||||
| export async function savePizzaList(pizzaList: Pizza[]): Promise<ClientData> { | ||||
|     await initIfNeeded(); | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     clientData.pizzaList = pizzaList; | ||||
|     clientData.pizzaListLastUpdate = formatDate(new Date()); | ||||
|     clientData.pizzaListLastUpdate = new Date(); | ||||
|     await storage.setData(today, clientData); | ||||
|     return clientData; | ||||
| } | ||||
| @ -42,14 +43,14 @@ export async function savePizzaList(pizzaList: Pizza[]): Promise<ClientData> { | ||||
|  */ | ||||
| export async function createPizzaDay(creator: string): Promise<ClientData> { | ||||
|     await initIfNeeded(); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den již existuje"); | ||||
|     } | ||||
|     // TODO berka rychlooprava, vyřešit lépe - stahovat jednou, na jediném místě!
 | ||||
|     const pizzaList = await getPizzaList(); | ||||
|     const data: ClientData = { pizzaDay: { state: PizzaDayState.CREATED, creator, orders: [] }, pizzaList, ...clientData }; | ||||
|     const today = formatDate(getToday()); | ||||
|     await storage.setData(today, data); | ||||
|     callNotifikace({ input: { udalost: UdalostEnum.ZAHAJENA_PIZZA, user: creator } }) | ||||
|     return data; | ||||
| @ -59,7 +60,8 @@ export async function createPizzaDay(creator: string): Promise<ClientData> { | ||||
|  * Smaže pizza day pro aktuální den. | ||||
|  */ | ||||
| export async function deletePizzaDay(login: string): Promise<ClientData> { | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -67,7 +69,6 @@ export async function deletePizzaDay(login: string): Promise<ClientData> { | ||||
|         throw Error("Login uživatele se neshoduje se zakladatelem Pizza Day"); | ||||
|     } | ||||
|     delete clientData.pizzaDay; | ||||
|     const today = formatDate(getToday()); | ||||
|     await storage.setData(today, clientData); | ||||
|     return clientData; | ||||
| } | ||||
| @ -81,31 +82,28 @@ export async function deletePizzaDay(login: string): Promise<ClientData> { | ||||
|  */ | ||||
| export async function addPizzaOrder(login: string, pizza: Pizza, size: PizzaSize) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
|     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { | ||||
|         throw Error("Pizza day není ve stavu " + PizzaDayState.CREATED); | ||||
|     } | ||||
|     let order: PizzaOrder | undefined = clientData.pizzaDay?.orders?.find(o => o.customer === login); | ||||
|     let order: Order | undefined = clientData.pizzaDay.orders.find(o => o.customer === login); | ||||
|     if (!order) { | ||||
|         order = { | ||||
|             customer: login, | ||||
|             pizzaList: [], | ||||
|             totalPrice: 0, | ||||
|             hasQr: false, | ||||
|         } | ||||
|         clientData.pizzaDay.orders ??= []; | ||||
|         clientData.pizzaDay.orders.push(order); | ||||
|     } | ||||
|     const pizzaOrder: PizzaVariant = { | ||||
|     const pizzaOrder: PizzaOrder = { | ||||
|         varId: size.varId, | ||||
|         name: pizza.name, | ||||
|         size: size.size, | ||||
|         price: size.price, | ||||
|     } | ||||
|     order.pizzaList ??= []; | ||||
|     order.pizzaList.push(pizzaOrder); | ||||
|     order.totalPrice += pizzaOrder.price; | ||||
|     await storage.setData(today, clientData); | ||||
| @ -118,26 +116,26 @@ export async function addPizzaOrder(login: string, pizza: Pizza, size: PizzaSize | ||||
|  * @param login login uživatele | ||||
|  * @param pizzaOrder objednávka pizzy | ||||
|  */ | ||||
| export async function removePizzaOrder(login: string, pizzaOrder: PizzaVariant) { | ||||
| export async function removePizzaOrder(login: string, pizzaOrder: PizzaOrder) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
|     const orderIndex = clientData.pizzaDay.orders!.findIndex(o => o.customer === login); | ||||
|     const orderIndex = clientData.pizzaDay.orders.findIndex(o => o.customer === login); | ||||
|     if (orderIndex < 0) { | ||||
|         throw Error("Nebyly nalezeny žádné objednávky pro uživatele " + login); | ||||
|     } | ||||
|     const order = clientData.pizzaDay.orders![orderIndex]; | ||||
|     const index = order.pizzaList!.findIndex(o => o.name === pizzaOrder.name && o.size === pizzaOrder.size); | ||||
|     const order = clientData.pizzaDay.orders[orderIndex]; | ||||
|     const index = order.pizzaList.findIndex(o => o.name === pizzaOrder.name && o.size === pizzaOrder.size); | ||||
|     if (index < 0) { | ||||
|         throw Error("Objednávka s danými parametry nebyla nalezena"); | ||||
|     } | ||||
|     const price = order.pizzaList![index].price; | ||||
|     order.pizzaList!.splice(index, 1); | ||||
|     const price = order.pizzaList[index].price; | ||||
|     order.pizzaList.splice(index, 1); | ||||
|     order.totalPrice -= price; | ||||
|     if (order.pizzaList!.length == 0) { | ||||
|         clientData.pizzaDay.orders!.splice(orderIndex, 1); | ||||
|     if (order.pizzaList.length == 0) { | ||||
|         clientData.pizzaDay.orders.splice(orderIndex, 1); | ||||
|     } | ||||
|     await storage.setData(today, clientData); | ||||
|     return clientData; | ||||
| @ -151,7 +149,7 @@ export async function removePizzaOrder(login: string, pizzaOrder: PizzaVariant) | ||||
|  */ | ||||
| export async function lockPizzaDay(login: string) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -174,7 +172,7 @@ export async function lockPizzaDay(login: string) { | ||||
|  */ | ||||
| export async function unlockPizzaDay(login: string) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -197,7 +195,7 @@ export async function unlockPizzaDay(login: string) { | ||||
|  */ | ||||
| export async function finishPizzaOrder(login: string) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -222,7 +220,7 @@ export async function finishPizzaOrder(login: string) { | ||||
|  */ | ||||
| export async function finishPizzaDelivery(login: string, bankAccount?: string, bankAccountHolder?: string) { | ||||
|     const today = formatDate(getToday()); | ||||
|     const clientData = await getClientData(getToday()); | ||||
|     const clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -236,9 +234,9 @@ export async function finishPizzaDelivery(login: string, bankAccount?: string, b | ||||
| 
 | ||||
|     // Vygenerujeme QR kód, pokud k tomu máme data
 | ||||
|     if (bankAccount?.length && bankAccountHolder?.length) { | ||||
|         for (const order of clientData.pizzaDay.orders!) { | ||||
|         for (const order of clientData.pizzaDay.orders) { | ||||
|             if (order.customer !== login) { // zatím platí creator = objednávající, a pro toho nemá QR kód smysl
 | ||||
|                 let message = order.pizzaList!.map(pizza => `Pizza ${pizza.name} (${pizza.size})`).join(', '); | ||||
|                 let message = order.pizzaList.map(pizza => `Pizza ${pizza.name} (${pizza.size})`).join(', '); | ||||
|                 await generateQr(order.customer, bankAccount, bankAccountHolder, order.totalPrice, message); | ||||
|                 order.hasQr = true; | ||||
|             } | ||||
| @ -257,15 +255,15 @@ export async function finishPizzaDelivery(login: string, bankAccount?: string, b | ||||
|  */ | ||||
| export async function updatePizzaDayNote(login: string, note?: string) { | ||||
|     const today = formatDate(getToday()); | ||||
|     let clientData = await getClientData(getToday()); | ||||
|     let clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
|     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { | ||||
|         throw Error("Pizza day není ve stavu " + PizzaDayState.CREATED); | ||||
|     } | ||||
|     const myOrder = clientData.pizzaDay.orders!.find(o => o.customer === login); | ||||
|     if (!myOrder?.pizzaList?.length) { | ||||
|     const myOrder = clientData.pizzaDay.orders.find(o => o.customer === login); | ||||
|     if (!myOrder || !myOrder.pizzaList.length) { | ||||
|         throw Error("Pizza day neobsahuje žádné objednávky uživatele " + login); | ||||
|     } | ||||
|     myOrder.note = note; | ||||
| @ -284,7 +282,7 @@ export async function updatePizzaDayNote(login: string, note?: string) { | ||||
|  */ | ||||
| export async function updatePizzaFee(login: string, targetLogin: string, text?: string, price?: number) { | ||||
|     const today = formatDate(getToday()); | ||||
|     let clientData = await getClientData(getToday()); | ||||
|     let clientData: DayData = await storage.getData(today); | ||||
|     if (!clientData.pizzaDay) { | ||||
|         throw Error("Pizza day pro dnešní den neexistuje"); | ||||
|     } | ||||
| @ -294,8 +292,8 @@ export async function updatePizzaFee(login: string, targetLogin: string, text?: | ||||
|     if (clientData.pizzaDay.creator !== login) { | ||||
|         throw Error("Příplatky může měnit pouze zakladatel Pizza day"); | ||||
|     } | ||||
|     const targetOrder = clientData.pizzaDay.orders!.find(o => o.customer === targetLogin); | ||||
|     if (!targetOrder?.pizzaList?.length) { | ||||
|     const targetOrder = clientData.pizzaDay.orders.find(o => o.customer === targetLogin); | ||||
|     if (!targetOrder || !targetOrder.pizzaList.length) { | ||||
|         throw Error(`Pizza day neobsahuje žádné objednávky uživatele ${targetLogin}`); | ||||
|     } | ||||
|     if (!price) { | ||||
| @ -304,7 +302,7 @@ export async function updatePizzaFee(login: string, targetLogin: string, text?: | ||||
|         targetOrder.fee = { text, price }; | ||||
|     } | ||||
|     // Přepočet ceny
 | ||||
|     targetOrder.totalPrice = targetOrder.pizzaList.reduce((price, pizzaOrder) => price + pizzaOrder.price, 0) + (targetOrder.fee?.price ?? 0); | ||||
|     targetOrder.totalPrice = targetOrder.pizzaList.reduce((price, pizzaOrder) => price + pizzaOrder.price, 0) + (targetOrder.fee?.price || 0); | ||||
|     await storage.setData(today, clientData); | ||||
|     return clientData; | ||||
| } | ||||
| @ -1,8 +1,7 @@ | ||||
| import axios from "axios"; | ||||
| import { load } from 'cheerio'; | ||||
| import { getMenuSladovnickaMock, getMenuTechTowerMock, getMenuUMotlikuMock, getMenuZastavkaUmichalaMock, getMenuSenkSerikovaMock } from "./mock"; | ||||
| import { formatDate } from "./utils"; | ||||
| import { Food } from "../../types/gen/types.gen"; | ||||
| import { DayOfWeek, DayOfWeekEnum, DayOfWeekIndex, Food, RestaurantWeeklyMenu } from "../../types"; | ||||
| import { getMenuSladovnickaMock, getMenuTechTowerMock, getMenuUMotlikuMock, getMenuZastavkaUmichalaMock } from "./mock"; | ||||
| 
 | ||||
| // Fráze v názvech jídel, které naznačují že se jedná o polévku
 | ||||
| const SOUP_NAMES = [ | ||||
| @ -23,7 +22,7 @@ const SOUP_NAMES = [ | ||||
| const DAYS_IN_WEEK = ['pondělí', 'úterý', 'středa', 'čtvrtek', 'pátek', 'sobota', 'neděle']; | ||||
| 
 | ||||
| // URL na týdenní menu jednotlivých restaurací
 | ||||
| const SLADOVNICKA_URL = 'https://sladovnicka.unasplzenchutna.cz/cz/#denni-nabidka'; | ||||
| const SLADOVNICKA_URL = 'https://sladovnicka.unasplzenchutna.cz/cz/denni-nabidka'; | ||||
| const U_MOTLIKU_URL = 'https://www.umotliku.cz/menu'; | ||||
| const TECHTOWER_URL = 'https://www.equifarm.cz/restaurace-techtower'; | ||||
| const ZASTAVKAUMICHALA_URL = 'https://www.zastavkaumichala.cz'; | ||||
| @ -53,28 +52,6 @@ const sanitizeText = (text: string): string => { | ||||
|     return text.replace('\t', '').replace(' , ', ', ').trim(); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Parsuje čísla alergenů z názvu jídla a vrací vyčištěný název spolu s polem alergenů. | ||||
|  * Alergeny jsou očekávány na konci názvu ve formátu číslic oddělených čárkami. | ||||
|  *  | ||||
|  * @param name původní název jídla | ||||
|  * @returns objekt obsahující vyčištěný název a pole alergenů | ||||
|  */ | ||||
| const parseAllergens = (name: string): { cleanName: string, allergens: number[] } => { | ||||
|     // Regex pro nalezení čísel na konci řetězce oddělených čárkami a případnými mezerami
 | ||||
|     const regex = /\s+(\d+(?:\s*,\s*\d+)*)\s*$/; | ||||
|     const match = regex.exec(name); | ||||
|      | ||||
|     if (match) { | ||||
|         const allergenString = match[1]; | ||||
|         const allergens = allergenString.split(',').map(num => parseInt(num.trim(), 10)).filter(num => !isNaN(num)); | ||||
|         const cleanName = name.replace(regex, '').trim(); | ||||
|         return { cleanName, allergens }; | ||||
|     } | ||||
|      | ||||
|     return { cleanName: name, allergens: [] }; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Stáhne a vrátí aktuální HTML z dané URL. | ||||
|  *  | ||||
| @ -92,7 +69,7 @@ const getHtml = async (url: string): Promise<any> => { | ||||
|  * @param mock zda vrátit mock data | ||||
|  * @returns seznam jídel pro daný týden | ||||
|  */ | ||||
| export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = false): Promise<Food[][]> => { | ||||
| export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = false): Promise<RestaurantWeeklyMenu> => { | ||||
|     if (mock) { | ||||
|         return getMenuSladovnickaMock(); | ||||
|     } | ||||
| @ -100,69 +77,84 @@ export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = f | ||||
|     const html = await getHtml(SLADOVNICKA_URL); | ||||
|     const $ = load(html); | ||||
| 
 | ||||
|     const menuContentElements = $('#daily-menu-content-list').children('[id^="daily-menu-content-"]'); | ||||
|     // Prozatím předpokládáme, že budou mít vždy elementy pro všech 5 dní v týdnu, i pokud bude zavřeno
 | ||||
|     if (menuContentElements.length < 5) { | ||||
|         throw Error("Neočekávaný počet dní v menu Sladovnické: " + menuContentElements.length + ", očekáváno 5 (možná je některý den zavřeno?)"); | ||||
|     } | ||||
| 
 | ||||
|     const result: Food[][] = []; | ||||
|     const list = $('ul.tab-links').children(); | ||||
|     const result: RestaurantWeeklyMenu = {}; | ||||
|     // TODO upravit až bude enum
 | ||||
|     for (let dayIndex = 0; dayIndex < 5; dayIndex++) { | ||||
|         const dayChildren = $(menuContentElements[dayIndex]).children(); | ||||
|         // Prozatím předpokládáme, že budou mít vždy polévku a hlavní jídla
 | ||||
|         if (dayChildren.length < 2) { | ||||
|             throw Error("Neočekávaný počet children v menu Sladovnické pro den " + dayIndex + ": " + dayChildren.length + ", očekávány alespoň 2 (polévka a hlavní jídlo)"); | ||||
|         const currentDate = new Date(firstDayOfWeek); | ||||
|         currentDate.setDate(firstDayOfWeek.getDate() + dayIndex); | ||||
|         const searchedDayText = `${currentDate.getDate()}.${currentDate.getMonth() + 1}.${capitalize(DAYS_IN_WEEK[dayIndex])}`; | ||||
|         // Najdeme index pro vstupní datum (např. při svátcích bude posunutý)
 | ||||
|         // TODO validovat, že vstupní datum je v aktuálním týdnu
 | ||||
|         // TODO tenhle způsob je zbytečně komplikovaný - stačilo by hledat rovnou v div.tab-content, protože každý den tam má datum taky (akorát je print-only)
 | ||||
|         let index = undefined; | ||||
|         list.each((i, dayRow) => { | ||||
|             const rowText = $(dayRow).first().text().trim(); | ||||
|             if (rowText === searchedDayText) { | ||||
|                 index = i; | ||||
|                 return; | ||||
|             } | ||||
|         }) | ||||
|         if (index === undefined) { | ||||
|             // Pravděpodobně svátek, nebo je zavřeno
 | ||||
|             const index: number = Object.keys(DayOfWeekEnum).indexOf('Casual'); // 1
 | ||||
|             result[dayIndex as DayOfWeekEnum] = [{ | ||||
|                 amount: undefined, | ||||
|                 name: "Pro daný den nebyla nalezena denní nabídka", | ||||
|                 price: "", | ||||
|                 isSoup: false, | ||||
|             }]; | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         // Parsování polévky
 | ||||
|         const soupElement = dayChildren.get(0); | ||||
|         const soupTable = $(soupElement).find('table tbody tr'); | ||||
|         const soupCells = soupTable.children('td'); | ||||
|         // Dle dohledaného indexu najdeme správný tabpanel
 | ||||
|         const rows = $('div.tab-content').children(); | ||||
|         if (index >= rows.length) { | ||||
|             throw Error("V HTML nebyl nalezen řádek menu pro index " + index); | ||||
|         } | ||||
|         const tabPanel = $(rows.get(index)); | ||||
| 
 | ||||
|         // Opětovná validace, že daný tabpanel je pro vstupní datum
 | ||||
|         const headers = tabPanel.find('h2'); | ||||
|         if (headers.length !== 3) { | ||||
|             throw Error("Neočekávaný počet elementů h2 v menu pro datum " + searchedDayText + ", očekávány 3, ale nalezeno bylo " + headers.length); | ||||
|         } | ||||
|         const dayText = $(headers.get(0)).text().trim(); | ||||
|         if (dayText !== searchedDayText) { | ||||
|             throw Error("Neočekávaný datum na řádce nalezeného dne: '" + dayText + "', ale očekáváno bylo '" + searchedDayText + "'"); | ||||
|         } | ||||
| 
 | ||||
|         // V tabpanelu očekáváme dvě tabulky - pro polévku a pro hlavní jídlo 
 | ||||
|         const tables = tabPanel.find('table'); | ||||
|         if (tables.length !== 2) { | ||||
|             throw Error("Neočekávaný počet tabulek na řádce nalezeného dne: " + tables.length + ", ale očekávány byly 2"); | ||||
|         } | ||||
|         const currentDayFood: Food[] = []; | ||||
|         // Polévka - div -> table -> tbody -> tr -> 3x td
 | ||||
|         const soupCells = $(tables.get(0)).children().first().children().first().children(); | ||||
|         if (soupCells.length !== 3) { | ||||
|             throw Error("Neočekávaný počet buněk v tabulce polévky: " + soupCells.length + ", ale očekávány byly 3"); | ||||
|         } | ||||
|          | ||||
|         const soupAmount = sanitizeText($(soupCells.get(0)).text()); | ||||
|         const soupNameRaw = sanitizeText($(soupCells.get(1)).text()); | ||||
|         const soupPrice = sanitizeText($(soupCells.get(2)).text().replace(' ', '\xA0')); | ||||
|         const soupParsed = parseAllergens(soupNameRaw); | ||||
|          | ||||
|         // Parsování hlavních jídel
 | ||||
|         const mainCourseElement = dayChildren.get(1); | ||||
|         const mainCourseTable = $(mainCourseElement).find('table tbody'); | ||||
|         const mainCourseRows = mainCourseTable.children('tr'); | ||||
|          | ||||
|         const currentDayFood: Food[] = []; | ||||
|          | ||||
|         // Přidáme polévku do seznamu jídel
 | ||||
|         currentDayFood.push({ | ||||
|             amount: soupAmount, | ||||
|             name: soupParsed.cleanName, | ||||
|             price: soupPrice, | ||||
|             amount: sanitizeText($(soupCells.get(0)).text()), | ||||
|             name: sanitizeText($(soupCells.get(1)).text()), | ||||
|             price: sanitizeText($(soupCells.get(2)).text().replace(' ', '\xA0')), | ||||
|             isSoup: true, | ||||
|             allergens: soupParsed.allergens.length > 0 ? soupParsed.allergens : undefined, | ||||
|         }); | ||||
|          | ||||
|         // Projdeme všechny řádky hlavních jídel
 | ||||
|         mainCourseRows.each((i, row) => { | ||||
|             const cells = $(row).children('td'); | ||||
|             const amount = sanitizeText($(cells.get(0)).text()); | ||||
|             const nameRaw = sanitizeText($(cells.get(1)).text()); | ||||
|             const price = sanitizeText($(cells.get(2)).text().replace(' ', '\xA0')); | ||||
|             const parsed = parseAllergens(nameRaw); | ||||
|              | ||||
|             // Přeskočíme prázdné řádky (první řádek může být prázdný)
 | ||||
|             if (parsed.cleanName.trim().length > 0) { | ||||
|                 currentDayFood.push({ | ||||
|                     amount, | ||||
|                     name: parsed.cleanName, | ||||
|                     price, | ||||
|                     isSoup: false, | ||||
|                     allergens: parsed.allergens.length > 0 ? parsed.allergens : undefined, | ||||
|         }); | ||||
|         // Hlavní jídla - div -> table -> tbody -> 3x tr
 | ||||
|         const mainCourseRows = $(tables.get(1)).children().first().children(); | ||||
|         mainCourseRows.each((i, foodRow) => { | ||||
|             const foodCells = $(foodRow).children(); | ||||
|             if (foodCells.length !== 3) { | ||||
|                 throw Error("Neočekávaný počet buněk v řádku jídla: " + foodCells.length + ", ale očekávány byly 3"); | ||||
|             } | ||||
|             currentDayFood.push({ | ||||
|                 amount: sanitizeText($(foodCells.get(0)).text()), | ||||
|                 name: sanitizeText($(foodCells.get(1)).text()), | ||||
|                 price: sanitizeText($(foodCells.get(2)).text().replace(' ', '\xA0')), | ||||
|                 isSoup: false, | ||||
|             }); | ||||
| 
 | ||||
|         }) | ||||
|         result[dayIndex] = currentDayFood; | ||||
|     } | ||||
|     return result; | ||||
| @ -301,7 +293,7 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal | ||||
| 
 | ||||
|     const result: Food[][] = []; | ||||
|     // TODO validovat, že v textu nalezeného <font> je rozsah, do kterého spadá vstupní datum
 | ||||
|     const siblings = secondTry ? $(font).parent().parent().parent().siblings('p') : $(font).parent().parent().siblings(); | ||||
|     const siblings = secondTry ? $(font).parent().siblings() : $(font).parent().parent().siblings(); | ||||
|     let parsing = false; | ||||
|     let currentDayIndex = 0; | ||||
|     for (let i = 0; i < siblings.length; i++) { | ||||
| @ -319,25 +311,21 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal | ||||
|                 continue; | ||||
|             } | ||||
|             let price = 'na\xA0váhu'; | ||||
|             let nameRaw = text.replace('•', ''); | ||||
|             let name = text.replace('•', ''); | ||||
|             if (text.toLowerCase().endsWith('kč')) { | ||||
|                 const tmp = text.replace('\xA0', ' ').split(' '); | ||||
|                 const split = [tmp.slice(0, -2).join(' ')].concat(tmp.slice(-2)); | ||||
|                 price = `${split.slice(1)[0]}\xA0Kč` | ||||
|                 nameRaw = split[0].replace('•', ''); | ||||
|                 name = split[0].replace('•', ''); | ||||
|             } | ||||
|             if (nameRaw.endsWith('–')) { | ||||
|                 nameRaw = nameRaw.slice(0, -1).trim(); | ||||
|             if (result[currentDayIndex] == null) { | ||||
|                 result[currentDayIndex] = []; | ||||
|             } | ||||
| 
 | ||||
|             const parsed = parseAllergens(nameRaw); | ||||
|             result[currentDayIndex] ??= []; | ||||
|             result[currentDayIndex].push({ | ||||
|                 amount: '-', | ||||
|                 name: parsed.cleanName, | ||||
|                 name, | ||||
|                 price, | ||||
|                 isSoup: isTextSoupName(parsed.cleanName), | ||||
|                 allergens: parsed.allergens.length > 0 ? parsed.allergens : undefined, | ||||
|                 isSoup: isTextSoupName(name), | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
| @ -356,33 +344,29 @@ export const getMenuZastavkaUmichala = async (firstDayOfWeek: Date, mock: boolea | ||||
|         return getMenuZastavkaUmichalaMock(); | ||||
|     } | ||||
| 
 | ||||
|     const today = new Date(); | ||||
|     today.setHours(0,0,0,0); | ||||
|     const headers = { | ||||
|         "Cookie": "_nss=1; PHPSESSID=9e37de17e0326b0942613d6e67a30e69", | ||||
|         "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/132.0.0.0 Safari/537.36", | ||||
|     }; | ||||
|     const nowDate = new Date().getDate(); | ||||
|     const result: Food[][] = []; | ||||
|     for (let dayIndex = 0; dayIndex < 5; dayIndex++) { | ||||
|         const currentDate = new Date(firstDayOfWeek); | ||||
|         currentDate.setDate(firstDayOfWeek.getDate() + dayIndex); | ||||
|         currentDate.setHours(0,0,0,0); | ||||
|         if (currentDate < today || (currentDate.getTime() === today.getTime() && new Date().getHours() >= 14)) { | ||||
| 
 | ||||
|         // if (currentDate < now) {
 | ||||
|         if (currentDate.getDate() !== nowDate) { | ||||
|             result[dayIndex] = [{ | ||||
|                 amount: undefined, | ||||
|                 name: "Pro tento den není uveřejněna nabídka jídel", | ||||
|                 price: "", | ||||
|                 isSoup: false, | ||||
|             }]; | ||||
|             continue; | ||||
|         } else { | ||||
|             const url = (currentDate.getTime() === today.getTime()) | ||||
|                 ? ZASTAVKAUMICHALA_URL | ||||
|                 : ZASTAVKAUMICHALA_URL + '/?do=dailyMenu-changeDate&dailyMenu-dateString=' + formatDate(currentDate, 'DD.MM.YYYY'); | ||||
|             const html = await axios.get(url, { | ||||
|                 headers, | ||||
|             }).then(res => res.data).then(content => content); | ||||
|             // let dateString = formatDate(currentDate, 'DD.MM.YYYY');
 | ||||
|             // const html = await getHtml(ZASTAVKAUMICHALA_URL + '/?do=dailyMenu-changeDate&dailyMenu-dateString=' + dateString);
 | ||||
|             const html = await getHtml(ZASTAVKAUMICHALA_URL); | ||||
|             const $ = load(html); | ||||
| 
 | ||||
|             // const row = $($('.foodsList li')[0]).text();
 | ||||
| 
 | ||||
|             const currentDayFood: Food[] = []; | ||||
|             $('.foodsList li').each((index, element) => { | ||||
|                 currentDayFood.push({ | ||||
| @ -397,56 +381,3 @@ export const getMenuZastavkaUmichala = async (firstDayOfWeek: Date, mock: boolea | ||||
|     } | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Získá obědovou nabídku SenkSerikova pro jeden týden. | ||||
|  * | ||||
|  * @param firstDayOfWeek první den v týdnu, pro který získat menu | ||||
|  * @param mock zda vrátit mock data | ||||
|  * @returns seznam jídel pro dané datum | ||||
|  */ | ||||
| export const getMenuSenkSerikova = async (firstDayOfWeek: Date, mock: boolean = false): Promise<Food[][]> => { | ||||
|     if (mock) { | ||||
|         return getMenuSenkSerikovaMock(); | ||||
|     } | ||||
| 
 | ||||
|     const decoder = new TextDecoder('windows-1250'); | ||||
|     const html = await axios.get(SENKSERIKOVA_URL, { | ||||
|         responseType: 'arraybuffer', | ||||
|         responseEncoding: 'binary' | ||||
|     }).then(res => decoder.decode(new Uint8Array(res.data))).then(content => content); | ||||
|     const $ = load(html); | ||||
| 
 | ||||
|     const today = new Date(); | ||||
|     today.setHours(0,0,0,0); | ||||
|     const currentDate = new Date(firstDayOfWeek); | ||||
|     const result: Food[][] = []; | ||||
|     let dayIndex = 0; | ||||
|     currentDate.setHours(0,0,0,0); | ||||
|     while (currentDate < today) { | ||||
|         result[dayIndex] = [{ | ||||
|             amount: undefined, | ||||
|             name: "Pro tento den není uveřejněna nabídka jídel", | ||||
|             price: "", | ||||
|             isSoup: false, | ||||
|         }]; | ||||
|         dayIndex = dayIndex + 1; | ||||
|         currentDate.setDate(firstDayOfWeek.getDate() + dayIndex); | ||||
|     } | ||||
| 
 | ||||
|     $('.menicka').each((i, element) => { | ||||
|         const currentDayFood: Food[] = []; | ||||
|         $(element).find('.popup-gallery li').each((j, element) => { | ||||
|             const rawName = $(element).children('div.polozka').text(); | ||||
|             const nameWithoutNumber = rawName.replace(/^\d+\.\s*/, ''); | ||||
|             currentDayFood.push({ | ||||
|                 amount: '-', | ||||
|                 name: nameWithoutNumber, | ||||
|                 price: $(element).children('div.cena').text().replace(/ /g, '\xA0'), | ||||
|                 isSoup: $(element).hasClass('polevka'), | ||||
|             }); | ||||
|         }); | ||||
|         result[dayIndex++] = currentDayFood; | ||||
|     }); | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| @ -1,11 +1,11 @@ | ||||
| import express, { NextFunction } from "express"; | ||||
| import { getLogin } from "../auth"; | ||||
| import { getLogin, getTrusted } from "../auth"; | ||||
| import { parseToken } from "../utils"; | ||||
| import path from "path"; | ||||
| import fs from "fs"; | ||||
| import { EasterEgg } from "../../../types/gen/types.gen"; | ||||
| import { EasterEgg } from "../../../types"; | ||||
| 
 | ||||
| const EASTER_EGGS_JSON_PATH = path.join(__dirname, "../../resources/.easter-eggs.json"); | ||||
| const EASTER_EGGS_JSON_PATH = path.join(__dirname, "../../.easter-eggs.json"); | ||||
| const IMAGES_PATH = '../../resources/easterEggs'; | ||||
| 
 | ||||
| type EasterEggsJson = { | ||||
| @ -34,12 +34,17 @@ function generateUrl() { | ||||
|  */ | ||||
| function getEasterEggImage(req: any, res: any, next: NextFunction) { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     try { | ||||
|         // TODO vrátit!
 | ||||
|         // if (trusted) {
 | ||||
|         if (true) { | ||||
|             if (login in easterEggs) { | ||||
|                 const imagePath = easterEggs[login][Math.floor(Math.random() * easterEggs[login].length)].path; | ||||
|                 res.sendFile(path.join(__dirname, IMAGES_PATH, imagePath)); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|         res.sendStatus(404); | ||||
|     } catch (e: any) { next(e) } | ||||
| } | ||||
| @ -119,7 +124,7 @@ let easterEggs: EasterEggsJson; | ||||
| if (fs.existsSync(EASTER_EGGS_JSON_PATH)) { | ||||
|     const content = fs.readFileSync(EASTER_EGGS_JSON_PATH, 'utf-8'); | ||||
|     easterEggs = JSON.parse(content); | ||||
|     for (const [_, eggs] of Object.entries(easterEggs)) { | ||||
|     for (const [key, eggs] of Object.entries(easterEggs)) { | ||||
|         for (const easterEgg of eggs) { | ||||
|             const url = generateUrl(); | ||||
|             easterEgg.url = url; | ||||
| @ -133,12 +138,17 @@ if (fs.existsSync(EASTER_EGGS_JSON_PATH)) { | ||||
| // Získání náhodného easter eggu pro přihlášeného uživatele
 | ||||
| router.get("/", async (req, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     try { | ||||
|         // TODO vrátit!
 | ||||
|         // if (trusted) {
 | ||||
|         if (true) { | ||||
|             if (easterEggs && login in easterEggs) { | ||||
|                 const randomEasterEgg = easterEggs[login][Math.floor(Math.random() * easterEggs[login].length)]; | ||||
|                 const { path, startOffset, endOffset, ...strippedEasterEgg } = randomEasterEgg; // Path klient k ničemu nepotřebuje a nemá ho znát
 | ||||
|                 return res.status(200).json({ ...strippedEasterEgg, ...getRandomPosition(startOffset, endOffset) }); | ||||
|             } | ||||
|         } | ||||
|         return res.status(200).send(); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
|  | ||||
| @ -1,51 +1,10 @@ | ||||
| import express, { Request, Response } from "express"; | ||||
| import express, { Request } from "express"; | ||||
| import { getLogin, getTrusted } from "../auth"; | ||||
| import { addChoice, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote, fetchRestaurantWeekMenuData, saveRestaurantWeekMenu } from "../service"; | ||||
| import { getDayOfWeekIndex, parseToken, getFirstWorkDayOfWeek } from "../utils"; | ||||
| import { addChoice, addVolatileData, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote } from "../service"; | ||||
| import { getDayOfWeekIndex, parseToken } from "../utils"; | ||||
| import { getWebsocket } from "../websocket"; | ||||
| import { callNotifikace } from "../notifikace"; | ||||
| import { AddChoiceData, ChangeDepartureTimeData, RemoveChoiceData, RemoveChoicesData, UdalostEnum, UpdateNoteData } from "../../../types/gen/types.gen"; | ||||
| 
 | ||||
| 
 | ||||
| // RateLimit na refresh endpoint
 | ||||
| interface RateLimitEntry { | ||||
|     count: number; | ||||
|     resetTime: number; | ||||
| } | ||||
| const rateLimits: Record<string, RateLimitEntry> = {}; | ||||
| const RATE_LIMIT = 1; // maximální počet požadavků za minutu
 | ||||
| const RATE_LIMIT_WINDOW = 30 * 60 * 1000; // je to v ms (x * 1min)
 | ||||
| 
 | ||||
| // Kontrola ratelimitu
 | ||||
| function checkRateLimit(key: string, limit: number = RATE_LIMIT): boolean { | ||||
|     const now = Date.now(); | ||||
| 
 | ||||
|     // Vyčištění starých záznamů
 | ||||
|     Object.keys(rateLimits).forEach(k => { | ||||
|         if (rateLimits[k].resetTime < now) { | ||||
|             delete rateLimits[k]; | ||||
|         } | ||||
|     }); | ||||
| 
 | ||||
|     // Kontrola, že záznam existuje a platí
 | ||||
|     if (rateLimits[key] && rateLimits[key].resetTime > now) { | ||||
|         // Záznam platí a kontroluje se limit
 | ||||
|         if (rateLimits[key].count >= limit) { | ||||
|             return false; // Překročen limit
 | ||||
|         } | ||||
| 
 | ||||
|         // ++ xd
 | ||||
|         rateLimits[key].count++; | ||||
|         return true; | ||||
|     } else { | ||||
|         // + klic
 | ||||
|         rateLimits[key] = { | ||||
|             count: 1, | ||||
|             resetTime: now + RATE_LIMIT_WINDOW | ||||
|         }; | ||||
|         return true; | ||||
|     } | ||||
| } | ||||
| import { AddChoiceRequest, ChangeDepartureTimeRequest, IDayIndex, RemoveChoiceRequest, RemoveChoicesRequest, UdalostEnum, UpdateNoteRequest } from "../../../types"; | ||||
| 
 | ||||
| /** | ||||
|  * Ověří a vrátí index dne v týdnu z požadavku, za předpokladu, že byl předán, a je zároveň | ||||
| @ -54,7 +13,7 @@ function checkRateLimit(key: string, limit: number = RATE_LIMIT): boolean { | ||||
|  * @param req request | ||||
|  * @returns index dne v týdnu | ||||
|  */ | ||||
| const parseValidateFutureDayIndex = (req: Request<{}, any, AddChoiceData["body"] | UpdateNoteData["body"]>) => { | ||||
| const parseValidateFutureDayIndex = (req: Request<{}, any, IDayIndex>) => { | ||||
|     if (req.body.dayIndex == null) { | ||||
|         throw Error(`Nebyl předán index dne v týdnu.`); | ||||
|     } | ||||
| @ -71,7 +30,7 @@ const parseValidateFutureDayIndex = (req: Request<{}, any, AddChoiceData["body"] | ||||
| 
 | ||||
| const router = express.Router(); | ||||
| 
 | ||||
| router.post("/addChoice", async (req: Request<{}, any, AddChoiceData["body"]>, res, next) => { | ||||
| router.post("/addChoice", async (req: Request<{}, any, AddChoiceRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     let date = undefined; | ||||
| @ -86,12 +45,12 @@ router.post("/addChoice", async (req: Request<{}, any, AddChoiceData["body"]>, r | ||||
|     } | ||||
|     try { | ||||
|         const data = await addChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         return res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["body"]>, res, next) => { | ||||
| router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     let date = undefined; | ||||
| @ -106,12 +65,12 @@ router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["bo | ||||
|     } | ||||
|     try { | ||||
|         const data = await removeChoices(login, trusted, req.body.locationKey, date); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body"]>, res, next) => { | ||||
| router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     let date = undefined; | ||||
| @ -126,12 +85,12 @@ router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body | ||||
|     } | ||||
|     try { | ||||
|         const data = await removeChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| router.post("/updateNote", async (req: Request<{}, any, UpdateNoteData["body"]>, res, next) => { | ||||
| router.post("/updateNote", async (req: Request<{}, any, UpdateNoteRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     const note = req.body.note; | ||||
| @ -150,12 +109,12 @@ router.post("/updateNote", async (req: Request<{}, any, UpdateNoteData["body"]>, | ||||
|             date = getDateForWeekIndex(dayIndex); | ||||
|         } | ||||
|         const data = await updateNote(login, trusted, note, date); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDepartureTimeData["body"]>, res, next) => { | ||||
| router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDepartureTimeRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     let date = undefined; | ||||
|     if (req.body.dayIndex != null) { | ||||
| @ -169,7 +128,7 @@ router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDeparture | ||||
|     } | ||||
|     try { | ||||
|         const data = await updateDepartureTime(login, req.body?.time, date); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| @ -177,90 +136,9 @@ router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDeparture | ||||
| router.post("/jdemeObed", async (req, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     try { | ||||
|         await callNotifikace({ input: { user: login, udalost: UdalostEnum.JDEME_NA_OBED }, gotify: false }) | ||||
|         await callNotifikace({ input: { user: login, udalost: UdalostEnum.JDEME_OBED }, gotify: false }) | ||||
|         res.status(200).json({}); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| // /api/food/refresh?type=week&heslo=docasnyheslo
 | ||||
| export const refreshMetoda = async (req: Request, res: Response) => { | ||||
|     const { type, heslo } = req.query as { type?: string; heslo?: string }; | ||||
|     if (heslo !== "docasnyheslo" && heslo !== "tohleheslopavelnesmizjistit123") { | ||||
|         return res.status(403).json({ error: "Neplatné heslo" }); | ||||
|     } | ||||
|     if (!checkRateLimit("refresh") && heslo !== "tohleheslopavelnesmizjistit123") { | ||||
|         return res.status(429).json({ error: "Refresh už se zavolal, chvíli počkej :))" }); | ||||
|     } | ||||
|     if (type !== "week" && type !== "day") { | ||||
|         return res.status(400).json({ error: "Neznámý typ refresh" }); | ||||
|     } | ||||
|     if (type === "day") { | ||||
|         return res.status(400).json({ error: "ještě neumim TODO..." }); | ||||
|     } | ||||
|     try { | ||||
|         // Pro všechny restaurace refreshni menu na aktuální týden
 | ||||
|         const restaurants = ["SLADOVNICKA", "TECHTOWER", "ZASTAVKAUMICHALA", "SENKSERIKOVA"] as const; | ||||
|         const firstDay = getFirstWorkDayOfWeek(getToday()); | ||||
|         const results: Record<string, any> = {}; | ||||
|         const successfulRestaurants: string[] = []; | ||||
|         const failedRestaurants: string[] = []; | ||||
| 
 | ||||
|         // Nejdříve načíst všechna data bez ukládání
 | ||||
|         for (const rest of restaurants) { | ||||
|             try { | ||||
|                 const weekData = await fetchRestaurantWeekMenuData(rest, firstDay); | ||||
|                 results[rest] = weekData; | ||||
| 
 | ||||
|                 // Kontrola validity dat
 | ||||
|                 if (weekData && weekData.length > 0 && | ||||
|                     weekData.some(dayMenu => dayMenu && dayMenu.length > 0)) { | ||||
|                     successfulRestaurants.push(rest); | ||||
|                 } else { | ||||
|                     failedRestaurants.push(rest); | ||||
|                     results[rest] = { error: "Žádná validní data" }; | ||||
|                 } | ||||
|             } catch (error) { | ||||
|                 failedRestaurants.push(rest); | ||||
|                 results[rest] = { error: `Chyba při načítání: ${error}` }; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // Pokud se nepodařilo načíst žádnou restauraci
 | ||||
|         if (successfulRestaurants.length === 0) { | ||||
|             return res.status(400).json({ | ||||
|                 error: "Nepodařilo se získat validní data z žádné restaurace", | ||||
|                 failed: failedRestaurants, | ||||
|                 results: results | ||||
|             }); | ||||
|         } | ||||
| 
 | ||||
|         // Uložit pouze validní data
 | ||||
|         for (const rest of successfulRestaurants) { | ||||
|             try { | ||||
|                 await saveRestaurantWeekMenu(rest as any, firstDay, results[rest]); | ||||
|             } catch (error) { | ||||
|                 console.error(`Chyba při ukládání dat pro ${rest}:`, error); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // Připravit odpověď
 | ||||
|         const response: any = { | ||||
|             ok: true, | ||||
|             refreshed: results, | ||||
|             successful: successfulRestaurants | ||||
|         }; | ||||
| 
 | ||||
|         if (failedRestaurants.length > 0) { | ||||
|             response.warning = `Nepodařilo se načíst: ${failedRestaurants.join(', ')}`; | ||||
|             response.failed = failedRestaurants; | ||||
|         } | ||||
| 
 | ||||
|         res.status(200).json(response); | ||||
|     } catch (e: any) { | ||||
|         res.status(500).json({ error: e?.message || "Chyba při refreshi" }); | ||||
|     } | ||||
| } | ||||
| router.get("/refresh", refreshMetoda); | ||||
| 
 | ||||
| 
 | ||||
| export default router; | ||||
| @ -3,7 +3,8 @@ import { getLogin } from "../auth"; | ||||
| import { createPizzaDay, deletePizzaDay, getPizzaList, addPizzaOrder, removePizzaOrder, lockPizzaDay, unlockPizzaDay, finishPizzaOrder, finishPizzaDelivery, updatePizzaDayNote, updatePizzaFee } from "../pizza"; | ||||
| import { parseToken } from "../utils"; | ||||
| import { getWebsocket } from "../websocket"; | ||||
| import { AddPizzaData, FinishDeliveryData, RemovePizzaData, UpdatePizzaDayNoteData, UpdatePizzaFeeData } from "../../../types"; | ||||
| import { addVolatileData } from "../service"; | ||||
| import { AddPizzaRequest, FinishDeliveryRequest, RemovePizzaRequest, UpdatePizzaDayNoteRequest, UpdatePizzaFeeRequest } from "../../../types"; | ||||
| 
 | ||||
| const router = express.Router(); | ||||
| 
 | ||||
| @ -12,17 +13,17 @@ router.post("/create", async (req: Request<{}, any, undefined>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await createPizzaDay(login); | ||||
|     res.status(200).json(data); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
| }); | ||||
| 
 | ||||
| /** Smaže pizza day pro aktuální den, za předpokladu že existuje. */ | ||||
| router.post("/delete", async (req: Request<{}, any, undefined>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await deletePizzaDay(login); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
| }); | ||||
| 
 | ||||
| router.post("/add", async (req: Request<{}, any, AddPizzaData["body"]>, res) => { | ||||
| router.post("/add", async (req: Request<{}, any, AddPizzaRequest>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     if (isNaN(req.body?.pizzaIndex)) { | ||||
|         throw Error("Nebyl předán index pizzy"); | ||||
| @ -43,68 +44,68 @@ router.post("/add", async (req: Request<{}, any, AddPizzaData["body"]>, res) => | ||||
|         throw Error("Neplatný index velikosti pizzy: " + pizzaSizeIndex); | ||||
|     } | ||||
|     const data = await addPizzaOrder(login, pizzy[pizzaIndex], pizzy[pizzaIndex].sizes[pizzaSizeIndex]); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/remove", async (req: Request<{}, any, RemovePizzaData["body"]>, res) => { | ||||
| router.post("/remove", async (req: Request<{}, any, RemovePizzaRequest>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     if (!req.body?.pizzaOrder) { | ||||
|         throw Error("Nebyla předána objednávka"); | ||||
|     } | ||||
|     const data = await removePizzaOrder(login, req.body?.pizzaOrder); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/lock", async (req: Request<{}, any, undefined>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await lockPizzaDay(login); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/unlock", async (req: Request<{}, any, undefined>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await unlockPizzaDay(login); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/finishOrder", async (req: Request<{}, any, undefined>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await finishPizzaOrder(login); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/finishDelivery", async (req: Request<{}, any, FinishDeliveryData["body"]>, res) => { | ||||
| router.post("/finishDelivery", async (req: Request<{}, any, FinishDeliveryRequest>, res) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await finishPizzaDelivery(login, req.body.bankAccount, req.body.bankAccountHolder); | ||||
|     getWebsocket().emit("message", data); | ||||
|     getWebsocket().emit("message", await addVolatileData(data)); | ||||
|     res.status(200).json({}); | ||||
| }); | ||||
| 
 | ||||
| router.post("/updatePizzaDayNote", async (req: Request<{}, any, UpdatePizzaDayNoteData["body"]>, res, next) => { | ||||
| router.post("/updatePizzaDayNote", async (req: Request<{}, any, UpdatePizzaDayNoteRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     try { | ||||
|         if (req.body.note && req.body.note.length > 70) { | ||||
|             throw Error("Poznámka může mít maximálně 70 znaků"); | ||||
|         } | ||||
|         const data = await updatePizzaDayNote(login, req.body.note); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
| 
 | ||||
| router.post("/updatePizzaFee", async (req: Request<{}, any, UpdatePizzaFeeData["body"]>, res, next) => { | ||||
| router.post("/updatePizzaFee", async (req: Request<{}, any, UpdatePizzaFeeRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     if (!req.body.login) { | ||||
|         return res.status(400).json({ error: "Nebyl předán login cílového uživatele" }); | ||||
|     } | ||||
|     try { | ||||
|         const data = await updatePizzaFee(login, req.body.login, req.body.text, req.body.price); | ||||
|         getWebsocket().emit("message", data); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| }); | ||||
|  | ||||
| @ -1,22 +0,0 @@ | ||||
| import express, { Request, Response } from "express"; | ||||
| import { getLogin } from "../auth"; | ||||
| import { parseToken } from "../utils"; | ||||
| import { getStats } from "../stats"; | ||||
| import { WeeklyStats } from "../../../types/gen/types.gen"; | ||||
| 
 | ||||
| const router = express.Router(); | ||||
| 
 | ||||
| router.get("/", async (req: Request<{}, any, undefined>, res: Response<WeeklyStats>) => { | ||||
|     getLogin(parseToken(req)); | ||||
|     if (typeof req.query.startDate === 'string' && typeof req.query.endDate === 'string') { | ||||
|         try { | ||||
|             const data = await getStats(req.query.startDate, req.query.endDate); | ||||
|             return res.status(200).json(data); | ||||
|         } catch (e) { | ||||
|             // necháme to zatím spadnout na 400
 | ||||
|         } | ||||
|     } | ||||
|     res.sendStatus(400); | ||||
| }); | ||||
| 
 | ||||
| export default router; | ||||
| @ -1,18 +1,18 @@ | ||||
| import express, { Request } from "express"; | ||||
| import express, { Request, Response } from "express"; | ||||
| import { getLogin } from "../auth"; | ||||
| import { parseToken } from "../utils"; | ||||
| import { getUserVotes, updateFeatureVote } from "../voting"; | ||||
| import { GetVotesData, UpdateVoteData } from "../../../types"; | ||||
| import { FeatureRequest, UpdateFeatureVoteRequest } from "../../../types"; | ||||
| 
 | ||||
| const router = express.Router(); | ||||
| 
 | ||||
| router.get("/getVotes", async (req: Request<{}, any, GetVotesData["body"]>, res) => { | ||||
| router.get("/getVotes", async (req: Request<{}, any, undefined>, res: Response<FeatureRequest[]>) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const data = await getUserVotes(login); | ||||
|     res.status(200).json(data); | ||||
| }); | ||||
| 
 | ||||
| router.post("/updateVote", async (req: Request<{}, any, UpdateVoteData["body"]>, res, next) => { | ||||
| router.post("/updateVote", async (req: Request<{}, any, UpdateFeatureVoteRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     if (req.body?.option == null || req.body?.active == null) { | ||||
|         res.status(400).json({ error: "Chybné parametry volání" }); | ||||
|  | ||||
| @ -1,8 +1,8 @@ | ||||
| import { InsufficientPermissions, formatDate, getDayOfWeekIndex, getFirstWorkDayOfWeek, getHumanDate, getIsWeekend, getWeekNumber } from "./utils"; | ||||
| import { ClientData, Restaurants, RestaurantDailyMenu, DepartureTime, DayData, WeekMenu, LocationKey, DayOfWeekIndex, daysOfWeeksIndices, DayOfWeekEnum, DayOfWeek } from "../../types"; | ||||
| import getStorage from "./storage"; | ||||
| import { getMenuSladovnicka, getMenuTechTower, getMenuZastavkaUmichala, getMenuSenkSerikova } from "./restaurants"; | ||||
| import { getMenuSladovnicka, getMenuTechTower, getMenuUMotliku, getMenuZastavkaUmichala } from "./restaurants"; | ||||
| import { getTodayMock } from "./mock"; | ||||
| import { ClientData, DepartureTime, LunchChoice, Restaurant, RestaurantDayMenu, WeekMenu } from "../../types/gen/types.gen"; | ||||
| 
 | ||||
| const storage = getStorage(); | ||||
| const MENU_PREFIX = 'menu'; | ||||
| @ -31,31 +31,44 @@ export const getDateForWeekIndex = (index: number) => { | ||||
| function getEmptyData(date?: Date): ClientData { | ||||
|     const usedDate = date || getToday(); | ||||
|     return { | ||||
|         todayDayIndex: getDayOfWeekIndex(getToday()), | ||||
|         date: getHumanDate(usedDate), | ||||
|         isWeekend: getIsWeekend(usedDate), | ||||
|         dayIndex: getDayOfWeekIndex(usedDate), | ||||
|         weekIndex: getDayOfWeekIndex(usedDate), | ||||
|         choices: {}, | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Přidá k datům "dopočítaná" data, která nejsou přímo uložena v databázi. | ||||
|  *  | ||||
|  * @param data data z databáze | ||||
|  * @returns obohacená data | ||||
|  */ | ||||
| export async function addVolatileData(data: ClientData): Promise<ClientData> { | ||||
|     data.todayWeekIndex = getDayOfWeekIndex(getToday()); | ||||
|     return data; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí veškerá klientská data pro předaný den, nebo aktuální den, pokud není předán. | ||||
|  */ | ||||
| export async function getData(date?: Date): Promise<ClientData> { | ||||
|     const clientData = await getClientData(date); | ||||
|     const targetDate = date ?? getToday(); | ||||
|     const dateString = formatDate(targetDate); | ||||
|     const data: DayData = await storage.getData(dateString) || getEmptyData(date); | ||||
|     let clientData: ClientData = { ...data }; | ||||
|     clientData.menus = { | ||||
|         SLADOVNICKA: await getRestaurantMenu('SLADOVNICKA', date), | ||||
|         // UMOTLIKU: await getRestaurantMenu('UMOTLIKU', date),
 | ||||
|         TECHTOWER: await getRestaurantMenu('TECHTOWER', date), | ||||
|         ZASTAVKAUMICHALA: await getRestaurantMenu('ZASTAVKAUMICHALA', date), | ||||
|         SENKSERIKOVA: await getRestaurantMenu('SENKSERIKOVA', date), | ||||
|         [Restaurants.SLADOVNICKA]: await getRestaurantMenu(Restaurants.SLADOVNICKA, date), | ||||
|         // [Restaurants.UMOTLIKU]: await getRestaurantMenu(Restaurants.UMOTLIKU, date),
 | ||||
|         [Restaurants.TECHTOWER]: await getRestaurantMenu(Restaurants.TECHTOWER, date), | ||||
|         [Restaurants.ZASTAVKAUMICHALA]: await getRestaurantMenu(Restaurants.ZASTAVKAUMICHALA, date), | ||||
|     } | ||||
|     clientData = await addVolatileData(clientData); | ||||
|     return clientData; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí klíč, pod kterým je uloženo menu pro týden příslušící předanému datu. | ||||
|  * Vrátí klíč, pod kterým je uloženo menu pro předané datum. | ||||
|  * | ||||
|  * @param date datum | ||||
|  * @returns databázový klíč | ||||
| @ -66,117 +79,25 @@ function getMenuKey(date: Date) { | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí menu všech podniků pro celý týden do kterého spadá předané datum, pokud již existují. | ||||
|  * Vrátí menu restaurací pro předané datum, pokud již existují. | ||||
|  *  | ||||
|  * @param date datum | ||||
|  * @returns menu restaurací pro týden příslušící předanému datu | ||||
|  * @returns menu restaurací pro předané datum | ||||
|  */ | ||||
| async function getMenu(date: Date): Promise<WeekMenu | undefined> { | ||||
|     return await storage.getData<WeekMenu | undefined>(getMenuKey(date)); | ||||
|     return await storage.getData(getMenuKey(date)); | ||||
| } | ||||
| 
 | ||||
| // TODO přesun do restaurants.ts
 | ||||
| /** | ||||
|  * Načte menu dané restaurace pro celý týden bez ukládání do storage. | ||||
|  * Používá se pro validaci dat před uložením. | ||||
|  *  | ||||
|  * @param restaurant restaurace | ||||
|  * @param firstDay první pracovní den týdne | ||||
|  * @returns pole menu pro jednotlivé dny týdne | ||||
|  */ | ||||
| export async function fetchRestaurantWeekMenuData(restaurant: Restaurant, firstDay: Date): Promise<any[]> { | ||||
|     return await fetchRestaurantWeekMenu(restaurant, firstDay); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Uloží týdenní menu restaurace do storage. | ||||
|  *  | ||||
|  * @param restaurant restaurace | ||||
|  * @param date datum z týdne, pro který ukládat menu | ||||
|  * @param weekData data týdenního menu | ||||
|  */ | ||||
| export async function saveRestaurantWeekMenu(restaurant: Restaurant, date: Date, weekData: any[]): Promise<void> { | ||||
|     const now = new Date().getTime(); | ||||
|     let weekMenu = await getMenu(date); | ||||
|     weekMenu ??= [{}, {}, {}, {}, {}]; | ||||
|      | ||||
|     // Inicializace struktury pro restauraci
 | ||||
|     for (let i = 0; i < 5; i++) { | ||||
|         weekMenu[i] ??= {}; | ||||
|         weekMenu[i][restaurant] ??= { | ||||
|             lastUpdate: now, | ||||
|             closed: false, | ||||
|             food: [], | ||||
|         }; | ||||
|     } | ||||
|      | ||||
|     // Uložení dat pro všechny dny
 | ||||
|     for (let i = 0; i < weekData.length && i < weekMenu.length; i++) { | ||||
|         weekMenu[i][restaurant]!.food = weekData[i]; | ||||
|         weekMenu[i][restaurant]!.lastUpdate = now; | ||||
|          | ||||
|         // Detekce uzavření pro každou restauraci
 | ||||
|         switch (restaurant) { | ||||
|             case 'SLADOVNICKA': | ||||
|                 if (weekData[i].length === 1 && weekData[i][0].name.toLowerCase() === 'pro daný den nebyla nalezena denní nabídka') { | ||||
|                     weekMenu[i][restaurant]!.closed = true; | ||||
|                 } | ||||
|                 break; | ||||
|             case 'TECHTOWER': | ||||
|                 if (weekData[i]?.length === 1 && weekData[i][0].name.toLowerCase() === 'svátek') { | ||||
|                     weekMenu[i][restaurant]!.closed = true; | ||||
|                 } | ||||
|                 break; | ||||
|             case 'ZASTAVKAUMICHALA': | ||||
|                 if (weekData[i]?.length === 1 && weekData[i][0].name === 'Pro tento den není uveřejněna nabídka jídel.') { | ||||
|                     weekMenu[i][restaurant]!.closed = true; | ||||
|                 } | ||||
|                 break; | ||||
|             case 'SENKSERIKOVA': | ||||
|                 if (weekData[i]?.length === 1 && weekData[i][0].name === 'Pro tento den nebylo zadáno menu.') { | ||||
|                     weekMenu[i][restaurant]!.closed = true; | ||||
|                 } | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     // Uložení do storage
 | ||||
|     await storage.setData(getMenuKey(date), weekMenu); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Načte menu dané restaurace pro celý týden bez ukládání do storage. | ||||
|  *  | ||||
|  * @param restaurant restaurace | ||||
|  * @param firstDay první pracovní den týdne | ||||
|  * @returns pole menu pro jednotlivé dny týdne | ||||
|  */ | ||||
| async function fetchRestaurantWeekMenu(restaurant: Restaurant, firstDay: Date): Promise<any[]> { | ||||
|     const mock = process.env.MOCK_DATA === 'true'; | ||||
|      | ||||
|     switch (restaurant) { | ||||
|         case 'SLADOVNICKA': | ||||
|             return await getMenuSladovnicka(firstDay, mock); | ||||
|         case 'TECHTOWER': | ||||
|             return await getMenuTechTower(firstDay, mock); | ||||
|         case 'ZASTAVKAUMICHALA': | ||||
|             return await getMenuZastavkaUmichala(firstDay, mock); | ||||
|         case 'SENKSERIKOVA': | ||||
|             return await getMenuSenkSerikova(firstDay, mock); | ||||
|         default: | ||||
|             throw new Error(`Nepodporovaná restaurace: ${restaurant}`); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí menu dané restaurace pro předaný den. | ||||
|  * Pokud neexistuje, provede stažení menu pro příslušný týden a uložení do DB. | ||||
|  *  | ||||
|  * @param restaurant restaurace | ||||
|  * @param date datum, ke kterému získat menu | ||||
|  * @param forceRefresh příznak vynuceného obnovení | ||||
|  * @param mock příznak, zda chceme pouze mock data | ||||
|  */ | ||||
| export async function getRestaurantMenu(restaurant: Restaurant, date?: Date, forceRefresh = false): Promise<RestaurantDayMenu> { | ||||
| export async function getRestaurantMenu(restaurant: Restaurants, date?: Date): Promise<RestaurantDailyMenu> { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     const dayOfWeekIndex = getDayOfWeekIndex(usedDate); | ||||
|     const now = new Date().getTime(); | ||||
| @ -188,59 +109,94 @@ export async function getRestaurantMenu(restaurant: Restaurant, date?: Date, for | ||||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     let weekMenu = await getMenu(usedDate); | ||||
|     weekMenu ??= [{}, {}, {}, {}, {}]; | ||||
|     for (let i = 0; i < 5; i++) { | ||||
|         weekMenu[i] ??= {}; | ||||
|         weekMenu[i][restaurant] ??= { | ||||
|     let menus = await getMenu(usedDate); | ||||
|     if (menus == null) { | ||||
|         menus = {}; | ||||
|     } | ||||
|     daysOfWeeksIndices.forEach(i => { | ||||
|         if (menus[i] == null) { | ||||
|             menus[i] = {}; | ||||
|         } | ||||
|         if (menus[i][restaurant] == null) { | ||||
|             menus[i][restaurant] = { | ||||
|                 lastUpdate: now, | ||||
|                 closed: false, | ||||
|                 food: [], | ||||
|             }; | ||||
|         } | ||||
|     if (forceRefresh || (!weekMenu[dayOfWeekIndex][restaurant]?.food?.length && !weekMenu[dayOfWeekIndex][restaurant]?.closed)) { | ||||
|     }) | ||||
|     if (!menus[dayOfWeekIndex]) { | ||||
|         menus[dayOfWeekIndex] = {}; | ||||
|     } | ||||
|     if (!menus[dayOfWeekIndex][restaurant]?.food?.length) { | ||||
|         const firstDay = getFirstWorkDayOfWeek(usedDate); | ||||
|          | ||||
|         try { | ||||
|             const restaurantWeekFood = await fetchRestaurantWeekMenu(restaurant, firstDay); | ||||
|              | ||||
|             // Aktualizace menu pro všechny dny
 | ||||
|             for (let i = 0; i < restaurantWeekFood.length && i < weekMenu.length; i++) { | ||||
|                 weekMenu[i][restaurant]!.food = restaurantWeekFood[i]; | ||||
|                 weekMenu[i][restaurant]!.lastUpdate = now; | ||||
|                  | ||||
|                 // Detekce uzavření pro každou restauraci
 | ||||
|         const mock = process.env.MOCK_DATA === 'true'; | ||||
|         switch (restaurant) { | ||||
|                     case 'SLADOVNICKA': | ||||
|                         if (restaurantWeekFood[i].length === 1 && restaurantWeekFood[i][0].name.toLowerCase() === 'pro daný den nebyla nalezena denní nabídka') { | ||||
|                             weekMenu[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                         break; | ||||
|                     case 'TECHTOWER': | ||||
|                         if (restaurantWeekFood[i]?.length === 1 && restaurantWeekFood[i][0].name.toLowerCase() === 'svátek') { | ||||
|                             weekMenu[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                         break; | ||||
|                     case 'ZASTAVKAUMICHALA': | ||||
|                         if (restaurantWeekFood[i]?.length === 1 && restaurantWeekFood[i][0].name === 'Pro tento den není uveřejněna nabídka jídel.') { | ||||
|                             weekMenu[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                         break; | ||||
|                     case 'SENKSERIKOVA': | ||||
|                         if (restaurantWeekFood[i]?.length === 1 && restaurantWeekFood[i][0].name === 'Pro tento den nebylo zadáno menu.') { | ||||
|                             weekMenu[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                         break; | ||||
|             case Restaurants.SLADOVNICKA: | ||||
|                 try { | ||||
|                     // TODO tady jsme v háji, protože z následujících metod vracíme arbitrárně dlouhé pole (musíme vracet omezené na maximálně 0-7 prvků)
 | ||||
|                     const sladovnickaFood = await getMenuSladovnicka(firstDay, mock); | ||||
|                     for (const i in DayOfWeekEnum) { | ||||
|                         menus[i][restaurant]!.food = sladovnickaFood[i]; | ||||
|                         // Velice chatrný a nespolehlivý způsob detekce uzavření...
 | ||||
|                         if (sladovnickaFood[i].length === 1 && sladovnickaFood[i][0].name.toLowerCase() === 'pro daný den nebyla nalezena denní nabídka') { | ||||
|                             menus[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                     } | ||||
|                     for (let i = 0; i < sladovnickaFood.length; i++) { | ||||
|                         menus[i][restaurant]!.food = sladovnickaFood[i]; | ||||
|                         // Velice chatrný a nespolehlivý způsob detekce uzavření...
 | ||||
|                         if (sladovnickaFood[i].length === 1 && sladovnickaFood[i][0].name.toLowerCase() === 'pro daný den nebyla nalezena denní nabídka') { | ||||
|                             menus[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                     } | ||||
|              | ||||
|             // Uložení do storage
 | ||||
|             await storage.setData(getMenuKey(usedDate), weekMenu); | ||||
|                 } catch (e: any) { | ||||
|             console.error(`Selhalo načtení jídel pro podnik ${restaurant}`, e); | ||||
|                     console.error("Selhalo načtení jídel pro podnik Sladovnická", e); | ||||
|                 } | ||||
|                 break; | ||||
|             // case Restaurants.UMOTLIKU:
 | ||||
|             //     try {
 | ||||
|             //         const uMotlikuFood = await getMenuUMotliku(firstDay, mock);
 | ||||
|             //         for (let i = 0; i < uMotlikuFood.length; i++) {
 | ||||
|             //             menus[i][restaurant]!.food = uMotlikuFood[i];
 | ||||
|             //             if (uMotlikuFood[i].length === 1 && uMotlikuFood[i][0].name.toLowerCase() === 'zavřeno') {
 | ||||
|             //                 menus[i][restaurant]!.closed = true;
 | ||||
|             //             }
 | ||||
|             //         }
 | ||||
|             //     } catch (e: any) {
 | ||||
|             //         console.error("Selhalo načtení jídel pro podnik U Motlíků", e);
 | ||||
|             //     }
 | ||||
|             //     break;
 | ||||
|             case Restaurants.TECHTOWER: | ||||
|                 try { | ||||
|                     const techTowerFood = await getMenuTechTower(firstDay, mock); | ||||
|                     for (let i = 0; i < techTowerFood.length; i++) { | ||||
|                         menus[i][restaurant]!.food = techTowerFood[i]; | ||||
|                         if (techTowerFood[i]?.length === 1 && techTowerFood[i][0].name.toLowerCase() === 'svátek') { | ||||
|                             menus[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                     } | ||||
|     return weekMenu[dayOfWeekIndex][restaurant]!; | ||||
|                     break; | ||||
|                 } catch (e: any) { | ||||
|                     console.error("Selhalo načtení jídel pro podnik TechTower", e); | ||||
|                 } | ||||
|             case Restaurants.ZASTAVKAUMICHALA: | ||||
|                 try { | ||||
|                     const zastavkaUmichalaFood = await getMenuZastavkaUmichala(firstDay, mock); | ||||
|                     for (let i = 0; i < zastavkaUmichalaFood.length; i++) { | ||||
|                         menus[i][restaurant]!.food = zastavkaUmichalaFood[i]; | ||||
|                         if (zastavkaUmichalaFood[i]?.length === 1 && zastavkaUmichalaFood[i][0].name === 'Pro tento den není uveřejněna nabídka jídel.') { | ||||
|                             menus[i][restaurant]!.closed = true; | ||||
|                         } | ||||
|                     } | ||||
|                     break; | ||||
|                 } catch (e: any) { | ||||
|                     console.error("Selhalo načtení jídel pro podnik Zastávka u Michala", e); | ||||
|                 } | ||||
|         } | ||||
|         await storage.setData(getMenuKey(usedDate), menus); | ||||
|     } | ||||
|     return menus[dayOfWeekIndex][restaurant]!; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
| @ -265,9 +221,9 @@ export async function initIfNeeded(date?: Date) { | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns  | ||||
|  */ | ||||
| export async function removeChoices(login: string, trusted: boolean, locationKey: LunchChoice, date?: Date) { | ||||
| export async function removeChoices(login: string, trusted: boolean, locationKey: LocationKey, date?: Date) { | ||||
|     const selectedDay = formatDate(date ?? getToday()); | ||||
|     let data = await getClientData(date); | ||||
|     let data: DayData = await storage.getData(selectedDay); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     if (locationKey in data.choices) { | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
| @ -292,15 +248,15 @@ export async function removeChoices(login: string, trusted: boolean, locationKey | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns  | ||||
|  */ | ||||
| export async function removeChoice(login: string, trusted: boolean, locationKey: LunchChoice, foodIndex: number, date?: Date) { | ||||
| export async function removeChoice(login: string, trusted: boolean, locationKey: LocationKey, foodIndex: number, date?: Date) { | ||||
|     const selectedDay = formatDate(date ?? getToday()); | ||||
|     let data = await getClientData(date); | ||||
|     let data: DayData = await storage.getData(selectedDay); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     if (locationKey in data.choices) { | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
|             const index = data.choices[locationKey][login].selectedFoods?.indexOf(foodIndex); | ||||
|             if (index != null && index > -1) { | ||||
|                 data.choices[locationKey][login].selectedFoods?.splice(index, 1); | ||||
|             const index = data.choices[locationKey][login].options.indexOf(foodIndex); | ||||
|             if (index > -1) { | ||||
|                 data.choices[locationKey][login].options.splice(index, 1) | ||||
|                 await storage.setData(selectedDay, data); | ||||
|             } | ||||
|         } | ||||
| @ -309,26 +265,20 @@ export async function removeChoice(login: string, trusted: boolean, locationKey: | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Odstraní kompletně volbu uživatele, vyjma ignoredLocationKey (pokud byla předána a existuje). | ||||
|  * Odstraní kompletně volbu uživatele. | ||||
|  *  | ||||
|  * @param login login uživatele | ||||
|  * @param date datum, ke kterému se volby vztahují | ||||
|  * @param ignoredLocationKey volba, která nebude odstraněna, pokud existuje | ||||
|  */ | ||||
| async function removeChoiceIfPresent(login: string, date?: Date, ignoredLocationKey?: LunchChoice) { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     let data = await getClientData(usedDate); | ||||
| async function removeChoiceIfPresent(login: string, date: string) { | ||||
|     let data: DayData = await storage.getData(date); | ||||
|     for (const key of Object.keys(data.choices)) { | ||||
|         const locationKey = key as LunchChoice; | ||||
|         if (ignoredLocationKey != null && ignoredLocationKey == locationKey) { | ||||
|             continue; | ||||
|         } | ||||
|         const locationKey = key as LocationKey; | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
|             delete data.choices[locationKey][login]; | ||||
|             if (Object.keys(data.choices[locationKey]).length === 0) { | ||||
|                 delete data.choices[locationKey]; | ||||
|             } | ||||
|             await storage.setData(formatDate(usedDate), data); | ||||
|             await storage.setData(date, data); | ||||
|         } | ||||
|     } | ||||
|     return data; | ||||
| @ -367,64 +317,36 @@ function validateTrusted(data: ClientData, login: string, trusted: boolean) { | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns aktuální data | ||||
|  */ | ||||
| export async function addChoice(login: string, trusted: boolean, locationKey: LunchChoice, foodIndex?: number, date?: Date) { | ||||
| export async function addChoice(login: string, trusted: boolean, locationKey: LocationKey, foodIndex?: number, date?: Date) { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     await initIfNeeded(usedDate); | ||||
|     let data = await getClientData(usedDate); | ||||
|     const selectedDate = formatDate(usedDate); | ||||
|     let data: DayData = await storage.getData(selectedDate); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     await validateFoodIndex(locationKey, foodIndex, date); | ||||
|     // Pokud měníme pouze lokaci, mažeme případné předchozí
 | ||||
|     if (foodIndex == null) { | ||||
|         data = await removeChoiceIfPresent(login, usedDate); | ||||
|     } else { | ||||
|         // Mažeme případné ostatní volby (měla by být maximálně jedna)
 | ||||
|         removeChoiceIfPresent(login, usedDate, locationKey); | ||||
|         data = await removeChoiceIfPresent(login, selectedDate); | ||||
|     } | ||||
|     // TODO vytáhnout inicializaci "prázdné struktury" do vlastní funkce
 | ||||
|     data.choices[locationKey] ??= {}; | ||||
|     if (!(data.choices[locationKey])) { | ||||
|         data.choices[locationKey] = {} | ||||
|     } | ||||
|     if (!(login in data.choices[locationKey])) { | ||||
|         if (!data.choices[locationKey]) { | ||||
|             data.choices[locationKey] = {} | ||||
|         } | ||||
|         data.choices[locationKey][login] = { | ||||
|             trusted, | ||||
|             selectedFoods: [] | ||||
|             options: [] | ||||
|         }; | ||||
|     } | ||||
|     if (foodIndex != null && !data.choices[locationKey][login].selectedFoods?.includes(foodIndex)) { | ||||
|         data.choices[locationKey][login].selectedFoods?.push(foodIndex); | ||||
|     if (foodIndex != null && !data.choices[locationKey][login].options.includes(foodIndex)) { | ||||
|         data.choices[locationKey][login].options.push(foodIndex); | ||||
|     } | ||||
|     const selectedDate = formatDate(usedDate); | ||||
|     await storage.setData(selectedDate, data); | ||||
|     return data; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Zvaliduje platnost indexu jídla pro vybranou lokalitu a datum. | ||||
|  *  | ||||
|  * @param locationKey vybraná lokalita | ||||
|  * @param foodIndex index jídla pro danou lokalitu | ||||
|  * @param date datum, pro které je validace prováděna | ||||
|  */ | ||||
| async function validateFoodIndex(locationKey: LunchChoice, foodIndex?: number, date?: Date) { | ||||
|     if (foodIndex != null) { | ||||
|         if (typeof foodIndex !== 'number') { | ||||
|             throw Error(`Neplatný index ${foodIndex} typu ${typeof foodIndex}`); | ||||
|         } | ||||
|         if (foodIndex < 0) { | ||||
|             throw Error(`Neplatný index ${foodIndex}`); | ||||
|         } | ||||
|         if (!Object.keys(Restaurant).includes(locationKey)) { | ||||
|             throw Error(`Neplatný index ${foodIndex} pro lokalitu ${locationKey} nepodporující indexy`); | ||||
|         } | ||||
|         const usedDate = date ?? getToday(); | ||||
|         const menu = await getRestaurantMenu(locationKey as Restaurant, usedDate); | ||||
|         if (menu.food?.length && foodIndex > (menu.food.length - 1)) { | ||||
|             throw new Error(`Neplatný index ${foodIndex} pro lokalitu ${locationKey}`); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Aktualizuje poznámku k aktuálně vybrané možnosti. | ||||
|  *  | ||||
| @ -436,16 +358,16 @@ async function validateFoodIndex(locationKey: LunchChoice, foodIndex?: number, d | ||||
| export async function updateNote(login: string, trusted: boolean, note?: string, date?: Date) { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     await initIfNeeded(usedDate); | ||||
|     let data = await getClientData(usedDate); | ||||
|     const selectedDate = formatDate(usedDate); | ||||
|     let data: DayData = await storage.getData(selectedDate); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     const userEntry = data.choices != null && Object.entries(data.choices).find(entry => entry[1][login] != null); | ||||
|     if (userEntry) { | ||||
|         if (!note?.length) { | ||||
|         if (!note || !note.length) { | ||||
|             delete userEntry[1][login].note; | ||||
|         } else { | ||||
|             userEntry[1][login].note = note; | ||||
|         } | ||||
|         const selectedDate = formatDate(usedDate); | ||||
|         await storage.setData(selectedDate, data); | ||||
|     } | ||||
|     return data; | ||||
| @ -459,8 +381,8 @@ export async function updateNote(login: string, trusted: boolean, note?: string, | ||||
|  * @param date datum, ke kterému se čas vztahuje | ||||
|  */ | ||||
| export async function updateDepartureTime(login: string, time?: string, date?: Date) { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     let clientData = await getClientData(usedDate); | ||||
|     const selectedDate = formatDate(date ?? getToday()); | ||||
|     let clientData: DayData = await storage.getData(selectedDate); | ||||
|     const found = Object.values(clientData.choices).find(location => login in location); | ||||
|     // TODO validace, že se jedná o restauraci
 | ||||
|     if (found) { | ||||
| @ -472,23 +394,7 @@ export async function updateDepartureTime(login: string, time?: string, date?: D | ||||
|             } | ||||
|             found[login].departureTime = time; | ||||
|         } | ||||
|         await storage.setData(formatDate(usedDate), clientData); | ||||
|         await storage.setData(selectedDate, clientData); | ||||
|     } | ||||
|     return clientData; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Vrátí data pro klienta pro předaný nebo aktuální den. | ||||
|  *  | ||||
|  * @param date datum pro který vrátit data, pokud není vyplněno, je použit dnešní den | ||||
|  * @returns data pro klienta | ||||
|  */ | ||||
| export async function getClientData(date?: Date): Promise<ClientData> { | ||||
|     const targetDate = date ?? getToday(); | ||||
|     const dateString = formatDate(targetDate); | ||||
|     const clientData = await storage.getData<ClientData>(dateString) || getEmptyData(date); | ||||
|     return { | ||||
|         ...clientData, | ||||
|         todayDayIndex: getDayOfWeekIndex(getToday()), | ||||
|     } | ||||
| } | ||||
| @ -1,48 +0,0 @@ | ||||
| import { DailyStats, LunchChoice, WeeklyStats } from "../../types/gen/types.gen"; | ||||
| import { getStatsMock } from "./mock"; | ||||
| import { getClientData } from "./service"; | ||||
| import getStorage from "./storage"; | ||||
| 
 | ||||
| const storage = getStorage(); | ||||
| 
 | ||||
| /** | ||||
|  * Vypočte a vrátí statistiky jednotlivých možností pro předaný rozsah dat. | ||||
|  *  | ||||
|  * @param startDate počáteční datum | ||||
|  * @param endDate koncové datum | ||||
|  * @returns statistiky pro zadaný rozsah dat | ||||
|  */ | ||||
| export async function getStats(startDate: string, endDate: string): Promise<WeeklyStats> { | ||||
|     if (process.env.MOCK_DATA === 'true') { | ||||
|         return getStatsMock(); | ||||
|     } | ||||
|     const start = new Date(startDate); | ||||
|     const end = new Date(endDate); | ||||
| 
 | ||||
|     // Dočasná validace, aby to někdo ručně neshodil
 | ||||
|     const daysDiff = ((end as any) - (start as any)) / (1000 * 60 * 60 * 24); | ||||
|     if (daysDiff > 4) { | ||||
|         throw Error('Neplatný rozsah'); | ||||
|     } | ||||
| 
 | ||||
|     const result = []; | ||||
|     for (const date = start; date <= end; date.setDate(date.getDate() + 1)) { | ||||
|         const locationsStats: DailyStats = { | ||||
|             // TODO vytáhnout do utils funkce
 | ||||
|             date: `${String(date.getDate()).padStart(2, "0")}.${String(date.getMonth() + 1).padStart(2, "0")}.`, | ||||
|             locations: {} | ||||
|         } | ||||
|         const data = await getClientData(date); | ||||
|         if (data?.choices) { | ||||
|             Object.keys(data.choices).forEach(locationKey => { | ||||
|                 if (!locationsStats.locations) { | ||||
|                     locationsStats.locations = {} | ||||
|                 } | ||||
|                 // TODO dořešit, tohle je zmatek a té hlášce Sonaru nerozumím
 | ||||
|                 locationsStats.locations[locationKey as LunchChoice] = Object.keys(data.choices[locationKey as LunchChoice]!).length; | ||||
|             }) | ||||
|         } | ||||
|         result.push(locationsStats); | ||||
|     } | ||||
|     return result as WeeklyStats; | ||||
| } | ||||
| @ -1,3 +1,5 @@ | ||||
| import { ClientData } from "../../../types"; | ||||
| 
 | ||||
| /** | ||||
|  * Interface pro úložiště dat. | ||||
|  *  | ||||
| @ -5,12 +7,6 @@ | ||||
|  * Postupem času lze předělat pro efektivnější využití Redis. | ||||
|  */ | ||||
| export interface StorageInterface { | ||||
| 
 | ||||
|     /** | ||||
|      * Inicializuje úložiště, pokud je potřeba (např. připojení k databázi). | ||||
|      */ | ||||
|     initialize?(): Promise<void>; | ||||
| 
 | ||||
|     /** | ||||
|      * Vrátí příznak, zda existují data pro předaný klíč. | ||||
|      * @param key klíč, pro který zjišťujeme data (typicky datum) | ||||
| @ -21,7 +17,7 @@ export interface StorageInterface { | ||||
|      * Vrátí veškerá data pro předaný klíč. | ||||
|      * @param key klíč, pro který vrátit data (typicky datum) | ||||
|      */ | ||||
|     getData<Type>(key: string): Promise<Type | undefined>; | ||||
|     getData<Type>(key: string): Promise<Type>; | ||||
| 
 | ||||
|     /** | ||||
|      * Uloží data pod předaný klíč. | ||||
|  | ||||
| @ -4,8 +4,8 @@ import { StorageInterface } from "./StorageInterface"; | ||||
| import JsonStorage from "./json"; | ||||
| import RedisStorage from "./redis"; | ||||
| 
 | ||||
| const ENVIRONMENT = process.env.NODE_ENV ?? 'production'; | ||||
| dotenv.config({ path: path.resolve(__dirname, `../../.env.${ENVIRONMENT}`) }); | ||||
| const ENVIRONMENT = process.env.NODE_ENV || 'production'; | ||||
| dotenv.config({ path: path.resolve(__dirname, `../.env.${ENVIRONMENT}`) }); | ||||
| 
 | ||||
| const JSON_KEY = 'json'; | ||||
| const REDIS_KEY = 'redis'; | ||||
| @ -19,13 +19,6 @@ if (!process.env.STORAGE || process.env.STORAGE?.toLowerCase() === JSON_KEY) { | ||||
|     throw Error("Nepodporovaná hodnota proměnné STORAGE: " + process.env.STORAGE + ", podporované jsou 'json' nebo 'redis'"); | ||||
| } | ||||
| 
 | ||||
| (async () => { | ||||
|     if (storage.initialize) { | ||||
|         await storage.initialize(); | ||||
|     } | ||||
| })(); | ||||
| 
 | ||||
| 
 | ||||
| export default function getStorage(): StorageInterface { | ||||
|     return storage; | ||||
| } | ||||
|  | ||||
| @ -1,17 +1,8 @@ | ||||
| import JSONdb from 'simple-json-db'; | ||||
| import { StorageInterface } from "./StorageInterface"; | ||||
| import * as fs from 'fs'; | ||||
| import * as path from 'path'; | ||||
| 
 | ||||
| const dbPath = path.resolve(__dirname, '../../data/db.json'); | ||||
| const dbDir = path.dirname(dbPath); | ||||
| const db = new JSONdb('./data.json'); | ||||
| 
 | ||||
| // Zajistěte, že adresář existuje
 | ||||
| if (!fs.existsSync(dbDir)) { | ||||
|     fs.mkdirSync(dbDir, { recursive: true }); | ||||
| } | ||||
| 
 | ||||
| const db = new JSONdb(dbPath); | ||||
| /** | ||||
|  * Implementace úložiště používající JSON soubor. | ||||
|  */ | ||||
|  | ||||
| @ -8,18 +8,15 @@ let client: RedisClientType; | ||||
|  */ | ||||
| export default class RedisStorage implements StorageInterface { | ||||
|     constructor() { | ||||
|         const HOST = process.env.REDIS_HOST ?? 'localhost'; | ||||
|         const PORT = process.env.REDIS_PORT ?? 6379; | ||||
|         const HOST = process.env.REDIS_HOST || 'localhost'; | ||||
|         const PORT = process.env.REDIS_PORT || 6379; | ||||
|         client = createClient({ url: `redis://${HOST}:${PORT}` }); | ||||
|     } | ||||
| 
 | ||||
|     async initialize() { | ||||
|         client.connect(); | ||||
|     } | ||||
| 
 | ||||
|     async hasData(key: string) { | ||||
|         const data = await client.json.get(key); | ||||
|         return (!!data); | ||||
|         return (data ? true : false); | ||||
|     } | ||||
| 
 | ||||
|     async getData<Type>(key: string) { | ||||
|  | ||||
| @ -1,6 +1,4 @@ | ||||
| import { LunchChoice, LunchChoices } from "../../types/gen/types.gen"; | ||||
| 
 | ||||
| const DAY_OF_WEEK_FORMAT = new Intl.DateTimeFormat(undefined, { weekday: 'long' }); | ||||
| import { Choices, DayOfWeekIndex, LocationKey } from "../../types"; | ||||
| 
 | ||||
| /** Vrátí datum v ISO formátu. */ | ||||
| export function formatDate(date: Date, format?: string) { | ||||
| @ -8,7 +6,7 @@ export function formatDate(date: Date, format?: string) { | ||||
|     let month = String(date.getMonth() + 1).padStart(2, "0"); | ||||
|     let year = String(date.getFullYear()); | ||||
| 
 | ||||
|     const f = format ?? 'YYYY-MM-DD'; | ||||
|     const f = (format === undefined) ? 'YYYY-MM-DD' : format; | ||||
|     return f.replace('DD', day).replace('MM', month).replace('YYYY', year); | ||||
| } | ||||
| 
 | ||||
| @ -17,7 +15,7 @@ export function getHumanDate(date: Date) { | ||||
|     let currentDay = String(date.getDate()).padStart(2, '0'); | ||||
|     let currentMonth = String(date.getMonth() + 1).padStart(2, "0"); | ||||
|     let currentYear = date.getFullYear(); | ||||
|     let currentDayOfWeek = DAY_OF_WEEK_FORMAT.format(date); | ||||
|     let currentDayOfWeek = date.toLocaleDateString("CZ-cs", { weekday: 'long' }); | ||||
|     return `${currentDay}.${currentMonth}.${currentYear} (${currentDayOfWeek})`; | ||||
| } | ||||
| 
 | ||||
| @ -34,9 +32,9 @@ export function getHumanTime(time: Date) { | ||||
|  * @param date datum | ||||
|  * @returns index dne v týdnu | ||||
|  */ | ||||
| export const getDayOfWeekIndex = (date: Date) => { | ||||
| export const getDayOfWeekIndex = (date: Date): DayOfWeekIndex => { | ||||
|     // https://stackoverflow.com/a/4467559
 | ||||
|     return (((date.getDay() - 1) % 7) + 7) % 7; | ||||
|     return ((((date.getDay() - 1) % 7) + 7) % 7) as DayOfWeekIndex; | ||||
| } | ||||
| 
 | ||||
| /** Vrátí true, pokud je předané datum o víkendu. */ | ||||
| @ -61,10 +59,10 @@ export function getLastWorkDayOfWeek(date: Date) { | ||||
| 
 | ||||
| /** Vrátí pořadové číslo týdne předaného data v roce dle ISO 8601. */ | ||||
| export function getWeekNumber(inputDate: Date) { | ||||
|     const date = new Date(inputDate.getTime()); | ||||
|     var date = new Date(inputDate.getTime()); | ||||
|     date.setHours(0, 0, 0, 0); | ||||
|     date.setDate(date.getDate() + 3 - (date.getDay() + 6) % 7); | ||||
|     const week1 = new Date(date.getFullYear(), 0, 4); | ||||
|     var week1 = new Date(date.getFullYear(), 0, 4); | ||||
|     return 1 + Math.round(((date.getTime() - week1.getTime()) / 86400000 - 3 + (week1.getDay() + 6) % 7) / 7); | ||||
| } | ||||
| 
 | ||||
| @ -114,13 +112,13 @@ export const checkBodyParams = (req: any, paramNames: string[]) => { | ||||
| // TODO umístit do samostatného souboru
 | ||||
| export class InsufficientPermissions extends Error { } | ||||
| 
 | ||||
| export const getUsersByLocation = (choices: LunchChoices, login?: string): string[] => { | ||||
| export const getUsersByLocation = (choices: Choices, login: string): string[] => { | ||||
|     const result: string[] = []; | ||||
| 
 | ||||
|     for (const location of Object.entries(choices)) { | ||||
|         const locationKey = location[0] as LunchChoice; | ||||
|         const locationKey = location[0] as LocationKey; | ||||
|         const locationValue = location[1]; | ||||
|         if (login && locationValue[login]) { | ||||
|         if (locationValue[login]) { | ||||
|             for (const username in choices[locationKey]) { | ||||
|                 if (choices[locationKey].hasOwnProperty(username)) { | ||||
|                     result.push(username); | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { FeatureRequest } from "../../types/gen/types.gen"; | ||||
| import { FeatureRequest } from "../../types"; | ||||
| import getStorage from "./storage"; | ||||
| 
 | ||||
| interface VotingData { | ||||
| @ -15,7 +15,7 @@ const STORAGE_KEY = 'voting'; | ||||
|  * @returns pole voleb | ||||
|  */ | ||||
| export async function getUserVotes(login: string) { | ||||
|     const data = await storage.getData<VotingData>(STORAGE_KEY); | ||||
|     const data: VotingData = await storage.getData(STORAGE_KEY); | ||||
|     return data?.[login] || []; | ||||
| } | ||||
| 
 | ||||
| @ -28,8 +28,10 @@ export async function getUserVotes(login: string) { | ||||
|  * @returns aktuální data | ||||
|  */ | ||||
| export async function updateFeatureVote(login: string, option: FeatureRequest, active: boolean): Promise<VotingData> { | ||||
|     let data = await storage.getData<VotingData>(STORAGE_KEY); | ||||
|     data ??= {}; | ||||
|     let data: VotingData = await storage.getData(STORAGE_KEY); | ||||
|     if (data == null) { | ||||
|         data = {}; | ||||
|     } | ||||
|     if (!(login in data)) { | ||||
|         data[login] = []; | ||||
|     } | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import { DefaultEventsMap, Server } from "socket.io"; | ||||
| import { Server } from "socket.io"; | ||||
| import { DefaultEventsMap } from "socket.io/dist/typed-events"; | ||||
| 
 | ||||
| let io: Server<DefaultEventsMap, DefaultEventsMap, DefaultEventsMap, any>; | ||||
| 
 | ||||
|  | ||||
| @ -4,11 +4,10 @@ | ||||
|         "../types/**/*" | ||||
|     ], | ||||
|     "compilerOptions": { | ||||
|         "target": "ES2022", | ||||
|         "module": "Node16", | ||||
|         "moduleResolution": "node16", | ||||
|         "target": "ES2016", | ||||
|         "module": "CommonJS", | ||||
|         "jsx": "react", | ||||
|         "esModuleInterop": true, | ||||
|         "skipLibCheck": true, | ||||
|         "forceConsistentCasingInFileNames": true, | ||||
|         "outDir": "./dist", | ||||
|         "rootDir": "../", | ||||
|  | ||||
							
								
								
									
										3572
									
								
								server/yarn.lock
									
									
									
									
									
								
							
							
						
						
									
										3572
									
								
								server/yarn.lock
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										56
									
								
								types/RequestTypes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								types/RequestTypes.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | ||||
| import { FeatureRequest, LocationKey, PizzaOrder } from "./Types"; | ||||
| 
 | ||||
| export type ILocationKey = { | ||||
|     locationKey: LocationKey, | ||||
| } | ||||
| 
 | ||||
| export type IDayIndex = { | ||||
|     dayIndex?: number, | ||||
| } | ||||
| 
 | ||||
| export type AddChoiceRequest = IDayIndex & ILocationKey & { | ||||
|     foodIndex?: number, | ||||
| } | ||||
| 
 | ||||
| export type RemoveChoicesRequest = IDayIndex & ILocationKey; | ||||
| 
 | ||||
| export type RemoveChoiceRequest = IDayIndex & ILocationKey & { | ||||
|     foodIndex: number, | ||||
| } | ||||
| 
 | ||||
| export type UpdateNoteRequest = IDayIndex & { | ||||
|     note?: string, | ||||
| } | ||||
| 
 | ||||
| export type ChangeDepartureTimeRequest = IDayIndex & { | ||||
|     time: string, | ||||
| } | ||||
| 
 | ||||
| export type FinishDeliveryRequest = { | ||||
|     bankAccount?: string, | ||||
|     bankAccountHolder?: string, | ||||
| } | ||||
| 
 | ||||
| export type AddPizzaRequest = { | ||||
|     pizzaIndex: number, | ||||
|     pizzaSizeIndex: number, | ||||
| } | ||||
| 
 | ||||
| export type RemovePizzaRequest = { | ||||
|     pizzaOrder: PizzaOrder, | ||||
| } | ||||
| 
 | ||||
| export type UpdatePizzaDayNoteRequest = { | ||||
|     note?: string, | ||||
| } | ||||
| 
 | ||||
| export type UpdatePizzaFeeRequest = { | ||||
|     login: string, | ||||
|     text?: string, | ||||
|     price?: number, | ||||
| } | ||||
| 
 | ||||
| export type UpdateFeatureVoteRequest = { | ||||
|     option: FeatureRequest, | ||||
|     active: boolean, | ||||
| } | ||||
							
								
								
									
										236
									
								
								types/Types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										236
									
								
								types/Types.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,236 @@ | ||||
| /** Výčtový typ pro restaurace, pro které umíme získat a parsovat obědové menu. */ | ||||
| export enum Restaurants { | ||||
|     SLADOVNICKA = 'sladovnicka', | ||||
|     // UMOTLIKU = 'uMotliku',
 | ||||
|     TECHTOWER = 'techTower', | ||||
|     ZASTAVKAUMICHALA = 'zastavkaUmichala', | ||||
| } | ||||
| 
 | ||||
| export type FoodChoices = { | ||||
|     trusted: boolean, | ||||
|     options: number[], | ||||
|     departureTime?: string, | ||||
|     note?: string, | ||||
| } | ||||
| 
 | ||||
| // TODO okomentovat / rozdělit
 | ||||
| export type Choices = { | ||||
|     [location in LocationKey]?: { | ||||
|         [login: string]: FoodChoices | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** Velikost konkrétní pizzy */ | ||||
| export type PizzaSize = { | ||||
|     varId: number, // unikátní ID varianty pizzy
 | ||||
|     size: string, // velikost pizzy, např. "30cm"
 | ||||
|     pizzaPrice: number, // cena samotné pizzy
 | ||||
|     boxPrice: number, // cena krabice
 | ||||
|     price: number, // celková cena (pizza + krabice)
 | ||||
| } | ||||
| 
 | ||||
| /** Jedna konkrétní pizza */ | ||||
| export type Pizza = { | ||||
|     name: string, // název pizzy
 | ||||
|     ingredients: string[], // seznam ingrediencí
 | ||||
|     sizes: PizzaSize[], // dostupné velikosti pizzy
 | ||||
| } | ||||
| 
 | ||||
| /** Objednávka jedné konkrétní pizzy */ | ||||
| export type PizzaOrder = { | ||||
|     varId: number, // unikátní ID varianty pizzy
 | ||||
|     name: string, // název pizzy
 | ||||
|     size: string, // velikost pizzy jako string (30cm)
 | ||||
|     price: number, // cena pizzy v Kč, včetně krabice
 | ||||
| } | ||||
| 
 | ||||
| /** Celková objednávka jednoho člověka */ | ||||
| export type Order = { | ||||
|     customer: string, // jméno objednatele
 | ||||
|     pizzaList: PizzaOrder[], // seznam objednaných pizz
 | ||||
|     fee?: { text?: string, price: number }, // příplatek (např. za extra ingredience)
 | ||||
|     totalPrice: number, // celková cena všech objednaných pizz, krabic a příplatků
 | ||||
|     hasQr?: boolean, // true, pokud je k objednávce vygenerován QR kód pro platbu
 | ||||
|     note?: string, // volitelná uživatelská poznámka k objednávce
 | ||||
| } | ||||
| 
 | ||||
| /** Stav pizza dne */ | ||||
| export enum PizzaDayState { | ||||
|     NOT_CREATED, // Pizza day nebyl založen
 | ||||
|     CREATED, // Pizza day je založen
 | ||||
|     LOCKED, // Objednávky uzamčeny
 | ||||
|     ORDERED, // Pizzy objednány
 | ||||
|     DELIVERED // Pizzy doručeny
 | ||||
| } | ||||
| 
 | ||||
| /** Informace o pizza day pro dnešní den */ | ||||
| interface PizzaDay { | ||||
|     state: PizzaDayState, // stav pizza dne
 | ||||
|     creator: string, // jméno zakladatele
 | ||||
|     orders: Order[], // seznam objednávek jednotlivých lidí
 | ||||
| } | ||||
| 
 | ||||
| /** Index dne v týdnu (0 = pondělí, 6 = neděle) */ | ||||
| // TODO tohle by měl být (seřazený) enum MONDAY-SUNDAY, ne číslo
 | ||||
| export const daysOfWeeksIndices = [0, 1, 2, 3, 4, 5, 6] as const; | ||||
| export type DayOfWeekIndex = typeof daysOfWeeksIndices[number] | ||||
| 
 | ||||
| const daysOfWeek = ['MONDAY', 'TUESDAY', 'WEDNESDAY', 'THURSDAY', 'FRIDAY', 'SATURDAY', 'SUNDAY'] as const; | ||||
| export type DayOfWeek = typeof daysOfWeek[number]; | ||||
| 
 | ||||
| /** Denní menu všech dostupných podniků. */ | ||||
| export type DailyMenu = { | ||||
|     [restaurant in Restaurants]?: RestaurantDailyMenu | ||||
| } | ||||
| 
 | ||||
| /** Týdenní menu jednotlivých restaurací. */ | ||||
| export type WeekMenu = { | ||||
|     [dayIndex in DayOfWeek]?: DailyMenu | ||||
| } | ||||
| 
 | ||||
| /** Týdenní menu jedné restaurace. */ | ||||
| export type RestaurantWeeklyMenu = { | ||||
|     [key in DayOfWeek]?: Food[] | ||||
| } | ||||
| 
 | ||||
| /** Data vztahující se k jednomu konkrétnímu dni. */ | ||||
| export type DayData = { | ||||
|     date: string, // datum dne
 | ||||
|     isWeekend: boolean, // příznak, zda je datum víkend
 | ||||
|     weekIndex: DayOfWeekIndex, // index dne v týdnu (0-6)
 | ||||
|     choices: Choices, // seznam voleb uživatelů
 | ||||
|     menus?: { [restaurant in Restaurants]?: RestaurantDailyMenu }, // menu jednotlivých restaurací
 | ||||
|     pizzaDay?: PizzaDay, // pizza day pro dnešní den, pokud existuje
 | ||||
|     pizzaList?: Pizza[], // seznam dostupných pizz pro dnešní den
 | ||||
|     pizzaListLastUpdate?: Date, // datum a čas poslední aktualizace pizz
 | ||||
| } | ||||
| 
 | ||||
| /** Veškerá data pro zobrazení na klientovi. */ | ||||
| export type ClientData = DayData & { | ||||
|     todayWeekIndex?: DayOfWeekIndex, // index dnešního dne v týdnu (0-6)
 | ||||
| } | ||||
| 
 | ||||
| /** Nabídka jídel jednoho podniku pro jeden konkrétní den. */ | ||||
| export type RestaurantDailyMenu = { | ||||
|     lastUpdate: number, // UNIX timestamp poslední aktualizace menu
 | ||||
|     closed: boolean, // příznak, zda je daný podnik v tento den zavřený
 | ||||
|     food: Food[], // seznam jídel v menu
 | ||||
| } | ||||
| 
 | ||||
| /** Jídlo z obědového menu restaurace. */ | ||||
| export type Food = { | ||||
|     amount?: string, // množství standardní porce, např. 0,33l nebo 150g
 | ||||
|     name: string, // název/popis jídla
 | ||||
|     price: string, // cena ve formátu '135 Kč'
 | ||||
|     isSoup: boolean, // příznak, zda se jedná o polévku
 | ||||
| } | ||||
| 
 | ||||
| // TODO tohle je dost špatné pojmenování, vzhledem k tomu co to obsahuje
 | ||||
| export enum Locations { | ||||
|     SLADOVNICKA = 'Sladovnická', | ||||
|     // UMOTLIKU = 'U Motlíků',
 | ||||
|     TECHTOWER = 'TechTower', | ||||
|     ZASTAVKAUMICHALA = 'Zastávka u Michala', | ||||
|     SPSE = 'SPŠE', | ||||
|     PIZZA = 'Pizza day', | ||||
|     OBJEDNAVAM = 'Budu objednávat', | ||||
|     NEOBEDVAM = 'Mám vlastní/neobědvám', | ||||
|     ROZHODUJI = 'Rozhoduji se', | ||||
| } | ||||
| 
 | ||||
| // TODO totéž
 | ||||
| export type LocationKey = keyof typeof Locations; | ||||
| 
 | ||||
| export enum UdalostEnum { | ||||
|     ZAHAJENA_PIZZA = "Zahájen pizza day", | ||||
|     OBJEDNANA_PIZZA = "Objednána pizza", | ||||
|     JDEME_OBED = "Jdeme oběd", | ||||
| } | ||||
| 
 | ||||
| export type NotififaceInput = { | ||||
|     udalost: UdalostEnum, | ||||
|     user: string, | ||||
| } | ||||
| 
 | ||||
| export type NotifikaceData = { | ||||
|     input: NotififaceInput, | ||||
|     gotify?: boolean, | ||||
|     teams?: boolean, | ||||
|     ntfy?: boolean, | ||||
| } | ||||
| 
 | ||||
| export type GotifyServer = { | ||||
|     server: string; | ||||
|     api_keys: string[]; | ||||
| } | ||||
| 
 | ||||
| /** Čas preferovaného odchodu na oběd. */ | ||||
| export enum DepartureTime { | ||||
|     T10_00 = "10:00", | ||||
|     T10_15 = "10:15", | ||||
|     T10_30 = "10:30", | ||||
|     T10_45 = "10:45", | ||||
|     T11_00 = "11:00", | ||||
|     T11_15 = "11:15", | ||||
|     T11_30 = "11:30", | ||||
|     T11_45 = "11:45", | ||||
|     T12_00 = "12:00", | ||||
|     T12_15 = "12:15", | ||||
|     T12_30 = "12:30", | ||||
|     T12_45 = "12:45", | ||||
|     T13_00 = "13:00", | ||||
| } | ||||
| 
 | ||||
| export enum FeatureRequest { | ||||
|     CUSTOM_QR = "Ruční generování QR kódů mimo Pizza day (např. při objednávání)", | ||||
|     FAVORITES = "Možnost označovat si jídla jako oblíbená (taková jídla by se uživateli následně zvýrazňovala)", | ||||
|     SINGLE_PAYMENT = "Možnost úhrady v podniku za všechny jednou osobou a následné generování QR ostatním", | ||||
|     NO_WEEKENDS = "Zrušení \"užívejte víkend\", místo toho umožnit zpětně náhled na uplynulý týden", | ||||
|     QR_FOREVER = "Umožnění zobrazení vygenerovaného QR kódu i po následující dny (dokud ho uživatel ručně \"nezavře\", např. tlačítkem \"Zaplatil jsem\")", | ||||
|     PIZZA_PICTURES = "Zobrazování náhledů (fotografií) pizz v rámci Pizza day", | ||||
|     STATISTICS = "Statistiky (nejoblíbenější podnik, nejpopulárnější jídla, nejobjednávanější pizzy, nejčastější uživatelé, ...)", | ||||
|     RESPONSIVITY = "Vylepšení responzivního designu", | ||||
|     SECURITY = "Zvýšení zabezpečení aplikace", | ||||
|     SAFETY = "Zvýšená ochrana proti chybám uživatele (potvrzovací dialogy, překliky, ...)", | ||||
|     UI = "Celkové vylepšení UI/UX", | ||||
|     DEVELOPMENT = "Zlepšení dokumentace/postupů pro ostatní vývojáře" | ||||
| } | ||||
| 
 | ||||
| export type EasterEgg = { | ||||
|     path: string; | ||||
|     url: string; | ||||
|     startOffset: number; | ||||
|     endOffset: number; | ||||
|     duration: number; | ||||
|     width?: string; | ||||
|     zIndex?: number; | ||||
|     position?: "absolute"; | ||||
|     animationName?: string; | ||||
|     animationDuration?: string; | ||||
|     animationTimingFunction?: string; | ||||
| } | ||||
| 
 | ||||
| // TODO aktuálně se k ničemu nepoužívá
 | ||||
| export type AnimationPosition = { | ||||
|     left?: string, | ||||
|     startLeft?: string, | ||||
|     "--start-left"?: string, | ||||
|     right?: string, | ||||
|     startRight?: string, | ||||
|     "--start-right"?: string, | ||||
|     top?: string, | ||||
|     startTop?: string, | ||||
|     "--start-top"?: string, | ||||
|     bottom?: string, | ||||
|     startBottom?: string, | ||||
|     "--start-bottom"?: string, | ||||
|     endLeft?: string, | ||||
|     "--end-left"?: string, | ||||
|     endRight?: string, | ||||
|     "--end-right"?: string, | ||||
|     endTop?: string, | ||||
|     "--end-top"?: string, | ||||
|     endBottom?: string, | ||||
|     "--end-bottom"?: string, | ||||
|     rotate?: string, | ||||
| } | ||||
| @ -1,84 +0,0 @@ | ||||
| openapi: 3.0.4 | ||||
| info: | ||||
|   title: Luncher API | ||||
|   version: 1.0.0 | ||||
| servers: | ||||
|   - url: /api | ||||
| paths: | ||||
|   # Obecné (/api) | ||||
|   /login: | ||||
|     $ref: "./paths/login.yml" | ||||
|   /qr: | ||||
|     $ref: "./paths/getPizzaQr.yml" | ||||
|   /data: | ||||
|     $ref: "./paths/getData.yml" | ||||
| 
 | ||||
|   # Restaurace a jídla (/api/food) | ||||
|   /food/addChoice: | ||||
|     $ref: "./paths/food/addChoice.yml" | ||||
|   /food/removeChoice: | ||||
|     $ref: "./paths/food/removeChoice.yml" | ||||
|   /food/updateNote: | ||||
|     $ref: "./paths/food/updateNote.yml" | ||||
|   /food/removeChoices: | ||||
|     $ref: "./paths/food/removeChoices.yml" | ||||
|   /food/changeDepartureTime: | ||||
|     $ref: "./paths/food/changeDepartureTime.yml" | ||||
|   /food/jdemeObed: | ||||
|     $ref: "./paths/food/jdemeObed.yml" | ||||
| 
 | ||||
|   # Pizza day (/api/pizzaDay) | ||||
|   /pizzaDay/create: | ||||
|     $ref: "./paths/pizzaDay/create.yml" | ||||
|   /pizzaDay/delete: | ||||
|     $ref: "./paths/pizzaDay/delete.yml" | ||||
|   /pizzaDay/lock: | ||||
|     $ref: "./paths/pizzaDay/lock.yml" | ||||
|   /pizzaDay/unlock: | ||||
|     $ref: "./paths/pizzaDay/unlock.yml" | ||||
|   /pizzaDay/finishOrder: | ||||
|     $ref: "./paths/pizzaDay/finishOrder.yml" | ||||
|   /pizzaDay/finishDelivery: | ||||
|     $ref: "./paths/pizzaDay/finishDelivery.yml" | ||||
|   /pizzaDay/add: | ||||
|     $ref: "./paths/pizzaDay/addPizza.yml" | ||||
|   /pizzaDay/remove: | ||||
|     $ref: "./paths/pizzaDay/removePizza.yml" | ||||
|   /pizzaDay/updatePizzaDayNote: | ||||
|     $ref: "./paths/pizzaDay/updatePizzaDayNote.yml" | ||||
|   /pizzaDay/updatePizzaFee: | ||||
|     $ref: "./paths/pizzaDay/updatePizzaFee.yml" | ||||
| 
 | ||||
|   # Easter eggy (/api/easterEggs) | ||||
|   /easterEggs: | ||||
|     $ref: "./paths/easterEggs/easterEggs.yml" | ||||
|   /easterEggs/{url}: | ||||
|     $ref: "./paths/easterEggs/easterEgg.yml" | ||||
| 
 | ||||
|   # Statistiky (/api/stats) | ||||
|   /stats: | ||||
|     $ref: "./paths/stats/stats.yml" | ||||
| 
 | ||||
|   # Hlasování (/api/voting) | ||||
|   /voting/getVotes: | ||||
|     $ref: "./paths/voting/getVotes.yml" | ||||
|   /voting/updateVote: | ||||
|     $ref: "./paths/voting/updateVote.yml" | ||||
| 
 | ||||
| components: | ||||
|   schemas: | ||||
|     $ref: "./schemas/_index.yml" | ||||
|   responses: | ||||
|     ClientDataResponse: | ||||
|       description: Aktuální data pro klienta | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             $ref: "./schemas/_index.yml#/ClientData" | ||||
|   securitySchemes: | ||||
|     bearerAuth: | ||||
|       type: http | ||||
|       scheme: bearer | ||||
|       bearerFormat: JWT | ||||
| security: | ||||
|   - bearerAuth: [] | ||||
| @ -1 +1,2 @@ | ||||
| export * from './gen'; | ||||
| export * from './Types'; | ||||
| export * from './RequestTypes'; | ||||
| @ -1,14 +0,0 @@ | ||||
| import { defaultPlugins } from '@hey-api/openapi-ts'; | ||||
| 
 | ||||
| export default { | ||||
|     input: 'api.yml', | ||||
|     output: 'gen', | ||||
|     plugins: [ | ||||
|         ...defaultPlugins, | ||||
|         '@hey-api/client-fetch', | ||||
|         { | ||||
|             enums: 'javascript', | ||||
|             name: '@hey-api/typescript', | ||||
|         }, | ||||
|     ], | ||||
| }; | ||||
| @ -1,11 +0,0 @@ | ||||
| { | ||||
|   "name": "@luncher/types", | ||||
|   "version": "1.0.0", | ||||
|   "license": "MIT", | ||||
|   "private": true, | ||||
|   "devDependencies": { | ||||
|     "@hey-api/client-fetch": "^0.8.2", | ||||
|     "@hey-api/openapi-ts": "^0.64.7", | ||||
|     "typescript": "^5.9.3" | ||||
|   } | ||||
| } | ||||
| @ -1,18 +0,0 @@ | ||||
| get: | ||||
|   operationId: getEasterEggImage | ||||
|   summary: Vrátí obrázek konkrétního easter eggu | ||||
|   parameters: | ||||
|     - in: path | ||||
|       name: url | ||||
|       required: true | ||||
|       schema: | ||||
|         type: string | ||||
|       description: URL easter eggu | ||||
|   responses: | ||||
|     "200": | ||||
|       content: | ||||
|         image/png: | ||||
|           description: Obrázek easter eggu | ||||
|           schema: | ||||
|             type: string | ||||
|             format: binary | ||||
| @ -1,9 +0,0 @@ | ||||
| get: | ||||
|   operationId: getEasterEgg | ||||
|   summary: Vrátí náhodně metadata jednoho z definovaných easter egg obrázků pro přihlášeného uživatele, nebo nic, pokud žádné definované nemá. | ||||
|   responses: | ||||
|     "200": | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             $ref: "../../schemas/_index.yml#/EasterEgg" | ||||
| @ -1,20 +0,0 @@ | ||||
| post: | ||||
|   operationId: addChoice | ||||
|   summary: Přidání či nahrazení volby uživatele pro zvolený den/podnik | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - locationKey | ||||
|           properties: | ||||
|             locationKey: | ||||
|               $ref: "../../schemas/_index.yml#/LunchChoice" | ||||
|             dayIndex: | ||||
|               $ref: "../../schemas/_index.yml#/DayIndex" | ||||
|             foodIndex: | ||||
|               $ref: "../../schemas/_index.yml#/FoodIndex" | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,16 +0,0 @@ | ||||
| post: | ||||
|   operationId: changeDepartureTime | ||||
|   summary: Úprava preferovaného času odchodu do aktuálně zvoleného podniku. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           properties: | ||||
|             dayIndex: | ||||
|               $ref: "../../schemas/_index.yml#/DayIndex" | ||||
|             time: | ||||
|               $ref: "../../schemas/_index.yml#/DepartureTime" | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: jdemeObed | ||||
|   summary: Odeslání notifikací "jdeme na oběd" dle konfigurace. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Notifikace byly odeslány. | ||||
| @ -1,21 +0,0 @@ | ||||
| post: | ||||
|   operationId: removeChoice | ||||
|   summary: Odstranění jednoho zvoleného jídla uživatele pro zvolený den/podnik | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - foodIndex | ||||
|             - locationKey | ||||
|           properties: | ||||
|             foodIndex: | ||||
|               $ref: "../../schemas/_index.yml#/FoodIndex" | ||||
|             locationKey: | ||||
|               $ref: "../../schemas/_index.yml#/LunchChoice" | ||||
|             dayIndex: | ||||
|               $ref: "../../schemas/_index.yml#/DayIndex" | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,18 +0,0 @@ | ||||
| post: | ||||
|   operationId: removeChoices | ||||
|   summary: Odstranění volby uživatele pro zvolený den/podnik, včetně případných jídel | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - locationKey | ||||
|           properties: | ||||
|             locationKey: | ||||
|               $ref: "../../schemas/_index.yml#/LunchChoice" | ||||
|             dayIndex: | ||||
|               $ref: "../../schemas/_index.yml#/DayIndex" | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,16 +0,0 @@ | ||||
| post: | ||||
|   operationId: updateNote | ||||
|   summary: Nastavení poznámky k volbě uživatele | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           properties: | ||||
|             dayIndex: | ||||
|               $ref: "../../schemas/_index.yml#/DayIndex" | ||||
|             note: | ||||
|               type: string | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,14 +0,0 @@ | ||||
| get: | ||||
|   operationId: getData | ||||
|   summary: Načtení klientských dat pro aktuální nebo předaný den | ||||
|   parameters: | ||||
|     - in: query | ||||
|       name: dayIndex | ||||
|       description: Index dne v týdnu. Pokud není předán, je použit aktuální den. | ||||
|       schema: | ||||
|         type: integer | ||||
|         minimum: 0 | ||||
|         maximum: 4 | ||||
|   responses: | ||||
|     "200": | ||||
|       $ref: "../api.yml#/components/responses/ClientDataResponse" | ||||
| @ -1,19 +0,0 @@ | ||||
| get: | ||||
|   operationId: getPizzaQr | ||||
|   summary: Získání QR kódu pro platbu za Pizza day | ||||
|   security: [] # Nevyžaduje autentizaci | ||||
|   parameters: | ||||
|     - in: query | ||||
|       name: login | ||||
|       schema: | ||||
|         type: string | ||||
|       required: true | ||||
|       description: Přihlašovací jméno uživatele, pro kterého bude vrácen QR kód | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Vygenerovaný QR kód pro platbu | ||||
|       content: | ||||
|         image/png: | ||||
|           schema: | ||||
|             type: string | ||||
|             format: binary | ||||
| @ -1,20 +0,0 @@ | ||||
| post: | ||||
|   operationId: login | ||||
|   summary: Přihlášení uživatele | ||||
|   security: [] # Nevyžaduje autentizaci | ||||
|   requestBody: | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           type: object | ||||
|           properties: | ||||
|             login: | ||||
|               type: string | ||||
|               description: Přihlašovací jméno uživatele. Vyžadováno pouze pokud není předáno pomocí hlaviček. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Přihlášení bylo úspěšné | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             $ref: "../schemas/_index.yml#/JWTToken" | ||||
| @ -1,21 +0,0 @@ | ||||
| post: | ||||
|   operationId: addPizza | ||||
|   summary: Přidání pizzy do objednávky. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - pizzaIndex | ||||
|             - pizzaSizeIndex | ||||
|           properties: | ||||
|             pizzaIndex: | ||||
|               description: Index pizzy v nabídce | ||||
|               type: integer | ||||
|             pizzaSizeIndex: | ||||
|               description: Index velikosti pizzy v nabídce variant | ||||
|               type: integer | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Přidání pizzy do objednávky proběhlo úspěšně. | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: createPizzaDay | ||||
|   summary: Založení pizza day. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl založen. | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: deletePizzaDay | ||||
|   summary: Smazání pizza day. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl smazán. | ||||
| @ -1,18 +0,0 @@ | ||||
| post: | ||||
|   operationId: finishDelivery | ||||
|   summary: Převod pizza day do stavu "Pizzy byly doručeny". Pokud má objednávající nastaveno číslo účtu, je ostatním uživatelům vygenerován a následně zobrazen QR kód pro úhradu jejich objednávky. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           properties: | ||||
|             bankAccount: | ||||
|               description: Číslo bankovního účtu objednávajícího | ||||
|               type: string | ||||
|             bankAccountHolder: | ||||
|               description: Jméno majitele bankovního účtu | ||||
|               type: string | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl přepnut do stavu "Pizzy doručeny". | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: finishOrder | ||||
|   summary: Přepnutí pizza day do stavu "Pizzy objednány". Není možné měnit objednávky, příslušným uživatelům je odeslána notifikace o provedené objednávce. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl přepnut do stavu "Pizzy objednány". | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: lockPizzaDay | ||||
|   summary: Uzamkne pizza day. Nebude možné přidávat či odebírat pizzy. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl uzamčen. | ||||
| @ -1,16 +0,0 @@ | ||||
| post: | ||||
|   operationId: removePizza | ||||
|   summary: Odstranění pizzy z objednávky. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - pizzaOrder | ||||
|           properties: | ||||
|             pizzaOrder: | ||||
|               $ref: "../../schemas/_index.yml#/PizzaVariant" | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Odstranění pizzy z objednávky proběhlo úspěšně. | ||||
| @ -1,6 +0,0 @@ | ||||
| post: | ||||
|   operationId: unlockPizzaDay | ||||
|   summary: Odemkne pizza day. Bude opět možné přidávat či odebírat pizzy. | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Pizza day byl odemčen. | ||||
| @ -1,15 +0,0 @@ | ||||
| post: | ||||
|   operationId: updatePizzaDayNote | ||||
|   summary: Nastavení poznámky k objednávkám pizz přihlášeného uživatele. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           properties: | ||||
|             note: | ||||
|               type: string | ||||
|               description: Poznámka k objednávkám pizz, např "bez oliv". | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Nastavení poznámky k objednávkám pizz proběhlo úspěšně. | ||||
| @ -1,23 +0,0 @@ | ||||
| post: | ||||
|   operationId: updatePizzaFee | ||||
|   summary: Nastavení přirážky/slevy k objednávce pizz uživatele. | ||||
|   requestBody: | ||||
|     required: true | ||||
|     content: | ||||
|       application/json: | ||||
|         schema: | ||||
|           required: | ||||
|             - login | ||||
|           properties: | ||||
|             login: | ||||
|               type: string | ||||
|               description: Login cíleného uživatele | ||||
|             text: | ||||
|               type: string | ||||
|               description: Textový popis přirážky/slevy | ||||
|             price: | ||||
|               type: number | ||||
|               description: Částka přirážky/slevy v Kč | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Nastavení přirážky/slevy proběhlo úspěšně. | ||||
| @ -1,23 +0,0 @@ | ||||
| get: | ||||
|   operationId: getStats | ||||
|   summary: Vrátí statistiky způsobu stravování pro předaný rozsah dat. | ||||
|   parameters: | ||||
|     - in: query | ||||
|       name: startDate | ||||
|       required: true | ||||
|       schema: | ||||
|         type: string | ||||
|       description: Počáteční datum pro načtení statistik | ||||
|     - in: query | ||||
|       name: endDate | ||||
|       required: true | ||||
|       schema: | ||||
|         type: string | ||||
|       description: Koncové datum pro načtení statistik | ||||
|   responses: | ||||
|     "200": | ||||
|       description: Statistiky způsobu stravování. Každý prvek v poli představuje statistiky pro jeden den z předaného rozsahu dat. | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             $ref: "../../schemas/_index.yml#/WeeklyStats" | ||||
| @ -1,11 +0,0 @@ | ||||
| get: | ||||
|   operationId: getVotes | ||||
|   summary: Vrátí statistiky hlasování o nových funkcích. | ||||
|   responses: | ||||
|     "200": | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             type: array | ||||
|             items: | ||||
|               $ref: "../../schemas/_index.yml#/FeatureRequest" | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user