Compare commits
	
		
			1 Commits
		
	
	
		
			master
			...
			feat/enumF
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| bc039a361d | 
							
								
								
									
										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 | 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 | # Builder | ||||||
| FROM ${NODE_VERSION} AS builder | FROM node:18-alpine3.18 AS builder | ||||||
| 
 | 
 | ||||||
| WORKDIR /build | 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 | # Zkopírování závislostí - server | ||||||
| COPY server/package.json ./server/ | COPY server/package.json ./server/ | ||||||
| COPY server/yarn.lock ./server/ | COPY server/yarn.lock ./server/ | ||||||
| @ -21,10 +11,6 @@ COPY server/yarn.lock ./server/ | |||||||
| COPY client/package.json ./client/ | COPY client/package.json ./client/ | ||||||
| COPY client/yarn.lock ./client/ | COPY client/yarn.lock ./client/ | ||||||
| 
 | 
 | ||||||
| # Instalace závislostí - OpenAPI generátor |  | ||||||
| WORKDIR /build/types |  | ||||||
| RUN yarn install --frozen-lockfile |  | ||||||
| 
 |  | ||||||
| # Instalace závislostí - server | # Instalace závislostí - server | ||||||
| WORKDIR /build/server | WORKDIR /build/server | ||||||
| RUN yarn install --frozen-lockfile | RUN yarn install --frozen-lockfile | ||||||
| @ -48,11 +34,7 @@ COPY client/src ./client/src | |||||||
| COPY client/public ./client/public | COPY client/public ./client/public | ||||||
| 
 | 
 | ||||||
| # Zkopírování společných typů | # Zkopírování společných typů | ||||||
| COPY types/index.ts ./types/ | COPY types ./types/ | ||||||
| 
 |  | ||||||
| # Vygenerování společných typů z OpenAPI |  | ||||||
| WORKDIR /build/types |  | ||||||
| RUN yarn openapi-ts |  | ||||||
| 
 | 
 | ||||||
| # Sestavení serveru | # Sestavení serveru | ||||||
| WORKDIR /build/server | WORKDIR /build/server | ||||||
| @ -63,30 +45,26 @@ WORKDIR /build/client | |||||||
| RUN yarn build | RUN yarn build | ||||||
| 
 | 
 | ||||||
| # Runner | # Runner | ||||||
| FROM ${NODE_VERSION} | FROM node:18-alpine3.18 | ||||||
| 
 | ENV LANG cs_CZ.UTF-8 | ||||||
| RUN apk add --no-cache tzdata | ENV NODE_ENV production | ||||||
| ENV TZ=Europe/Prague \ |  | ||||||
|     LC_ALL=cs_CZ.UTF-8 \ |  | ||||||
|     NODE_ENV=production |  | ||||||
| 
 | 
 | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| 
 | 
 | ||||||
| # Vykopírování sestaveného serveru | # Vykopírování sestaveného serveru | ||||||
| COPY --from=builder /build/server/node_modules ./server/node_modules | COPY --from=builder /build/server/node_modules ./server/node_modules | ||||||
| COPY --from=builder /build/server/dist ./ | COPY --from=builder /build/server/dist ./ | ||||||
|  | COPY server/resources ./server/resources | ||||||
| 
 | 
 | ||||||
| # Vykopírování sestaveného klienta | # Vykopírování sestaveného klienta | ||||||
| COPY --from=builder /build/client/dist ./public | COPY --from=builder /build/client/dist ./public | ||||||
| 
 | 
 | ||||||
| # Zkopírování produkčních .env serveru | # 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ů | # Zkopírování konfigurace easter eggů | ||||||
| RUN if [ -f /server/.easter-eggs.json ]; then cp /server/.easter-eggs.json ./server/; fi | # TODO tohle spadne když nebude existovat! | ||||||
| 
 | COPY /server/.easter-eggs.json ./server/ | ||||||
| # Export /data/db.json do složky /data |  | ||||||
| VOLUME ["/data"] |  | ||||||
| 
 | 
 | ||||||
| EXPOSE 3000 | 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 | # Luncher | ||||||
| Aplikace pro profesionální management obědů. | Aplikace pro profesionální management obědů. | ||||||
| 
 | 
 | ||||||
| Aplikace sestává ze tří modulů. | Aplikace sestává ze dvou modulů + společných TypeScript definic (adresář `types`). | ||||||
| - types |  | ||||||
|   - OpenAPI definice společných typů, generované přes [openapi-ts](https://github.com/hey-api/openapi-ts) |  | ||||||
| - server | - server | ||||||
|   - backend psaný v [node.js](https://nodejs.dev) |   - backend psaný v [node.js](https://nodejs.dev) | ||||||
| - client | - client | ||||||
| @ -12,27 +10,19 @@ Aplikace sestává ze tří modulů. | |||||||
| ## Spuštění pro vývoj | ## Spuštění pro vývoj | ||||||
| ### Závislosti | ### Závislosti | ||||||
| #### Klient/server | #### 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) | - [Yarn 1.22.x (Classic)](https://classic.yarnpkg.com) | ||||||
| 
 | 
 | ||||||
| ### Spuštění na *nix platformách | ### Spuštění na *nix platformách | ||||||
| - Nainstalovat závislosti viz předchozí bod | - Nainstalovat závislosti viz předchozí bod | ||||||
| - Zkopírovat `server/.env.template` do `server/.env.development` a upravit dle potřeby | - Zkopírovat `server/.env.template` do `server/.env.development` a upravit dle potřeby | ||||||
| - Vygenerovat společné TypeScript typy | - Spustit `./run_dev.sh`. Na jiných platformách se lze inspirovat jeho obsahem, postup by měl být víceméně stejný. | ||||||
|   - `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` |  | ||||||
| 
 | 
 | ||||||
| ## Sestavení a spuštění produkční verze v Docker | ## Sestavení a spuštění produkční verze v Docker | ||||||
| ### Závislosti | ### Závislosti | ||||||
| - [Docker](https://www.docker.com) | - [Docker](https://www.docker.com) | ||||||
| - [Docker Compose](https://docs.docker.com/compose) | - [Docker Compose](https://docs.docker.com/compose) | ||||||
| 
 | 
 | ||||||
| ### Spuštění |  | ||||||
| - `docker compose up --build -d` |  | ||||||
| 
 |  | ||||||
| ### Spuštení s traefik | ### Spuštení s traefik | ||||||
| - `docker compose -f compose-traefik.yml up --build -d` | - `docker compose -f compose-traefik.yml up --build -d` | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								client/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								client/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +1,3 @@ | |||||||
| build | build | ||||||
| dist | dist | ||||||
|  | src/types | ||||||
| @ -6,36 +6,34 @@ | |||||||
|   "type": "module", |   "type": "module", | ||||||
|   "homepage": ".", |   "homepage": ".", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@fortawesome/fontawesome-svg-core": "^7.1.0", |     "@fortawesome/fontawesome-svg-core": "^6.4.0", | ||||||
|     "@fortawesome/free-regular-svg-icons": "^7.1.0", |     "@fortawesome/free-regular-svg-icons": "^6.4.0", | ||||||
|     "@fortawesome/free-solid-svg-icons": "^7.1.0", |     "@fortawesome/free-solid-svg-icons": "^6.4.0", | ||||||
|     "@fortawesome/react-fontawesome": "^3.1.0", |     "@fortawesome/react-fontawesome": "^0.2.0", | ||||||
|     "@types/jest": "^30.0.0", |     "@types/jest": "^29.5.12", | ||||||
|     "@types/node": "^24.7.1", |     "@types/node": "^20.11.20", | ||||||
|     "@types/react": "^19.2.2", |     "@types/react": "^19.0.0", | ||||||
|     "@types/react-dom": "^19.2.1", |     "@types/react-dom": "^19.0.0", | ||||||
|     "@vitejs/plugin-react": "^5.0.4", |     "@vitejs/plugin-react": "^4.3.4", | ||||||
|     "bootstrap": "^5.3.8", |     "bootstrap": "^5.2.3", | ||||||
|     "react": "^19.2.0", |     "react": "^19.0.0", | ||||||
|     "react-bootstrap": "^2.10.10", |     "react-bootstrap": "^2.7.2", | ||||||
|     "react-dom": "^19.2.0", |     "react-dom": "^19.0.0", | ||||||
|     "react-jwt": "^1.3.0", |     "react-jwt": "^1.2.0", | ||||||
|     "react-modal": "^3.16.3", |     "react-modal": "^3.16.1", | ||||||
|     "react-router": "^7.9.4", |  | ||||||
|     "react-router-dom": "^7.9.4", |  | ||||||
|     "react-select-search": "^4.1.6", |     "react-select-search": "^4.1.6", | ||||||
|     "react-snowfall": "^2.3.0", |     "react-snowfall": "^2.2.0", | ||||||
|     "react-toastify": "^11.0.5", |     "react-toastify": "^10.0.4", | ||||||
|     "recharts": "^3.2.1", |     "sass": "^1.80.6", | ||||||
|     "sass": "^1.93.2", |  | ||||||
|     "socket.io-client": "^4.6.1", |     "socket.io-client": "^4.6.1", | ||||||
|     "typescript": "^5.9.3", |     "typescript": "^5.3.3", | ||||||
|     "vite": "^7.1.9", |     "vite": "^6.0.3", | ||||||
|     "vite-tsconfig-paths": "^5.1.4" |     "vite-tsconfig-paths": "^5.1.4" | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "yarn vite", |     "copy-types": "cp -r ../types ./src", | ||||||
|     "build": "tsc --noEmit && yarn vite build" |     "start": "yarn copy-types && vite", | ||||||
|  |     "build": "yarn copy-types && vite build" | ||||||
|   }, |   }, | ||||||
|   "eslintConfig": { |   "eslintConfig": { | ||||||
|     "extends": [ |     "extends": [ | ||||||
| @ -56,6 +54,6 @@ | |||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@babel/plugin-proposal-private-property-in-object": "^7.21.11", |     "@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; |   font-size: 64px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* Sticky footer layout */ |  | ||||||
| html, |  | ||||||
| body, |  | ||||||
| #root { |  | ||||||
|   height: 100%; |  | ||||||
|   margin: 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .app-container { |  | ||||||
|   min-height: 100vh; |  | ||||||
|   display: flex; |  | ||||||
|   flex-direction: column; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .wrapper { | .wrapper { | ||||||
|   padding: 20px; |   padding: 20px; | ||||||
|   flex: 1; |  | ||||||
|   /* Zabere zbytek místa */ |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .title { | .title { | ||||||
|  | |||||||
| @ -1,27 +1,29 @@ | |||||||
| import React, { useContext, useEffect, useMemo, useRef, useState, useCallback } from 'react'; | import React, { useContext, useEffect, useMemo, useRef, useState, useCallback } from 'react'; | ||||||
| import 'bootstrap/dist/css/bootstrap.min.css'; | import 'bootstrap/dist/css/bootstrap.min.css'; | ||||||
| import { EVENT_DISCONNECT, EVENT_MESSAGE, SocketContext } from './context/socket'; | 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 { useAuth } from './context/auth'; | ||||||
| import Login from './Login'; | import Login from './Login'; | ||||||
| import { Alert, Button, Col, Form, Row, Table } from 'react-bootstrap'; | import { Alert, Button, Col, Form, Row, Table } from 'react-bootstrap'; | ||||||
| import Header from './components/Header'; | import Header from './components/Header'; | ||||||
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | ||||||
| import PizzaOrderList from './components/PizzaOrderList'; | 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 'react-select-search/style.css'; | ||||||
| import './App.scss'; | import './App.scss'; | ||||||
|  | import { SelectSearchOption } from 'react-select-search'; | ||||||
| import { faCircleCheck, faNoteSticky, faTrashCan } from '@fortawesome/free-regular-svg-icons'; | import { faCircleCheck, faNoteSticky, faTrashCan } from '@fortawesome/free-regular-svg-icons'; | ||||||
| import { useSettings } from './context/settings'; | import { useSettings } from './context/settings'; | ||||||
|  | import { ClientData, Restaurants, Food, Order, Locations, PizzaOrder, PizzaDayState, FoodChoices, DayMenu, DepartureTime } from './types'; | ||||||
| import Footer from './components/Footer'; | import Footer from './components/Footer'; | ||||||
| import { faChainBroken, faChevronLeft, faChevronRight, faGear, faSatelliteDish, faSearch } from '@fortawesome/free-solid-svg-icons'; | import { faChainBroken, faChevronLeft, faChevronRight, faGear, faSatelliteDish, faSearch } from '@fortawesome/free-solid-svg-icons'; | ||||||
| import Loader from './components/Loader'; | import Loader from './components/Loader'; | ||||||
| import { getHumanDateTime, isInTheFuture } from './Utils'; | import { getData, errorHandler, getQrUrl } from './api/Api'; | ||||||
|  | import { addChoice, removeChoices, removeChoice, changeDepartureTime, jdemeObed, updateNote } from './api/FoodApi'; | ||||||
|  | import { getHumanDateTime } from './Utils'; | ||||||
| import NoteModal from './components/modals/NoteModal'; | import NoteModal from './components/modals/NoteModal'; | ||||||
| import { useEasterEgg } from './context/eggs'; | 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 { getImage } from './api/EasterEggApi'; | ||||||
| import { getLunchChoiceName } from './enums'; |  | ||||||
| import FallingLeaves, { LEAF_PRESETS, LEAF_COLOR_THEMES } from './FallingLeaves'; |  | ||||||
| import './FallingLeaves.scss'; |  | ||||||
| 
 | 
 | ||||||
| const EVENT_CONNECT = "connect" | const EVENT_CONNECT = "connect" | ||||||
| 
 | 
 | ||||||
| @ -32,37 +34,17 @@ const EASTER_EGG_STYLE = { | |||||||
|   animationTimingFunction: "ease" |   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
 | // 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; | const EASTER_EGG_DEFAULT_DURATION = 0.75; | ||||||
| 
 | 
 | ||||||
| function App() { | function App() { | ||||||
|   const auth = useAuth(); |   const auth = useAuth(); | ||||||
|   const settings = useSettings(); |   const settings = useSettings(); | ||||||
|   const [easterEgg, _] = useEasterEgg(auth); |   const [easterEgg, easterEggLoading] = useEasterEgg(auth); | ||||||
|   const [isConnected, setIsConnected] = useState<boolean>(false); |   const [isConnected, setIsConnected] = useState<boolean>(false); | ||||||
|   const [data, setData] = useState<ClientData>(); |   const [data, setData] = useState<ClientData>(); | ||||||
|   const [food, setFood] = useState<RestaurantDayMenuMap>(); |   const [food, setFood] = useState<{ [key in Restaurants]?: DayMenu }>(); | ||||||
|   const [myOrder, setMyOrder] = useState<PizzaOrder>(); |   const [myOrder, setMyOrder] = useState<Order>(); | ||||||
|   const [foodChoiceList, setFoodChoiceList] = useState<Food[]>(); |   const [foodChoiceList, setFoodChoiceList] = useState<Food[]>(); | ||||||
|   const [closed, setClosed] = useState<boolean>(false); |   const [closed, setClosed] = useState<boolean>(false); | ||||||
|   const socket = useContext(SocketContext); |   const socket = useContext(SocketContext); | ||||||
| @ -82,17 +64,14 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   // Načtení dat po přihlášení
 |   // Načtení dat po přihlášení
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (!auth?.login) { |     if (!auth || !auth.login) { | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     getData().then(response => { |     getData().then((data: ClientData) => { | ||||||
|       const data = response.data |       setData(data); | ||||||
|       if (data) { |       setDayIndex(data.weekIndex); | ||||||
|         setData(data); |       dayIndexRef.current = data.weekIndex; | ||||||
|         setDayIndex(data.dayIndex); |       setFood(data.menus); | ||||||
|         dayIndexRef.current = data.dayIndex; |  | ||||||
|         setFood(data.menus); |  | ||||||
|       } |  | ||||||
|     }).catch(e => { |     }).catch(e => { | ||||||
|       setFailure(true); |       setFailure(true); | ||||||
|     }) |     }) | ||||||
| @ -100,15 +79,12 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   // Přenačtení pro zvolený den
 |   // Přenačtení pro zvolený den
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (!auth?.login) { |     if (!auth || !auth.login) { | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     getData({ query: { dayIndex: dayIndex } }).then(response => { |     getData(dayIndex).then((data: ClientData) => { | ||||||
|       const data = response.data; |  | ||||||
|       setData(data); |       setData(data); | ||||||
|       if (data) { |       setFood(data.menus); | ||||||
|         setFood(data.menus); |  | ||||||
|       } |  | ||||||
|     }).catch(e => { |     }).catch(e => { | ||||||
|       setFailure(true); |       setFailure(true); | ||||||
|     }) |     }) | ||||||
| @ -117,15 +93,17 @@ function App() { | |||||||
|   // Registrace socket eventů
 |   // Registrace socket eventů
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     socket.on(EVENT_CONNECT, () => { |     socket.on(EVENT_CONNECT, () => { | ||||||
|  |       // console.log("Connected!");
 | ||||||
|       setIsConnected(true); |       setIsConnected(true); | ||||||
|     }); |     }); | ||||||
|     socket.on(EVENT_DISCONNECT, () => { |     socket.on(EVENT_DISCONNECT, () => { | ||||||
|  |       // console.log("Disconnected!");
 | ||||||
|       setIsConnected(false); |       setIsConnected(false); | ||||||
|     }); |     }); | ||||||
|     socket.on(EVENT_MESSAGE, (newData: ClientData) => { |     socket.on(EVENT_MESSAGE, (newData: ClientData) => { | ||||||
|       // console.log("Přijata nová data ze socketu", newData);
 |       // console.log("Přijata nová data ze socketu", newData);
 | ||||||
|       // Aktualizujeme pouze, pokud jsme dostali data pro den, který máme aktuálně zobrazený
 |       // 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); |         setData(newData); | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
| @ -138,7 +116,7 @@ function App() { | |||||||
|   }, [socket]); |   }, [socket]); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (!auth?.login) { |     if (!auth || !auth.login) { | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     // TODO tohle občas náhodně nezafunguje, nutno přepsat, viz https://medium.com/@teh_builder/ref-objects-inside-useeffect-hooks-eb7c15198780
 |     // 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,13 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (choiceRef?.current?.value && choiceRef.current.value !== "") { |     if (choiceRef?.current?.value && choiceRef.current.value !== "") { | ||||||
|       const locationKey = choiceRef.current.value as LunchChoice; |       // TODO: wtf, cos pil, když jsi tohle psal?
 | ||||||
|       const restaurantKey = Object.keys(Restaurant).indexOf(locationKey); |       const key = choiceRef?.current?.value; | ||||||
|  |       const locationIndex = Object.keys(Locations).indexOf(key as unknown as Locations); | ||||||
|  |       const locationsKey = Object.keys(Locations)[locationIndex]; | ||||||
|  |       const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||||
|       if (restaurantKey > -1 && food) { |       if (restaurantKey > -1 && food) { | ||||||
|         const restaurant = Object.keys(Restaurant)[restaurantKey] as Restaurant; |         const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||||
|         setFoodChoiceList(food[restaurant]?.food); |         setFoodChoiceList(food[restaurant]?.food); | ||||||
|         setClosed(food[restaurant]?.closed ?? false); |         setClosed(food[restaurant]?.closed ?? false); | ||||||
|       } else { |       } else { | ||||||
| @ -198,9 +179,9 @@ function App() { | |||||||
|   // Stažení a nastavení easter egg obrázku
 |   // Stažení a nastavení easter egg obrázku
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (auth?.login && easterEgg?.url && !eggImage) { |     if (auth?.login && easterEgg?.url && !eggImage) { | ||||||
|       getEasterEggImage({ path: { url: easterEgg.url } }).then(response => { |       getImage(easterEgg.url).then(data => { | ||||||
|         if (response.data) { |         if (data) { | ||||||
|           setEggImage(response.data); |           setEggImage(data); | ||||||
|           // Smazání obrázku z DOMu po animaci
 |           // Smazání obrázku z DOMu po animaci
 | ||||||
|           setTimeout(() => { |           setTimeout(() => { | ||||||
|             if (eggRef?.current) { |             if (eggRef?.current) { | ||||||
| @ -212,18 +193,10 @@ function App() { | |||||||
|     } |     } | ||||||
|   }, [auth?.login, easterEgg?.duration, easterEgg?.url, eggImage]); |   }, [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 doAddChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||||
|     const locationKey = event.target.value as LunchChoice; |     const locationKey = event.target.value as Locations; | ||||||
|     if (auth?.login) { |     if (auth?.login) { | ||||||
|       await addChoice({ body: { locationKey, dayIndex } }); |       await errorHandler(() => addChoice(locationKey, undefined, dayIndex)); | ||||||
|       if (foodChoiceRef.current?.value) { |       if (foodChoiceRef.current?.value) { | ||||||
|         foodChoiceRef.current.value = ""; |         foodChoiceRef.current.value = ""; | ||||||
|       } |       } | ||||||
| @ -238,16 +211,16 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   const doAddFoodChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { |   const doAddFoodChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||||
|     if (event.target.value && foodChoiceList?.length && choiceRef.current?.value) { |     if (event.target.value && foodChoiceList?.length && choiceRef.current?.value) { | ||||||
|       const locationKey = choiceRef.current.value as LunchChoice; |       const locationKey = choiceRef.current.value as Locations; | ||||||
|       if (auth?.login) { |       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: Locations) => { | ||||||
|     if (auth?.login) { |     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
 |       // Vyresetujeme výběr, aby bylo jasné pro který případně vybíráme jídlo
 | ||||||
|       if (choiceRef?.current?.value) { |       if (choiceRef?.current?.value) { | ||||||
|         choiceRef.current.value = ""; |         choiceRef.current.value = ""; | ||||||
| @ -258,9 +231,9 @@ function App() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const doRemoveFoodChoice = async (locationKey: LunchChoice, foodIndex: number) => { |   const doRemoveFoodChoice = async (locationKey: Locations, foodIndex: number) => { | ||||||
|     if (auth?.login) { |     if (auth?.login) { | ||||||
|       await removeChoice({ body: { locationKey, foodIndex, dayIndex } }); |       await errorHandler(() => removeChoice(locationKey, foodIndex, dayIndex)); | ||||||
|       if (choiceRef?.current?.value) { |       if (choiceRef?.current?.value) { | ||||||
|         choiceRef.current.value = ""; |         choiceRef.current.value = ""; | ||||||
|       } |       } | ||||||
| @ -272,7 +245,7 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   const saveNote = async (note?: string) => { |   const saveNote = async (note?: string) => { | ||||||
|     if (auth?.login) { |     if (auth?.login) { | ||||||
|       await updateNote({ body: { note, dayIndex } }); |       await errorHandler(() => updateNote(note, dayIndex)); | ||||||
|       setNoteModalOpen(false); |       setNoteModalOpen(false); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @ -296,18 +269,18 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   const handlePizzaChange = async (value: SelectedOptionValue | SelectedOptionValue[]) => { |   const handlePizzaChange = async (value: SelectedOptionValue | SelectedOptionValue[]) => { | ||||||
|     if (auth?.login && data?.pizzaList) { |     if (auth?.login && data?.pizzaList) { | ||||||
|       if (typeof value !== 'string') { |       if (!(typeof value === 'string')) { | ||||||
|         throw Error('Nepodporovaný typ hodnoty'); |         throw Error('Nepodporovaný typ hodnoty'); | ||||||
|       } |       } | ||||||
|       const s = value.split('|'); |       const s = value.split('|'); | ||||||
|       const pizzaIndex = Number.parseInt(s[0]); |       const pizzaIndex = Number.parseInt(s[0]); | ||||||
|       const pizzaSizeIndex = Number.parseInt(s[1]); |       const pizzaSizeIndex = Number.parseInt(s[1]); | ||||||
|       await addPizza({ body: { pizzaIndex, pizzaSizeIndex } }); |       await addPizza(pizzaIndex, pizzaSizeIndex); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const handlePizzaDelete = async (pizzaOrder: PizzaVariant) => { |   const handlePizzaDelete = async (pizzaOrder: PizzaOrder) => { | ||||||
|     await removePizza({ body: { pizzaOrder } }); |     await removePizza(pizzaOrder); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const handlePizzaPoznamkaChange = async () => { |   const handlePizzaPoznamkaChange = async () => { | ||||||
| @ -315,7 +288,7 @@ function App() { | |||||||
|       alert("Poznámka může mít maximálně 70 znaků"); |       alert("Poznámka může mít maximálně 70 znaků"); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|     updatePizzaDayNote({ body: { note: pizzaPoznamkaRef.current?.value } }); |     updatePizzaDayNote(pizzaPoznamkaRef.current?.value); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // const addToCart = async () => {
 |   // const addToCart = async () => {
 | ||||||
| @ -344,7 +317,7 @@ function App() { | |||||||
| 
 | 
 | ||||||
|   const handleChangeDepartureTime = async (event: React.ChangeEvent<HTMLSelectElement>) => { |   const handleChangeDepartureTime = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||||
|     if (foodChoiceList?.length && choiceRef.current?.value) { |     if (foodChoiceList?.length && choiceRef.current?.value) { | ||||||
|       await changeDepartureTime({ body: { time: event.target.value as DepartureTime, dayIndex } }); |       await changeDepartureTime(event.target.value, dayIndex); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -362,32 +335,17 @@ function App() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const renderFoodTable = (location: Restaurant, menu: RestaurantDayMenu) => { |   const renderFoodTable = (name: string, menu: DayMenu) => { | ||||||
|     let content; |     let content; | ||||||
|     if (menu?.closed) { |     if (menu?.closed) { | ||||||
|       content = <h3>Zavřeno</h3> |       content = <h3>Zavřeno</h3> | ||||||
|     } else if (menu?.food?.length && menu.food.length > 0) { |     } else if (menu?.food?.length > 0) { | ||||||
|       const hideSoups = settings?.hideSoups; |  | ||||||
|       content = <Table striped bordered hover> |       content = <Table striped bordered hover> | ||||||
|         <tbody style={{ cursor: 'pointer' }}> |         <tbody> | ||||||
|           {menu.food.map((f: Food, index: number) => |           {menu.food.filter(f => (settings?.hideSoups ? !f.isSoup : true)).map((f: any, index: number) => | ||||||
|             (!hideSoups || !f.isSoup) && |             <tr key={index}> | ||||||
|             <tr key={f.name} onClick={() => doAddClickFoodChoice(location, index)}> |  | ||||||
|               <td>{f.amount}</td> |               <td>{f.amount}</td> | ||||||
|               <td> |               <td>{f.name}</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.price}</td> |               <td>{f.price}</td> | ||||||
|             </tr> |             </tr> | ||||||
|           )} |           )} | ||||||
| @ -396,14 +354,14 @@ function App() { | |||||||
|     } else { |     } else { | ||||||
|       content = <h3>Chyba načtení dat</h3> |       content = <h3>Chyba načtení dat</h3> | ||||||
|     } |     } | ||||||
|     return <Col md={12} lg={3} className='mt-3'> |     return <Col md={12} lg={6} className='mt-3'> | ||||||
|       <h3 style={{ cursor: 'pointer' }} onClick={() => doAddClickFoodChoice(location)}>{getLunchChoiceName(location)}</h3> |       <h3>{name}</h3> | ||||||
|       {menu?.lastUpdate && <small>Poslední aktualizace: {getHumanDateTime(new Date(menu.lastUpdate))}</small>} |       {menu?.lastUpdate && <small>Poslední aktualizace: {getHumanDateTime(new Date(menu.lastUpdate))}</small>} | ||||||
|       {content} |       {content} | ||||||
|     </Col> |     </Col> | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (!auth?.login) { |   if (!auth || !auth.login) { | ||||||
|     return <Login />; |     return <Login />; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -432,72 +390,68 @@ function App() { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const noOrders = data?.pizzaDay?.orders?.length === 0; |   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 || {}; |   const { path, url, startOffset, endOffset, duration, ...style } = easterEgg || {}; | ||||||
| 
 | 
 | ||||||
|   return ( |   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` }} />} |       {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 /> |       <Header /> | ||||||
|       <div className='wrapper'> |       <div className='wrapper'> | ||||||
|         {data.isWeekend ? <h4>Užívejte víkend :)</h4> : <> |         {data.isWeekend ? <h4>Užívejte víkend :)</h4> : <> | ||||||
|           <Alert variant={'primary'}> |           <Alert variant={'primary'}> | ||||||
|             {/* <img alt="" src='hat.png' style={{ position: "absolute", width: "70px", rotate: "-45deg", left: -40, top: -58 }} /> |             <img 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='snowman.png' style={{ position: "absolute", height: "110px", right: 10, top: 5 }} /> | ||||||
|             Poslední změny: |             Poslední změny: | ||||||
|             <ul> |             <ul> | ||||||
|               <li>Zobrazení alergenu při najetí myší a proklik na seznam alergenů</li> |               <li>Zimní atmosféra</li> | ||||||
|               <li>Přesun přenačtení menu do samostatného dialogu</li> |               <li>Odstranění podniku U Motlíků</li> | ||||||
|               <li>Podzimní atmosféra</li> |  | ||||||
|             </ul> |             </ul> | ||||||
|           </Alert> |           </Alert> | ||||||
|           {dayIndex != null && |           {dayIndex != null && | ||||||
|             <div className='day-navigator'> |             <div className='day-navigator'> | ||||||
|               <span title='Předchozí den'> |               <FontAwesomeIcon title="Předchozí den" icon={faChevronLeft} style={{ cursor: "pointer", visibility: dayIndex > 0 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex - 1)} /> | ||||||
|                 <FontAwesomeIcon 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> | ||||||
|               </span> |               <FontAwesomeIcon title="Následující den" icon={faChevronRight} style={{ cursor: "pointer", visibility: dayIndex < 4 ? "initial" : "hidden" }} onClick={() => handleDayChange(dayIndex + 1)} /> | ||||||
|               <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> |  | ||||||
|             </div> |             </div> | ||||||
|           } |           } | ||||||
|           <Row className='food-tables'> |           <Row className='food-tables'> | ||||||
|             {/* TODO zjednodušit, stačí iterovat klíče typu Restaurant */} |             {food[Restaurants.SLADOVNICKA] && renderFoodTable('Sladovnická', food[Restaurants.SLADOVNICKA])} | ||||||
|             {food['SLADOVNICKA'] && renderFoodTable('SLADOVNICKA', food['SLADOVNICKA'])} |             {/* {food[Restaurants.UMOTLIKU] && renderFoodTable('U Motlíků', food[Restaurants.UMOTLIKU])} */} | ||||||
|             {food['TECHTOWER'] && renderFoodTable('TECHTOWER', food['TECHTOWER'])} |             {food[Restaurants.TECHTOWER] && renderFoodTable('TechTower', food[Restaurants.TECHTOWER])} | ||||||
|             {food['ZASTAVKAUMICHALA'] && renderFoodTable('ZASTAVKAUMICHALA', food['ZASTAVKAUMICHALA'])} |  | ||||||
|             {food['SENKSERIKOVA'] && renderFoodTable('SENKSERIKOVA', food['SENKSERIKOVA'])} |  | ||||||
|           </Row> |           </Row> | ||||||
|           <div className='content-wrapper'> |           <div className='content-wrapper'> | ||||||
|             <div className='content'> |             <div className='content'> | ||||||
|               {canChangeChoice && <> |               {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}> |                 <Form.Select ref={choiceRef} onChange={doAddChoice}> | ||||||
|                   <option></option> |                   <option></option> | ||||||
|                   {Object.entries(LunchChoice) |                   {Object.entries(Locations) | ||||||
|                     .filter(entry => { |                     .filter(entry => { | ||||||
|                       const locationKey = entry[0] as Restaurant; |                       // TODO: wtf, cos pil, když jsi tohle psal? v2
 | ||||||
|                       return !food[locationKey]?.closed; |                       const key = entry[0]; | ||||||
|  |                       const locationIndex = Object.keys(Locations).indexOf(key as unknown as Locations); | ||||||
|  |                       const locationsKey = Object.keys(Locations)[locationIndex]; | ||||||
|  |                       const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||||
|  |                       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> |                 </Form.Select> | ||||||
|                 <small>Je možné vybrat jen jednu možnost. Výběr jiné odstraní předchozí.</small> |                 <small>Je možné vybrat jen jednu možnost. Výběr jiné odstraní předchozí.</small> | ||||||
|                 {foodChoiceList && !closed && <> |                 {foodChoiceList && !closed && <> | ||||||
|                   <p style={{ marginTop: "10px" }}>Na co dobrého? <small>(nepovinné)</small></p> |                   <p style={{ marginTop: "10px" }}>Na co dobrého? <small>(nepovinné)</small></p> | ||||||
|                   <Form.Select ref={foodChoiceRef} onChange={doAddFoodChoice}> |                   <Form.Select ref={foodChoiceRef} onChange={doAddFoodChoice}> | ||||||
|                     <option></option> |                     <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> |                   </Form.Select> | ||||||
|                 </>} |                 </>} | ||||||
|                 {foodChoiceList && !closed && <> |                 {foodChoiceList && !closed && <> | ||||||
|                   <p style={{ marginTop: "10px" }}>V kolik hodin preferuješ odchod?</p> |                   <p style={{ marginTop: "10px" }}>V kolik hodin preferuješ odchod?</p> | ||||||
|                   <Form.Select ref={departureChoiceRef} onChange={handleChangeDepartureTime}> |                   <Form.Select ref={departureChoiceRef} onChange={handleChangeDepartureTime}> | ||||||
|                     <option></option> |                     <option></option> | ||||||
|                     {Object.values(DepartureTime) |                     {Object.values(DepartureTime).map(time => <option key={time} value={time}>{time}</option>)} | ||||||
|                       .filter(time => isInTheFuture(time)) |  | ||||||
|                       .map(time => <option key={time} value={time}>{time}</option>)} |  | ||||||
|                   </Form.Select> |                   </Form.Select> | ||||||
|                 </>} |                 </>} | ||||||
|               </>} |               </>} | ||||||
| @ -505,60 +459,56 @@ function App() { | |||||||
|                 <Table bordered className='mt-5'> |                 <Table bordered className='mt-5'> | ||||||
|                   <tbody> |                   <tbody> | ||||||
|                     {Object.keys(data.choices).map(key => { |                     {Object.keys(data.choices).map(key => { | ||||||
|                       const locationKey = key as LunchChoice; |                       const locationKey = key as keyof typeof Locations; | ||||||
|                       const locationName = getLunchChoiceName(locationKey); |                       console.log("Mapuji location key", locationKey); | ||||||
|  |                       const locationName = Locations[locationKey]; | ||||||
|  |                       console.log("Location name", locationName); | ||||||
|  |                       console.log("Obsah data.choices", data.choices); | ||||||
|                       const loginObject = data.choices[locationKey]; |                       const loginObject = data.choices[locationKey]; | ||||||
|  |                       // TODO toto je hovnokód, mělo by to být napsané tak, aby si na to TypeScript nemohl stěžovat
 | ||||||
|                       if (!loginObject) { |                       if (!loginObject) { | ||||||
|                         return; |                         return; | ||||||
|                       } |                       } | ||||||
|                       const locationLoginList = Object.entries(loginObject); |                       const locationLoginList = Object.entries(loginObject); | ||||||
|                       const locationPickCount = locationLoginList.length |                       console.log("Entries", locationLoginList); | ||||||
|                       return ( |                       return ( | ||||||
|                         <tr key={key}> |                         <tr key={key}> | ||||||
|                           {(locationPickCount ?? 0) > 1 ? ( |                           <td>{locationName}</td> | ||||||
|                             <td>{locationName} ({locationPickCount})</td> |  | ||||||
|                           ) : ( |  | ||||||
|                             <td>{locationName}</td>)} |  | ||||||
|                           <td className='p-0'> |                           <td className='p-0'> | ||||||
|                             <Table> |                             <Table> | ||||||
|                               <tbody> |                               <tbody> | ||||||
|                                 {locationLoginList.map((entry: [string, UserLunchChoice], index) => { |                                 {locationLoginList.map((entry: [string, FoodChoices], index) => { | ||||||
|                                   const login = entry[0]; |                                   const login = entry[0]; | ||||||
|                                   const userPayload = entry[1]; |                                   const userPayload = entry[1]; | ||||||
|                                   const userChoices = userPayload?.selectedFoods; |                                   const userChoices = userPayload?.options; | ||||||
|                                   const trusted = userPayload?.trusted || false; |                                   const trusted = userPayload?.trusted || false; | ||||||
|                                   return <tr key={entry[0]}> |                                   return <tr key={index}> | ||||||
|                                     <td> |                                     <td> | ||||||
|                                       {trusted && <span className='trusted-icon' title='Uživatel ověřený doménovým přihlášením'> |                                       {trusted && <span className='trusted-icon'> | ||||||
|                                         <FontAwesomeIcon icon={faCircleCheck} style={{ cursor: "help" }} /> |                                         <FontAwesomeIcon title='Uživatel ověřený doménovým přihlášením' icon={faCircleCheck} style={{ cursor: "help" }} /> | ||||||
|                                       </span>} |                                       </span>} | ||||||
|                                       {login} |                                       {login} | ||||||
|                                       {userPayload.departureTime && <small> ({userPayload.departureTime})</small>} |                                       {userPayload.departureTime && <small> ({userPayload.departureTime})</small>} | ||||||
|                                       {userPayload.note && <small style={{ overflowWrap: 'anywhere' }}> ({userPayload.note})</small>} |                                       {userPayload.note && <small style={{ overflowWrap: 'anywhere' }}> ({userPayload.note})</small>} | ||||||
|                                       {login === auth.login && canChangeChoice && <span title='Upravit poznámku'> |                                       {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||||
|                                         <FontAwesomeIcon onClick={() => { |  | ||||||
|                                         setNoteModalOpen(true); |                                         setNoteModalOpen(true); | ||||||
|                                       }} className='action-icon' icon={faNoteSticky} /> |                                       }} title='Upravit poznámku' className='action-icon' icon={faNoteSticky} />} | ||||||
|                                       </span>} |                                       {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||||
|                                       {login === auth.login && canChangeChoice && <span title={`Odstranit volbu ${locationName}, včetně případných zvolených jídel`}> |                                         doRemoveChoices(key as Locations); // TODO dořešit
 | ||||||
|                                         <FontAwesomeIcon onClick={() => { |                                       }} title={`Odstranit volbu ${locationName}, včetně případných zvolených jídel`} className='action-icon' icon={faTrashCan} />} | ||||||
|                                         doRemoveChoices(key as LunchChoice); |  | ||||||
|                                       }} className='action-icon' icon={faTrashCan} /> |  | ||||||
|                                       </span>} |  | ||||||
|                                     </td> |                                     </td> | ||||||
|                                     {userChoices?.length && food ? <td> |                                     {userChoices?.length && food ? <td> | ||||||
|                                       <ul> |                                       <ul> | ||||||
|                                         {userChoices?.map(foodIndex => { |                                         {userChoices?.map(foodIndex => { | ||||||
|                                           const restaurantKey = key as Restaurant; |                                           const locationsKey = Object.keys(Locations)[Number(key)] | ||||||
|                                           const foodName = food[restaurantKey]?.food?.[foodIndex].name; |                                           const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||||
|  |                                           const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||||
|  |                                           const foodName = food[restaurant]?.food[foodIndex].name; | ||||||
|                                           return <li key={foodIndex}> |                                           return <li key={foodIndex}> | ||||||
|                                             {foodName} |                                             {foodName} | ||||||
|                                             {login === auth.login && canChangeChoice && |                                             {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||||
|                                             <span title={`Odstranit ${foodName}`}> |                                               doRemoveFoodChoice(key as Locations, foodIndex); // TODO dořešit
 | ||||||
|                                               <FontAwesomeIcon onClick={() => { |                                             }} title={`Odstranit ${foodName}`} className='action-icon' icon={faTrashCan} />} | ||||||
|                                               doRemoveFoodChoice(restaurantKey, foodIndex); |  | ||||||
|                                             }} className='action-icon' icon={faTrashCan} /> |  | ||||||
|                                             </span>} |  | ||||||
|                                           </li> |                                           </li> | ||||||
|                                         })} |                                         })} | ||||||
|                                       </ul> |                                       </ul> | ||||||
| @ -577,7 +527,7 @@ function App() { | |||||||
|                 : <div className='mt-5'><i>Zatím nikdo nehlasoval...</i></div> |                 : <div className='mt-5'><i>Zatím nikdo nehlasoval...</i></div> | ||||||
|               } |               } | ||||||
|             </div> |             </div> | ||||||
|             {dayIndex === data.todayDayIndex && |             {dayIndex === data.todayWeekIndex && | ||||||
|               <div className='mt-5'> |               <div className='mt-5'> | ||||||
|                 {!data.pizzaDay && |                 {!data.pizzaDay && | ||||||
|                   <div style={{ textAlign: 'center' }}> |                   <div style={{ textAlign: 'center' }}> | ||||||
| @ -630,6 +580,9 @@ function App() { | |||||||
|                               <Button className='danger mb-3' title="Umožní znovu editovat objednávky." onClick={async () => { |                               <Button className='danger mb-3' title="Umožní znovu editovat objednávky." onClick={async () => { | ||||||
|                                 await unlockPizzaDay(); |                                 await unlockPizzaDay(); | ||||||
|                               }}>Odemknout</Button> |                               }}>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 () => { |                               <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(); |                                 await finishOrder(); | ||||||
|                               }}>Objednáno</Button> |                               }}>Objednáno</Button> | ||||||
| @ -647,7 +600,7 @@ function App() { | |||||||
|                                 await lockPizzaDay(); |                                 await lockPizzaDay(); | ||||||
|                               }}>Vrátit do "uzamčeno"</Button> |                               }}>Vrátit do "uzamčeno"</Button> | ||||||
|                               <Button className='danger mb-3' style={{ marginLeft: '20px' }} title="Nastaví stav na 'Doručeno' - koncový stav." onClick={async () => { |                               <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> |                               }}>Doručeno</Button> | ||||||
|                             </div> |                             </div> | ||||||
|                           } |                           } | ||||||
| @ -684,28 +637,24 @@ function App() { | |||||||
|                         </Button> |                         </Button> | ||||||
|                       </div> |                       </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'> |                       <div className='qr-code'> | ||||||
|                           <h3>QR platba</h3> |                         <h3>QR platba</h3> | ||||||
|                           <img src={`/api/qr?login=${auth.login}`} alt='QR kód' /> |                         <img src={getQrUrl(auth.login)} alt='QR kód' /> | ||||||
|                         </div> : null |                       </div> | ||||||
|                     } |                     } | ||||||
|                   </div> |                   </div> | ||||||
|                 } |                 } | ||||||
|               </div> |               </div> | ||||||
|             } |             } | ||||||
|           </div> |           </div> | ||||||
|         </> || "Jejda, něco se nám nepovedlo :("} |         </>} | ||||||
|       </div> |       </div> | ||||||
|       <FallingLeaves |  | ||||||
|         numLeaves={LEAF_PRESETS.NORMAL} |  | ||||||
|         leafVariants={LEAF_COLOR_THEMES.AUTUMN} |  | ||||||
|       /> |  | ||||||
|       <Footer /> |       <Footer /> | ||||||
|       <NoteModal isOpen={noteModalOpen} onClose={() => setNoteModalOpen(false)} onSave={saveNote} /> |       <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 React, { useCallback, useEffect, useRef } from 'react'; | ||||||
| import { Button } from 'react-bootstrap'; | import { Button } from 'react-bootstrap'; | ||||||
| import { useAuth } from './context/auth'; | import { useAuth } from './context/auth'; | ||||||
| import { login } from '../../types'; | import { login } from './api/Api'; | ||||||
| import './Login.css'; | import './Login.css'; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @ -14,10 +14,9 @@ export default function Login() { | |||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (auth && !auth.login) { |     if (auth && !auth.login) { | ||||||
|       // Vyzkoušíme přihlášení "naprázdno", pokud projde, přihlásili nás trusted headers
 |       // Vyzkoušíme přihlášení "naprázdno", pokud projde, přihlásili nás trusted headers
 | ||||||
|       login().then(response => { |       login().then(token => { | ||||||
|         const token = response.data; |  | ||||||
|         if (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 => { |       }).catch(error => { | ||||||
|         // nezajímá nás
 |         // nezajímá nás
 | ||||||
| @ -26,16 +25,17 @@ export default function Login() { | |||||||
|   }, [auth]); |   }, [auth]); | ||||||
| 
 | 
 | ||||||
|   const doLogin = useCallback(async () => { |   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) { |     if (length) { | ||||||
|       const response = await login({ body: { login: loginRef.current?.value } }); |       // TODO odchytávat cokoliv mimo 200
 | ||||||
|       if (response.data) { |       const token = await login(loginRef.current.value); | ||||||
|         auth?.setToken(response.data as unknown as string); // TODO vyřešit
 |       if (token) { | ||||||
|  |         auth?.setToken(token); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }, [auth]); |   }, [auth]); | ||||||
| 
 | 
 | ||||||
|   if (!auth?.login) { |   if (!auth || !auth.login) { | ||||||
|     return <div className='login'> |     return <div className='login'> | ||||||
|       <h1>Luncher</h1> |       <h1>Luncher</h1> | ||||||
|       <h4 style={{ marginBottom: "50px" }}>Aplikace pro profesionální management obědů</h4> |       <h4 style={{ marginBottom: "50px" }}>Aplikace pro profesionální management obědů</h4> | ||||||
|  | |||||||
| @ -1,5 +1,3 @@ | |||||||
| import { DepartureTime } from "../../types"; |  | ||||||
| 
 |  | ||||||
| const TOKEN_KEY = "token"; | const TOKEN_KEY = "token"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @ -16,8 +14,8 @@ export const storeToken = (token: string) => { | |||||||
|  *  |  *  | ||||||
|  * @returns token nebo null |  * @returns token nebo null | ||||||
|  */ |  */ | ||||||
| export const getToken = (): string | undefined => { | export const getToken = (): string | null => { | ||||||
|     return localStorage.getItem(TOKEN_KEY) ?? undefined; |     return localStorage.getItem(TOKEN_KEY); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @ -45,62 +43,3 @@ export function getHumanDateTime(datetime: Date) { | |||||||
|         return `${day}.${month}.${year} ${hours}:${minutes}`; |         return `${day}.${month}.${year} ${hours}:${minutes}`; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Vrátí true, pokud je předaný čas větší než aktuální čas. |  | ||||||
|  */ |  | ||||||
| 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, Locations, RemoveChoiceRequest, RemoveChoicesRequest, UpdateNoteRequest } from "../types"; | ||||||
|  | import { api } from "./Api"; | ||||||
|  | 
 | ||||||
|  | const FOOD_API_PREFIX = '/api/food'; | ||||||
|  | 
 | ||||||
|  | export const addChoice = async (locationKey: keyof typeof Locations, foodIndex?: number, dayIndex?: number) => { | ||||||
|  |     return await api.post<AddChoiceRequest, void>(`${FOOD_API_PREFIX}/addChoice`, { locationKey, foodIndex, dayIndex }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const removeChoices = async (locationKey: keyof typeof Locations, dayIndex?: number) => { | ||||||
|  |     return await api.post<RemoveChoicesRequest, void>(`${FOOD_API_PREFIX}/removeChoices`, { locationKey, dayIndex }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const removeChoice = async (locationKey: keyof typeof Locations, 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"; | import { Navbar } from "react-bootstrap"; | ||||||
| 
 | 
 | ||||||
| export default function Footer() { | export default function Footer() { | ||||||
|     return <Navbar className="text-light" variant='dark' expand="lg" style={{ |     return <Navbar className="text-light" variant='dark' expand="lg" style={{ display: "flex", justifyContent: "center" }}> | ||||||
|         display: "flex", |  | ||||||
|         justifyContent: "center", |  | ||||||
|         marginTop: "auto", // Pushne footer na spodek
 |  | ||||||
|         flexShrink: 0 // Zabrání zmenšování při malém obsahu
 |  | ||||||
|     }}> |  | ||||||
|         <span>🄯 Žádná práva nevyhrazena. TODO a zdrojové kódy dostupné <a href="https://gitea.melancholik.eu/mates/Luncher">zde</a>.</span> |         <span>🄯 Žádná práva nevyhrazena. TODO a zdrojové kódy dostupné <a href="https://gitea.melancholik.eu/mates/Luncher">zde</a>.</span> | ||||||
|     </Navbar > |     </Navbar > | ||||||
| } | } | ||||||
| @ -4,26 +4,24 @@ import { useAuth } from "../context/auth"; | |||||||
| import SettingsModal from "./modals/SettingsModal"; | import SettingsModal from "./modals/SettingsModal"; | ||||||
| import { useSettings } from "../context/settings"; | import { useSettings } from "../context/settings"; | ||||||
| import FeaturesVotingModal from "./modals/FeaturesVotingModal"; | 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 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() { | export default function Header() { | ||||||
|     const auth = useAuth(); |     const auth = useAuth(); | ||||||
|     const settings = useSettings(); |     const settings = useSettings(); | ||||||
|     const navigate = useNavigate(); |  | ||||||
|     const [settingsModalOpen, setSettingsModalOpen] = useState<boolean>(false); |     const [settingsModalOpen, setSettingsModalOpen] = useState<boolean>(false); | ||||||
|     const [votingModalOpen, setVotingModalOpen] = useState<boolean>(false); |     const [votingModalOpen, setVotingModalOpen] = useState<boolean>(false); | ||||||
|     const [pizzaModalOpen, setPizzaModalOpen] = useState<boolean>(false); |     const [pizzaModalOpen, setPizzaModalOpen] = useState<boolean>(false); | ||||||
|     const [refreshMenuModalOpen, setRefreshMenuModalOpen] = useState<boolean>(false); |     const [featureVotes, setFeatureVotes] = useState<FeatureRequest[]>([]); | ||||||
|     const [featureVotes, setFeatureVotes] = useState<FeatureRequest[] | undefined>([]); |  | ||||||
| 
 | 
 | ||||||
|     useEffect(() => { |     useEffect(() => { | ||||||
|         if (auth?.login) { |         if (auth?.login) { | ||||||
|             getVotes().then(response => { |             getFeatureVotes().then(votes => { | ||||||
|                 setFeatureVotes(response.data); |                 setFeatureVotes(votes); | ||||||
|             }) |             }) | ||||||
|         } |         } | ||||||
|     }, [auth?.login]); |     }, [auth?.login]); | ||||||
| @ -40,10 +38,6 @@ export default function Header() { | |||||||
|         setPizzaModalOpen(false); |         setPizzaModalOpen(false); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const closeRefreshMenuModal = () => { |  | ||||||
|         setRefreshMenuModalOpen(false); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     const isValidInteger = (str: string) => { |     const isValidInteger = (str: string) => { | ||||||
|         str = str.trim(); |         str = str.trim(); | ||||||
|         if (!str) { |         if (!str) { | ||||||
| @ -82,7 +76,7 @@ export default function Header() { | |||||||
|                     cislo = cislo.padStart(16, '0'); |                     cislo = cislo.padStart(16, '0'); | ||||||
|                 } |                 } | ||||||
|                 let sum = 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 char = cislo.charAt(i); | ||||||
|                     const order = (cislo.length - 1) - i; |                     const order = (cislo.length - 1) - i; | ||||||
|                     const weight = (2 ** order) % 11; |                     const weight = (2 ** order) % 11; | ||||||
| @ -103,8 +97,8 @@ export default function Header() { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const saveFeatureVote = async (option: FeatureRequest, active: boolean) => { |     const saveFeatureVote = async (option: FeatureRequest, active: boolean) => { | ||||||
|         await updateVote({ body: { option, active } }); |         await errorHandler(() => updateFeatureVote(option, active)); | ||||||
|         const votes = [...featureVotes || []]; |         const votes = [...featureVotes]; | ||||||
|         if (active) { |         if (active) { | ||||||
|             votes.push(option); |             votes.push(option); | ||||||
|         } else { |         } else { | ||||||
| @ -114,23 +108,20 @@ export default function Header() { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return <Navbar variant='dark' expand="lg"> |     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.Toggle aria-controls="basic-navbar-nav" /> | ||||||
|         <Navbar.Collapse id="basic-navbar-nav"> |         <Navbar.Collapse id="basic-navbar-nav"> | ||||||
|             <Nav className="nav"> |             <Nav className="nav"> | ||||||
|                 <NavDropdown align="end" title={auth?.login} id="basic-nav-dropdown"> |                 <NavDropdown align="end" title={auth?.login} id="basic-nav-dropdown"> | ||||||
|                     <NavDropdown.Item onClick={() => setSettingsModalOpen(true)}>Nastavení</NavDropdown.Item> |                     <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={() => setVotingModalOpen(true)}>Hlasovat o nových funkcích</NavDropdown.Item> | ||||||
|                     <NavDropdown.Item onClick={() => setPizzaModalOpen(true)}>Pizza kalkulačka</NavDropdown.Item> |                     <NavDropdown.Item onClick={() => setPizzaModalOpen(true)}>Pizza kalkulačka</NavDropdown.Item> | ||||||
|                     <NavDropdown.Item onClick={() => navigate(STATS_URL)}>Statistiky</NavDropdown.Item> |  | ||||||
|                     <NavDropdown.Divider /> |                     <NavDropdown.Divider /> | ||||||
|                     <NavDropdown.Item onClick={auth?.logout}>Odhlásit se</NavDropdown.Item> |                     <NavDropdown.Item onClick={auth?.logout}>Odhlásit se</NavDropdown.Item> | ||||||
|                 </NavDropdown> |                 </NavDropdown> | ||||||
|             </Nav> |             </Nav> | ||||||
|         </Navbar.Collapse> |         </Navbar.Collapse> | ||||||
|         <SettingsModal isOpen={settingsModalOpen} onClose={closeSettingsModal} onSave={saveSettings} /> |         <SettingsModal isOpen={settingsModalOpen} onClose={closeSettingsModal} onSave={saveSettings} /> | ||||||
|         <RefreshMenuModal isOpen={refreshMenuModalOpen} onClose={closeRefreshMenuModal} /> |  | ||||||
|         <FeaturesVotingModal isOpen={votingModalOpen} onClose={closeVotingModal} onChange={saveFeatureVote} initialValues={featureVotes} /> |         <FeaturesVotingModal isOpen={votingModalOpen} onClose={closeVotingModal} onChange={saveFeatureVote} initialValues={featureVotes} /> | ||||||
|         <PizzaCalculatorModal isOpen={pizzaModalOpen} onClose={closePizzaModal} /> |         <PizzaCalculatorModal isOpen={pizzaModalOpen} onClose={closePizzaModal} /> | ||||||
|     </Navbar> |     </Navbar> | ||||||
|  | |||||||
| @ -2,15 +2,15 @@ import { IconDefinition } from '@fortawesome/free-solid-svg-icons'; | |||||||
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' | ||||||
| 
 | 
 | ||||||
| type Props = { | type Props = { | ||||||
|   title?: string, |   title?: String, | ||||||
|   icon: IconDefinition, |   icon: IconDefinition, | ||||||
|   description: string, |   description: String, | ||||||
|   animation?: string, |   animation?: String, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function Loader(props: Readonly<Props>) { | function Loader(props: Props) { | ||||||
|   return <div className='loader'> |   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 ?? '')} /> |     <FontAwesomeIcon icon={props.icon} className={`loader-icon mb-3 ` + (props.animation ?? '')} /> | ||||||
|     <p>{props.description}</p> |     <p>{props.description}</p> | ||||||
|   </div> |   </div> | ||||||
|  | |||||||
| @ -1,17 +1,18 @@ | |||||||
| import { Table } from "react-bootstrap"; | import { Table } from "react-bootstrap"; | ||||||
|  | import { Order, PizzaDayState, PizzaOrder } from "../types"; | ||||||
| import PizzaOrderRow from "./PizzaOrderRow"; | import PizzaOrderRow from "./PizzaOrderRow"; | ||||||
| import { PizzaDayState, PizzaOrder, PizzaVariant, updatePizzaFee } from "../../../types"; | import { updatePizzaFee } from "../api/PizzaDayApi"; | ||||||
| 
 | 
 | ||||||
| type Props = { | type Props = { | ||||||
|     state: PizzaDayState, |     state: PizzaDayState, | ||||||
|     orders: PizzaOrder[], |     orders: Order[], | ||||||
|     onDelete: (pizzaOrder: PizzaVariant) => void, |     onDelete: (pizzaOrder: PizzaOrder) => void, | ||||||
|     creator: string, |     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) => { |     const saveFees = async (customer: string, text?: string, price?: number) => { | ||||||
|         await updatePizzaFee({ body: { login: customer, text, price } }); |         await updatePizzaFee(customer, text, price); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (!orders?.length) { |     if (!orders?.length) { | ||||||
| @ -20,24 +21,26 @@ export default function PizzaOrderList({ state, orders, onDelete, creator }: Rea | |||||||
| 
 | 
 | ||||||
|     const total = orders.reduce((total, order) => total + order.totalPrice, 0); |     const total = orders.reduce((total, order) => total + order.totalPrice, 0); | ||||||
| 
 | 
 | ||||||
|     return <Table className="mt-3" striped bordered hover> |     return <> | ||||||
|         <thead> |         <Table className="mt-3" striped bordered hover> | ||||||
|             <tr> |             <thead> | ||||||
|                 <th>Jméno</th> |                 <tr> | ||||||
|                 <th>Objednávka</th> |                     <th>Jméno</th> | ||||||
|                 <th>Poznámka</th> |                     <th>Objednávka</th> | ||||||
|                 <th>Příplatek</th> |                     <th>Poznámka</th> | ||||||
|                 <th>Cena</th> |                     <th>Příplatek</th> | ||||||
|             </tr> |                     <th>Cena</th> | ||||||
|         </thead> |                 </tr> | ||||||
|         <tbody> |             </thead> | ||||||
|             {orders.map(order => <tr key={order.customer}> |             <tbody> | ||||||
|                 <PizzaOrderRow creator={creator} state={state} order={order} onDelete={onDelete} onFeeModalSave={saveFees} /> |                 {orders.map(order => <tr key={order.customer}> | ||||||
|             </tr>)} |                     <PizzaOrderRow creator={creator} state={state} order={order} onDelete={onDelete} onFeeModalSave={saveFees} /> | ||||||
|             <tr style={{ fontWeight: 'bold' }}> |                 </tr>)} | ||||||
|                 <td colSpan={4}>Celkem</td> |                 <tr style={{ fontWeight: 'bold' }}> | ||||||
|                 <td>{`${total} Kč`}</td> |                     <td colSpan={4}>Celkem</td> | ||||||
|             </tr> |                     <td>{`${total} Kč`}</td> | ||||||
|         </tbody> |                 </tr> | ||||||
|     </Table> |             </tbody> | ||||||
|  |         </Table> | ||||||
|  |     </> | ||||||
| } | } | ||||||
| @ -2,46 +2,44 @@ import React, { useState } from "react"; | |||||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||||
| import { faMoneyBill1, faTrashCan } from "@fortawesome/free-regular-svg-icons"; | import { faMoneyBill1, faTrashCan } from "@fortawesome/free-regular-svg-icons"; | ||||||
| import { useAuth } from "../context/auth"; | import { useAuth } from "../context/auth"; | ||||||
|  | import { Order, PizzaDayState, PizzaOrder } from "../types"; | ||||||
| import PizzaAdditionalFeeModal from "./modals/PizzaAdditionalFeeModal"; | import PizzaAdditionalFeeModal from "./modals/PizzaAdditionalFeeModal"; | ||||||
| import { PizzaDayState, PizzaOrder, PizzaVariant } from "../../../types"; |  | ||||||
| 
 | 
 | ||||||
| type Props = { | type Props = { | ||||||
|     creator: string, |     creator: string, | ||||||
|     order: PizzaOrder, |     order: Order, | ||||||
|     state: PizzaDayState, |     state: PizzaDayState, | ||||||
|     onDelete: (order: PizzaVariant) => void, |     onDelete: (order: PizzaOrder) => void, | ||||||
|     onFeeModalSave: (customer: string, name?: string, price?: number) => 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 auth = useAuth(); | ||||||
|     const [isFeeModalOpen, setIsFeeModalOpen] = useState<boolean>(false); |     const [isFeeModalOpen, setFeeModalOpen] = useState<boolean>(false); | ||||||
| 
 | 
 | ||||||
|     const saveFees = (customer: string, text?: string, price?: number) => { |     const saveFees = (customer: string, text?: string, price?: number) => { | ||||||
|         onFeeModalSave(customer, text, price); |         onFeeModalSave(customer, text, price); | ||||||
|         setIsFeeModalOpen(false); |         setFeeModalOpen(false); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return <> |     return <> | ||||||
|         <td>{order.customer}</td> |         <td>{order.customer}</td> | ||||||
|         <td>{order.pizzaList!.map<React.ReactNode>(pizzaOrder => |         <td>{order.pizzaList.map<React.ReactNode>((pizzaOrder, index) => | ||||||
|             <span key={pizzaOrder.name}> |             <span key={index}> | ||||||
|                 {`${pizzaOrder.name}, ${pizzaOrder.size} (${pizzaOrder.price} Kč)`} |                 {`${pizzaOrder.name}, ${pizzaOrder.size} (${pizzaOrder.price} Kč)`} | ||||||
|                 {auth?.login === order.customer && state === PizzaDayState.CREATED && |                 {auth?.login === order.customer && state === PizzaDayState.CREATED && | ||||||
|                     <span title='Odstranit'> |                     <FontAwesomeIcon onClick={() => { | ||||||
|                         <FontAwesomeIcon onClick={() => { |                         onDelete(pizzaOrder); | ||||||
|                             onDelete(pizzaOrder); |                     }} title='Odstranit' className='action-icon' icon={faTrashCan} /> | ||||||
|                         }} className='action-icon' icon={faTrashCan} /> |  | ||||||
|                     </span> |  | ||||||
|                 } |                 } | ||||||
|             </span>) |             </span>) | ||||||
|             .reduce((prev, curr, index) => [prev, <br key={`br-${index}`} />, curr])} |             .reduce((prev, curr, index) => [prev, <br key={`br-${index}`} />, curr])} | ||||||
|         </td> |         </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 style={{ maxWidth: "200px" }}>{order.fee?.price ? `${order.fee.price} Kč${order.fee.text ? ` (${order.fee.text})` : ''}` : '-'}</td> | ||||||
|         <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> |         </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 { Modal, Button, Form } from "react-bootstrap" | ||||||
| import { FeatureRequest } from "../../../../types"; | import { FeatureRequest } from "../../types"; | ||||||
| 
 | 
 | ||||||
| type Props = { | type Props = { | ||||||
|     isOpen: boolean, |     isOpen: boolean, | ||||||
| @ -9,7 +9,7 @@ type Props = { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** Modální dialog pro hlasování o nových funkcích. */ | /** 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>) => { |     const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { | ||||||
|         onChange(e.currentTarget.value as FeatureRequest, e.currentTarget.checked); |         onChange(e.currentTarget.value as FeatureRequest, e.currentTarget.checked); | ||||||
| @ -31,7 +31,7 @@ export default function FeaturesVotingModal({ isOpen, onClose, onChange, initial | |||||||
|                     label={FeatureRequest[key]} |                     label={FeatureRequest[key]} | ||||||
|                     onChange={handleChange} |                     onChange={handleChange} | ||||||
|                     value={key} |                     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> |             <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. */ | /** 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 note = useRef<HTMLInputElement>(null); | ||||||
| 
 | 
 | ||||||
|     const save = () => { |     const save = () => { | ||||||
|  | |||||||
| @ -10,17 +10,17 @@ type Props = { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** Modální dialog pro nastavení příplatků za pizzu. */ | /** 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 textRef = useRef<HTMLInputElement>(null); | ||||||
|     const priceRef = useRef<HTMLInputElement>(null); |     const priceRef = useRef<HTMLInputElement>(null); | ||||||
| 
 | 
 | ||||||
|     const doSubmit = () => { |     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>) => { |     const handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => { | ||||||
|         if (e.key === 'Enter') { |         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. */ | /** 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 diameter1Ref = useRef<HTMLInputElement>(null); | ||||||
|     const price1Ref = useRef<HTMLInputElement>(null); |     const price1Ref = useRef<HTMLInputElement>(null); | ||||||
|     const diameter2Ref = useRef<HTMLInputElement>(null); |     const diameter2Ref = useRef<HTMLInputElement>(null); | ||||||
| @ -37,7 +37,9 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props | |||||||
|         // 1. pizza
 |         // 1. pizza
 | ||||||
|         if (diameter1Ref.current?.value) { |         if (diameter1Ref.current?.value) { | ||||||
|             const diameter1 = parseInt(diameter1Ref.current?.value); |             const diameter1 = parseInt(diameter1Ref.current?.value); | ||||||
|             r.pizza1 ??= {}; |             if (!r.pizza1) { | ||||||
|  |                 r.pizza1 = {}; | ||||||
|  |             } | ||||||
|             if (diameter1 && diameter1 > 0) { |             if (diameter1 && diameter1 > 0) { | ||||||
|                 r.pizza1.diameter = diameter1; |                 r.pizza1.diameter = diameter1; | ||||||
|                 r.pizza1.area = Math.PI * Math.pow(diameter1 / 2, 2); |                 r.pizza1.area = Math.PI * Math.pow(diameter1 / 2, 2); | ||||||
| @ -57,7 +59,9 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props | |||||||
|         // 2. pizza
 |         // 2. pizza
 | ||||||
|         if (diameter2Ref.current?.value) { |         if (diameter2Ref.current?.value) { | ||||||
|             const diameter2 = parseInt(diameter2Ref.current?.value); |             const diameter2 = parseInt(diameter2Ref.current?.value); | ||||||
|             r.pizza2 ??= {}; |             if (!r.pizza2) { | ||||||
|  |                 r.pizza2 = {}; | ||||||
|  |             } | ||||||
|             if (diameter2 && diameter2 > 0) { |             if (diameter2 && diameter2 > 0) { | ||||||
|                 r.pizza2.diameter = diameter2; |                 r.pizza2.diameter = diameter2; | ||||||
|                 r.pizza2.area = Math.PI * Math.pow(diameter2 / 2, 2); |                 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í. */ | /** 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 settings = useSettings(); | ||||||
|     const bankAccountRef = useRef<HTMLInputElement>(null); |     const bankAccountRef = useRef<HTMLInputElement>(null); | ||||||
|     const nameRef = 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 { useJwt } from "react-jwt"; | ||||||
| import { deleteToken, getToken, storeToken } from "../Utils"; | import { deleteToken, getToken, storeToken } from "../Utils"; | ||||||
| 
 | 
 | ||||||
| @ -15,7 +16,7 @@ type ContextProps = { | |||||||
| 
 | 
 | ||||||
| const authContext = React.createContext<AuthContextProps | null>(null); | const authContext = React.createContext<AuthContextProps | null>(null); | ||||||
| 
 | 
 | ||||||
| export function ProvideAuth(props: Readonly<ContextProps>) { | export function ProvideAuth(props: ContextProps) { | ||||||
|   const auth = useProvideAuth(); |   const auth = useProvideAuth(); | ||||||
|   return <authContext.Provider value={auth}>{props.children}</authContext.Provider> |   return <authContext.Provider value={auth}>{props.children}</authContext.Provider> | ||||||
| } | } | ||||||
| @ -27,8 +28,8 @@ export const useAuth = () => { | |||||||
| function useProvideAuth(): AuthContextProps { | function useProvideAuth(): AuthContextProps { | ||||||
|   const [loginName, setLoginName] = useState<string | undefined>(); |   const [loginName, setLoginName] = useState<string | undefined>(); | ||||||
|   const [trusted, setTrusted] = useState<boolean | undefined>(); |   const [trusted, setTrusted] = useState<boolean | undefined>(); | ||||||
|   const [token, setToken] = useState<string | undefined>(getToken()); |   const [token, setToken] = useState<string | null>(getToken()); | ||||||
|   const { decodedToken } = useJwt(token ?? ''); |   const { decodedToken } = useJwt(token || ''); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (token && token.length > 0) { |     if (token && token.length > 0) { | ||||||
| @ -51,7 +52,7 @@ function useProvideAuth(): AuthContextProps { | |||||||
|   function logout() { |   function logout() { | ||||||
|     const trusted = (decodedToken as any).trusted; |     const trusted = (decodedToken as any).trusted; | ||||||
|     const logoutUrl = (decodedToken as any).logoutUrl; |     const logoutUrl = (decodedToken as any).logoutUrl; | ||||||
|     setToken(undefined); |     setToken(null); | ||||||
|     setLoginName(undefined); |     setLoginName(undefined); | ||||||
|     setTrusted(undefined); |     setTrusted(undefined); | ||||||
|     if (trusted && logoutUrl?.length) { |     if (trusted && logoutUrl?.length) { | ||||||
|  | |||||||
| @ -1,6 +1,7 @@ | |||||||
| import { useEffect, useState } from "react"; | import { useEffect, useState } from "react"; | ||||||
|  | import { getEasterEgg } from "../api/EasterEggApi"; | ||||||
| import { AuthContextProps } from "./auth"; | import { AuthContextProps } from "./auth"; | ||||||
| import { EasterEgg, getEasterEgg } from "../../../types"; | import { EasterEgg } from "../types"; | ||||||
| 
 | 
 | ||||||
| export const useEasterEgg = (auth?: AuthContextProps | null): [EasterEgg | undefined, boolean] => { | export const useEasterEgg = (auth?: AuthContextProps | null): [EasterEgg | undefined, boolean] => { | ||||||
|   const [result, setResult] = useState<EasterEgg | undefined>(); |   const [result, setResult] = useState<EasterEgg | undefined>(); | ||||||
| @ -10,7 +11,7 @@ export const useEasterEgg = (auth?: AuthContextProps | null): [EasterEgg | undef | |||||||
|     async function fetchEasterEgg() { |     async function fetchEasterEgg() { | ||||||
|       if (auth?.login) { |       if (auth?.login) { | ||||||
|         setLoading(true); |         setLoading(true); | ||||||
|         const egg = (await getEasterEgg())?.data; |         const egg = await getEasterEgg(); | ||||||
|         setResult(egg); |         setResult(egg); | ||||||
|         setLoading(false); |         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_NUMBER_KEY = 'bank_account_number'; | ||||||
| const BANK_ACCOUNT_HOLDER_KEY = 'bank_account_holder_name'; | const BANK_ACCOUNT_HOLDER_KEY = 'bank_account_holder_name'; | ||||||
| @ -19,7 +20,7 @@ type ContextProps = { | |||||||
| 
 | 
 | ||||||
| const settingsContext = React.createContext<SettingsContextProps | null>(null); | const settingsContext = React.createContext<SettingsContextProps | null>(null); | ||||||
| 
 | 
 | ||||||
| export function ProvideSettings(props: Readonly<ContextProps>) { | export function ProvideSettings(props: ContextProps) { | ||||||
|   const settings = useProvideSettings(); |   const settings = useProvideSettings(); | ||||||
|   return <settingsContext.Provider value={settings}>{props.children}</settingsContext.Provider> |   return <settingsContext.Provider value={settings}>{props.children}</settingsContext.Provider> | ||||||
| } | } | ||||||
| @ -44,7 +45,7 @@ function useProvideSettings(): SettingsContextProps { | |||||||
|     } |     } | ||||||
|     const hideSoups = localStorage.getItem(HIDE_SOUPS_KEY); |     const hideSoups = localStorage.getItem(HIDE_SOUPS_KEY); | ||||||
|     if (hideSoups !== null) { |     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_CONNECT = 'connect'; | ||||||
| export const EVENT_DISCONNECT = 'disconnect'; | export const EVENT_DISCONNECT = 'disconnect'; | ||||||
| export const EVENT_MESSAGE = 'message'; | 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 React from 'react'; | ||||||
| import ReactDOM from 'react-dom/client'; | import ReactDOM from 'react-dom/client'; | ||||||
|  | import App from './App'; | ||||||
|  | import { SocketContext, socket } from './context/socket'; | ||||||
| import { ProvideAuth } from './context/auth'; | import { ProvideAuth } from './context/auth'; | ||||||
|  | import { ToastContainer } from 'react-toastify'; | ||||||
|  | import { ProvideSettings } from './context/settings'; | ||||||
| import 'react-toastify/dist/ReactToastify.css'; | import 'react-toastify/dist/ReactToastify.css'; | ||||||
| import './index.css'; | import './index.css'; | ||||||
| import AppRoutes from './AppRoutes'; | import Snowfall from 'react-snowfall'; | ||||||
| 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; |  | ||||||
| }); |  | ||||||
| 
 | 
 | ||||||
| const root = ReactDOM.createRoot( | const root = ReactDOM.createRoot( | ||||||
|   document.getElementById('root') as HTMLElement |   document.getElementById('root') as HTMLElement | ||||||
| ); | ); | ||||||
| root.render( | root.render( | ||||||
|   <React.StrictMode> |   <React.StrictMode> | ||||||
|     <BrowserRouter> |     <ProvideAuth> | ||||||
|       <ProvideAuth> |       <ProvideSettings> | ||||||
|         <AppRoutes /> |         <SocketContext.Provider value={socket}> | ||||||
|       </ProvideAuth> |           <> | ||||||
|     </BrowserRouter> |             <Snowfall style={{ | ||||||
|  |               zIndex: 2, | ||||||
|  |               position: 'fixed', | ||||||
|  |               width: '100vw', | ||||||
|  |               height: '100vh'}} /> | ||||||
|  |             <App /> | ||||||
|  |           </> | ||||||
|  |           <ToastContainer /> | ||||||
|  |         </SocketContext.Provider> | ||||||
|  |       </ProvideSettings> | ||||||
|  |     </ProvideAuth> | ||||||
|   </React.StrictMode> |   </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": { |   "compilerOptions": { | ||||||
|  |     "target": "ESNext", | ||||||
|     "lib": [ |     "lib": [ | ||||||
|       "dom", |       "dom", | ||||||
|       "dom.iterable", |       "dom.iterable", | ||||||
| @ -15,12 +16,10 @@ | |||||||
|     "strict": true, |     "strict": true, | ||||||
|     "forceConsistentCasingInFileNames": true, |     "forceConsistentCasingInFileNames": true, | ||||||
|     "noFallthroughCasesInSwitch": true, |     "noFallthroughCasesInSwitch": true, | ||||||
|     "moduleResolution": "bundler", |     "module": "esnext", | ||||||
|     "module": "ESNext", |     "moduleResolution": "node", | ||||||
|     "target": "ESNext", |  | ||||||
|     "resolveJsonModule": true, |     "resolveJsonModule": true, | ||||||
|     "isolatedModules": true, |     "isolatedModules": true, | ||||||
|     "allowImportingTsExtensions": true, |  | ||||||
|     "noEmit": true, |     "noEmit": true, | ||||||
|     "jsx": "react-jsx" |     "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 | export NODE_ENV=development | ||||||
| # Spustí server a klienta v samostatných panelech uvnitř stejného tmux okna. | yarn install | ||||||
| # Pokud už daná tmux session existuje, pouze se k ní připojí. | cd server && yarn start & | ||||||
| 
 | cd client && yarn start & | ||||||
| SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) | wait | ||||||
| 
 |  | ||||||
| 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 |  | ||||||
							
								
								
									
										6
									
								
								server/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								server/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,7 +1,7 @@ | |||||||
| /data | /node_modules | ||||||
| /dist | /dist | ||||||
| /resources/easterEggs | data.json | ||||||
| /src/gen |  | ||||||
| .env.production | .env.production | ||||||
| .env.development | .env.development | ||||||
| .easter-eggs.json | .easter-eggs.json | ||||||
|  | /resources/easterEggs | ||||||
| @ -11,29 +11,29 @@ | |||||||
|     "test": "jest" |     "test": "jest" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@babel/core": "^7.28.4", |     "@babel/core": "^7.23.0", | ||||||
|     "@babel/preset-env": "^7.28.3", |     "@babel/preset-env": "^7.22.20", | ||||||
|     "@babel/preset-typescript": "^7.27.1", |     "@babel/preset-typescript": "^7.23.0", | ||||||
|     "@types/express": "^5.0.3", |     "@types/express": "^4.17.17", | ||||||
|     "@types/jest": "^30.0.0", |     "@types/jest": "^29.5.14", | ||||||
|     "@types/jsonwebtoken": "^9.0.10", |     "@types/jsonwebtoken": "^9.0.6", | ||||||
|     "@types/node": "^24.7.1", |     "@types/node": "^20.11.20", | ||||||
|     "@types/request-promise": "^4.1.48", |     "@types/request-promise": "^4.1.48", | ||||||
|     "babel-jest": "^30.2.0", |     "babel-jest": "^29.7.0", | ||||||
|     "jest": "^30.2.0", |     "jest": "^29.7.0", | ||||||
|     "nodemon": "^3.1.10", |     "nodemon": "^3.1.0", | ||||||
|     "ts-node": "^10.9.1", |     "ts-node": "^10.9.1", | ||||||
|     "typescript": "^5.9.3" |     "typescript": "^5.0.2" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "axios": "^1.12.2", |     "axios": "^1.4.0", | ||||||
|     "cheerio": "^1.1.2", |     "cheerio": "^1.0.0-rc.12", | ||||||
|     "cors": "^2.8.5", |     "cors": "^2.8.5", | ||||||
|     "dotenv": "^17.2.3", |     "dotenv": "^16.4.5", | ||||||
|     "express": "^5.1.0", |     "express": "^4.18.2", | ||||||
|     "jsonwebtoken": "^9.0.0", |     "jsonwebtoken": "^9.0.0", | ||||||
|     "redis": "^5.8.3", |     "redis": "^4.6.7", | ||||||
|     "simple-json-db": "^2.0.0", |     "simple-json-db": "^2.0.0", | ||||||
|     "socket.io": "^4.6.1" |     "socket.io": "^4.6.1" | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -49,16 +49,16 @@ export async function downloadPizzy(mock: boolean): Promise<Pizza[]> { | |||||||
|     const $ = load(html); |     const $ = load(html); | ||||||
|     const links = $('.vypisproduktu > div > h4 > a'); |     const links = $('.vypisproduktu > div > h4 > a'); | ||||||
|     const urls = []; |     const urls = []; | ||||||
|     for (const element of links) { |     for (let i = 0; i < links.length; i++) { | ||||||
|         if (element.name === 'a' && element.attribs?.href) { |         if (links[i].name === 'a' && links[i].attribs?.href) { | ||||||
|             const pizzaUrl = element.attribs?.href; |             const pizzaUrl = links[i].attribs?.href; | ||||||
|             urls.push(buildPizzaUrl(pizzaUrl)); |             urls.push(buildPizzaUrl(pizzaUrl)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     // Scrapneme jednotlivé pizzy
 |     // Scrapneme jednotlivé pizzy
 | ||||||
|     const result: Pizza[] = []; |     const result: Pizza[] = []; | ||||||
|     for (const element of urls) { |     for (let i = 0; i < urls.length; i++) { | ||||||
|         const pizzaUrl = element; |         const pizzaUrl = urls[i]; | ||||||
|         const pizzaHtml = await axios.get(pizzaUrl).then(res => res.data); |         const pizzaHtml = await axios.get(pizzaUrl).then(res => res.data); | ||||||
|         // Název
 |         // Název
 | ||||||
|         const name = $('.produkt > h2', pizzaHtml).first().text() |         const name = $('.produkt > h2', pizzaHtml).first().text() | ||||||
|  | |||||||
| @ -9,13 +9,12 @@ import { generateToken, verify } from "./auth"; | |||||||
| import { InsufficientPermissions } from "./utils"; | import { InsufficientPermissions } from "./utils"; | ||||||
| import { initWebsocket } from "./websocket"; | import { initWebsocket } from "./websocket"; | ||||||
| import pizzaDayRoutes from "./routes/pizzaDayRoutes"; | import pizzaDayRoutes from "./routes/pizzaDayRoutes"; | ||||||
| import foodRoutes, { refreshMetoda } from "./routes/foodRoutes"; | import foodRoutes from "./routes/foodRoutes"; | ||||||
| import votingRoutes from "./routes/votingRoutes"; | import votingRoutes from "./routes/votingRoutes"; | ||||||
| import easterEggRoutes from "./routes/easterEggRoutes"; | import easterEggRoutes from "./routes/easterEggRoutes"; | ||||||
| import statsRoutes from "./routes/statsRoutes"; |  | ||||||
| 
 | 
 | ||||||
| const ENVIRONMENT = process.env.NODE_ENV ?? 'production'; | const ENVIRONMENT = process.env.NODE_ENV || 'production'; | ||||||
| dotenv.config({ path: path.resolve(__dirname, `../.env.${ENVIRONMENT}`) }); | dotenv.config({ path: path.resolve(__dirname, `./.env.${ENVIRONMENT}`) }); | ||||||
| 
 | 
 | ||||||
| // Validace nastavení JWT tokenu - nemá bez něj smysl vůbec povolit server spustit
 | // Validace nastavení JWT tokenu - nemá bez něj smysl vůbec povolit server spustit
 | ||||||
| if (!process.env.JWT_SECRET) { | if (!process.env.JWT_SECRET) { | ||||||
| @ -35,7 +34,7 @@ app.use(cors({ | |||||||
| 
 | 
 | ||||||
| // Zapínatelný login přes hlavičky - pokud je zapnutý nepovolí "basicauth"
 | // 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_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 (HTTP_REMOTE_USER_ENABLED) { | ||||||
|     if (!process.env.HTTP_REMOTE_TRUSTED_IPS) { |     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.'); |         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) { |     if (!HTTP_REMOTE_USER_ENABLED) { | ||||||
|         res.status(403).json({ error: 'Není zapnuté přihlášení z hlaviček' }); |         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)); |     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')
 |     if (HTTP_REMOTE_USER_ENABLED) { // je rovno app.enabled('trust proxy')
 | ||||||
|         // Autentizace pomocí trusted headers
 |         // Autentizace pomocí trusted headers
 | ||||||
|         const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME); |         const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME); | ||||||
|         //const remoteName = req.header('remote-name');
 |         const remoteName = req.header('remote-name'); | ||||||
|         if (remoteUser && remoteUser.length > 0 ) { |         if (remoteUser && remoteUser.length > 0 && remoteName && remoteName.length > 0) { | ||||||
|             res.status(200).json(generateToken(Buffer.from(remoteUser, 'latin1').toString(), true)); |             res.status(200).json(generateToken(Buffer.from(remoteName, 'latin1').toString(), true)); | ||||||
|         } else { |         } 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 { |     } else { | ||||||
|         // Klasická autentizace loginem
 |         // 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
 | // TODO dočasné řešení - QR se zobrazuje přes <img>, nemáme sem jak dostat token
 | ||||||
| app.get("/api/qr", (req, res) => { | app.get("/api/qr", (req, res) => { | ||||||
|  |     // const login = getLogin(parseToken(req));
 | ||||||
|     if (!req.query?.login) { |     if (!req.query?.login) { | ||||||
|         throw Error("Nebyl předán 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 */ | /** Middleware ověřující JWT token */ | ||||||
| app.use("/api/", (req, res, next) => { | app.use("/api/", (req, res, next) => { | ||||||
|     if (HTTP_REMOTE_USER_ENABLED) { |     if (HTTP_REMOTE_USER_ENABLED) { | ||||||
|         // Autentizace pomocí trusted headers
 |         const userHeader = req.header(HTTP_REMOTE_USER_HEADER_NAME); | ||||||
|         const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME); |         const nameHeader = req.header('remote-name'); | ||||||
|         if(process.env.ENABLE_HEADERS_LOGGING === 'yes'){ |         const emailHeader = req.header('remote-email'); | ||||||
|             delete req.headers["cookie"] |         if (userHeader !== undefined && nameHeader !== undefined) { | ||||||
|             console.log(req.headers) |             const remoteName = Buffer.from(nameHeader, 'latin1').toString(); | ||||||
|         } |             console.log("Tvuj username, name a email: %s, %s, %s.", userHeader, remoteName, emailHeader); | ||||||
|         if (remoteUser && remoteUser.length > 0) { |  | ||||||
|             const remoteName = Buffer.from(remoteUser, 'latin1').toString(); |  | ||||||
|             if (ENVIRONMENT !== "production") { |  | ||||||
|                 console.log("Tvuj username: %s.", remoteName); |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     if (!req.headers.authorization) { |     if (!req.headers.authorization) { | ||||||
| @ -142,10 +130,7 @@ app.use("/api/pizzaDay", pizzaDayRoutes); | |||||||
| app.use("/api/food", foodRoutes); | app.use("/api/food", foodRoutes); | ||||||
| app.use("/api/voting", votingRoutes); | app.use("/api/voting", votingRoutes); | ||||||
| app.use("/api/easterEggs", easterEggRoutes); | app.use("/api/easterEggs", easterEggRoutes); | ||||||
| app.use("/api/stats", statsRoutes); | app.use(express.static('public')) | ||||||
| 
 |  | ||||||
| app.use('/stats', express.static('public')); |  | ||||||
| app.use(express.static('public')); |  | ||||||
| 
 | 
 | ||||||
| // Middleware pro zpracování chyb
 | // Middleware pro zpracování chyb
 | ||||||
| app.use((err: any, req: any, res: any, next: any) => { | 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(); |     next(); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| const PORT = process.env.PORT ?? 3001; | const PORT = process.env.PORT || 3001; | ||||||
| const HOST = process.env.HOST ?? '0.0.0.0'; | const HOST = process.env.HOST || '0.0.0.0'; | ||||||
| 
 | 
 | ||||||
| server.listen(PORT, () => { | server.listen(PORT, () => { | ||||||
|     console.log(`Server listening on ${HOST}, port ${PORT}`); |     console.log(`Server listening on ${HOST}, port ${PORT}`); | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| import { WeeklyStats, LunchChoice } from "../../types/gen/types.gen"; | import { getDayOfWeekIndex } from "./utils"; | ||||||
| 
 | 
 | ||||||
| // Mockovací data pro podporované podniky, na jeden týden
 | // Mockovací data pro podporované podniky, na jeden týden
 | ||||||
| const MOCK_DATA = { | const MOCK_DATA = { | ||||||
| @ -6,31 +6,79 @@ const MOCK_DATA = { | |||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "0,25l", |                 amount: "0,25l", | ||||||
|                 name: "Česnečka s uzeným masem a krutony", |                 name: "Kulajda", | ||||||
|                 price: "35\xA0Kč", |                 price: "35\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [1, 3, 7, 9] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "250g", |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 9, 10] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "145\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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, | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         [ | ||||||
|  |             { | ||||||
|  |                 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, | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         [ | ||||||
|  |             { | ||||||
|  |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7, 9] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
| @ -39,118 +87,50 @@ const MOCK_DATA = { | |||||||
|                 name: "Kuřecí vývar s nudlemi", |                 name: "Kuřecí vývar s nudlemi", | ||||||
|                 price: "35\xA0Kč", |                 price: "35\xA0Kč", | ||||||
|                 isSoup: true, |                 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", |                 amount: "150g", | ||||||
|                 name: "Krůtí roláda se sušenými rajčaty , mozzarellou a bramborovou kaší", |                 name: "Kovbojské fazole s klobásou a chlebem", | ||||||
|                 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", |  | ||||||
|                 price: "125\xA0Kč", |                 price: "125\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [3, 7] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "0,25l", |                 amount: "0,25l", | ||||||
|                 name: "Čočková polévka", |                 name: "Dršťková polévka", | ||||||
|                 price: "35\xA0Kč", |                 price: "35\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [9, 12] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "135\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7, 9, 10] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "150g", |                 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č", |                 price: "145\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 9] |  | ||||||
|             } |             } | ||||||
|         ] |         ] | ||||||
|     ], |     ], | ||||||
| @ -290,362 +270,164 @@ const MOCK_DATA = { | |||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Batátový krém s chilli a kokosovým mlékem", |                 name: "Uzený vývar s kapustou", | ||||||
|                 price: "40\xA0Kč", |                 price: "40\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [1, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Kuřecí stehno na paprice, knedlík", |                 name: "Čočka na kyselo, opečená klobása, okurka, chléb", | ||||||
|                 price: "130\xA0Kč", |                 price: "130\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Ovarová", |                 name: "Slepičí s nudlemi", | ||||||
|                 price: "40\xA0Kč", |                 price: "40\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [1] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Zapečené těstoviny s uzeným masem, okurka", |                 name: "Zvěřinový guláš, knedlík", | ||||||
|                 price: "130\xA0Kč", |                 price: "130\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 6, 11] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [7] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Hovězí s hráškem a rýží", |                 name: "Dýňový krém se smetanou", | ||||||
|                 price: "40\xA0Kč", |                 price: "40\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [9] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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č", |                 price: "130\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [7, 9] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Smažené rýžové nudle Pad thai s arašídy, zeleninou a vejcem", |                 name: "Tvarohové knedlíky s meruňkami, strouhaný tvaroh, máslo, cukr", | ||||||
|                 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", |  | ||||||
|                 price: "na\xA0váhu", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Dýňová", |                 name: "Zeleninová s jáhly", | ||||||
|                 price: "40\xA0Kč", |                 price: "40\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [1, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Uzená plec, křenová omáčka, knedlík", |                 name: "Rizoto s vepřovým masem, okurka", | ||||||
|                 price: "130\xA0Kč", |                 price: "130\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Palačinky s marmeládou přelité čokoládou, sypané cukrem", |                 name: "Tortellini s parmezánovou omáčkou", | ||||||
|                 price: "na\xA0váhu", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [7] |  | ||||||
|             } |             } | ||||||
|         ], |         ], | ||||||
|         [ |         [ | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Hovězí vývar s játrovými knedlíčky", |                 name: "Fazolová s uzeninou", | ||||||
|                 price: "40\xA0Kč", |                 price: "40\xA0Kč", | ||||||
|                 isSoup: true, |                 isSoup: true, | ||||||
|                 allergens: [1, 3, 7, 9] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Kuřecí Kung-pao, jasmínová rýže", |                 name: "Krůtí perkelt, těstoviny", | ||||||
|                 price: "130\xA0Kč", |                 price: "130\xA0Kč", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 5, 6] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 amount: "-", | ||||||
|                 name: "Sýrové tortelliny s pažitkovou omáčkou", |                 name: "Grilovaný hermelín, bulgurový salát se zeleninou", | ||||||
|                 price: "na\xA0váhu", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 6, 7, 11] |  | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 amount: "-", |                 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", |                 price: "na\xA0váhu", | ||||||
|                 isSoup: false, |                 isSoup: false, | ||||||
|                 allergens: [1, 3, 7] |  | ||||||
|             } |             } | ||||||
|         ] |         ] | ||||||
|     ], |     ] | ||||||
|     'zastavkaUmichala': [ |  | ||||||
|         [ |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Fazolačka s klobásou & zakysačkou", |  | ||||||
|                 price: "39\xA0Kč", |  | ||||||
|                 isSoup: true, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Zeleninová musaka – lilek, cuketa, tomatové sugo & sýrový bešamel", |  | ||||||
|                 price: "135\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Slovácké strapačky s uzenou slaninou, zelím, mletým pepřem & sekanou petrželkou", |  | ||||||
|                 price: "140\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Hovězí guláš s vejcem, zeleninovou garniturkou & žemlovými knedlíky", |  | ||||||
|                 price: "145\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Kuřecí roláda s kaštanovou nádivkou, demi-glace & smetanovou bramborovou kaší", |  | ||||||
|                 price: "150\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             } |  | ||||||
|         ], |  | ||||||
|         [ |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Hovězí vývar se zeleninou a játrovou rýží", |  | ||||||
|                 price: "39\xA0Kč", |  | ||||||
|                 isSoup: true, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Pečené vepřové koleno, křen, hořčice, chléb", |  | ||||||
|                 price: "320\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             } |  | ||||||
|         ], |  | ||||||
|         [ |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Zeleninová polévka s kuskusem", |  | ||||||
|                 price: "39\xA0Kč", |  | ||||||
|                 isSoup: true, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Poutine (trhané vepřové, hranolky, sýr, čalamáda, pikantní omáčka)", |  | ||||||
|                 price: "190\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             } |  | ||||||
|         ], |  | ||||||
|         [ |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Hrachová polévka s uzeninou", |  | ||||||
|                 price: "39\xA0Kč", |  | ||||||
|                 isSoup: true, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Vepřový řízek z kotlety, domácí bramborový salát", |  | ||||||
|                 price: "170\xA0Kč", |  | ||||||
|                 isSoup: false, |  | ||||||
|             } |  | ||||||
|         ], |  | ||||||
|         [ |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Cibulačka se sýrem", |  | ||||||
|                 price: "39\xA0Kč", |  | ||||||
|                 isSoup: true, |  | ||||||
|             }, |  | ||||||
|             { |  | ||||||
|                 amount: "-", |  | ||||||
|                 name: "Burger z Chuck rollu, hranolky, tatarská omáčka", |  | ||||||
|                 price: "200\xA0Kč", |  | ||||||
|                 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
 | // Mockovací data pro Pizza day
 | ||||||
| @ -1398,11 +1180,8 @@ const MOCK_PIZZA_LIST = [ | |||||||
|     } |     } | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| /** | export const getTodayMock = () => { | ||||||
|  * Funkce vrací mock datu ve formátu YYYY-MM-DD |     return '2023-05-31'; // středa
 | ||||||
|  */ |  | ||||||
| export const getTodayMock = (): Date => { |  | ||||||
|     return new Date('2025-01-10'); // pátek
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export const getMenuSladovnickaMock = () => { | export const getMenuSladovnickaMock = () => { | ||||||
| @ -1417,39 +1196,6 @@ export const getMenuTechTowerMock = () => { | |||||||
|     return MOCK_DATA['techTower']; |     return MOCK_DATA['techTower']; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export const getMenuZastavkaUmichalaMock = () => { |  | ||||||
|     return MOCK_DATA['zastavkaUmichala']; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export const getMenuSenkSerikovaMock = () => { |  | ||||||
|     return MOCK_DATA['senkSerikova']; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export const getPizzaListMock = () => { | export const getPizzaListMock = () => { | ||||||
|     return MOCK_PIZZA_LIST; |     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 dotenv from 'dotenv'; | ||||||
| import path from 'path'; | import path from 'path'; | ||||||
| import { getClientData, getToday } from "./service"; | import { getToday } from "./service"; | ||||||
| import { getUsersByLocation, getHumanTime } from "./utils"; | import { formatDate, getUsersByLocation } from "./utils"; | ||||||
| import { NotifikaceData, NotifikaceInput } from '../../types'; | 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}`) }); | 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 url = process.env.NTFY_HOST | ||||||
|     const username = process.env.NTFY_USERNAME; |     const username = process.env.NTFY_USERNAME; | ||||||
|     const password = process.env.NTFY_PASSWD; |     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") |         console.log("NTFY_PASSWD není definován v env") | ||||||
|         return |         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 userByCLocation = getUsersByLocation(clientData.choices, data.user) | ||||||
| 
 | 
 | ||||||
|     const token = Buffer.from(`${username}:${password}`, 'utf8').toString('base64'); |     const token = Buffer.from(`${username}:${password}`, 'utf8').toString('base64'); | ||||||
| @ -50,64 +97,27 @@ export const ntfyCall = async (data: NotifikaceInput) => { | |||||||
|     return promises; |     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*/ | /** 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) => { | export const callNotifikace = async ({ input, teams = true, gotify = false, ntfy = true }: NotifikaceData) => { | ||||||
|     const notifications = []; |     const notifications = []; | ||||||
| 
 |  | ||||||
|     if (ntfy) { |     if (ntfy) { | ||||||
|         const ntfyPromises = await ntfyCall(input); |         const ntfyPromises = await ntfyCall(input); | ||||||
|         if (ntfyPromises) { |         if (ntfyPromises) { | ||||||
|             notifications.push(...ntfyPromises); |             notifications.push(...ntfyPromises); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     /* Zatím není | ||||||
|     if (teams) { |     if (teams) { | ||||||
|         const teamsPromises = await teamsCall(input); |          notifications.push(teamsCall(input)); | ||||||
|         if (teamsPromises) { |      }*/ | ||||||
|             notifications.push(teamsPromises); | 
 | ||||||
|         } |     // Add more notifications as necessary
 | ||||||
|  | 
 | ||||||
|  |     //gotify bych řekl, že už je deprecated
 | ||||||
|  |     if (gotify) { | ||||||
|  |         const gotifyPromises = await gotifyCall(input, gotifyData); | ||||||
|  |         notifications.push(...gotifyPromises); | ||||||
|     } |     } | ||||||
|     // gotify bych řekl, že už je deprecated
 |  | ||||||
|     // if (gotify) {
 |  | ||||||
|     //     const gotifyPromises = await gotifyCall(input, gotifyData);
 |  | ||||||
|     //     notifications.push(...gotifyPromises);
 |  | ||||||
|     // }
 |  | ||||||
| 
 | 
 | ||||||
|     try { |     try { | ||||||
|         const results = await Promise.all(notifications); |         const results = await Promise.all(notifications); | ||||||
|  | |||||||
| @ -1,10 +1,10 @@ | |||||||
| import { formatDate } from "./utils"; | import { formatDate } from "./utils"; | ||||||
| import { callNotifikace } from "./notifikace"; | import { callNotifikace } from "./notifikace"; | ||||||
| import { generateQr } from "./qr"; | import { generateQr } from "./qr"; | ||||||
|  | import { ClientData, PizzaDayState, UdalostEnum, Pizza, PizzaSize, Order, PizzaOrder, DayData } from "../../types"; | ||||||
| import getStorage from "./storage"; | import getStorage from "./storage"; | ||||||
| import { downloadPizzy } from "./chefie"; | import { downloadPizzy } from "./chefie"; | ||||||
| import { getClientData, getToday, initIfNeeded } from "./service"; | import { getToday, initIfNeeded } from "./service"; | ||||||
| import { Pizza, ClientData, PizzaDayState, PizzaSize, PizzaOrder, PizzaVariant, UdalostEnum } from "../../types/gen/types.gen"; |  | ||||||
| 
 | 
 | ||||||
| const storage = getStorage(); | const storage = getStorage(); | ||||||
| 
 | 
 | ||||||
| @ -14,7 +14,8 @@ const storage = getStorage(); | |||||||
|  */ |  */ | ||||||
| export async function getPizzaList(): Promise<Pizza[] | undefined> { | export async function getPizzaList(): Promise<Pizza[] | undefined> { | ||||||
|     await initIfNeeded(); |     await initIfNeeded(); | ||||||
|     let clientData = await getClientData(getToday()); |     const today = formatDate(getToday()); | ||||||
|  |     let clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaList) { |     if (!clientData.pizzaList) { | ||||||
|         const mock = process.env.MOCK_DATA === 'true'; |         const mock = process.env.MOCK_DATA === 'true'; | ||||||
|         clientData = await savePizzaList(await downloadPizzy(mock)); |         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> { | export async function savePizzaList(pizzaList: Pizza[]): Promise<ClientData> { | ||||||
|     await initIfNeeded(); |     await initIfNeeded(); | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     clientData.pizzaList = pizzaList; |     clientData.pizzaList = pizzaList; | ||||||
|     clientData.pizzaListLastUpdate = formatDate(new Date()); |     clientData.pizzaListLastUpdate = new Date(); | ||||||
|     await storage.setData(today, clientData); |     await storage.setData(today, clientData); | ||||||
|     return clientData; |     return clientData; | ||||||
| } | } | ||||||
| @ -42,14 +43,14 @@ export async function savePizzaList(pizzaList: Pizza[]): Promise<ClientData> { | |||||||
|  */ |  */ | ||||||
| export async function createPizzaDay(creator: string): Promise<ClientData> { | export async function createPizzaDay(creator: string): Promise<ClientData> { | ||||||
|     await initIfNeeded(); |     await initIfNeeded(); | ||||||
|     const clientData = await getClientData(getToday()); |     const today = formatDate(getToday()); | ||||||
|  |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (clientData.pizzaDay) { |     if (clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den již existuje"); |         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ě!
 |     // TODO berka rychlooprava, vyřešit lépe - stahovat jednou, na jediném místě!
 | ||||||
|     const pizzaList = await getPizzaList(); |     const pizzaList = await getPizzaList(); | ||||||
|     const data: ClientData = { pizzaDay: { state: PizzaDayState.CREATED, creator, orders: [] }, pizzaList, ...clientData }; |     const data: ClientData = { pizzaDay: { state: PizzaDayState.CREATED, creator, orders: [] }, pizzaList, ...clientData }; | ||||||
|     const today = formatDate(getToday()); |  | ||||||
|     await storage.setData(today, data); |     await storage.setData(today, data); | ||||||
|     callNotifikace({ input: { udalost: UdalostEnum.ZAHAJENA_PIZZA, user: creator } }) |     callNotifikace({ input: { udalost: UdalostEnum.ZAHAJENA_PIZZA, user: creator } }) | ||||||
|     return data; |     return data; | ||||||
| @ -59,7 +60,8 @@ export async function createPizzaDay(creator: string): Promise<ClientData> { | |||||||
|  * Smaže pizza day pro aktuální den. |  * Smaže pizza day pro aktuální den. | ||||||
|  */ |  */ | ||||||
| export async function deletePizzaDay(login: string): Promise<ClientData> { | 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) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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"); |         throw Error("Login uživatele se neshoduje se zakladatelem Pizza Day"); | ||||||
|     } |     } | ||||||
|     delete clientData.pizzaDay; |     delete clientData.pizzaDay; | ||||||
|     const today = formatDate(getToday()); |  | ||||||
|     await storage.setData(today, clientData); |     await storage.setData(today, clientData); | ||||||
|     return 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) { | export async function addPizzaOrder(login: string, pizza: Pizza, size: PizzaSize) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         throw Error("Pizza day pro dnešní den neexistuje"); | ||||||
|     } |     } | ||||||
|     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { |     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { | ||||||
|         throw Error("Pizza day není ve stavu " + 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) { |     if (!order) { | ||||||
|         order = { |         order = { | ||||||
|             customer: login, |             customer: login, | ||||||
|             pizzaList: [], |             pizzaList: [], | ||||||
|             totalPrice: 0, |             totalPrice: 0, | ||||||
|             hasQr: false, |  | ||||||
|         } |         } | ||||||
|         clientData.pizzaDay.orders ??= []; |  | ||||||
|         clientData.pizzaDay.orders.push(order); |         clientData.pizzaDay.orders.push(order); | ||||||
|     } |     } | ||||||
|     const pizzaOrder: PizzaVariant = { |     const pizzaOrder: PizzaOrder = { | ||||||
|         varId: size.varId, |         varId: size.varId, | ||||||
|         name: pizza.name, |         name: pizza.name, | ||||||
|         size: size.size, |         size: size.size, | ||||||
|         price: size.price, |         price: size.price, | ||||||
|     } |     } | ||||||
|     order.pizzaList ??= []; |  | ||||||
|     order.pizzaList.push(pizzaOrder); |     order.pizzaList.push(pizzaOrder); | ||||||
|     order.totalPrice += pizzaOrder.price; |     order.totalPrice += pizzaOrder.price; | ||||||
|     await storage.setData(today, clientData); |     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 login login uživatele | ||||||
|  * @param pizzaOrder objednávka pizzy |  * @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 today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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) { |     if (orderIndex < 0) { | ||||||
|         throw Error("Nebyly nalezeny žádné objednávky pro uživatele " + login); |         throw Error("Nebyly nalezeny žádné objednávky pro uživatele " + login); | ||||||
|     } |     } | ||||||
|     const order = clientData.pizzaDay.orders![orderIndex]; |     const order = clientData.pizzaDay.orders[orderIndex]; | ||||||
|     const index = order.pizzaList!.findIndex(o => o.name === pizzaOrder.name && o.size === pizzaOrder.size); |     const index = order.pizzaList.findIndex(o => o.name === pizzaOrder.name && o.size === pizzaOrder.size); | ||||||
|     if (index < 0) { |     if (index < 0) { | ||||||
|         throw Error("Objednávka s danými parametry nebyla nalezena"); |         throw Error("Objednávka s danými parametry nebyla nalezena"); | ||||||
|     } |     } | ||||||
|     const price = order.pizzaList![index].price; |     const price = order.pizzaList[index].price; | ||||||
|     order.pizzaList!.splice(index, 1); |     order.pizzaList.splice(index, 1); | ||||||
|     order.totalPrice -= price; |     order.totalPrice -= price; | ||||||
|     if (order.pizzaList!.length == 0) { |     if (order.pizzaList.length == 0) { | ||||||
|         clientData.pizzaDay.orders!.splice(orderIndex, 1); |         clientData.pizzaDay.orders.splice(orderIndex, 1); | ||||||
|     } |     } | ||||||
|     await storage.setData(today, clientData); |     await storage.setData(today, clientData); | ||||||
|     return clientData; |     return clientData; | ||||||
| @ -151,7 +149,7 @@ export async function removePizzaOrder(login: string, pizzaOrder: PizzaVariant) | |||||||
|  */ |  */ | ||||||
| export async function lockPizzaDay(login: string) { | export async function lockPizzaDay(login: string) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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) { | export async function unlockPizzaDay(login: string) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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) { | export async function finishPizzaOrder(login: string) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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) { | export async function finishPizzaDelivery(login: string, bankAccount?: string, bankAccountHolder?: string) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     const clientData = await getClientData(getToday()); |     const clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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
 |     // Vygenerujeme QR kód, pokud k tomu máme data
 | ||||||
|     if (bankAccount?.length && bankAccountHolder?.length) { |     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
 |             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); |                 await generateQr(order.customer, bankAccount, bankAccountHolder, order.totalPrice, message); | ||||||
|                 order.hasQr = true; |                 order.hasQr = true; | ||||||
|             } |             } | ||||||
| @ -257,15 +255,15 @@ export async function finishPizzaDelivery(login: string, bankAccount?: string, b | |||||||
|  */ |  */ | ||||||
| export async function updatePizzaDayNote(login: string, note?: string) { | export async function updatePizzaDayNote(login: string, note?: string) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     let clientData = await getClientData(getToday()); |     let clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         throw Error("Pizza day pro dnešní den neexistuje"); | ||||||
|     } |     } | ||||||
|     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { |     if (clientData.pizzaDay.state !== PizzaDayState.CREATED) { | ||||||
|         throw Error("Pizza day není ve stavu " + PizzaDayState.CREATED); |         throw Error("Pizza day není ve stavu " + PizzaDayState.CREATED); | ||||||
|     } |     } | ||||||
|     const myOrder = clientData.pizzaDay.orders!.find(o => o.customer === login); |     const myOrder = clientData.pizzaDay.orders.find(o => o.customer === login); | ||||||
|     if (!myOrder?.pizzaList?.length) { |     if (!myOrder || !myOrder.pizzaList.length) { | ||||||
|         throw Error("Pizza day neobsahuje žádné objednávky uživatele " + login); |         throw Error("Pizza day neobsahuje žádné objednávky uživatele " + login); | ||||||
|     } |     } | ||||||
|     myOrder.note = note; |     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) { | export async function updatePizzaFee(login: string, targetLogin: string, text?: string, price?: number) { | ||||||
|     const today = formatDate(getToday()); |     const today = formatDate(getToday()); | ||||||
|     let clientData = await getClientData(getToday()); |     let clientData: DayData = await storage.getData(today); | ||||||
|     if (!clientData.pizzaDay) { |     if (!clientData.pizzaDay) { | ||||||
|         throw Error("Pizza day pro dnešní den neexistuje"); |         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) { |     if (clientData.pizzaDay.creator !== login) { | ||||||
|         throw Error("Příplatky může měnit pouze zakladatel Pizza day"); |         throw Error("Příplatky může měnit pouze zakladatel Pizza day"); | ||||||
|     } |     } | ||||||
|     const targetOrder = clientData.pizzaDay.orders!.find(o => o.customer === targetLogin); |     const targetOrder = clientData.pizzaDay.orders.find(o => o.customer === targetLogin); | ||||||
|     if (!targetOrder?.pizzaList?.length) { |     if (!targetOrder || !targetOrder.pizzaList.length) { | ||||||
|         throw Error(`Pizza day neobsahuje žádné objednávky uživatele ${targetLogin}`); |         throw Error(`Pizza day neobsahuje žádné objednávky uživatele ${targetLogin}`); | ||||||
|     } |     } | ||||||
|     if (!price) { |     if (!price) { | ||||||
| @ -304,7 +302,7 @@ export async function updatePizzaFee(login: string, targetLogin: string, text?: | |||||||
|         targetOrder.fee = { text, price }; |         targetOrder.fee = { text, price }; | ||||||
|     } |     } | ||||||
|     // Přepočet ceny
 |     // 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); |     await storage.setData(today, clientData); | ||||||
|     return clientData; |     return clientData; | ||||||
| } | } | ||||||
| @ -1,8 +1,7 @@ | |||||||
| import axios from "axios"; | import axios from "axios"; | ||||||
| import { load } from 'cheerio'; | import { load } from 'cheerio'; | ||||||
| import { getMenuSladovnickaMock, getMenuTechTowerMock, getMenuUMotlikuMock, getMenuZastavkaUmichalaMock, getMenuSenkSerikovaMock } from "./mock"; | import { Food } from "../../types"; | ||||||
| import { formatDate } from "./utils"; | import { getMenuSladovnickaMock, getMenuTechTowerMock, getMenuUMotlikuMock } from "./mock"; | ||||||
| import { Food } from "../../types/gen/types.gen"; |  | ||||||
| 
 | 
 | ||||||
| // Fráze v názvech jídel, které naznačují že se jedná o polévku
 | // Fráze v názvech jídel, které naznačují že se jedná o polévku
 | ||||||
| const SOUP_NAMES = [ | const SOUP_NAMES = [ | ||||||
| @ -14,20 +13,18 @@ const SOUP_NAMES = [ | |||||||
|     'fazolová', |     'fazolová', | ||||||
|     'cuketový krém', |     'cuketový krém', | ||||||
|     'boršč', |     'boršč', | ||||||
|     'slepičí s ', |     'slepičí s', | ||||||
|     'zeleninová s ', |     'zeleninová s', | ||||||
|     'hovězí s ', |     'hovězí s', | ||||||
|     'kachní kaldoun', |     'kachní kaldoun', | ||||||
|     'dršťková' |     'dršťková' | ||||||
| ]; | ]; | ||||||
| const DAYS_IN_WEEK = ['pondělí', 'úterý', 'středa', 'čtvrtek', 'pátek', 'sobota', 'neděle']; | const DAYS_IN_WEEK = ['pondělí', 'úterý', 'středa', 'čtvrtek', 'pátek', 'sobota', 'neděle']; | ||||||
| 
 | 
 | ||||||
| // URL na týdenní menu jednotlivých restaurací
 | // 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 U_MOTLIKU_URL = 'https://www.umotliku.cz/menu'; | ||||||
| const TECHTOWER_URL = 'https://www.equifarm.cz/restaurace-techtower'; | const TECHTOWER_URL = 'https://www.equifarm.cz/restaurace-techtower'; | ||||||
| const ZASTAVKAUMICHALA_URL = 'https://www.zastavkaumichala.cz'; |  | ||||||
| const SENKSERIKOVA_URL = 'https://www.menicka.cz/6561-pivovarsky-senk-serikova.html'; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Vrátí true, pokud předaný text obsahuje některé ze slov, které naznačuje, že se jedná o polévku. |  * Vrátí true, pokud předaný text obsahuje některé ze slov, které naznačuje, že se jedná o polévku. | ||||||
| @ -53,28 +50,6 @@ const sanitizeText = (text: string): string => { | |||||||
|     return text.replace('\t', '').replace(' , ', ', ').trim(); |     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. |  * Stáhne a vrátí aktuální HTML z dané URL. | ||||||
|  *  |  *  | ||||||
| @ -100,69 +75,82 @@ export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = f | |||||||
|     const html = await getHtml(SLADOVNICKA_URL); |     const html = await getHtml(SLADOVNICKA_URL); | ||||||
|     const $ = load(html); |     const $ = load(html); | ||||||
| 
 | 
 | ||||||
|     const menuContentElements = $('#daily-menu-content-list').children('[id^="daily-menu-content-"]'); |     const list = $('ul.tab-links').children(); | ||||||
|     // 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 result: Food[][] = []; | ||||||
|     for (let dayIndex = 0; dayIndex < 5; dayIndex++) { |     for (let dayIndex = 0; dayIndex < 5; dayIndex++) { | ||||||
|         const dayChildren = $(menuContentElements[dayIndex]).children(); |         const currentDate = new Date(firstDayOfWeek); | ||||||
|         // Prozatím předpokládáme, že budou mít vždy polévku a hlavní jídla
 |         currentDate.setDate(firstDayOfWeek.getDate() + dayIndex); | ||||||
|         if (dayChildren.length < 2) { |         const searchedDayText = `${currentDate.getDate()}.${currentDate.getMonth() + 1}.${capitalize(DAYS_IN_WEEK[dayIndex])}`; | ||||||
|             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)"); |         // 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
 | ||||||
|  |             result[dayIndex] = [{ | ||||||
|  |                 amount: undefined, | ||||||
|  |                 name: "Pro daný den nebyla nalezena denní nabídka", | ||||||
|  |                 price: "", | ||||||
|  |                 isSoup: false, | ||||||
|  |             }]; | ||||||
|  |             continue; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // Parsování polévky
 |         // Dle dohledaného indexu najdeme správný tabpanel
 | ||||||
|         const soupElement = dayChildren.get(0); |         const rows = $('div.tab-content').children(); | ||||||
|         const soupTable = $(soupElement).find('table tbody tr'); |         if (index >= rows.length) { | ||||||
|         const soupCells = soupTable.children('td'); |             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) { |         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"); |             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({ |         currentDayFood.push({ | ||||||
|             amount: soupAmount, |             amount: sanitizeText($(soupCells.get(0)).text()), | ||||||
|             name: soupParsed.cleanName, |             name: sanitizeText($(soupCells.get(1)).text()), | ||||||
|             price: soupPrice, |             price: sanitizeText($(soupCells.get(2)).text().replace(' ', '\xA0')), | ||||||
|             isSoup: true, |             isSoup: true, | ||||||
|             allergens: soupParsed.allergens.length > 0 ? soupParsed.allergens : undefined, |  | ||||||
|         }); |         }); | ||||||
|          |         // Hlavní jídla - div -> table -> tbody -> 3x tr
 | ||||||
|         // Projdeme všechny řádky hlavních jídel
 |         const mainCourseRows = $(tables.get(1)).children().first().children(); | ||||||
|         mainCourseRows.each((i, row) => { |         mainCourseRows.each((i, foodRow) => { | ||||||
|             const cells = $(row).children('td'); |             const foodCells = $(foodRow).children(); | ||||||
|             const amount = sanitizeText($(cells.get(0)).text()); |             if (foodCells.length !== 3) { | ||||||
|             const nameRaw = sanitizeText($(cells.get(1)).text()); |                 throw Error("Neočekávaný počet buněk v řádku jídla: " + foodCells.length + ", ale očekávány byly 3"); | ||||||
|             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, |  | ||||||
|                 }); |  | ||||||
|             } |             } | ||||||
|         }); |             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; |         result[dayIndex] = currentDayFood; | ||||||
|     } |     } | ||||||
|     return result; |     return result; | ||||||
| @ -301,7 +289,7 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal | |||||||
| 
 | 
 | ||||||
|     const result: Food[][] = []; |     const result: Food[][] = []; | ||||||
|     // TODO validovat, že v textu nalezeného <font> je rozsah, do kterého spadá vstupní datum
 |     // 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 parsing = false; | ||||||
|     let currentDayIndex = 0; |     let currentDayIndex = 0; | ||||||
|     for (let i = 0; i < siblings.length; i++) { |     for (let i = 0; i < siblings.length; i++) { | ||||||
| @ -319,134 +307,23 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal | |||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             let price = 'na\xA0váhu'; |             let price = 'na\xA0váhu'; | ||||||
|             let nameRaw = text.replace('•', ''); |             let name = text.replace('•', ''); | ||||||
|             if (text.toLowerCase().endsWith('kč')) { |             if (text.toLowerCase().endsWith('kč')) { | ||||||
|                 const tmp = text.replace('\xA0', ' ').split(' '); |                 const tmp = text.replace('\xA0', ' ').split(' '); | ||||||
|                 const split = [tmp.slice(0, -2).join(' ')].concat(tmp.slice(-2)); |                 const split = [tmp.slice(0, -2).join(' ')].concat(tmp.slice(-2)); | ||||||
|                 price = `${split.slice(1)[0]}\xA0Kč` |                 price = `${split.slice(1)[0]}\xA0Kč` | ||||||
|                 nameRaw = split[0].replace('•', ''); |                 name = split[0].replace('•', ''); | ||||||
|             } |             } | ||||||
|             if (nameRaw.endsWith('–')) { |             if (result[currentDayIndex] == null) { | ||||||
|                 nameRaw = nameRaw.slice(0, -1).trim(); |                 result[currentDayIndex] = []; | ||||||
|             } |             } | ||||||
| 
 |  | ||||||
|             const parsed = parseAllergens(nameRaw); |  | ||||||
|             result[currentDayIndex] ??= []; |  | ||||||
|             result[currentDayIndex].push({ |             result[currentDayIndex].push({ | ||||||
|                 amount: '-', |                 amount: '-', | ||||||
|                 name: parsed.cleanName, |                 name, | ||||||
|                 price, |                 price, | ||||||
|                 isSoup: isTextSoupName(parsed.cleanName), |                 isSoup: isTextSoupName(name), | ||||||
|                 allergens: parsed.allergens.length > 0 ? parsed.allergens : undefined, |  | ||||||
|             }) |             }) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     return result; |     return result; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Získá obědovou nabídku ZastavkaUmichala 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 getMenuZastavkaUmichala = async (firstDayOfWeek: Date, mock: boolean = false): Promise<Food[][]> => { |  | ||||||
|     if (mock) { |  | ||||||
|         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 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)) { |  | ||||||
|             result[dayIndex] = [{ |  | ||||||
|                 amount: undefined, |  | ||||||
|                 name: "Pro tento den není uveřejněna nabídka jídel", |  | ||||||
|                 price: "", |  | ||||||
|                 isSoup: false, |  | ||||||
|             }]; |  | ||||||
|         } 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); |  | ||||||
|             const $ = load(html); |  | ||||||
| 
 |  | ||||||
|             const currentDayFood: Food[] = []; |  | ||||||
|             $('.foodsList li').each((index, element) => { |  | ||||||
|                 currentDayFood.push({ |  | ||||||
|                     amount: '-', |  | ||||||
|                     name: sanitizeText($(element).contents().not('span').text()), |  | ||||||
|                     price: sanitizeText($(element).find('span').text()), |  | ||||||
|                     isSoup: (index === 0), |  | ||||||
|                 }); |  | ||||||
|             }); |  | ||||||
|             result[dayIndex] = currentDayFood; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     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 express, { NextFunction } from "express"; | ||||||
| import { getLogin } from "../auth"; | import { getLogin, getTrusted } from "../auth"; | ||||||
| import { parseToken } from "../utils"; | import { parseToken } from "../utils"; | ||||||
| import path from "path"; | import path from "path"; | ||||||
| import fs from "fs"; | 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'; | const IMAGES_PATH = '../../resources/easterEggs'; | ||||||
| 
 | 
 | ||||||
| type EasterEggsJson = { | type EasterEggsJson = { | ||||||
| @ -34,11 +34,16 @@ function generateUrl() { | |||||||
|  */ |  */ | ||||||
| function getEasterEggImage(req: any, res: any, next: NextFunction) { | function getEasterEggImage(req: any, res: any, next: NextFunction) { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|  |     const trusted = getTrusted(parseToken(req)); | ||||||
|     try { |     try { | ||||||
|         if (login in easterEggs) { |         // TODO vrátit!
 | ||||||
|             const imagePath = easterEggs[login][Math.floor(Math.random() * easterEggs[login].length)].path; |         // if (trusted) {
 | ||||||
|             res.sendFile(path.join(__dirname, IMAGES_PATH, imagePath)); |         if (true) { | ||||||
|             return; |             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); |         res.sendStatus(404); | ||||||
|     } catch (e: any) { next(e) } |     } catch (e: any) { next(e) } | ||||||
| @ -119,7 +124,7 @@ let easterEggs: EasterEggsJson; | |||||||
| if (fs.existsSync(EASTER_EGGS_JSON_PATH)) { | if (fs.existsSync(EASTER_EGGS_JSON_PATH)) { | ||||||
|     const content = fs.readFileSync(EASTER_EGGS_JSON_PATH, 'utf-8'); |     const content = fs.readFileSync(EASTER_EGGS_JSON_PATH, 'utf-8'); | ||||||
|     easterEggs = JSON.parse(content); |     easterEggs = JSON.parse(content); | ||||||
|     for (const [_, eggs] of Object.entries(easterEggs)) { |     for (const [key, eggs] of Object.entries(easterEggs)) { | ||||||
|         for (const easterEgg of eggs) { |         for (const easterEgg of eggs) { | ||||||
|             const url = generateUrl(); |             const url = generateUrl(); | ||||||
|             easterEgg.url = url; |             easterEgg.url = url; | ||||||
| @ -133,11 +138,16 @@ if (fs.existsSync(EASTER_EGGS_JSON_PATH)) { | |||||||
| // Získání náhodného easter eggu pro přihlášeného uživatele
 | // Získání náhodného easter eggu pro přihlášeného uživatele
 | ||||||
| router.get("/", async (req, res, next) => { | router.get("/", async (req, res, next) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|  |     const trusted = getTrusted(parseToken(req)); | ||||||
|     try { |     try { | ||||||
|         if (easterEggs && login in easterEggs) { |         // TODO vrátit!
 | ||||||
|             const randomEasterEgg = easterEggs[login][Math.floor(Math.random() * easterEggs[login].length)]; |         // if (trusted) {
 | ||||||
|             const { path, startOffset, endOffset, ...strippedEasterEgg } = randomEasterEgg; // Path klient k ničemu nepotřebuje a nemá ho znát
 |         if (true) { | ||||||
|             return res.status(200).json({ ...strippedEasterEgg, ...getRandomPosition(startOffset, endOffset) }); |             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(); |         return res.status(200).send(); | ||||||
|     } catch (e: any) { next(e) } |     } 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 { getLogin, getTrusted } from "../auth"; | ||||||
| import { addChoice, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote, fetchRestaurantWeekMenuData, saveRestaurantWeekMenu } from "../service"; | import { addChoice, addVolatileData, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote } from "../service"; | ||||||
| import { getDayOfWeekIndex, parseToken, getFirstWorkDayOfWeek } from "../utils"; | import { getDayOfWeekIndex, parseToken } from "../utils"; | ||||||
| import { getWebsocket } from "../websocket"; | import { getWebsocket } from "../websocket"; | ||||||
| import { callNotifikace } from "../notifikace"; | import { callNotifikace } from "../notifikace"; | ||||||
| import { AddChoiceData, ChangeDepartureTimeData, RemoveChoiceData, RemoveChoicesData, UdalostEnum, UpdateNoteData } from "../../../types/gen/types.gen"; | import { AddChoiceRequest, ChangeDepartureTimeRequest, IDayIndex, RemoveChoiceRequest, RemoveChoicesRequest, UdalostEnum, UpdateNoteRequest } from "../../../types"; | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| // 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; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Ověří a vrátí index dne v týdnu z požadavku, za předpokladu, že byl předán, a je zároveň |  * 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 |  * @param req request | ||||||
|  * @returns index dne v týdnu |  * @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) { |     if (req.body.dayIndex == null) { | ||||||
|         throw Error(`Nebyl předán index dne v týdnu.`); |         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(); | 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 login = getLogin(parseToken(req)); | ||||||
|     const trusted = getTrusted(parseToken(req)); |     const trusted = getTrusted(parseToken(req)); | ||||||
|     let date = undefined; |     let date = undefined; | ||||||
| @ -86,12 +45,12 @@ router.post("/addChoice", async (req: Request<{}, any, AddChoiceData["body"]>, r | |||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|         const data = await addChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); |         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); |         return res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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 login = getLogin(parseToken(req)); | ||||||
|     const trusted = getTrusted(parseToken(req)); |     const trusted = getTrusted(parseToken(req)); | ||||||
|     let date = undefined; |     let date = undefined; | ||||||
| @ -106,12 +65,12 @@ router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["bo | |||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|         const data = await removeChoices(login, trusted, req.body.locationKey, date); |         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); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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 login = getLogin(parseToken(req)); | ||||||
|     const trusted = getTrusted(parseToken(req)); |     const trusted = getTrusted(parseToken(req)); | ||||||
|     let date = undefined; |     let date = undefined; | ||||||
| @ -126,12 +85,12 @@ router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body | |||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|         const data = await removeChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); |         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); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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 login = getLogin(parseToken(req)); | ||||||
|     const trusted = getTrusted(parseToken(req)); |     const trusted = getTrusted(parseToken(req)); | ||||||
|     const note = req.body.note; |     const note = req.body.note; | ||||||
| @ -150,12 +109,12 @@ router.post("/updateNote", async (req: Request<{}, any, UpdateNoteData["body"]>, | |||||||
|             date = getDateForWeekIndex(dayIndex); |             date = getDateForWeekIndex(dayIndex); | ||||||
|         } |         } | ||||||
|         const data = await updateNote(login, trusted, note, date); |         const data = await updateNote(login, trusted, note, date); | ||||||
|         getWebsocket().emit("message", data); |         getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|         res.status(200).json(data); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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)); |     const login = getLogin(parseToken(req)); | ||||||
|     let date = undefined; |     let date = undefined; | ||||||
|     if (req.body.dayIndex != null) { |     if (req.body.dayIndex != null) { | ||||||
| @ -169,7 +128,7 @@ router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDeparture | |||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|         const data = await updateDepartureTime(login, req.body?.time, date); |         const data = await updateDepartureTime(login, req.body?.time, date); | ||||||
|         getWebsocket().emit("message", data); |         getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|         res.status(200).json(data); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } catch (e: any) { next(e) } | ||||||
| }); | }); | ||||||
| @ -177,90 +136,9 @@ router.post("/changeDepartureTime", async (req: Request<{}, any, ChangeDeparture | |||||||
| router.post("/jdemeObed", async (req, res, next) => { | router.post("/jdemeObed", async (req, res, next) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     try { |     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({}); |         res.status(200).json({}); | ||||||
|     } catch (e: any) { next(e) } |     } 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; | 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 { createPizzaDay, deletePizzaDay, getPizzaList, addPizzaOrder, removePizzaOrder, lockPizzaDay, unlockPizzaDay, finishPizzaOrder, finishPizzaDelivery, updatePizzaDayNote, updatePizzaFee } from "../pizza"; | ||||||
| import { parseToken } from "../utils"; | import { parseToken } from "../utils"; | ||||||
| import { getWebsocket } from "../websocket"; | 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(); | const router = express.Router(); | ||||||
| 
 | 
 | ||||||
| @ -12,17 +13,17 @@ router.post("/create", async (req: Request<{}, any, undefined>, res) => { | |||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     const data = await createPizzaDay(login); |     const data = await createPizzaDay(login); | ||||||
|     res.status(200).json(data); |     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. */ | /** Smaže pizza day pro aktuální den, za předpokladu že existuje. */ | ||||||
| router.post("/delete", async (req: Request<{}, any, undefined>, res) => { | router.post("/delete", async (req: Request<{}, any, undefined>, res) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     const data = await deletePizzaDay(login); |     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)); |     const login = getLogin(parseToken(req)); | ||||||
|     if (isNaN(req.body?.pizzaIndex)) { |     if (isNaN(req.body?.pizzaIndex)) { | ||||||
|         throw Error("Nebyl předán index pizzy"); |         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); |         throw Error("Neplatný index velikosti pizzy: " + pizzaSizeIndex); | ||||||
|     } |     } | ||||||
|     const data = await addPizzaOrder(login, pizzy[pizzaIndex], pizzy[pizzaIndex].sizes[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({}); |     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)); |     const login = getLogin(parseToken(req)); | ||||||
|     if (!req.body?.pizzaOrder) { |     if (!req.body?.pizzaOrder) { | ||||||
|         throw Error("Nebyla předána objednávka"); |         throw Error("Nebyla předána objednávka"); | ||||||
|     } |     } | ||||||
|     const data = await removePizzaOrder(login, req.body?.pizzaOrder); |     const data = await removePizzaOrder(login, req.body?.pizzaOrder); | ||||||
|     getWebsocket().emit("message", data); |     getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|     res.status(200).json({}); |     res.status(200).json({}); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| router.post("/lock", async (req: Request<{}, any, undefined>, res) => { | router.post("/lock", async (req: Request<{}, any, undefined>, res) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     const data = await lockPizzaDay(login); |     const data = await lockPizzaDay(login); | ||||||
|     getWebsocket().emit("message", data); |     getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|     res.status(200).json({}); |     res.status(200).json({}); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| router.post("/unlock", async (req: Request<{}, any, undefined>, res) => { | router.post("/unlock", async (req: Request<{}, any, undefined>, res) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     const data = await unlockPizzaDay(login); |     const data = await unlockPizzaDay(login); | ||||||
|     getWebsocket().emit("message", data); |     getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|     res.status(200).json({}); |     res.status(200).json({}); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| router.post("/finishOrder", async (req: Request<{}, any, undefined>, res) => { | router.post("/finishOrder", async (req: Request<{}, any, undefined>, res) => { | ||||||
|     const login = getLogin(parseToken(req)); |     const login = getLogin(parseToken(req)); | ||||||
|     const data = await finishPizzaOrder(login); |     const data = await finishPizzaOrder(login); | ||||||
|     getWebsocket().emit("message", data); |     getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|     res.status(200).json({}); |     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 login = getLogin(parseToken(req)); | ||||||
|     const data = await finishPizzaDelivery(login, req.body.bankAccount, req.body.bankAccountHolder); |     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({}); |     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)); |     const login = getLogin(parseToken(req)); | ||||||
|     try { |     try { | ||||||
|         if (req.body.note && req.body.note.length > 70) { |         if (req.body.note && req.body.note.length > 70) { | ||||||
|             throw Error("Poznámka může mít maximálně 70 znaků"); |             throw Error("Poznámka může mít maximálně 70 znaků"); | ||||||
|         } |         } | ||||||
|         const data = await updatePizzaDayNote(login, req.body.note); |         const data = await updatePizzaDayNote(login, req.body.note); | ||||||
|         getWebsocket().emit("message", data); |         getWebsocket().emit("message", await addVolatileData(data)); | ||||||
|         res.status(200).json(data); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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)); |     const login = getLogin(parseToken(req)); | ||||||
|     if (!req.body.login) { |     if (!req.body.login) { | ||||||
|         return res.status(400).json({ error: "Nebyl předán login cílového uživatele" }); |         return res.status(400).json({ error: "Nebyl předán login cílového uživatele" }); | ||||||
|     } |     } | ||||||
|     try { |     try { | ||||||
|         const data = await updatePizzaFee(login, req.body.login, req.body.text, req.body.price); |         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); |         res.status(200).json(data); | ||||||
|     } catch (e: any) { next(e) } |     } 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 { getLogin } from "../auth"; | ||||||
| import { parseToken } from "../utils"; | import { parseToken } from "../utils"; | ||||||
| import { getUserVotes, updateFeatureVote } from "../voting"; | import { getUserVotes, updateFeatureVote } from "../voting"; | ||||||
| import { GetVotesData, UpdateVoteData } from "../../../types"; | import { FeatureRequest, UpdateFeatureVoteRequest } from "../../../types"; | ||||||
| 
 | 
 | ||||||
| const router = express.Router(); | 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 login = getLogin(parseToken(req)); | ||||||
|     const data = await getUserVotes(login); |     const data = await getUserVotes(login); | ||||||
|     res.status(200).json(data); |     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)); |     const login = getLogin(parseToken(req)); | ||||||
|     if (req.body?.option == null || req.body?.active == null) { |     if (req.body?.option == null || req.body?.active == null) { | ||||||
|         res.status(400).json({ error: "Chybné parametry volání" }); |         res.status(400).json({ error: "Chybné parametry volání" }); | ||||||
|  | |||||||
| @ -1,8 +1,8 @@ | |||||||
| import { InsufficientPermissions, formatDate, getDayOfWeekIndex, getFirstWorkDayOfWeek, getHumanDate, getIsWeekend, getWeekNumber } from "./utils"; | import { InsufficientPermissions, formatDate, getDayOfWeekIndex, getFirstWorkDayOfWeek, getHumanDate, getHumanTime, getIsWeekend, getLastWorkDayOfWeek, getWeekNumber } from "./utils"; | ||||||
|  | import { ClientData, Locations, Restaurants, DayMenu, DepartureTime, DayData, WeekMenu } from "../../types"; | ||||||
| import getStorage from "./storage"; | import getStorage from "./storage"; | ||||||
| import { getMenuSladovnicka, getMenuTechTower, getMenuZastavkaUmichala, getMenuSenkSerikova } from "./restaurants"; | import { getMenuSladovnicka, getMenuTechTower, getMenuUMotliku } from "./restaurants"; | ||||||
| import { getTodayMock } from "./mock"; | import { getTodayMock } from "./mock"; | ||||||
| import { ClientData, DepartureTime, LunchChoice, Restaurant, RestaurantDayMenu, WeekMenu } from "../../types/gen/types.gen"; |  | ||||||
| 
 | 
 | ||||||
| const storage = getStorage(); | const storage = getStorage(); | ||||||
| const MENU_PREFIX = 'menu'; | const MENU_PREFIX = 'menu'; | ||||||
| @ -10,7 +10,7 @@ const MENU_PREFIX = 'menu'; | |||||||
| /** Vrátí dnešní datum, případně fiktivní datum pro účely vývoje a testování. */ | /** Vrátí dnešní datum, případně fiktivní datum pro účely vývoje a testování. */ | ||||||
| export function getToday(): Date { | export function getToday(): Date { | ||||||
|     if (process.env.MOCK_DATA === 'true') { |     if (process.env.MOCK_DATA === 'true') { | ||||||
|         return getTodayMock(); |         return new Date(getTodayMock()); | ||||||
|     } |     } | ||||||
|     return new Date(); |     return new Date(); | ||||||
| } | } | ||||||
| @ -31,31 +31,43 @@ export const getDateForWeekIndex = (index: number) => { | |||||||
| function getEmptyData(date?: Date): ClientData { | function getEmptyData(date?: Date): ClientData { | ||||||
|     const usedDate = date || getToday(); |     const usedDate = date || getToday(); | ||||||
|     return { |     return { | ||||||
|         todayDayIndex: getDayOfWeekIndex(getToday()), |  | ||||||
|         date: getHumanDate(usedDate), |         date: getHumanDate(usedDate), | ||||||
|         isWeekend: getIsWeekend(usedDate), |         isWeekend: getIsWeekend(usedDate), | ||||||
|         dayIndex: getDayOfWeekIndex(usedDate), |         weekIndex: getDayOfWeekIndex(usedDate), | ||||||
|         choices: {}, |         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. |  * 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> { | 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 = { |     clientData.menus = { | ||||||
|         SLADOVNICKA: await getRestaurantMenu('SLADOVNICKA', date), |         [Restaurants.SLADOVNICKA]: await getRestaurantMenu(Restaurants.SLADOVNICKA, date), | ||||||
|         // UMOTLIKU: await getRestaurantMenu('UMOTLIKU', date),
 |         // [Restaurants.UMOTLIKU]: await getRestaurantMenu(Restaurants.UMOTLIKU, date),
 | ||||||
|         TECHTOWER: await getRestaurantMenu('TECHTOWER', date), |         [Restaurants.TECHTOWER]: await getRestaurantMenu(Restaurants.TECHTOWER, date), | ||||||
|         ZASTAVKAUMICHALA: await getRestaurantMenu('ZASTAVKAUMICHALA', date), |  | ||||||
|         SENKSERIKOVA: await getRestaurantMenu('SENKSERIKOVA', date), |  | ||||||
|     } |     } | ||||||
|  |     clientData = await addVolatileData(clientData); | ||||||
|     return 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 |  * @param date datum | ||||||
|  * @returns databázový klíč |  * @returns databázový klíč | ||||||
| @ -66,117 +78,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 |  * @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> { | 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
 | // 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. |  * 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. |  * Pokud neexistuje, provede stažení menu pro příslušný týden a uložení do DB. | ||||||
|  *  |  *  | ||||||
|  * @param restaurant restaurace |  * @param restaurant restaurace | ||||||
|  * @param date datum, ke kterému získat menu |  * @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<DayMenu> { | ||||||
|     const usedDate = date ?? getToday(); |     const usedDate = date ?? getToday(); | ||||||
|     const dayOfWeekIndex = getDayOfWeekIndex(usedDate); |     const dayOfWeekIndex = getDayOfWeekIndex(usedDate); | ||||||
|     const now = new Date().getTime(); |     const now = new Date().getTime(); | ||||||
| @ -188,59 +108,70 @@ export async function getRestaurantMenu(restaurant: Restaurant, date?: Date, for | |||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let weekMenu = await getMenu(usedDate); |     let menus = await getMenu(usedDate); | ||||||
|     weekMenu ??= [{}, {}, {}, {}, {}]; |     if (menus == null) { | ||||||
|     for (let i = 0; i < 5; i++) { |         menus = []; | ||||||
|         weekMenu[i] ??= {}; |  | ||||||
|         weekMenu[i][restaurant] ??= { |  | ||||||
|             lastUpdate: now, |  | ||||||
|             closed: false, |  | ||||||
|             food: [], |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|     if (forceRefresh || (!weekMenu[dayOfWeekIndex][restaurant]?.food?.length && !weekMenu[dayOfWeekIndex][restaurant]?.closed)) { |     for (let i = 0; i < 5; i++) { | ||||||
|         const firstDay = getFirstWorkDayOfWeek(usedDate); |         if (menus[i] == null) { | ||||||
|          |             menus[i] = {}; | ||||||
|         try { |         } | ||||||
|             const restaurantWeekFood = await fetchRestaurantWeekMenu(restaurant, firstDay); |         if (menus[i][restaurant] == null) { | ||||||
|              |             menus[i][restaurant] = { | ||||||
|             // Aktualizace menu pro všechny dny
 |                 lastUpdate: now, | ||||||
|             for (let i = 0; i < restaurantWeekFood.length && i < weekMenu.length; i++) { |                 closed: false, | ||||||
|                 weekMenu[i][restaurant]!.food = restaurantWeekFood[i]; |                 food: [], | ||||||
|                 weekMenu[i][restaurant]!.lastUpdate = now; |             }; | ||||||
|                  |  | ||||||
|                 // Detekce uzavření pro každou restauraci
 |  | ||||||
|                 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; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|              |  | ||||||
|             // Uložení do storage
 |  | ||||||
|             await storage.setData(getMenuKey(usedDate), weekMenu); |  | ||||||
|         } catch (e: any) { |  | ||||||
|             console.error(`Selhalo načtení jídel pro podnik ${restaurant}`, e); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     return weekMenu[dayOfWeekIndex][restaurant]!; |     if (!menus[dayOfWeekIndex][restaurant]?.food?.length) { | ||||||
|  |         const firstDay = getFirstWorkDayOfWeek(usedDate); | ||||||
|  |         const mock = process.env.MOCK_DATA === 'true'; | ||||||
|  |         switch (restaurant) { | ||||||
|  |             case Restaurants.SLADOVNICKA: | ||||||
|  |                 try { | ||||||
|  |                     const sladovnickaFood = await getMenuSladovnicka(firstDay, mock); | ||||||
|  |                     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; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } catch (e: any) { | ||||||
|  |                     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; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     break; | ||||||
|  |                 } catch (e: any) { | ||||||
|  |                     console.error("Selhalo načtení jídel pro podnik TechTower", e); | ||||||
|  |                 } | ||||||
|  |         } | ||||||
|  |         await storage.setData(getMenuKey(usedDate), menus); | ||||||
|  |     } | ||||||
|  |     return menus[dayOfWeekIndex][restaurant]!; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @ -265,9 +196,9 @@ export async function initIfNeeded(date?: Date) { | |||||||
|  * @param date datum, ke kterému se volba vztahuje |  * @param date datum, ke kterému se volba vztahuje | ||||||
|  * @returns  |  * @returns  | ||||||
|  */ |  */ | ||||||
| export async function removeChoices(login: string, trusted: boolean, locationKey: LunchChoice, date?: Date) { | export async function removeChoices(login: string, trusted: boolean, locationKey: keyof typeof Locations, date?: Date) { | ||||||
|     const selectedDay = formatDate(date ?? getToday()); |     const selectedDay = formatDate(date ?? getToday()); | ||||||
|     let data = await getClientData(date); |     let data: DayData = await storage.getData(selectedDay); | ||||||
|     validateTrusted(data, login, trusted); |     validateTrusted(data, login, trusted); | ||||||
|     if (locationKey in data.choices) { |     if (locationKey in data.choices) { | ||||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { |         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||||
| @ -292,15 +223,15 @@ export async function removeChoices(login: string, trusted: boolean, locationKey | |||||||
|  * @param date datum, ke kterému se volba vztahuje |  * @param date datum, ke kterému se volba vztahuje | ||||||
|  * @returns  |  * @returns  | ||||||
|  */ |  */ | ||||||
| export async function removeChoice(login: string, trusted: boolean, locationKey: LunchChoice, foodIndex: number, date?: Date) { | export async function removeChoice(login: string, trusted: boolean, locationKey: keyof typeof Locations, foodIndex: number, date?: Date) { | ||||||
|     const selectedDay = formatDate(date ?? getToday()); |     const selectedDay = formatDate(date ?? getToday()); | ||||||
|     let data = await getClientData(date); |     let data: DayData = await storage.getData(selectedDay); | ||||||
|     validateTrusted(data, login, trusted); |     validateTrusted(data, login, trusted); | ||||||
|     if (locationKey in data.choices) { |     if (locationKey in data.choices) { | ||||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { |         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||||
|             const index = data.choices[locationKey][login].selectedFoods?.indexOf(foodIndex); |             const index = data.choices[locationKey][login].options.indexOf(foodIndex); | ||||||
|             if (index != null && index > -1) { |             if (index > -1) { | ||||||
|                 data.choices[locationKey][login].selectedFoods?.splice(index, 1); |                 data.choices[locationKey][login].options.splice(index, 1) | ||||||
|                 await storage.setData(selectedDay, data); |                 await storage.setData(selectedDay, data); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -309,26 +240,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 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) { | async function removeChoiceIfPresent(login: string, date: string) { | ||||||
|     const usedDate = date ?? getToday(); |     let data: DayData = await storage.getData(date); | ||||||
|     let data = await getClientData(usedDate); |  | ||||||
|     for (const key of Object.keys(data.choices)) { |     for (const key of Object.keys(data.choices)) { | ||||||
|         const locationKey = key as LunchChoice; |         const locationKey = key as keyof typeof Locations; | ||||||
|         if (ignoredLocationKey != null && ignoredLocationKey == locationKey) { |  | ||||||
|             continue; |  | ||||||
|         } |  | ||||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { |         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||||
|             delete data.choices[locationKey][login]; |             delete data.choices[locationKey][login]; | ||||||
|             if (Object.keys(data.choices[locationKey]).length === 0) { |             if (Object.keys(data.choices[locationKey]).length === 0) { | ||||||
|                 delete data.choices[locationKey]; |                 delete data.choices[locationKey]; | ||||||
|             } |             } | ||||||
|             await storage.setData(formatDate(usedDate), data); |             await storage.setData(date, data); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     return data; |     return data; | ||||||
| @ -367,64 +292,43 @@ function validateTrusted(data: ClientData, login: string, trusted: boolean) { | |||||||
|  * @param date datum, ke kterému se volba vztahuje |  * @param date datum, ke kterému se volba vztahuje | ||||||
|  * @returns aktuální data |  * @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: keyof typeof Locations, foodIndex?: number, date?: Date) { | ||||||
|     const usedDate = date ?? getToday(); |     const usedDate = date ?? getToday(); | ||||||
|     await initIfNeeded(usedDate); |     await initIfNeeded(usedDate); | ||||||
|     let data = await getClientData(usedDate); |     const selectedDate = formatDate(usedDate); | ||||||
|  |     let data: DayData = await storage.getData(selectedDate); | ||||||
|     validateTrusted(data, login, trusted); |     validateTrusted(data, login, trusted); | ||||||
|     await validateFoodIndex(locationKey, foodIndex, date); |  | ||||||
|     // Pokud měníme pouze lokaci, mažeme případné předchozí
 |     // Pokud měníme pouze lokaci, mažeme případné předchozí
 | ||||||
|     if (foodIndex == null) { |     if (foodIndex == null) { | ||||||
|         data = await removeChoiceIfPresent(login, usedDate); |         data = await removeChoiceIfPresent(login, selectedDate); | ||||||
|     } else { |  | ||||||
|         // Mažeme případné ostatní volby (měla by být maximálně jedna)
 |  | ||||||
|         removeChoiceIfPresent(login, usedDate, locationKey); |  | ||||||
|     } |     } | ||||||
|     // TODO vytáhnout inicializaci "prázdné struktury" do vlastní funkce
 |     if (!data.choices) { | ||||||
|     data.choices[locationKey] ??= {}; |         console.log("Klíč", Locations[locationKey]); // TODO smazat
 | ||||||
|     if (!(login in data.choices[locationKey])) { |         data.choices = { | ||||||
|         if (!data.choices[locationKey]) { |             [Locations[locationKey]]: {} | ||||||
|             data.choices[locationKey] = {} |  | ||||||
|         } |         } | ||||||
|         data.choices[locationKey][login] = { |  | ||||||
|             trusted, |  | ||||||
|             selectedFoods: [] |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|     if (foodIndex != null && !data.choices[locationKey][login].selectedFoods?.includes(foodIndex)) { |     console.log("Máme choices", data.choices); | ||||||
|         data.choices[locationKey][login].selectedFoods?.push(foodIndex); |     console.log("Hodnota locationKey", locationKey); | ||||||
|     } |     // if (!(data.choices[locationKey])) {
 | ||||||
|     const selectedDate = formatDate(usedDate); |     //     data?.choices[locationKey] = {}
 | ||||||
|  |     // }
 | ||||||
|  |     // if (!(login in data.choices[locationKey])) {
 | ||||||
|  |     //     if (!data.choices[locationKey]) {
 | ||||||
|  |     //         data.choices[locationKey] = {}
 | ||||||
|  |     //     }
 | ||||||
|  |     //     data.choices[locationKey][login] = {
 | ||||||
|  |     //         trusted,
 | ||||||
|  |     //         options: []
 | ||||||
|  |     //     };
 | ||||||
|  |     // }
 | ||||||
|  |     // if (foodIndex != null && !data.choices[locationKey][login].options.includes(foodIndex)) {
 | ||||||
|  |     //     data.choices[locationKey][login].options.push(foodIndex);
 | ||||||
|  |     // }
 | ||||||
|     await storage.setData(selectedDate, data); |     await storage.setData(selectedDate, data); | ||||||
|     return 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. |  * Aktualizuje poznámku k aktuálně vybrané možnosti. | ||||||
|  *  |  *  | ||||||
| @ -436,16 +340,16 @@ async function validateFoodIndex(locationKey: LunchChoice, foodIndex?: number, d | |||||||
| export async function updateNote(login: string, trusted: boolean, note?: string, date?: Date) { | export async function updateNote(login: string, trusted: boolean, note?: string, date?: Date) { | ||||||
|     const usedDate = date ?? getToday(); |     const usedDate = date ?? getToday(); | ||||||
|     await initIfNeeded(usedDate); |     await initIfNeeded(usedDate); | ||||||
|     let data = await getClientData(usedDate); |     const selectedDate = formatDate(usedDate); | ||||||
|  |     let data: DayData = await storage.getData(selectedDate); | ||||||
|     validateTrusted(data, login, trusted); |     validateTrusted(data, login, trusted); | ||||||
|     const userEntry = data.choices != null && Object.entries(data.choices).find(entry => entry[1][login] != null); |     const userEntry = data.choices != null && Object.entries(data.choices).find(entry => entry[1][login] != null); | ||||||
|     if (userEntry) { |     if (userEntry) { | ||||||
|         if (!note?.length) { |         if (!note || !note.length) { | ||||||
|             delete userEntry[1][login].note; |             delete userEntry[1][login].note; | ||||||
|         } else { |         } else { | ||||||
|             userEntry[1][login].note = note; |             userEntry[1][login].note = note; | ||||||
|         } |         } | ||||||
|         const selectedDate = formatDate(usedDate); |  | ||||||
|         await storage.setData(selectedDate, data); |         await storage.setData(selectedDate, data); | ||||||
|     } |     } | ||||||
|     return data; |     return data; | ||||||
| @ -459,8 +363,8 @@ export async function updateNote(login: string, trusted: boolean, note?: string, | |||||||
|  * @param date datum, ke kterému se čas vztahuje |  * @param date datum, ke kterému se čas vztahuje | ||||||
|  */ |  */ | ||||||
| export async function updateDepartureTime(login: string, time?: string, date?: Date) { | export async function updateDepartureTime(login: string, time?: string, date?: Date) { | ||||||
|     const usedDate = date ?? getToday(); |     const selectedDate = formatDate(date ?? getToday()); | ||||||
|     let clientData = await getClientData(usedDate); |     let clientData: DayData = await storage.getData(selectedDate); | ||||||
|     const found = Object.values(clientData.choices).find(location => login in location); |     const found = Object.values(clientData.choices).find(location => login in location); | ||||||
|     // TODO validace, že se jedná o restauraci
 |     // TODO validace, že se jedná o restauraci
 | ||||||
|     if (found) { |     if (found) { | ||||||
| @ -472,23 +376,7 @@ export async function updateDepartureTime(login: string, time?: string, date?: D | |||||||
|             } |             } | ||||||
|             found[login].departureTime = time; |             found[login].departureTime = time; | ||||||
|         } |         } | ||||||
|         await storage.setData(formatDate(usedDate), clientData); |         await storage.setData(selectedDate, clientData); | ||||||
|     } |     } | ||||||
|     return 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. |  * Interface pro úložiště dat. | ||||||
|  *  |  *  | ||||||
| @ -5,12 +7,6 @@ | |||||||
|  * Postupem času lze předělat pro efektivnější využití Redis. |  * Postupem času lze předělat pro efektivnější využití Redis. | ||||||
|  */ |  */ | ||||||
| export interface StorageInterface { | 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íč. |      * Vrátí příznak, zda existují data pro předaný klíč. | ||||||
|      * @param key klíč, pro který zjišťujeme data (typicky datum) |      * @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íč. |      * Vrátí veškerá data pro předaný klíč. | ||||||
|      * @param key klíč, pro který vrátit data (typicky datum) |      * @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íč. |      * Uloží data pod předaný klíč. | ||||||
|  | |||||||
| @ -4,8 +4,8 @@ import { StorageInterface } from "./StorageInterface"; | |||||||
| import JsonStorage from "./json"; | import JsonStorage from "./json"; | ||||||
| import RedisStorage from "./redis"; | import RedisStorage from "./redis"; | ||||||
| 
 | 
 | ||||||
| const ENVIRONMENT = process.env.NODE_ENV ?? 'production'; | const ENVIRONMENT = process.env.NODE_ENV || 'production'; | ||||||
| dotenv.config({ path: path.resolve(__dirname, `../../.env.${ENVIRONMENT}`) }); | dotenv.config({ path: path.resolve(__dirname, `../.env.${ENVIRONMENT}`) }); | ||||||
| 
 | 
 | ||||||
| const JSON_KEY = 'json'; | const JSON_KEY = 'json'; | ||||||
| const REDIS_KEY = 'redis'; | 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'"); |     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 { | export default function getStorage(): StorageInterface { | ||||||
|     return storage; |     return storage; | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,17 +1,8 @@ | |||||||
| import JSONdb from 'simple-json-db'; | import JSONdb from 'simple-json-db'; | ||||||
| import { StorageInterface } from "./StorageInterface"; | import { StorageInterface } from "./StorageInterface"; | ||||||
| import * as fs from 'fs'; |  | ||||||
| import * as path from 'path'; |  | ||||||
| 
 | 
 | ||||||
| const dbPath = path.resolve(__dirname, '../../data/db.json'); | const db = new JSONdb('./data.json'); | ||||||
| const dbDir = path.dirname(dbPath); |  | ||||||
| 
 | 
 | ||||||
| // 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. |  * Implementace úložiště používající JSON soubor. | ||||||
|  */ |  */ | ||||||
|  | |||||||
| @ -8,18 +8,15 @@ let client: RedisClientType; | |||||||
|  */ |  */ | ||||||
| export default class RedisStorage implements StorageInterface { | export default class RedisStorage implements StorageInterface { | ||||||
|     constructor() { |     constructor() { | ||||||
|         const HOST = process.env.REDIS_HOST ?? 'localhost'; |         const HOST = process.env.REDIS_HOST || 'localhost'; | ||||||
|         const PORT = process.env.REDIS_PORT ?? 6379; |         const PORT = process.env.REDIS_PORT || 6379; | ||||||
|         client = createClient({ url: `redis://${HOST}:${PORT}` }); |         client = createClient({ url: `redis://${HOST}:${PORT}` }); | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     async initialize() { |  | ||||||
|         client.connect(); |         client.connect(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async hasData(key: string) { |     async hasData(key: string) { | ||||||
|         const data = await client.json.get(key); |         const data = await client.json.get(key); | ||||||
|         return (!!data); |         return (data ? true : false); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async getData<Type>(key: string) { |     async getData<Type>(key: string) { | ||||||
|  | |||||||
| @ -1,15 +1,11 @@ | |||||||
| import { LunchChoice, LunchChoices } from "../../types/gen/types.gen"; | import { Choices, Locations } from "../../types"; | ||||||
| 
 |  | ||||||
| const DAY_OF_WEEK_FORMAT = new Intl.DateTimeFormat(undefined, { weekday: 'long' }); |  | ||||||
| 
 | 
 | ||||||
| /** Vrátí datum v ISO formátu. */ | /** Vrátí datum v ISO formátu. */ | ||||||
| export function formatDate(date: Date, format?: string) { | export function formatDate(date: Date) { | ||||||
|     let day = String(date.getDate()).padStart(2, '0'); |     let currentDay = String(date.getDate()).padStart(2, '0'); | ||||||
|     let month = String(date.getMonth() + 1).padStart(2, "0"); |     let currentMonth = String(date.getMonth() + 1).padStart(2, "0"); | ||||||
|     let year = String(date.getFullYear()); |     let currentYear = date.getFullYear(); | ||||||
| 
 |     return `${currentYear}-${currentMonth}-${currentDay}`; | ||||||
|     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í. */ | /** Vrátí human-readable reprezentaci předaného data pro zobrazení. */ | ||||||
| @ -17,7 +13,7 @@ export function getHumanDate(date: Date) { | |||||||
|     let currentDay = String(date.getDate()).padStart(2, '0'); |     let currentDay = String(date.getDate()).padStart(2, '0'); | ||||||
|     let currentMonth = String(date.getMonth() + 1).padStart(2, "0"); |     let currentMonth = String(date.getMonth() + 1).padStart(2, "0"); | ||||||
|     let currentYear = date.getFullYear(); |     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})`; |     return `${currentDay}.${currentMonth}.${currentYear} (${currentDayOfWeek})`; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -61,10 +57,10 @@ export function getLastWorkDayOfWeek(date: Date) { | |||||||
| 
 | 
 | ||||||
| /** Vrátí pořadové číslo týdne předaného data v roce dle ISO 8601. */ | /** Vrátí pořadové číslo týdne předaného data v roce dle ISO 8601. */ | ||||||
| export function getWeekNumber(inputDate: Date) { | export function getWeekNumber(inputDate: Date) { | ||||||
|     const date = new Date(inputDate.getTime()); |     var date = new Date(inputDate.getTime()); | ||||||
|     date.setHours(0, 0, 0, 0); |     date.setHours(0, 0, 0, 0); | ||||||
|     date.setDate(date.getDate() + 3 - (date.getDay() + 6) % 7); |     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); |     return 1 + Math.round(((date.getTime() - week1.getTime()) / 86400000 - 3 + (week1.getDay() + 6) % 7) / 7); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -114,20 +110,22 @@ export const checkBodyParams = (req: any, paramNames: string[]) => { | |||||||
| // TODO umístit do samostatného souboru
 | // TODO umístit do samostatného souboru
 | ||||||
| export class InsufficientPermissions extends Error { } | export class InsufficientPermissions extends Error { } | ||||||
| 
 | 
 | ||||||
| export const getUsersByLocation = (choices: LunchChoices, login?: string): string[] => { | export const getUsersByLocation = (choices: Choices, login: string): string[] => { | ||||||
|     const result: string[] = []; |     const result: string[] = []; | ||||||
| 
 | 
 | ||||||
|     for (const location of Object.entries(choices)) { |     for (const location of Object.entries(choices)) { | ||||||
|         const locationKey = location[0] as LunchChoice; |         const locationKey = location[0]; | ||||||
|         const locationValue = location[1]; |         const locationValue = location[1]; | ||||||
|         if (login && locationValue[login]) { |         console.log("locationKey", locationKey); | ||||||
|             for (const username in choices[locationKey]) { |         console.log("locationValue", locationValue); | ||||||
|                 if (choices[locationKey].hasOwnProperty(username)) { |         // if (locationValues[login]) {
 | ||||||
|                     result.push(username); |         //     for (const username in choices[locationKey]) {
 | ||||||
|                 } |         //         if (choices[locationKey].hasOwnProperty(username)) {
 | ||||||
|             } |         //             result.push(username);
 | ||||||
|             break; |         //         }
 | ||||||
|         } |         //     }
 | ||||||
|  |         //     break;
 | ||||||
|  |         // }
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return result; |     return result; | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| import { FeatureRequest } from "../../types/gen/types.gen"; | import { FeatureRequest } from "../../types"; | ||||||
| import getStorage from "./storage"; | import getStorage from "./storage"; | ||||||
| 
 | 
 | ||||||
| interface VotingData { | interface VotingData { | ||||||
| @ -15,7 +15,7 @@ const STORAGE_KEY = 'voting'; | |||||||
|  * @returns pole voleb |  * @returns pole voleb | ||||||
|  */ |  */ | ||||||
| export async function getUserVotes(login: string) { | 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] || []; |     return data?.[login] || []; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -28,8 +28,10 @@ export async function getUserVotes(login: string) { | |||||||
|  * @returns aktuální data |  * @returns aktuální data | ||||||
|  */ |  */ | ||||||
| export async function updateFeatureVote(login: string, option: FeatureRequest, active: boolean): Promise<VotingData> { | export async function updateFeatureVote(login: string, option: FeatureRequest, active: boolean): Promise<VotingData> { | ||||||
|     let data = await storage.getData<VotingData>(STORAGE_KEY); |     let data: VotingData = await storage.getData(STORAGE_KEY); | ||||||
|     data ??= {}; |     if (data == null) { | ||||||
|  |         data = {}; | ||||||
|  |     } | ||||||
|     if (!(login in data)) { |     if (!(login in data)) { | ||||||
|         data[login] = []; |         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>; | let io: Server<DefaultEventsMap, DefaultEventsMap, DefaultEventsMap, any>; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4,11 +4,10 @@ | |||||||
|         "../types/**/*" |         "../types/**/*" | ||||||
|     ], |     ], | ||||||
|     "compilerOptions": { |     "compilerOptions": { | ||||||
|         "target": "ES2022", |         "target": "ES2016", | ||||||
|         "module": "Node16", |         "module": "CommonJS", | ||||||
|         "moduleResolution": "node16", |         "jsx": "react", | ||||||
|         "esModuleInterop": true, |         "esModuleInterop": true, | ||||||
|         "skipLibCheck": true, |  | ||||||
|         "forceConsistentCasingInFileNames": true, |         "forceConsistentCasingInFileNames": true, | ||||||
|         "outDir": "./dist", |         "outDir": "./dist", | ||||||
|         "rootDir": "../", |         "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, Locations, PizzaOrder } from "./Types"; | ||||||
|  | 
 | ||||||
|  | export type ILocationKey = { | ||||||
|  |     locationKey: keyof typeof Locations, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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, | ||||||
|  | } | ||||||
							
								
								
									
										214
									
								
								types/Types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										214
									
								
								types/Types.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,214 @@ | |||||||
|  | /** 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', | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export type FoodChoices = { | ||||||
|  |     trusted: boolean, | ||||||
|  |     options: number[], | ||||||
|  |     departureTime?: string, | ||||||
|  |     note?: string, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export type Choices = { | ||||||
|  |     [location in keyof typeof Locations]?: { | ||||||
|  |         [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í
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Týdenní menu jednotlivých restaurací. */ | ||||||
|  | export type WeekMenu = { | ||||||
|  |     [dayIndex: number]: { | ||||||
|  |         [restaurant in Restaurants]?: DayMenu | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 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: number, // index dne v týdnu (0-6)
 | ||||||
|  |     choices: Choices, // seznam voleb uživatelů
 | ||||||
|  |     menus?: { [restaurant in Restaurants]?: DayMenu }, // 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?: number, // index dnešního dne v týdnu (0-6)
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Nabídka jídel jednoho podniku pro jeden konkrétní den. */ | ||||||
|  | export type DayMenu = { | ||||||
|  |     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', | ||||||
|  |     SPSE = 'SPŠE', | ||||||
|  |     PIZZA = 'Pizza day', | ||||||
|  |     OBJEDNAVAM = 'Budu objednávat', | ||||||
|  |     NEOBEDVAM = 'Mám vlastní/neobědvám', | ||||||
|  |     ROZHODUJI = 'Rozhoduji se', | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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