Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
0179afca75
|
@@ -1,261 +0,0 @@
|
|||||||
name: CI
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- "**"
|
|
||||||
pull_request:
|
|
||||||
|
|
||||||
concurrency:
|
|
||||||
group: ${{ github.ref }}
|
|
||||||
cancel-in-progress: true
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
|
|
||||||
# ─── 1. Generate OpenAPI types ────────────────────────────────────────────
|
|
||||||
|
|
||||||
generate-types:
|
|
||||||
name: Generate TypeScript types
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- run: corepack enable
|
|
||||||
|
|
||||||
- run: cd types && yarn install --frozen-lockfile && yarn openapi-ts
|
|
||||||
|
|
||||||
- uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: types-gen
|
|
||||||
path: types/gen
|
|
||||||
|
|
||||||
# ─── 2a. Server unit tests ────────────────────────────────────────────────
|
|
||||||
|
|
||||||
server-test:
|
|
||||||
name: Server unit tests
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: generate-types
|
|
||||||
env:
|
|
||||||
NODE_ENV: test
|
|
||||||
JWT_SECRET: test-secret-min-32-chars-aaaaaaa!
|
|
||||||
MOCK_DATA: "true"
|
|
||||||
STORAGE: json
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- run: corepack enable
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: types-gen
|
|
||||||
path: types/gen
|
|
||||||
|
|
||||||
- run: cd server && yarn install --frozen-lockfile && yarn test
|
|
||||||
|
|
||||||
# ─── 2b. Build server ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
server-build:
|
|
||||||
name: Build server
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: generate-types
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- run: corepack enable
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: types-gen
|
|
||||||
path: types/gen
|
|
||||||
|
|
||||||
- run: cd types && yarn install --frozen-lockfile
|
|
||||||
|
|
||||||
- run: cd server && yarn install --frozen-lockfile && yarn build
|
|
||||||
|
|
||||||
- uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: server-dist
|
|
||||||
path: server/dist
|
|
||||||
|
|
||||||
# ─── 2c. Build client ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
client-build:
|
|
||||||
name: Build client
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: generate-types
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- run: corepack enable
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: types-gen
|
|
||||||
path: types/gen
|
|
||||||
|
|
||||||
- run: cd types && yarn install --frozen-lockfile
|
|
||||||
|
|
||||||
- run: cd client && yarn install --frozen-lockfile && yarn build
|
|
||||||
|
|
||||||
- uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: client-dist
|
|
||||||
path: client/dist
|
|
||||||
|
|
||||||
# ─── 3. Playwright E2E tests ──────────────────────────────────────────────
|
|
||||||
|
|
||||||
e2e:
|
|
||||||
name: Playwright E2E tests
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: [ server-build, client-build ]
|
|
||||||
container: mcr.microsoft.com/playwright:v1.59.1-jammy
|
|
||||||
services:
|
|
||||||
redis:
|
|
||||||
image: redis/redis-stack-server:7.4.0-v1
|
|
||||||
env:
|
|
||||||
REDIS_ARGS: "--save '' --loglevel warning"
|
|
||||||
env:
|
|
||||||
CI: "true"
|
|
||||||
NODE_ENV: test
|
|
||||||
JWT_SECRET: test-secret-min-32-chars-aaaaaaa!
|
|
||||||
MOCK_DATA: "true"
|
|
||||||
STORAGE: redis
|
|
||||||
REDIS_HOST: redis
|
|
||||||
REDIS_PORT: "6379"
|
|
||||||
HTTP_REMOTE_USER_ENABLED: "true"
|
|
||||||
HTTP_REMOTE_USER_HEADER_NAME: remote-user
|
|
||||||
HTTP_REMOTE_TRUSTED_IPS: "127.0.0.1,::1,::ffff:127.0.0.1"
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: server-dist
|
|
||||||
path: server/dist
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: client-dist
|
|
||||||
path: client/dist
|
|
||||||
|
|
||||||
- name: Install server dependencies
|
|
||||||
run: cd server && yarn install --frozen-lockfile
|
|
||||||
|
|
||||||
- name: Copy client build into server/public
|
|
||||||
run: cp -r client/dist server/public
|
|
||||||
|
|
||||||
- name: Install e2e dependencies and browsers
|
|
||||||
run: cd e2e && yarn install --frozen-lockfile && yarn playwright install firefox
|
|
||||||
--with-deps
|
|
||||||
|
|
||||||
- name: Run Playwright tests
|
|
||||||
run: cd e2e && yarn test
|
|
||||||
|
|
||||||
- uses: actions/upload-artifact@v3
|
|
||||||
if: failure()
|
|
||||||
with:
|
|
||||||
name: playwright-report
|
|
||||||
path: |
|
|
||||||
e2e/playwright-report
|
|
||||||
e2e/test-results
|
|
||||||
|
|
||||||
# ─── 4. Build and push Docker image (master only) ─────────────────────────
|
|
||||||
|
|
||||||
docker-build:
|
|
||||||
name: Build and push Docker image
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: [ server-build, client-build, server-test, e2e ]
|
|
||||||
if: github.event_name == 'push' && github.ref == 'refs/heads/master'
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- run: corepack enable
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: server-dist
|
|
||||||
path: server/dist
|
|
||||||
|
|
||||||
- uses: actions/download-artifact@v3
|
|
||||||
with:
|
|
||||||
name: client-dist
|
|
||||||
path: client/dist
|
|
||||||
|
|
||||||
- name: Install server production dependencies
|
|
||||||
run: cd server && yarn install --frozen-lockfile --production
|
|
||||||
|
|
||||||
- uses: docker/setup-buildx-action@v3
|
|
||||||
|
|
||||||
- uses: docker/login-action@v3
|
|
||||||
with:
|
|
||||||
registry: ${{ secrets.REPO_URL }}
|
|
||||||
username: ${{ secrets.REPO_USERNAME }}
|
|
||||||
password: ${{ secrets.REPO_PASSWORD }}
|
|
||||||
|
|
||||||
- uses: docker/build-push-action@v5
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
file: Dockerfile
|
|
||||||
target: runner-prebuilt
|
|
||||||
platforms: linux/amd64
|
|
||||||
push: true
|
|
||||||
tags: ${{ secrets.REPO_URL }}/${{ secrets.REPO_NAME }}:latest
|
|
||||||
|
|
||||||
# ─── 5. Notifications ────────────────────────
|
|
||||||
|
|
||||||
notify:
|
|
||||||
name: Notify
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: [ server-build, client-build, server-test, e2e, docker-build ]
|
|
||||||
if: always() && github.event_name == 'push'
|
|
||||||
steps:
|
|
||||||
- name: Send webhook
|
|
||||||
env:
|
|
||||||
DISCORD_WEBHOOK_ID: ${{ secrets.DISCORD_WEBHOOK_ID }}
|
|
||||||
DISCORD_WEBHOOK_TOKEN: ${{ secrets.DISCORD_WEBHOOK_TOKEN }}
|
|
||||||
NTFY_URL: ${{ secrets.NTFY_URL }}
|
|
||||||
BUILD_RESULT: ${{ needs.docker-build.result }}
|
|
||||||
RUN_NUMBER: ${{ github.run_number }}
|
|
||||||
RUN_URL: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{
|
|
||||||
github.run_id }}
|
|
||||||
COMMIT_MESSAGE: ${{ github.event.head_commit.message }}
|
|
||||||
COMMIT_AUTHOR: ${{ github.event.head_commit.author.name }}
|
|
||||||
run: |
|
|
||||||
if [ "$BUILD_RESULT" = "success" ]; then
|
|
||||||
MSG="✅ Sestavení #${RUN_NUMBER} proběhlo úspěšně."
|
|
||||||
NTFY_TAGS="white_check_mark"
|
|
||||||
else
|
|
||||||
MSG="❌ Sestavení #${RUN_NUMBER} selhalo."
|
|
||||||
NTFY_TAGS="x"
|
|
||||||
fi
|
|
||||||
FULL_MSG="$(printf '%s\n\nPipeline: %s\nPoslední commit: %sAutor: %s' \
|
|
||||||
"$MSG" "$RUN_URL" "$COMMIT_MESSAGE" "$COMMIT_AUTHOR")"
|
|
||||||
curl -s -X POST \
|
|
||||||
"https://discord.com/api/webhooks/${DISCORD_WEBHOOK_ID}/${DISCORD_WEBHOOK_TOKEN}" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
--data "$(jq -n --arg content "$FULL_MSG" '{content: $content}')"
|
|
||||||
curl -s -X POST "${NTFY_URL}" \
|
|
||||||
-H "Title: Luncher CI #${RUN_NUMBER}" \
|
|
||||||
-H "Tags: ${NTFY_TAGS}" \
|
|
||||||
-H "Click: ${RUN_URL}" \
|
|
||||||
-d "${FULL_MSG}"
|
|
||||||
@@ -1,6 +1,2 @@
|
|||||||
node_modules
|
node_modules
|
||||||
types/gen
|
types/gen
|
||||||
**.DS_Store
|
|
||||||
.mcp.json
|
|
||||||
.claude/settings.local.json
|
|
||||||
server/public/
|
|
||||||
|
|||||||
Vendored
-67
@@ -1,67 +0,0 @@
|
|||||||
{
|
|
||||||
"version": "2.0.0",
|
|
||||||
"tasks": [
|
|
||||||
{
|
|
||||||
"label": "types: openapi-ts",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "yarn openapi-ts",
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/types"
|
|
||||||
},
|
|
||||||
"presentation": {
|
|
||||||
"reveal": "silent",
|
|
||||||
"panel": "dedicated"
|
|
||||||
},
|
|
||||||
"problemMatcher": []
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "server: startReload",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "yarn startReload",
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/server",
|
|
||||||
"env": {
|
|
||||||
"NODE_ENV": "development"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"isBackground": true,
|
|
||||||
"presentation": {
|
|
||||||
"reveal": "always",
|
|
||||||
"panel": "dedicated",
|
|
||||||
"group": "dev"
|
|
||||||
},
|
|
||||||
"problemMatcher": []
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "client: vite",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "yarn start",
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/client"
|
|
||||||
},
|
|
||||||
"isBackground": true,
|
|
||||||
"presentation": {
|
|
||||||
"reveal": "always",
|
|
||||||
"panel": "dedicated",
|
|
||||||
"group": "dev"
|
|
||||||
},
|
|
||||||
"problemMatcher": []
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "dev: server+client",
|
|
||||||
"dependsOn": [
|
|
||||||
"server: startReload",
|
|
||||||
"client: vite"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "dev: all",
|
|
||||||
"dependsOrder": "sequence",
|
|
||||||
"dependsOn": [
|
|
||||||
"types: openapi-ts",
|
|
||||||
"dev: server+client"
|
|
||||||
],
|
|
||||||
"problemMatcher": []
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
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}}"
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
# CLAUDE.md
|
|
||||||
|
|
||||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
|
||||||
|
|
||||||
## Project Overview
|
|
||||||
|
|
||||||
Luncher is a lunch management app for teams — daily restaurant menus, food ordering, pizza day events, and payment QR codes. Czech-language UI. Full-stack TypeScript monorepo.
|
|
||||||
|
|
||||||
## Monorepo Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
types/ → Shared OpenAPI-generated TypeScript types (source of truth: types/api.yml)
|
|
||||||
server/ → Express 5 backend (Node.js 22, ts-node)
|
|
||||||
client/ → React 19 frontend (Vite 7, React Bootstrap)
|
|
||||||
e2e/ → Playwright E2E tests (separate package)
|
|
||||||
```
|
|
||||||
|
|
||||||
Each of the four directories has its own `package.json`. Package manager: **Yarn Classic**.
|
|
||||||
|
|
||||||
Deployment files at repo root: `Dockerfile` (multi-stage, dva runner targety: výchozí `runner` pro lokální build, `runner-prebuilt` pro CI s předem sestavenými artefakty), `compose.yml`, `compose-traefik.yml`.
|
|
||||||
|
|
||||||
## Development Commands
|
|
||||||
|
|
||||||
### Initial setup
|
|
||||||
```bash
|
|
||||||
cd types && yarn install && yarn openapi-ts # Generate API types first
|
|
||||||
cd ../server && yarn install
|
|
||||||
cd ../client && yarn install
|
|
||||||
cd ../e2e && yarn install
|
|
||||||
```
|
|
||||||
|
|
||||||
### Running dev environment
|
|
||||||
```bash
|
|
||||||
# All-in-one (tmux):
|
|
||||||
./run_dev.sh
|
|
||||||
|
|
||||||
# Or manually in separate terminals:
|
|
||||||
cd server && NODE_ENV=development yarn startReload # Port 3001, nodemon watch
|
|
||||||
cd client && yarn start # Port 3000, proxies /api → 3001
|
|
||||||
```
|
|
||||||
|
|
||||||
### Building
|
|
||||||
```bash
|
|
||||||
cd types && yarn openapi-ts # Regenerate types from api.yml
|
|
||||||
cd server && yarn build # tsc → server/dist
|
|
||||||
cd client && yarn build # tsc --noEmit + vite build → client/dist
|
|
||||||
```
|
|
||||||
|
|
||||||
### Tests
|
|
||||||
```bash
|
|
||||||
# Server unit tests (Jest)
|
|
||||||
cd server && yarn test # All tests in server/src/tests/
|
|
||||||
cd server && yarn test dates # Run one file by name
|
|
||||||
cd server && yarn test -t "name" # Run by test name pattern
|
|
||||||
|
|
||||||
# E2E (Playwright) — requires prebuilt server
|
|
||||||
cd server && yarn build
|
|
||||||
cd e2e && yarn test # chromium + firefox, baseURL 127.0.0.1:3001
|
|
||||||
cd e2e && yarn test:ui # interactive UI mode
|
|
||||||
cd e2e && yarn report # open last HTML report
|
|
||||||
```
|
|
||||||
|
|
||||||
Jest setup (`server/src/tests/setupEnv.ts`) forces `STORAGE=memory`, deletes `MOCK_DATA`, and sets a fixed `JWT_SECRET`. Playwright auto-starts the prebuilt server and authenticates via the `remote-user: e2e-user` trusted-header path; locally uses `STORAGE=json` + `MOCK_DATA=true`, CI uses `STORAGE=redis`.
|
|
||||||
|
|
||||||
### CI pipeline
|
|
||||||
Gitea Actions — `.gitea/workflows/ci.yaml` (no `.github/` equivalent):
|
|
||||||
|
|
||||||
1. `generate-types` — runs `yarn openapi-ts`, uploads artifact
|
|
||||||
2. `server-test` — Jest
|
|
||||||
3. `server-build` + `client-build` — parallel tsc/vite builds
|
|
||||||
4. `e2e` — Playwright in `mcr.microsoft.com/playwright:v1.59.1-jammy` with a Redis service container; only Firefox installed in CI
|
|
||||||
5. `docker-build` — master branch only, uses `Dockerfile` with `--target runner-prebuilt` (skládá image z artefaktů `server-build` + `client-build`)
|
|
||||||
6. `notify` — Discord + ntfy webhooks
|
|
||||||
|
|
||||||
### Formatting
|
|
||||||
Prettier is installed in `client/` (devDependency only, no script or config) — invoke via `yarn prettier <path>` with defaults.
|
|
||||||
|
|
||||||
## Architecture
|
|
||||||
|
|
||||||
### API Types (types/)
|
|
||||||
- OpenAPI 3.0 spec in `types/api.yml` — all API endpoints and DTOs defined here
|
|
||||||
- `api.yml` is a thin aggregator — actual endpoint specs live in `types/paths/<domain>/*.yml`, shared schemas in `types/schemas/_index.yml`
|
|
||||||
- `yarn openapi-ts` generates `types/gen/` (client.gen.ts, sdk.gen.ts, types.gen.ts)
|
|
||||||
- Both server and client import from these generated types
|
|
||||||
- **When changing API contracts: update api.yml first, then regenerate**
|
|
||||||
|
|
||||||
### Server (server/src/)
|
|
||||||
- **Entry:** `index.ts` — Express app + Socket.io setup
|
|
||||||
- **Routes:** `routes/` — 9 modular Express route handlers (food, pizzaDay, voting, notifications, qr, stats, easterEgg, dev, changelog)
|
|
||||||
- **Services:** domain logic files at root level (`service.ts`, `pizza.ts`, `restaurants.ts`, `chefie.ts`, `voting.ts`, `stats.ts`, `qr.ts`, `notifikace.ts`)
|
|
||||||
- **Helpers:** `mock.ts` (fake menu data for `MOCK_DATA=true`), `pushReminder.ts` (push notification reminders), `utils.ts` (shared utilities)
|
|
||||||
- **Auth:** `auth.ts` — JWT + optional trusted-header authentication
|
|
||||||
- **Storage:** `storage/index.ts` factory selects implementation; backends: `json.ts` (file-based, dev), `redis.ts` (production), `memory.ts` (tests). Data keyed by date (YYYY-MM-DD).
|
|
||||||
- **Restaurant scrapers:** Cheerio-based HTML parsing for daily menus from multiple restaurants
|
|
||||||
- **Pizza Chefie integration:** `chefie.ts` scrapes pizzachefie.cz for Pizza day menus and salads (only active when a Pizza day is open)
|
|
||||||
- **WebSocket:** `websocket.ts` — Socket.io for real-time client updates
|
|
||||||
- **Mock mode:** `MOCK_DATA=true` env var returns fake menus (useful for weekend/holiday dev)
|
|
||||||
- **Config:** `.env.development` / `.env.production` (see `.env.template` for all options)
|
|
||||||
|
|
||||||
### Client (client/src/)
|
|
||||||
- **Entry:** `index.tsx` → `App.tsx` → `AppRoutes.tsx`; `Login.tsx` is the auth screen; `FallingLeaves.tsx` is a seasonal visual effect
|
|
||||||
- **Pages:** `pages/` (StatsPage)
|
|
||||||
- **Components:** `components/` (Header, Footer, Loader, PizzaOrderList, PizzaOrderRow, and modals in `components/modals/`)
|
|
||||||
- **Context providers:** `context/` — `auth.tsx`, `settings.tsx`, `socket.js`, `eggs.tsx` (note: `socket.js` is the only non-TSX context file)
|
|
||||||
- **Hooks:** `hooks/` (`usePushReminder.ts`)
|
|
||||||
- **Utils:** `utils/` (`parsePrice.ts`)
|
|
||||||
- **Styling:** Bootstrap 5 + React Bootstrap + custom SCSS files (co-located with components)
|
|
||||||
- **API calls:** use OpenAPI-generated SDK from `types/gen/`
|
|
||||||
- **Routing:** React Router DOM v7
|
|
||||||
|
|
||||||
### Data Flow
|
|
||||||
1. Client calls API via generated SDK → Express routes
|
|
||||||
2. Server scrapes restaurant websites or returns cached data
|
|
||||||
3. Storage: Redis (production) or JSON file (development)
|
|
||||||
4. Socket.io broadcasts changes to all connected clients
|
|
||||||
|
|
||||||
## Environment
|
|
||||||
|
|
||||||
- **Server env files:** `server/.env.development`, `server/.env.production` (see `server/.env.template`)
|
|
||||||
- Key vars: `JWT_SECRET`, `STORAGE` (json/redis/memory), `MOCK_DATA`, `REDIS_HOST`, `REDIS_PORT`
|
|
||||||
- **Docker:** multi-stage Dockerfile, `compose.yml` for app + Redis. Timezone: Europe/Prague.
|
|
||||||
|
|
||||||
## Conventions
|
|
||||||
|
|
||||||
- Czech naming for domain variables and UI strings; English for infrastructure code
|
|
||||||
- TypeScript strict mode in both client and server
|
|
||||||
- Server module resolution: Node16; Client: ESNext/bundler
|
|
||||||
- `TODO.md` tracks open bugs and roadmap items — worth scanning before starting non-trivial work
|
|
||||||
+8
-36
@@ -1,6 +1,6 @@
|
|||||||
ARG NODE_VERSION="node:22-alpine"
|
ARG NODE_VERSION="node:22-alpine"
|
||||||
|
|
||||||
# ─── Builder ──────────────────────────────────────────────────────────────────
|
# Builder
|
||||||
FROM ${NODE_VERSION} AS builder
|
FROM ${NODE_VERSION} AS builder
|
||||||
|
|
||||||
WORKDIR /build
|
WORKDIR /build
|
||||||
@@ -62,9 +62,8 @@ RUN yarn build
|
|||||||
WORKDIR /build/client
|
WORKDIR /build/client
|
||||||
RUN yarn build
|
RUN yarn build
|
||||||
|
|
||||||
# ─── Runner base ──────────────────────────────────────────────────────────────
|
# Runner
|
||||||
# Společný základ pro oba runner targety – nastaví prostředí a metadata běhu.
|
FROM ${NODE_VERSION}
|
||||||
FROM ${NODE_VERSION} AS runner-base
|
|
||||||
|
|
||||||
RUN apk add --no-cache tzdata
|
RUN apk add --no-cache tzdata
|
||||||
ENV TZ=Europe/Prague \
|
ENV TZ=Europe/Prague \
|
||||||
@@ -73,17 +72,6 @@ ENV TZ=Europe/Prague \
|
|||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
# Export /data/db.json do složky /data
|
|
||||||
VOLUME ["/data"]
|
|
||||||
|
|
||||||
EXPOSE 3000
|
|
||||||
|
|
||||||
CMD [ "node", "./server/src/index.js" ]
|
|
||||||
|
|
||||||
# ─── Runner (default) ─────────────────────────────────────────────────────────
|
|
||||||
# Použití: docker build . (lokální sestavení – vše se buildí uvnitř image)
|
|
||||||
FROM runner-base AS runner
|
|
||||||
|
|
||||||
# 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 ./
|
||||||
@@ -94,28 +82,12 @@ 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
|
||||||
|
|
||||||
# Zkopírování changelogů (seznamu novinek)
|
|
||||||
COPY /server/changelogs ./server/changelogs
|
|
||||||
|
|
||||||
# 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
|
RUN if [ -f /server/.easter-eggs.json ]; then cp /server/.easter-eggs.json ./server/; fi
|
||||||
|
|
||||||
# ─── Runner (prebuilt) ────────────────────────────────────────────────────────
|
# Export /data/db.json do složky /data
|
||||||
# Použití: docker build --target runner-prebuilt .
|
VOLUME ["/data"]
|
||||||
# Očekává předem sestavené artefakty v build kontextu (server/dist,
|
|
||||||
# client/dist, server/node_modules) – využívá Gitea Actions, kde se
|
|
||||||
# server i klient buildí v separátních jobech a sem se jen kopírují.
|
|
||||||
FROM runner-base AS runner-prebuilt
|
|
||||||
|
|
||||||
# Vykopírování sestaveného serveru
|
EXPOSE 3000
|
||||||
COPY ./server/node_modules ./server/node_modules
|
|
||||||
COPY ./server/dist ./
|
|
||||||
|
|
||||||
# Vykopírování sestaveného klienta
|
CMD [ "node", "./server/src/index.js" ]
|
||||||
COPY ./client/dist ./public
|
|
||||||
|
|
||||||
# Zkopírování changelogů (seznamu novinek)
|
|
||||||
COPY ./server/changelogs ./server/changelogs
|
|
||||||
|
|
||||||
# Zkopírování konfigurace easter eggů
|
|
||||||
RUN if [ -f ./server/.easter-eggs.json ]; then cp ./server/.easter-eggs.json ./server/; fi
|
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
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" ]
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
# TODO
|
||||||
|
- [ ] HTTP_REMOTE_TRUSTED_IPS se nikde nevalidují, hlavičky jsou přijímány odkudkoli
|
||||||
|
- [ ] V případě zapnutí přihlašování přes trusted headers nefunguje standardní přihlášení (nevrátí žádnou odpověď)
|
||||||
|
- [ ] Nemělo by se jít dostat na přihlašovací formulář (měla by tam být nanejvýš hláška nebo přesměrování)
|
||||||
|
- [ ] Možnost úhrady celé útraty jednou osobou
|
||||||
|
- Základní myšlenka: jedna osoba uhradí celou útratu (v zájmu rychlosti odbavení), ostatním se automaticky vygeneruje QR kód, kterým následně uhradí svoji část útraty
|
||||||
|
- Obecně to bude problém např. pokud si někdo objedná něco navíc (pití apod.)
|
||||||
|
- [ ] Tlačítko "Uhradit" u každého řádku podniku - platí ten, kdo kliknul
|
||||||
|
- [ ] Zobrazeno bude pouze, pokud má daný uživatel nastaveno číslo účtu
|
||||||
|
- [ ] Dialog pro zadání spropitného, které se následně rozpočte rovnoměrně všem strávníkům
|
||||||
|
- [ ] Generování a zobrazení QR kódů ostatním strávníkům
|
||||||
|
- [ ] Umožnit u každého strávníka připočíst vlastní částku (např. za pití)
|
||||||
|
- [ ] Umožnit (např. zaškrtávátky) vybrat, za koho bude zaplaceno (pokud někdo bude platit zvlášť)
|
||||||
|
- [ ] Podpora pro notifikace v externích systémech (Gotify, Discord, MS Teams)
|
||||||
|
- [ ] Umožnit zadat URL/tokeny uživatelem
|
||||||
|
- [ ] Umožnit uživatelsky konfigurovat typy notifikací, které se budou odesílat
|
||||||
|
- [ ] Zavést notifikace typu "Jdeme na oběd"
|
||||||
|
- [ ] Notifikaci dostanou pouze uživatelé, kteří mají vybranou stejnou lokalitu
|
||||||
|
- [ ] Vylepšit parsery restaurací
|
||||||
|
- [ ] Sladovnická
|
||||||
|
- [ ] Zbytečná prvotní validace indexu, datum konkrétního dne je i v samotné tabulce s jídly, viz TODO v parseru
|
||||||
|
- [ ] U Motlíků
|
||||||
|
- [ ] Validovat, že vstupní datum je zahrnuto v rozsahu uvedeném nad tabulkou (např. '12.6.-16.6.')
|
||||||
|
- [ ] Jídelní lístek se stahuje jednou každý den, teoreticky by stačilo jednou týdně (za předpokladu, že se během týdne nemění)
|
||||||
|
- [ ] TechTower
|
||||||
|
- [ ] Validovat, že vstupní datum je zahrnuto v rozsahu uvedeném nad tabulkou (typicky 'Obědy 12. 6. - 16. 6. 2023 (každý den vždy i obědový bufet)')
|
||||||
|
- [ ] Jídelní lístek se stahuje v rámci prvního požadavku daný den, ale často se jídelní lístek na stránkách aktualizuje až v průběhu pondělního dopoledne a ten zobrazený je proto neaktuální
|
||||||
|
- Stránka neposílá hlavičku o času poslední modifikace, takže o to se nelze opřít
|
||||||
|
- Nevím aktuálně jak řešit jinak, než častějším scrapováním celé stránky
|
||||||
|
- [X] Někdy jsou v názvech jídel přebytečné mezery kolem čárek ( , )
|
||||||
|
- [ ] Nasazení nové verze v Docker smaže veškerá data (protože data.json není vystrčený ven z kontejneru)
|
||||||
|
- [ ] Zavést složku /data
|
||||||
|
- [ ] Mazat z databáze data z minulosti, aktuálně je to k ničemu
|
||||||
|
- [ ] Skripty pro snadné spuštění vývoje na Windows (ekvivalent ./run_dev.sh)
|
||||||
|
- [ ] Implementovat Pizza day
|
||||||
|
- [ ] Zobrazit upozornění před smazáním/zamknutím/odemknutím pizza day
|
||||||
|
- [ ] Pizzy se samy budou při naklikání přidávat do košíku
|
||||||
|
- [ ] Nutno nejprve vyřešit předávání PHPSESSIONID cookie na pizzachefie.cz pomocí fetch()
|
||||||
|
- [ ] Ceny krabic za pizzu jsou napevno v kódu - problém, pokud se někdy změní
|
||||||
|
- [X] Umožnit u Pizza day ručně připočíst cenu za přísady
|
||||||
|
- [X] Prvotní načtení pizz při založení Pizza Day trvá a nic se během toho nezobrazuje (např. loader)
|
||||||
|
- [X] Po doručení zobrazit komu zaplatit (kdo objednával)
|
||||||
|
- [x] Zbytečně nescrapovat každý den pizzy z Pizza Chefie, dokud není založen Pizza Day
|
||||||
|
- [x] Umožnit uzamčení objednávek zakladatelem
|
||||||
|
- [x] Možnost uložení čísla účtu
|
||||||
|
- [x] Automatické generování a zobrazení QR kódů
|
||||||
|
- [x] https://qr-platba.cz/pro-vyvojare/restful-api/
|
||||||
|
- [x] Zobrazovat celkovou cenu objednávky pod tabulkou objednávek
|
||||||
|
- [x] Umožnit přidat k objednávce poznámku (např. "bez oliv")
|
||||||
|
- [x] Negenerovat QR kód pro objednávajícího
|
||||||
|
- [X] Možnost náhledu na ostatní dny v týdnu (např. pomocí šipek)
|
||||||
|
- [X] Možnost výběru oběda na následující dny v týdnu
|
||||||
|
- [X] Umožnit vybrat libovolný čas odchodu
|
||||||
|
- [X] Validace zadání smysluplného času (ideálně i klientská)
|
||||||
|
- [x] Umožnit smazání aktuální volby "popelnicí", místo nutnosti vybrat prázdnou položku v selectu
|
||||||
|
- [x] Přívětivější možnost odhlašování
|
||||||
|
- [x] Vyřešit responzivní design pro použití na mobilu
|
||||||
|
- [x] Vyndat URL na Food API do .env
|
||||||
|
- [x] Neselhat při nedostupnosti nebo chybě z Food API
|
||||||
|
- [x] Dokončit docker-compose pro kompletní funkčnost
|
||||||
|
- [x] Vylepšit dokumentaci projektu
|
||||||
|
- [x] Popsat závislosti, co je nutné provést před vývojem a postup spuštění pro vývoj
|
||||||
|
- [x] Popsat dostupné env
|
||||||
|
- [x] Přesunout autentizaci na server (JWT?)
|
||||||
|
- [x] Zavést .env.template a přidat .env do .gitignore
|
||||||
|
- [x] Zkrášlit dialog pro vyplnění čísla účtu, vypadá mizerně
|
||||||
|
- [x] Zbavit se Food API, potřebnou funkcionalitu zahrnout do serveru
|
||||||
|
- [x] Vyřešit API mezi serverem a klientem, aby nebyl v obou projektech duplicitní kód (viz types.ts a Types.tsx)
|
||||||
|
- [X] Vybraná jídla strávníků zobrazovat v samostatném sloupci
|
||||||
|
- [X] Umožnit výběr/zadání preferovaného času odchodu na oběd
|
||||||
|
- Hodí se např. pokud má někdo schůzky
|
||||||
|
- [X] Ukládat dostupné pizzy do DB místo souborů
|
||||||
|
- [X] Ukládat jídla do DB místo souborů
|
||||||
@@ -10,26 +10,6 @@
|
|||||||
<link rel="apple-touch-icon" href="/logo192.png" />
|
<link rel="apple-touch-icon" href="/logo192.png" />
|
||||||
<link rel="manifest" href="/manifest.json" />
|
<link rel="manifest" href="/manifest.json" />
|
||||||
<title>Luncher</title>
|
<title>Luncher</title>
|
||||||
<script>
|
|
||||||
(function() {
|
|
||||||
try {
|
|
||||||
var saved = localStorage.getItem('theme_preference');
|
|
||||||
var theme;
|
|
||||||
if (saved === 'dark') {
|
|
||||||
theme = 'dark';
|
|
||||||
} else if (saved === 'light') {
|
|
||||||
theme = 'light';
|
|
||||||
} else {
|
|
||||||
// 'system' nebo neuloženo - použij systémové nastavení
|
|
||||||
theme = window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';
|
|
||||||
}
|
|
||||||
document.documentElement.setAttribute('data-bs-theme', theme);
|
|
||||||
} catch (e) {
|
|
||||||
// Fallback pokud localStorage není dostupný
|
|
||||||
document.documentElement.setAttribute('data-bs-theme', 'light');
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
</script>
|
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
|
|||||||
@@ -24,7 +24,6 @@
|
|||||||
"react-router": "^7.9.5",
|
"react-router": "^7.9.5",
|
||||||
"react-router-dom": "^7.9.5",
|
"react-router-dom": "^7.9.5",
|
||||||
"react-select-search": "^4.1.6",
|
"react-select-search": "^4.1.6",
|
||||||
"react-snow-overlay": "^1.0.14",
|
|
||||||
"react-snowfall": "^2.3.0",
|
"react-snowfall": "^2.3.0",
|
||||||
"react-toastify": "^11.0.5",
|
"react-toastify": "^11.0.5",
|
||||||
"recharts": "^3.4.1",
|
"recharts": "^3.4.1",
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
// Service Worker pro Web Push notifikace (připomínka výběru oběda)
|
|
||||||
|
|
||||||
self.addEventListener('push', (event) => {
|
|
||||||
const data = event.data?.json() ?? { title: 'Luncher', body: 'Ještě nemáte zvolený oběd!' };
|
|
||||||
event.waitUntil(
|
|
||||||
self.registration.showNotification(data.title, {
|
|
||||||
body: data.body,
|
|
||||||
icon: '/favicon.ico',
|
|
||||||
tag: 'lunch-reminder',
|
|
||||||
data: { login: data.login, token: data.token },
|
|
||||||
actions: [
|
|
||||||
{ action: 'neobedvam', title: 'Mám vlastní/neobědvám' },
|
|
||||||
],
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
self.addEventListener('notificationclick', (event) => {
|
|
||||||
event.notification.close();
|
|
||||||
|
|
||||||
if (event.action === 'neobedvam') {
|
|
||||||
const { login, token } = event.notification.data ?? {};
|
|
||||||
if (login && token) {
|
|
||||||
event.waitUntil(
|
|
||||||
fetch('/api/notifications/push/quickChoice', {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify({ login, token }),
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
event.waitUntil(
|
|
||||||
self.clients.matchAll({ type: 'window' }).then((clientList) => {
|
|
||||||
for (const client of clientList) {
|
|
||||||
if (client.url.includes(self.location.origin) && 'focus' in client) {
|
|
||||||
return client.focus();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return self.clients.openWindow('/');
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
+92
-1035
File diff suppressed because it is too large
Load Diff
+261
-501
File diff suppressed because it is too large
Load Diff
@@ -1,28 +1,17 @@
|
|||||||
import { Routes, Route } from "react-router-dom";
|
import { Routes, Route } from "react-router-dom";
|
||||||
import { ProvideSettings } from "./context/settings";
|
import { ProvideSettings } from "./context/settings";
|
||||||
// import Snowfall from "react-snowfall";
|
// import Snowfall from "react-snowfall";
|
||||||
import { SnowOverlay } from 'react-snow-overlay';
|
|
||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
import { SocketContext, socket } from "./context/socket";
|
import { SocketContext, socket } from "./context/socket";
|
||||||
import StatsPage from "./pages/StatsPage";
|
import StatsPage from "./pages/StatsPage";
|
||||||
import OrderGroupsPage from "./pages/OrderGroupsPage";
|
|
||||||
import App from "./App";
|
import App from "./App";
|
||||||
|
|
||||||
export const STATS_URL = '/stats';
|
export const STATS_URL = '/stats';
|
||||||
export const OBJEDNANI_URL = '/objednani';
|
|
||||||
|
|
||||||
export default function AppRoutes() {
|
export default function AppRoutes() {
|
||||||
return (
|
return (
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path={STATS_URL} element={<StatsPage />} />
|
<Route path={STATS_URL} element={<StatsPage />} />
|
||||||
<Route path={OBJEDNANI_URL} element={
|
|
||||||
<ProvideSettings>
|
|
||||||
<SocketContext.Provider value={socket}>
|
|
||||||
<OrderGroupsPage />
|
|
||||||
<ToastContainer />
|
|
||||||
</SocketContext.Provider>
|
|
||||||
</ProvideSettings>
|
|
||||||
} />
|
|
||||||
<Route path="/" element={
|
<Route path="/" element={
|
||||||
<ProvideSettings>
|
<ProvideSettings>
|
||||||
<SocketContext.Provider value={socket}>
|
<SocketContext.Provider value={socket}>
|
||||||
@@ -33,7 +22,6 @@ export default function AppRoutes() {
|
|||||||
width: '100vw',
|
width: '100vw',
|
||||||
height: '100vh'
|
height: '100vh'
|
||||||
}} /> */}
|
}} /> */}
|
||||||
<SnowOverlay color={'rgba(240, 240, 240, 0.9)'} disabledOnSingleCpuDevices={true} />
|
|
||||||
<App />
|
<App />
|
||||||
</>
|
</>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
|
|||||||
+8
-84
@@ -1,89 +1,13 @@
|
|||||||
.login-page {
|
.login {
|
||||||
min-height: 100vh;
|
height: 100%;
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
background: var(--luncher-bg);
|
|
||||||
padding: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-card {
|
|
||||||
background: var(--luncher-bg-card);
|
|
||||||
border-radius: var(--luncher-radius-xl);
|
|
||||||
box-shadow: var(--luncher-shadow-lg);
|
|
||||||
padding: 48px;
|
|
||||||
max-width: 420px;
|
|
||||||
width: 100%;
|
|
||||||
text-align: center;
|
|
||||||
border: 1px solid var(--luncher-border-light);
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-logo {
|
|
||||||
font-size: 2.5rem;
|
|
||||||
font-weight: 700;
|
|
||||||
color: var(--luncher-text);
|
|
||||||
margin-bottom: 8px;
|
|
||||||
letter-spacing: -0.5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-subtitle {
|
|
||||||
color: var(--luncher-text-secondary);
|
|
||||||
font-size: 1rem;
|
|
||||||
margin-bottom: 40px;
|
|
||||||
line-height: 1.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 20px;
|
text-align: center;
|
||||||
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.login-form label {
|
.login-inner {
|
||||||
display: block;
|
display: flex;
|
||||||
text-align: left;
|
flex-direction: column;
|
||||||
font-weight: 500;
|
align-items: center;
|
||||||
color: var(--luncher-text);
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form .hint {
|
|
||||||
font-size: 0.85rem;
|
|
||||||
color: var(--luncher-text-muted);
|
|
||||||
margin-top: 8px;
|
|
||||||
text-align: left;
|
|
||||||
line-height: 1.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form input[type="text"] {
|
|
||||||
width: 100%;
|
|
||||||
padding: 14px 18px;
|
|
||||||
font-size: 1rem;
|
|
||||||
border: 2px solid var(--luncher-border);
|
|
||||||
border-radius: var(--luncher-radius-sm);
|
|
||||||
background: var(--luncher-bg);
|
|
||||||
color: var(--luncher-text);
|
|
||||||
transition: var(--luncher-transition);
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form input[type="text"]:hover {
|
|
||||||
border-color: var(--luncher-text-muted);
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form input[type="text"]:focus {
|
|
||||||
border-color: var(--luncher-primary);
|
|
||||||
box-shadow: 0 0 0 3px var(--luncher-primary-light);
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form input[type="text"]::placeholder {
|
|
||||||
color: var(--luncher-text-muted);
|
|
||||||
}
|
|
||||||
|
|
||||||
.login-form .btn {
|
|
||||||
width: 100%;
|
|
||||||
padding: 14px 24px;
|
|
||||||
font-size: 1rem;
|
|
||||||
font-weight: 600;
|
|
||||||
margin-top: 8px;
|
|
||||||
}
|
}
|
||||||
+15
-29
@@ -26,7 +26,7 @@ export default function Login() {
|
|||||||
}, [auth]);
|
}, [auth]);
|
||||||
|
|
||||||
const doLogin = useCallback(async () => {
|
const doLogin = useCallback(async () => {
|
||||||
const length = loginRef?.current?.value.length && loginRef.current.value.replaceAll(/\s/g, '').length
|
const length = loginRef?.current?.value.length && loginRef.current.value.replace(/\s/g, '').length
|
||||||
if (length) {
|
if (length) {
|
||||||
const response = await login({ body: { login: loginRef.current?.value } });
|
const response = await login({ body: { login: loginRef.current?.value } });
|
||||||
if (response.data) {
|
if (response.data) {
|
||||||
@@ -36,35 +36,21 @@ export default function Login() {
|
|||||||
}, [auth]);
|
}, [auth]);
|
||||||
|
|
||||||
if (!auth?.login) {
|
if (!auth?.login) {
|
||||||
return (
|
return <div className='login'>
|
||||||
<div className='login-page'>
|
<h1>Luncher</h1>
|
||||||
<div className='login-card'>
|
<h4 style={{ marginBottom: "50px" }}>Aplikace pro profesionální management obědů</h4>
|
||||||
<h1 className='login-logo'>Luncher</h1>
|
<div className='login-inner'>
|
||||||
<p className='login-subtitle'>Aplikace pro profesionální management obědů</p>
|
<p style={{ fontSize: "12px", marginTop: "10px" }}>
|
||||||
<div className='login-form'>
|
Zobrazované jméno by mělo být vaše jméno nebo přezdívka, pod kterou vás kolegové dokáží snadno identifikovat. Jméno je možné kdykoli změnit.
|
||||||
<div>
|
</p>
|
||||||
<label htmlFor="login-input">Zobrazované jméno</label>
|
Zobrazované jméno: <input style={{ marginTop: "10px" }} ref={loginRef} type='text' onKeyDown={event => {
|
||||||
<input
|
if (event.key === 'Enter') {
|
||||||
id="login-input"
|
doLogin()
|
||||||
ref={loginRef}
|
}
|
||||||
type='text'
|
}} />
|
||||||
placeholder="Např. Jan Novák"
|
<Button onClick={doLogin} style={{ marginTop: "20px" }}>Uložit</Button>
|
||||||
onKeyDown={event => {
|
|
||||||
if (event.key === 'Enter') {
|
|
||||||
doLogin()
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<p className='hint'>
|
|
||||||
Zadejte jméno nebo přezdívku, pod kterou vás kolegové snadno identifikují.
|
|
||||||
Jméno je možné kdykoli změnit.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<Button onClick={doLogin}>Pokračovat</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
</div>
|
||||||
}
|
}
|
||||||
return <div>Neplatný stav</div>
|
return <div>Neplatný stav</div>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,16 +73,22 @@ export const getDayOfWeekIndex = (date: Date) => {
|
|||||||
return (((date.getDay() - 1) % 7) + 7) % 7;
|
return (((date.getDay() - 1) % 7) + 7) % 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Vrátí true, pokud je předané datum o víkendu. */
|
||||||
|
export function getIsWeekend(date: Date) {
|
||||||
|
const index = getDayOfWeekIndex(date);
|
||||||
|
return index == 5 || index == 6;
|
||||||
|
}
|
||||||
|
|
||||||
/** Vrátí první pracovní den v týdnu předaného data. */
|
/** Vrátí první pracovní den v týdnu předaného data. */
|
||||||
export function getFirstWorkDayOfWeek(date: Date) {
|
export function getFirstWorkDayOfWeek(date: Date) {
|
||||||
const firstDay = new Date(date);
|
const firstDay = new Date(date.getTime());
|
||||||
firstDay.setDate(date.getDate() - getDayOfWeekIndex(date));
|
firstDay.setDate(date.getDate() - getDayOfWeekIndex(date));
|
||||||
return firstDay;
|
return firstDay;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Vrátí poslední pracovní den v týdnu předaného data. */
|
/** Vrátí poslední pracovní den v týdnu předaného data. */
|
||||||
export function getLastWorkDayOfWeek(date: Date) {
|
export function getLastWorkDayOfWeek(date: Date) {
|
||||||
const lastDay = new Date(date);
|
const lastDay = new Date(date.getTime());
|
||||||
lastDay.setDate(date.getDate() + (4 - getDayOfWeekIndex(date)));
|
lastDay.setDate(date.getDate() + (4 - getDayOfWeekIndex(date)));
|
||||||
return lastDay;
|
return lastDay;
|
||||||
}
|
}
|
||||||
@@ -104,9 +110,3 @@ export function getHumanDate(date: Date) {
|
|||||||
let currentYear = date.getFullYear();
|
let currentYear = date.getFullYear();
|
||||||
return `${currentDay}.${currentMonth}.${currentYear}`;
|
return `${currentDay}.${currentMonth}.${currentYear}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Převede datum ve formátu YYYY-MM-DD na DD.MM.YYYY */
|
|
||||||
export function formatDateString(dateString: string): string {
|
|
||||||
const [year, month, day] = dateString.split('-');
|
|
||||||
return `${day}.${month}.${year}`;
|
|
||||||
}
|
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
|
import { Navbar } from "react-bootstrap";
|
||||||
|
|
||||||
export default function Footer() {
|
export default function Footer() {
|
||||||
return (
|
return <Navbar className="text-light" variant='dark' expand="lg" style={{
|
||||||
<footer className="footer">
|
display: "flex",
|
||||||
<span>
|
justifyContent: "center",
|
||||||
Zdroj. kódy dostupné na{' '}
|
marginTop: "auto", // Pushne footer na spodek
|
||||||
<a href="https://gitea.melancholik.eu/mates/Luncher" target="_blank" rel="noopener noreferrer">
|
flexShrink: 0 // Zabrání zmenšování při malém obsahu
|
||||||
Gitea
|
}}>
|
||||||
</a>
|
<span>🄯 Žádná práva nevyhrazena. TODO a zdrojové kódy dostupné <a href="https://gitea.melancholik.eu/mates/Luncher">zde</a>.</span>
|
||||||
</span>
|
</Navbar >
|
||||||
</footer>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
@@ -1,31 +1,16 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { Navbar, Nav, NavDropdown, Modal, Button } from "react-bootstrap";
|
import { Navbar, Nav, NavDropdown } from "react-bootstrap";
|
||||||
import { useAuth } from "../context/auth";
|
import { useAuth } from "../context/auth";
|
||||||
import SettingsModal from "./modals/SettingsModal";
|
import SettingsModal from "./modals/SettingsModal";
|
||||||
import { useSettings, ThemePreference } from "../context/settings";
|
import { useSettings } from "../context/settings";
|
||||||
import FeaturesVotingModal from "./modals/FeaturesVotingModal";
|
import FeaturesVotingModal from "./modals/FeaturesVotingModal";
|
||||||
import PizzaCalculatorModal from "./modals/PizzaCalculatorModal";
|
import PizzaCalculatorModal from "./modals/PizzaCalculatorModal";
|
||||||
import RefreshMenuModal from "./modals/RefreshMenuModal";
|
import RefreshMenuModal from "./modals/RefreshMenuModal";
|
||||||
import GenerateQrModal from "./modals/GenerateQrModal";
|
|
||||||
import GenerateMockDataModal from "./modals/GenerateMockDataModal";
|
|
||||||
import ClearMockDataModal from "./modals/ClearMockDataModal";
|
|
||||||
import { useNavigate } from "react-router";
|
import { useNavigate } from "react-router";
|
||||||
import { STATS_URL, OBJEDNANI_URL } from "../AppRoutes";
|
import { STATS_URL } from "../AppRoutes";
|
||||||
import { FeatureRequest, getVotes, updateVote, LunchChoices, getChangelogs } from "../../../types";
|
import { FeatureRequest, getVotes, updateVote } from "../../../types";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
|
||||||
import { faSun, faMoon } from "@fortawesome/free-solid-svg-icons";
|
|
||||||
import { formatDateString } from "../Utils";
|
|
||||||
|
|
||||||
const LAST_SEEN_CHANGELOG_KEY = "lastChangelogDate";
|
export default function Header() {
|
||||||
|
|
||||||
const IS_DEV = process.env.NODE_ENV === 'development';
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
choices?: LunchChoices;
|
|
||||||
dayIndex?: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function Header({ choices, dayIndex }: Props) {
|
|
||||||
const auth = useAuth();
|
const auth = useAuth();
|
||||||
const settings = useSettings();
|
const settings = useSettings();
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
@@ -33,33 +18,8 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
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 [refreshMenuModalOpen, setRefreshMenuModalOpen] = useState<boolean>(false);
|
||||||
const [changelogModalOpen, setChangelogModalOpen] = useState<boolean>(false);
|
|
||||||
const [changelogEntries, setChangelogEntries] = useState<Record<string, string[]>>({});
|
|
||||||
const [qrModalOpen, setQrModalOpen] = useState<boolean>(false);
|
|
||||||
const [generateMockModalOpen, setGenerateMockModalOpen] = useState<boolean>(false);
|
|
||||||
const [clearMockModalOpen, setClearMockModalOpen] = useState<boolean>(false);
|
|
||||||
const [featureVotes, setFeatureVotes] = useState<FeatureRequest[] | undefined>([]);
|
const [featureVotes, setFeatureVotes] = useState<FeatureRequest[] | undefined>([]);
|
||||||
|
|
||||||
// Zjistíme aktuální efektivní téma (pro zobrazení správné ikony)
|
|
||||||
const [effectiveTheme, setEffectiveTheme] = useState<'light' | 'dark'>('light');
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const updateEffectiveTheme = () => {
|
|
||||||
if (settings?.themePreference === 'system') {
|
|
||||||
const isDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
||||||
setEffectiveTheme(isDark ? 'dark' : 'light');
|
|
||||||
} else {
|
|
||||||
setEffectiveTheme(settings?.themePreference || 'light');
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
updateEffectiveTheme();
|
|
||||||
|
|
||||||
const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
|
||||||
mediaQuery.addEventListener('change', updateEffectiveTheme);
|
|
||||||
return () => mediaQuery.removeEventListener('change', updateEffectiveTheme);
|
|
||||||
}, [settings?.themePreference]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (auth?.login) {
|
if (auth?.login) {
|
||||||
getVotes().then(response => {
|
getVotes().then(response => {
|
||||||
@@ -68,19 +28,6 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
}
|
}
|
||||||
}, [auth?.login]);
|
}, [auth?.login]);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!auth?.login) return;
|
|
||||||
const lastSeen = localStorage.getItem(LAST_SEEN_CHANGELOG_KEY) ?? undefined;
|
|
||||||
getChangelogs({ query: lastSeen ? { since: lastSeen } : {} }).then(response => {
|
|
||||||
const entries = response.data;
|
|
||||||
if (!entries || Object.keys(entries).length === 0) return;
|
|
||||||
setChangelogEntries(entries);
|
|
||||||
setChangelogModalOpen(true);
|
|
||||||
const newestDate = Object.keys(entries).sort((a, b) => b.localeCompare(a))[0];
|
|
||||||
localStorage.setItem(LAST_SEEN_CHANGELOG_KEY, newestDate);
|
|
||||||
});
|
|
||||||
}, [auth?.login]);
|
|
||||||
|
|
||||||
const closeSettingsModal = () => {
|
const closeSettingsModal = () => {
|
||||||
setSettingsModalOpen(false);
|
setSettingsModalOpen(false);
|
||||||
}
|
}
|
||||||
@@ -97,24 +44,6 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
setRefreshMenuModalOpen(false);
|
setRefreshMenuModalOpen(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
const closeQrModal = () => {
|
|
||||||
setQrModalOpen(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleQrMenuClick = () => {
|
|
||||||
if (!settings?.bankAccount || !settings?.holderName) {
|
|
||||||
alert('Pro generování QR kódů je nutné mít v nastavení vyplněné číslo účtu a jméno držitele účtu.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setQrModalOpen(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
const toggleTheme = () => {
|
|
||||||
// Přepínáme mezi light a dark (ignorujeme system pro jednoduchost)
|
|
||||||
const newTheme: ThemePreference = effectiveTheme === 'dark' ? 'light' : 'dark';
|
|
||||||
settings?.setThemePreference(newTheme);
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValidInteger = (str: string) => {
|
const isValidInteger = (str: string) => {
|
||||||
str = str.trim();
|
str = str.trim();
|
||||||
if (!str) {
|
if (!str) {
|
||||||
@@ -125,19 +54,19 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
return n !== Infinity && String(n) === str && n >= 0;
|
return n !== Infinity && String(n) === str && n >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
const saveSettings = (bankAccountNumber?: string, bankAccountHolderName?: string, hideSoupsOption?: boolean, themePreference?: ThemePreference) => {
|
const saveSettings = (bankAccountNumber?: string, bankAccountHolderName?: string, hideSoupsOption?: boolean) => {
|
||||||
if (bankAccountNumber) {
|
if (bankAccountNumber) {
|
||||||
try {
|
try {
|
||||||
// Validace kódu banky
|
// Validace kódu banky
|
||||||
if (!bankAccountNumber.includes('/')) {
|
if (bankAccountNumber.indexOf('/') < 0) {
|
||||||
throw new Error("Číslo účtu neobsahuje lomítko/kód banky")
|
throw Error("Číslo účtu neobsahuje lomítko/kód banky")
|
||||||
}
|
}
|
||||||
const split = bankAccountNumber.split("/");
|
const split = bankAccountNumber.split("/");
|
||||||
if (split[1].length !== 4) {
|
if (split[1].length !== 4) {
|
||||||
throw new Error("Kód banky musí být 4 číslice")
|
throw Error("Kód banky musí být 4 číslice")
|
||||||
}
|
}
|
||||||
if (!isValidInteger(split[1])) {
|
if (!isValidInteger(split[1])) {
|
||||||
throw new Error("Kód banky není číslo")
|
throw Error("Kód banky není číslo")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validace čísla a předčíslí
|
// Validace čísla a předčíslí
|
||||||
@@ -147,7 +76,7 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
cislo = cislo.replace('-', '');
|
cislo = cislo.replace('-', '');
|
||||||
}
|
}
|
||||||
if (!isValidInteger(cislo)) {
|
if (!isValidInteger(cislo)) {
|
||||||
throw new Error("Předčíslí nebo číslo účtu neobsahuje pouze číslice")
|
throw Error("Předčíslí nebo číslo účtu neobsahuje pouze číslice")
|
||||||
}
|
}
|
||||||
if (cislo.length < 16) {
|
if (cislo.length < 16) {
|
||||||
cislo = cislo.padStart(16, '0');
|
cislo = cislo.padStart(16, '0');
|
||||||
@@ -160,7 +89,7 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
sum += Number.parseInt(char) * weight
|
sum += Number.parseInt(char) * weight
|
||||||
}
|
}
|
||||||
if (sum % 11 !== 0) {
|
if (sum % 11 !== 0) {
|
||||||
throw new Error("Číslo účtu je neplatné")
|
throw Error("Číslo účtu je neplatné")
|
||||||
}
|
}
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
alert(e.message)
|
alert(e.message)
|
||||||
@@ -170,9 +99,6 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
settings?.setBankAccountNumber(bankAccountNumber);
|
settings?.setBankAccountNumber(bankAccountNumber);
|
||||||
settings?.setBankAccountHolderName(bankAccountHolderName);
|
settings?.setBankAccountHolderName(bankAccountHolderName);
|
||||||
settings?.setHideSoupsOption(hideSoupsOption);
|
settings?.setHideSoupsOption(hideSoupsOption);
|
||||||
if (themePreference) {
|
|
||||||
settings?.setThemePreference(themePreference);
|
|
||||||
}
|
|
||||||
closeSettingsModal();
|
closeSettingsModal();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -192,40 +118,12 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
<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">
|
||||||
<button
|
|
||||||
className="theme-toggle"
|
|
||||||
onClick={toggleTheme}
|
|
||||||
title={effectiveTheme === 'dark' ? 'Přepnout na světlý režim' : 'Přepnout na tmavý režim'}
|
|
||||||
aria-label="Přepnout barevný motiv"
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={effectiveTheme === 'dark' ? faSun : faMoon} />
|
|
||||||
</button>
|
|
||||||
<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={() => 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={handleQrMenuClick}>Generování QR kódů</NavDropdown.Item>
|
|
||||||
<NavDropdown.Item onClick={() => navigate(STATS_URL)}>Statistiky</NavDropdown.Item>
|
<NavDropdown.Item onClick={() => navigate(STATS_URL)}>Statistiky</NavDropdown.Item>
|
||||||
<NavDropdown.Item onClick={() => navigate(OBJEDNANI_URL)}>Objednání</NavDropdown.Item>
|
|
||||||
<NavDropdown.Item onClick={() => {
|
|
||||||
getChangelogs().then(response => {
|
|
||||||
const entries = response.data ?? {};
|
|
||||||
setChangelogEntries(entries);
|
|
||||||
setChangelogModalOpen(true);
|
|
||||||
const dates = Object.keys(entries).sort((a, b) => b.localeCompare(a));
|
|
||||||
if (dates.length > 0) {
|
|
||||||
localStorage.setItem(LAST_SEEN_CHANGELOG_KEY, dates[0]);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}}>Novinky</NavDropdown.Item>
|
|
||||||
{IS_DEV && (
|
|
||||||
<>
|
|
||||||
<NavDropdown.Divider />
|
|
||||||
<NavDropdown.Item onClick={() => setGenerateMockModalOpen(true)}>🔧 Generovat mock data</NavDropdown.Item>
|
|
||||||
<NavDropdown.Item onClick={() => setClearMockModalOpen(true)}>🔧 Smazat data dne</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>
|
||||||
@@ -235,53 +133,5 @@ export default function Header({ choices, dayIndex }: Props) {
|
|||||||
<RefreshMenuModal isOpen={refreshMenuModalOpen} onClose={closeRefreshMenuModal} />
|
<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} />
|
||||||
{choices && settings?.bankAccount && settings?.holderName && (
|
|
||||||
<GenerateQrModal
|
|
||||||
isOpen={qrModalOpen}
|
|
||||||
onClose={closeQrModal}
|
|
||||||
choices={choices}
|
|
||||||
bankAccount={settings.bankAccount}
|
|
||||||
bankAccountHolder={settings.holderName}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{IS_DEV && (
|
|
||||||
<>
|
|
||||||
<GenerateMockDataModal
|
|
||||||
isOpen={generateMockModalOpen}
|
|
||||||
onClose={() => setGenerateMockModalOpen(false)}
|
|
||||||
currentDayIndex={dayIndex}
|
|
||||||
/>
|
|
||||||
<ClearMockDataModal
|
|
||||||
isOpen={clearMockModalOpen}
|
|
||||||
onClose={() => setClearMockModalOpen(false)}
|
|
||||||
currentDayIndex={dayIndex}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<Modal show={changelogModalOpen} onHide={() => setChangelogModalOpen(false)} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Novinky</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{Object.keys(changelogEntries).sort((a, b) => b.localeCompare(a)).map(date => (
|
|
||||||
<div key={date}>
|
|
||||||
<strong>{formatDateString(date)}</strong>
|
|
||||||
<ul>
|
|
||||||
{changelogEntries[date].map((item, index) => (
|
|
||||||
<li key={index}>{item}</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
{Object.keys(changelogEntries).length === 0 && (
|
|
||||||
<p>Žádné novinky.</p>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={() => setChangelogModalOpen(false)}>
|
|
||||||
Zavřít
|
|
||||||
</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
</Navbar>
|
</Navbar>
|
||||||
}
|
}
|
||||||
@@ -9,13 +9,11 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function Loader(props: Readonly<Props>) {
|
function Loader(props: Readonly<Props>) {
|
||||||
return (
|
return <div className='loader'>
|
||||||
<div className='loader'>
|
<h1>{props.title ?? 'Prosím čekejte...'}</h1>
|
||||||
<FontAwesomeIcon icon={props.icon} className={`loader-icon ${props.animation ?? ''}`} />
|
<FontAwesomeIcon icon={props.icon} className={`loader-icon mb-3 ` + (props.animation ?? '')} />
|
||||||
<h2 className='loader-title'>{props.title ?? 'Prosím čekejte...'}</h2>
|
<p>{props.description}</p>
|
||||||
<p className='loader-description'>{props.description}</p>
|
</div>
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Loader;
|
export default Loader;
|
||||||
|
|||||||
@@ -15,43 +15,29 @@ export default function PizzaOrderList({ state, orders, onDelete, creator }: Rea
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!orders?.length) {
|
if (!orders?.length) {
|
||||||
return <p className="mt-4" style={{ color: 'var(--luncher-text-muted)', fontStyle: 'italic' }}>Zatím žádné objednávky...</p>
|
return <p className="mt-3"><i>Zatím žádné objednávky...</i></p>
|
||||||
}
|
}
|
||||||
|
|
||||||
const total = orders.reduce((total, order) => total + order.totalPrice, 0);
|
const total = orders.reduce((total, order) => total + order.totalPrice, 0);
|
||||||
|
|
||||||
return (
|
return <Table className="mt-3" striped bordered hover>
|
||||||
<div className="mt-4" style={{
|
<thead>
|
||||||
background: 'var(--luncher-bg-card)',
|
<tr>
|
||||||
borderRadius: 'var(--luncher-radius-lg)',
|
<th>Jméno</th>
|
||||||
overflow: 'hidden',
|
<th>Objednávka</th>
|
||||||
border: '1px solid var(--luncher-border-light)',
|
<th>Poznámka</th>
|
||||||
boxShadow: 'var(--luncher-shadow)'
|
<th>Příplatek</th>
|
||||||
}}>
|
<th>Cena</th>
|
||||||
<Table className="mb-0" style={{ color: 'var(--luncher-text)' }}>
|
</tr>
|
||||||
<thead style={{ background: 'var(--luncher-primary-light)' }}>
|
</thead>
|
||||||
<tr>
|
<tbody>
|
||||||
<th style={{ padding: '16px 20px', color: 'var(--luncher-primary)', fontWeight: 600, border: 'none' }}>Jméno</th>
|
{orders.map(order => <tr key={order.customer}>
|
||||||
<th style={{ padding: '16px 20px', color: 'var(--luncher-primary)', fontWeight: 600, border: 'none' }}>Objednávka</th>
|
<PizzaOrderRow creator={creator} state={state} order={order} onDelete={onDelete} onFeeModalSave={saveFees} />
|
||||||
<th style={{ padding: '16px 20px', color: 'var(--luncher-primary)', fontWeight: 600, border: 'none' }}>Poznámka</th>
|
</tr>)}
|
||||||
<th style={{ padding: '16px 20px', color: 'var(--luncher-primary)', fontWeight: 600, border: 'none' }}>Příplatek</th>
|
<tr style={{ fontWeight: 'bold' }}>
|
||||||
<th style={{ padding: '16px 20px', color: 'var(--luncher-primary)', fontWeight: 600, border: 'none', textAlign: 'right' }}>Cena</th>
|
<td colSpan={4}>Celkem</td>
|
||||||
</tr>
|
<td>{`${total} Kč`}</td>
|
||||||
</thead>
|
</tr>
|
||||||
<tbody>
|
</tbody>
|
||||||
{orders.map(order => <tr key={order.customer} style={{ borderColor: 'var(--luncher-border-light)' }}>
|
</Table>
|
||||||
<PizzaOrderRow creator={creator} state={state} order={order} onDelete={onDelete} onFeeModalSave={saveFees} />
|
|
||||||
</tr>)}
|
|
||||||
<tr style={{
|
|
||||||
fontWeight: 700,
|
|
||||||
background: 'var(--luncher-bg-hover)',
|
|
||||||
borderTop: '2px solid var(--luncher-border)'
|
|
||||||
}}>
|
|
||||||
<td colSpan={4} style={{ padding: '16px 20px', border: 'none' }}>Celkem</td>
|
|
||||||
<td style={{ padding: '16px 20px', border: 'none', textAlign: 'right', color: 'var(--luncher-primary)' }}>{`${total / 100} Kč`}</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</Table>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
@@ -26,7 +26,7 @@ export default function PizzaOrderRow({ creator, order, state, onDelete, onFeeMo
|
|||||||
<td>{order.customer}</td>
|
<td>{order.customer}</td>
|
||||||
<td>{order.pizzaList!.map<React.ReactNode>(pizzaOrder =>
|
<td>{order.pizzaList!.map<React.ReactNode>(pizzaOrder =>
|
||||||
<span key={pizzaOrder.name}>
|
<span key={pizzaOrder.name}>
|
||||||
{`${pizzaOrder.name}, ${pizzaOrder.size} (${pizzaOrder.price / 100} 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'>
|
<span title='Odstranit'>
|
||||||
<FontAwesomeIcon onClick={() => {
|
<FontAwesomeIcon onClick={() => {
|
||||||
@@ -38,10 +38,10 @@ export default function PizzaOrderRow({ creator, order, state, onDelete, onFeeMo
|
|||||||
.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 / 100} 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 / 100} 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 && <span title='Nastavit příplatek'><FontAwesomeIcon onClick={() => { setIsFeeModalOpen(true) }} className='action-icon' icon={faMoneyBill1} /></span>}
|
||||||
</td>
|
</td>
|
||||||
<PizzaAdditionalFeeModal customerName={order.customer} isOpen={isFeeModalOpen} onClose={() => setIsFeeModalOpen(false)} onSave={saveFees} initialValues={{ text: order.fee?.text, price: order.fee?.price != null ? String(order.fee.price / 100) : undefined }} />
|
<PizzaAdditionalFeeModal customerName={order.customer} isOpen={isFeeModalOpen} onClose={() => setIsFeeModalOpen(false)} onSave={saveFees} initialValues={{ text: order.fee?.text, price: order.fee?.price?.toString() }} />
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
@@ -1,104 +0,0 @@
|
|||||||
import { useState } from "react";
|
|
||||||
import { Modal, Button, Alert } from "react-bootstrap";
|
|
||||||
import { clearMockData, DayIndex } from "../../../../types";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
currentDayIndex?: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
const DAY_NAMES = ['Pondělí', 'Úterý', 'Středa', 'Čtvrtek', 'Pátek'];
|
|
||||||
|
|
||||||
/** Modální dialog pro smazání mock dat (pouze DEV). */
|
|
||||||
export default function ClearMockDataModal({ isOpen, onClose, currentDayIndex }: Readonly<Props>) {
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [success, setSuccess] = useState(false);
|
|
||||||
|
|
||||||
const handleClear = async () => {
|
|
||||||
setError(null);
|
|
||||||
setLoading(true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const body: any = {};
|
|
||||||
if (currentDayIndex !== undefined) {
|
|
||||||
body.dayIndex = currentDayIndex as DayIndex;
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await clearMockData({ body });
|
|
||||||
if (response.error) {
|
|
||||||
setError((response.error as any).error || 'Nastala chyba při mazání dat');
|
|
||||||
} else {
|
|
||||||
setSuccess(true);
|
|
||||||
setTimeout(() => {
|
|
||||||
onClose();
|
|
||||||
setSuccess(false);
|
|
||||||
}, 1500);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba při mazání dat');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = () => {
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
const dayName = currentDayIndex !== undefined ? DAY_NAMES[currentDayIndex] : 'aktuální den';
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={handleClose}>
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Smazat data</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{success ? (
|
|
||||||
<Alert variant="success">
|
|
||||||
Data byla úspěšně smazána!
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Alert variant="warning">
|
|
||||||
<strong>DEV režim</strong> - Tato funkce je dostupná pouze ve vývojovém prostředí.
|
|
||||||
</Alert>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<p>
|
|
||||||
Opravdu chcete smazat všechny volby stravování pro <strong>{dayName}</strong>?
|
|
||||||
</p>
|
|
||||||
<p className="text-muted">
|
|
||||||
Tato akce je nevratná.
|
|
||||||
</p>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
{!success && (
|
|
||||||
<>
|
|
||||||
<Button variant="secondary" onClick={handleClose} disabled={loading}>
|
|
||||||
Ne, zrušit
|
|
||||||
</Button>
|
|
||||||
<Button variant="danger" onClick={handleClear} disabled={loading}>
|
|
||||||
{loading ? 'Mažu...' : 'Ano, smazat'}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{success && (
|
|
||||||
<Button variant="secondary" onClick={handleClose}>
|
|
||||||
Zavřít
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
import { Modal, Button } from "react-bootstrap";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
title: string;
|
|
||||||
message: string;
|
|
||||||
confirmLabel?: string;
|
|
||||||
confirmVariant?: string;
|
|
||||||
onConfirm: () => void;
|
|
||||||
onClose: () => void;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function ConfirmModal({ isOpen, title, message, confirmLabel = "Potvrdit", confirmVariant = "primary", onConfirm, onClose }: Readonly<Props>) {
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose}>
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title>{title}</Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>{message}</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={onClose}>Zrušit</Button>
|
|
||||||
<Button variant={confirmVariant} onClick={onConfirm}>{confirmLabel}</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,197 +0,0 @@
|
|||||||
import { useState, useEffect } from "react";
|
|
||||||
import { Modal, Button, Form, Table, Alert } from "react-bootstrap";
|
|
||||||
import { updateGroupFees, OrderGroup, OrderGroupMember } from "../../../../types";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
group: OrderGroup;
|
|
||||||
onSaved: (data: any) => void;
|
|
||||||
};
|
|
||||||
|
|
||||||
function parseHal(s: string): number {
|
|
||||||
const n = parseFloat(s.replace(',', '.'));
|
|
||||||
return isNaN(n) || n < 0 ? 0 : Math.round(n * 100);
|
|
||||||
}
|
|
||||||
|
|
||||||
function parsePercent(s: string): number {
|
|
||||||
const n = parseFloat(s.replace(',', '.'));
|
|
||||||
return isNaN(n) || n < 0 ? 0 : Math.round(n);
|
|
||||||
}
|
|
||||||
|
|
||||||
function computeMemberTotal(member: OrderGroupMember, feeShare: number, discountType: string, discountValue: number, memberCount: number): number {
|
|
||||||
const base = member.amount ?? 0;
|
|
||||||
const surcharge = member.surchargeAmount ?? 0;
|
|
||||||
const discount = discountType === 'percent'
|
|
||||||
? Math.round((base + surcharge) * discountValue / 100)
|
|
||||||
: Math.round(discountValue / memberCount);
|
|
||||||
return base + surcharge + feeShare - discount;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function EditGroupFeesModal({ isOpen, onClose, group, onSaved }: Readonly<Props>) {
|
|
||||||
const [fees, setFees] = useState('');
|
|
||||||
const [shipping, setShipping] = useState('');
|
|
||||||
const [tip, setTip] = useState('');
|
|
||||||
const [discountType, setDiscountType] = useState<'percent' | 'fixed'>('percent');
|
|
||||||
const [discountValue, setDiscountValue] = useState('');
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isOpen) return;
|
|
||||||
setFees(group.fees ? String(group.fees / 100) : '');
|
|
||||||
setShipping(group.shipping ? String(group.shipping / 100) : '');
|
|
||||||
setTip(group.tip ? String(group.tip / 100) : '');
|
|
||||||
setDiscountType((group.discountType as 'percent' | 'fixed') ?? 'percent');
|
|
||||||
setDiscountValue(group.discountValue
|
|
||||||
? ((group.discountType as string) === 'fixed' ? String(group.discountValue / 100) : String(group.discountValue))
|
|
||||||
: '');
|
|
||||||
setError(null);
|
|
||||||
}, [isOpen, group]);
|
|
||||||
|
|
||||||
const memberEntries = Object.entries(group.members) as [string, OrderGroupMember][];
|
|
||||||
const memberCount = memberEntries.length;
|
|
||||||
|
|
||||||
const feesNum = parseHal(fees);
|
|
||||||
const shippingNum = parseHal(shipping);
|
|
||||||
const tipNum = parseHal(tip);
|
|
||||||
const discountNum = discountType === 'percent' ? parsePercent(discountValue) : parseHal(discountValue);
|
|
||||||
const totalFees = feesNum + shippingNum + tipNum;
|
|
||||||
const feeShare = memberCount > 0 ? Math.round(totalFees / memberCount) : 0;
|
|
||||||
|
|
||||||
const handleSave = async () => {
|
|
||||||
setError(null);
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const res = await updateGroupFees({
|
|
||||||
body: {
|
|
||||||
id: group.id,
|
|
||||||
fees: feesNum,
|
|
||||||
shipping: shippingNum,
|
|
||||||
tip: tipNum,
|
|
||||||
discountType: discountNum > 0 ? discountType : undefined,
|
|
||||||
discountValue: discountNum > 0 ? discountNum : undefined,
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (res.error) {
|
|
||||||
setError((res.error as any).error || 'Nastala chyba');
|
|
||||||
} else {
|
|
||||||
onSaved(res.data);
|
|
||||||
onClose();
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Poplatky skupiny — {group.name}</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>{error}</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div className="d-flex gap-3 flex-wrap mb-3">
|
|
||||||
<Form.Group>
|
|
||||||
<Form.Label>Poplatky (Kč)</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
type="number" min={0} step={0.01}
|
|
||||||
value={fees} onChange={e => setFees(e.target.value)}
|
|
||||||
placeholder="0" style={{ width: 110 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</Form.Group>
|
|
||||||
<Form.Group>
|
|
||||||
<Form.Label>Doprava (Kč)</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
type="number" min={0} step={0.01}
|
|
||||||
value={shipping} onChange={e => setShipping(e.target.value)}
|
|
||||||
placeholder="0" style={{ width: 110 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</Form.Group>
|
|
||||||
<Form.Group>
|
|
||||||
<Form.Label>Spropitné (Kč)</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
type="number" min={0} step={0.01}
|
|
||||||
value={tip} onChange={e => setTip(e.target.value)}
|
|
||||||
placeholder="0" style={{ width: 110 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</Form.Group>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="d-flex gap-3 align-items-end flex-wrap mb-3">
|
|
||||||
<Form.Group>
|
|
||||||
<Form.Label>Sleva</Form.Label>
|
|
||||||
<div className="d-flex gap-2 align-items-center">
|
|
||||||
<Form.Select
|
|
||||||
value={discountType}
|
|
||||||
onChange={e => setDiscountType(e.target.value as 'percent' | 'fixed')}
|
|
||||||
style={{ width: 160 }}
|
|
||||||
>
|
|
||||||
<option value="percent">Procentuální (%)</option>
|
|
||||||
<option value="fixed">Pevná částka (Kč)</option>
|
|
||||||
</Form.Select>
|
|
||||||
<Form.Control
|
|
||||||
type="number" min={0} step={discountType === 'percent' ? 1 : 0.01}
|
|
||||||
value={discountValue} onChange={e => setDiscountValue(e.target.value)}
|
|
||||||
placeholder="0" style={{ width: 100 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<span className="text-muted">{discountType === 'percent' ? '%' : 'Kč'}</span>
|
|
||||||
</div>
|
|
||||||
</Form.Group>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<hr />
|
|
||||||
<h6>Náhled celkových částek ({memberCount} členů, {feeShare > 0 ? `poplatek ${feeShare / 100} Kč/os.` : 'bez poplatku'})</h6>
|
|
||||||
<Table size="sm" bordered>
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>Člen</th>
|
|
||||||
<th className="text-end">Základ</th>
|
|
||||||
<th className="text-end">Příplatek</th>
|
|
||||||
<th className="text-end">Poplatek</th>
|
|
||||||
<th className="text-end">Sleva</th>
|
|
||||||
<th className="text-end fw-bold">Celkem</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{memberEntries.map(([login, member]) => {
|
|
||||||
const base = member.amount ?? 0;
|
|
||||||
const surcharge = member.surchargeAmount ?? 0;
|
|
||||||
const discount = discountNum > 0
|
|
||||||
? (discountType === 'percent'
|
|
||||||
? Math.round((base + surcharge) * discountNum / 100)
|
|
||||||
: Math.round(discountNum / memberCount))
|
|
||||||
: 0;
|
|
||||||
const total = computeMemberTotal(member, feeShare, discountType, discountNum, memberCount);
|
|
||||||
return (
|
|
||||||
<tr key={login}>
|
|
||||||
<td><strong>{login}</strong></td>
|
|
||||||
<td className="text-end">{base > 0 ? `${base / 100} Kč` : '—'}</td>
|
|
||||||
<td className="text-end">{surcharge > 0 ? `${surcharge / 100} Kč` : '—'}</td>
|
|
||||||
<td className="text-end">{feeShare > 0 ? `${feeShare / 100} Kč` : '—'}</td>
|
|
||||||
<td className="text-end text-danger">{discount > 0 ? `-${discount / 100} Kč` : '—'}</td>
|
|
||||||
<td className="text-end fw-bold">{total > 0 ? `${total / 100} Kč` : '—'}</td>
|
|
||||||
</tr>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</tbody>
|
|
||||||
</Table>
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={onClose} disabled={loading}>Storno</Button>
|
|
||||||
<Button variant="primary" onClick={handleSave} disabled={loading}>
|
|
||||||
{loading ? 'Ukládám...' : 'Uložit'}
|
|
||||||
</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,140 +0,0 @@
|
|||||||
import { useState } from "react";
|
|
||||||
import { Modal, Button, Form, Alert } from "react-bootstrap";
|
|
||||||
import { generateMockData, DayIndex } from "../../../../types";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
currentDayIndex?: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
const DAY_NAMES = ['Pondělí', 'Úterý', 'Středa', 'Čtvrtek', 'Pátek'];
|
|
||||||
|
|
||||||
/** Modální dialog pro generování mock dat (pouze DEV). */
|
|
||||||
export default function GenerateMockDataModal({ isOpen, onClose, currentDayIndex }: Readonly<Props>) {
|
|
||||||
const [dayIndex, setDayIndex] = useState<number | undefined>(currentDayIndex);
|
|
||||||
const [count, setCount] = useState<string>('');
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [success, setSuccess] = useState(false);
|
|
||||||
|
|
||||||
const handleGenerate = async () => {
|
|
||||||
setError(null);
|
|
||||||
setLoading(true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const body: any = {};
|
|
||||||
if (dayIndex !== undefined) {
|
|
||||||
body.dayIndex = dayIndex as DayIndex;
|
|
||||||
}
|
|
||||||
if (count && count.trim() !== '') {
|
|
||||||
const countNum = parseInt(count, 10);
|
|
||||||
if (isNaN(countNum) || countNum < 1 || countNum > 100) {
|
|
||||||
setError('Počet musí být číslo mezi 1 a 100');
|
|
||||||
setLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
body.count = countNum;
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await generateMockData({ body });
|
|
||||||
if (response.error) {
|
|
||||||
setError((response.error as any).error || 'Nastala chyba při generování dat');
|
|
||||||
} else {
|
|
||||||
setSuccess(true);
|
|
||||||
setTimeout(() => {
|
|
||||||
onClose();
|
|
||||||
setSuccess(false);
|
|
||||||
setCount('');
|
|
||||||
}, 1500);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba při generování dat');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = () => {
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
setCount('');
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={handleClose}>
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Generovat mock data</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{success ? (
|
|
||||||
<Alert variant="success">
|
|
||||||
Mock data byla úspěšně vygenerována!
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Alert variant="warning">
|
|
||||||
<strong>DEV režim</strong> - Tato funkce je dostupná pouze ve vývojovém prostředí.
|
|
||||||
</Alert>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Den</Form.Label>
|
|
||||||
<Form.Select
|
|
||||||
value={dayIndex ?? ''}
|
|
||||||
onChange={e => setDayIndex(e.target.value === '' ? undefined : parseInt(e.target.value, 10))}
|
|
||||||
>
|
|
||||||
<option value="">Aktuální den</option>
|
|
||||||
{DAY_NAMES.map((name, index) => (
|
|
||||||
<option key={index} value={index}>{name}</option>
|
|
||||||
))}
|
|
||||||
</Form.Select>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Pokud není vybráno, použije se aktuální den.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Počet záznamů</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
type="number"
|
|
||||||
placeholder="Náhodný (5-20)"
|
|
||||||
value={count}
|
|
||||||
onChange={e => setCount(e.target.value)}
|
|
||||||
min={1}
|
|
||||||
max={100}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Pokud není zadáno, vybere se náhodný počet 5-20.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
{!success && (
|
|
||||||
<>
|
|
||||||
<Button variant="secondary" onClick={handleClose} disabled={loading}>
|
|
||||||
Storno
|
|
||||||
</Button>
|
|
||||||
<Button variant="primary" onClick={handleGenerate} disabled={loading}>
|
|
||||||
{loading ? 'Generuji...' : 'Generovat'}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{success && (
|
|
||||||
<Button variant="secondary" onClick={handleClose}>
|
|
||||||
Zavřít
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,255 +0,0 @@
|
|||||||
import { useState, useEffect, useCallback } from "react";
|
|
||||||
import { Modal, Button, Form, Table, Alert } from "react-bootstrap";
|
|
||||||
import { generateQr, LunchChoices, QrRecipient } from "../../../../types";
|
|
||||||
|
|
||||||
type UserEntry = {
|
|
||||||
login: string;
|
|
||||||
selected: boolean;
|
|
||||||
purpose: string;
|
|
||||||
amount: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
choices: LunchChoices;
|
|
||||||
bankAccount: string;
|
|
||||||
bankAccountHolder: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** Modální dialog pro generování QR kódů pro platbu. */
|
|
||||||
export default function GenerateQrModal({ isOpen, onClose, choices, bankAccount, bankAccountHolder }: Readonly<Props>) {
|
|
||||||
const [users, setUsers] = useState<UserEntry[]>([]);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [success, setSuccess] = useState(false);
|
|
||||||
|
|
||||||
// Při otevření modálu načteme seznam uživatelů z choices
|
|
||||||
useEffect(() => {
|
|
||||||
if (isOpen && choices) {
|
|
||||||
const userLogins = new Set<string>();
|
|
||||||
// Projdeme všechny lokace a získáme unikátní loginy
|
|
||||||
Object.values(choices).forEach(locationChoices => {
|
|
||||||
if (locationChoices) {
|
|
||||||
Object.keys(locationChoices).forEach(login => {
|
|
||||||
userLogins.add(login);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// Vytvoříme seznam uživatelů
|
|
||||||
const userList: UserEntry[] = Array.from(userLogins)
|
|
||||||
.sort((a, b) => a.localeCompare(b, 'cs'))
|
|
||||||
.map(login => ({
|
|
||||||
login,
|
|
||||||
selected: false,
|
|
||||||
purpose: '',
|
|
||||||
amount: '',
|
|
||||||
}));
|
|
||||||
setUsers(userList);
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
}
|
|
||||||
}, [isOpen, choices]);
|
|
||||||
|
|
||||||
const handleCheckboxChange = useCallback((login: string, checked: boolean) => {
|
|
||||||
setUsers(prev => prev.map(u =>
|
|
||||||
u.login === login ? { ...u, selected: checked } : u
|
|
||||||
));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handlePurposeChange = useCallback((login: string, value: string) => {
|
|
||||||
setUsers(prev => prev.map(u =>
|
|
||||||
u.login === login ? { ...u, purpose: value } : u
|
|
||||||
));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handleAmountChange = useCallback((login: string, value: string) => {
|
|
||||||
// Povolíme pouze čísla, tečku a čárku
|
|
||||||
const sanitized = value.replace(/[^0-9.,]/g, '').replace(',', '.');
|
|
||||||
setUsers(prev => prev.map(u =>
|
|
||||||
u.login === login ? { ...u, amount: sanitized } : u
|
|
||||||
));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const validateAmount = (amountStr: string): number | null => {
|
|
||||||
if (!amountStr || amountStr.trim().length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const amount = parseFloat(amountStr);
|
|
||||||
if (isNaN(amount) || amount <= 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
// Max 2 desetinná místa
|
|
||||||
const parts = amountStr.split('.');
|
|
||||||
if (parts.length === 2 && parts[1].length > 2) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return Math.round(amount * 100) / 100; // Zaokrouhlíme na 2 desetinná místa
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleGenerate = async () => {
|
|
||||||
setError(null);
|
|
||||||
const selectedUsers = users.filter(u => u.selected);
|
|
||||||
|
|
||||||
if (selectedUsers.length === 0) {
|
|
||||||
setError("Nebyl vybrán žádný uživatel");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validace
|
|
||||||
const recipients: QrRecipient[] = [];
|
|
||||||
for (const user of selectedUsers) {
|
|
||||||
if (!user.purpose || user.purpose.trim().length === 0) {
|
|
||||||
setError(`Uživatel ${user.login} nemá vyplněný účel platby`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const amount = validateAmount(user.amount);
|
|
||||||
if (amount === null) {
|
|
||||||
setError(`Uživatel ${user.login} má neplatnou částku (musí být kladné číslo s max. 2 desetinnými místy)`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
recipients.push({
|
|
||||||
login: user.login,
|
|
||||||
purpose: user.purpose.trim(),
|
|
||||||
amount,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const response = await generateQr({
|
|
||||||
body: {
|
|
||||||
recipients,
|
|
||||||
bankAccount,
|
|
||||||
bankAccountHolder,
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (response.error) {
|
|
||||||
setError((response.error as any).error || 'Nastala chyba při generování QR kódů');
|
|
||||||
} else {
|
|
||||||
setSuccess(true);
|
|
||||||
// Po 2 sekundách zavřeme modal
|
|
||||||
setTimeout(() => {
|
|
||||||
onClose();
|
|
||||||
}, 2000);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba při generování QR kódů');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = () => {
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
const selectedCount = users.filter(u => u.selected).length;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={handleClose} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Generování QR kódů</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{success ? (
|
|
||||||
<Alert variant="success">
|
|
||||||
QR kódy byly úspěšně vygenerovány! Uživatelé je uvidí v sekci "Nevyřízené platby".
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<p>
|
|
||||||
Vyberte uživatele, kterým chcete vygenerovat QR kód pro platbu.
|
|
||||||
QR kódy se uživatelům zobrazí v sekci "Nevyřízené platby".
|
|
||||||
</p>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{users.length === 0 ? (
|
|
||||||
<Alert variant="info">
|
|
||||||
V tento den nemá žádný uživatel zvolenou možnost stravování.
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<Table striped bordered hover responsive>
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th style={{ width: '50px' }}></th>
|
|
||||||
<th>Uživatel</th>
|
|
||||||
<th>Účel platby</th>
|
|
||||||
<th style={{ width: '120px' }}>Částka (Kč)</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{users.map(user => (
|
|
||||||
<tr key={user.login} className={user.selected ? '' : 'text-muted'}>
|
|
||||||
<td className="text-center">
|
|
||||||
<Form.Check
|
|
||||||
type="checkbox"
|
|
||||||
checked={user.selected}
|
|
||||||
onChange={e => handleCheckboxChange(user.login, e.target.checked)}
|
|
||||||
/>
|
|
||||||
</td>
|
|
||||||
<td>{user.login}</td>
|
|
||||||
<td>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="např. Pizza prosciutto"
|
|
||||||
value={user.purpose}
|
|
||||||
onChange={e => handlePurposeChange(user.login, e.target.value)}
|
|
||||||
disabled={!user.selected}
|
|
||||||
size="sm"
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="0.00"
|
|
||||||
value={user.amount}
|
|
||||||
onChange={e => handleAmountChange(user.login, e.target.value)}
|
|
||||||
disabled={!user.selected}
|
|
||||||
size="sm"
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
))}
|
|
||||||
</tbody>
|
|
||||||
</Table>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
{!success && (
|
|
||||||
<>
|
|
||||||
<span className="me-auto text-muted">
|
|
||||||
Vybráno: {selectedCount} / {users.length}
|
|
||||||
</span>
|
|
||||||
<Button variant="secondary" onClick={handleClose} disabled={loading}>
|
|
||||||
Storno
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="primary"
|
|
||||||
onClick={handleGenerate}
|
|
||||||
disabled={loading || selectedCount === 0}
|
|
||||||
>
|
|
||||||
{loading ? 'Generuji...' : 'Generovat'}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{success && (
|
|
||||||
<Button variant="secondary" onClick={handleClose}>
|
|
||||||
Zavřít
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,305 +0,0 @@
|
|||||||
import { useState, useEffect, useCallback } from "react";
|
|
||||||
import { Modal, Button, Form, Table, Alert } from "react-bootstrap";
|
|
||||||
import { generateQr, LunchChoice, LocationLunchChoicesMap, RestaurantDayMenu, QrRecipient } from "../../../../types";
|
|
||||||
import { parsePriceCzk } from "../../utils/parsePrice";
|
|
||||||
|
|
||||||
type DinerEntry = {
|
|
||||||
login: string;
|
|
||||||
selectedFoods: number[];
|
|
||||||
baseAmount: number;
|
|
||||||
baseAmountParseFailed: boolean;
|
|
||||||
surchargeText: string;
|
|
||||||
surchargeAmount: string;
|
|
||||||
included: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
locationKey: LunchChoice;
|
|
||||||
locationName: string;
|
|
||||||
locationChoices: LocationLunchChoicesMap;
|
|
||||||
menu: RestaurantDayMenu | undefined;
|
|
||||||
payerLogin: string;
|
|
||||||
bankAccount: string;
|
|
||||||
bankAccountHolder: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
function sanitizeAmount(value: string): string {
|
|
||||||
return value.replace(/[^0-9.,]/g, '').replace(',', '.');
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseAmount(s: string): number | null {
|
|
||||||
if (!s || s.trim().length === 0) return null;
|
|
||||||
const n = parseFloat(s);
|
|
||||||
if (isNaN(n) || n < 0) return null;
|
|
||||||
return Math.round(n * 100);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function PayForAllModal({ isOpen, onClose, locationName, locationChoices, menu, payerLogin, bankAccount, bankAccountHolder }: Readonly<Props>) {
|
|
||||||
const [diners, setDiners] = useState<DinerEntry[]>([]);
|
|
||||||
const [tipTotal, setTipTotal] = useState('');
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [success, setSuccess] = useState(false);
|
|
||||||
|
|
||||||
const hasMenu = !!menu;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isOpen) return;
|
|
||||||
const entries: DinerEntry[] = Object.entries(locationChoices).map(([login, choice]) => {
|
|
||||||
const selectedFoods = choice.selectedFoods ?? [];
|
|
||||||
let baseAmount = 0;
|
|
||||||
let baseAmountParseFailed = false;
|
|
||||||
if (menu) {
|
|
||||||
for (const idx of selectedFoods) {
|
|
||||||
const priceKc = parsePriceCzk(menu.food?.[idx]?.price);
|
|
||||||
if (priceKc === null) {
|
|
||||||
baseAmountParseFailed = true;
|
|
||||||
} else {
|
|
||||||
baseAmount += Math.round(priceKc * 100);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
login,
|
|
||||||
selectedFoods,
|
|
||||||
baseAmount,
|
|
||||||
baseAmountParseFailed,
|
|
||||||
surchargeText: '',
|
|
||||||
surchargeAmount: '',
|
|
||||||
included: login !== payerLogin,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
setDiners(entries);
|
|
||||||
setTipTotal('');
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
}, [isOpen, locationChoices, menu, payerLogin]);
|
|
||||||
|
|
||||||
const includedDiners = diners.filter(d => d.included && d.login !== payerLogin);
|
|
||||||
const tipPerPerson = (() => {
|
|
||||||
if (includedDiners.length === 0) return 0;
|
|
||||||
const tip = parseAmount(tipTotal);
|
|
||||||
if (tip === null || tip === 0) return 0;
|
|
||||||
const totalPeople = includedDiners.length + 1;
|
|
||||||
return Math.round(tip / totalPeople);
|
|
||||||
})();
|
|
||||||
const payerTipShare = (() => {
|
|
||||||
const tip = parseAmount(tipTotal);
|
|
||||||
if (!tip) return 0;
|
|
||||||
return tip - tipPerPerson * includedDiners.length;
|
|
||||||
})();
|
|
||||||
|
|
||||||
const getTotal = (d: DinerEntry): number => {
|
|
||||||
const surcharge = parseAmount(d.surchargeAmount) ?? 0;
|
|
||||||
const tip = d.login === payerLogin ? payerTipShare : tipPerPerson;
|
|
||||||
return d.baseAmount + surcharge + tip;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleInclude = useCallback((login: string, checked: boolean) => {
|
|
||||||
setDiners(prev => prev.map(d => d.login === login ? { ...d, included: checked } : d));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handleSurchargeText = useCallback((login: string, value: string) => {
|
|
||||||
setDiners(prev => prev.map(d => d.login === login ? { ...d, surchargeText: value } : d));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handleSurchargeAmount = useCallback((login: string, value: string) => {
|
|
||||||
setDiners(prev => prev.map(d => d.login === login ? { ...d, surchargeAmount: sanitizeAmount(value) } : d));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handleGenerate = async () => {
|
|
||||||
setError(null);
|
|
||||||
const recipients: QrRecipient[] = [];
|
|
||||||
|
|
||||||
for (const d of diners) {
|
|
||||||
if (!d.included || d.login === payerLogin) continue;
|
|
||||||
const total = getTotal(d);
|
|
||||||
if (total <= 0) {
|
|
||||||
setError(`Celková částka pro ${d.login} musí být kladná`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const foods = d.selectedFoods.map(i => menu?.food?.[i]?.name).filter(Boolean).join(', ');
|
|
||||||
const purposeBase = `Oběd ${locationName}${foods ? ` — ${foods}` : ''}`;
|
|
||||||
recipients.push({
|
|
||||||
login: d.login,
|
|
||||||
purpose: purposeBase.substring(0, 60),
|
|
||||||
amount: total,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (recipients.length === 0) {
|
|
||||||
setError("Nebyl vybrán žádný příjemce");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const response = await generateQr({
|
|
||||||
body: { recipients, bankAccount, bankAccountHolder },
|
|
||||||
});
|
|
||||||
if (response.error) {
|
|
||||||
setError((response.error as any).error || 'Nastala chyba při generování QR kódů');
|
|
||||||
} else {
|
|
||||||
setSuccess(true);
|
|
||||||
setTimeout(() => onClose(), 2000);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba při generování QR kódů');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const anyParseFailed = diners.some(d => d.baseAmountParseFailed && d.included);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Zaplatit za všechny — {locationName}</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{success ? (
|
|
||||||
<Alert variant="success">
|
|
||||||
QR kódy byly úspěšně vygenerovány! Uživatelé je uvidí v sekci „Nevyřízené platby".
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<p>Zaplatili jste za skupinu v restauraci. Nastavte příplatky a společné poplatky, poté vygenerujte QR kódy pro ostatní.</p>
|
|
||||||
|
|
||||||
{!hasMenu && (
|
|
||||||
<Alert variant="info">
|
|
||||||
Pro tuto skupinu nejsou k dispozici ceny jídel — vyplňte příplatky ručně.
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{anyParseFailed && (
|
|
||||||
<Alert variant="warning">
|
|
||||||
U některých jídel se nepodařilo načíst cenu — doplňte ji ručně v sloupci Příplatek.
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Table striped bordered hover responsive size="sm">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th style={{ width: 40 }}></th>
|
|
||||||
<th>Strávník</th>
|
|
||||||
<th>Jídla</th>
|
|
||||||
<th style={{ width: 220 }}>Příplatek</th>
|
|
||||||
<th style={{ width: 90 }}>Poplatek</th>
|
|
||||||
<th style={{ width: 90 }}>Celkem</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{diners.map(d => {
|
|
||||||
const isPayer = d.login === payerLogin;
|
|
||||||
const foodNames = d.selectedFoods.map(i => menu?.food?.[i]?.name).filter(Boolean).join(', ');
|
|
||||||
const total = getTotal(d);
|
|
||||||
return (
|
|
||||||
<tr key={d.login} className={!d.included && !isPayer ? 'text-muted' : ''}>
|
|
||||||
<td className="text-center">
|
|
||||||
{isPayer ? (
|
|
||||||
<small className="text-muted">plátce</small>
|
|
||||||
) : (
|
|
||||||
<Form.Check
|
|
||||||
type="checkbox"
|
|
||||||
checked={d.included}
|
|
||||||
onChange={e => handleInclude(d.login, e.target.checked)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td><strong>{d.login}</strong></td>
|
|
||||||
<td>
|
|
||||||
<small>
|
|
||||||
{foodNames || <span className="text-muted">—</span>}
|
|
||||||
{hasMenu && d.baseAmount > 0 && <span className="text-muted"> ({d.baseAmount / 100} Kč)</span>}
|
|
||||||
{d.baseAmountParseFailed && <span className="text-warning"> ⚠</span>}
|
|
||||||
</small>
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
<div className="d-flex gap-1">
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="popis"
|
|
||||||
value={d.surchargeText}
|
|
||||||
onChange={e => handleSurchargeText(d.login, e.target.value)}
|
|
||||||
disabled={!isPayer && !d.included}
|
|
||||||
size="sm"
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="Kč"
|
|
||||||
value={d.surchargeAmount}
|
|
||||||
onChange={e => handleSurchargeAmount(d.login, e.target.value)}
|
|
||||||
disabled={!isPayer && !d.included}
|
|
||||||
size="sm"
|
|
||||||
style={{ width: 70 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td className="text-end">
|
|
||||||
{(() => { const s = isPayer ? payerTipShare : tipPerPerson; return s > 0 ? `${s / 100} Kč` : '—'; })()}
|
|
||||||
</td>
|
|
||||||
<td className="text-end fw-bold">
|
|
||||||
{`${total / 100} Kč`}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</tbody>
|
|
||||||
</Table>
|
|
||||||
|
|
||||||
<div className="d-flex align-items-center gap-2 mt-2">
|
|
||||||
<label className="mb-0 text-nowrap">Poplatky celkem (Kč):</label>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="0"
|
|
||||||
value={tipTotal}
|
|
||||||
onChange={e => setTipTotal(sanitizeAmount(e.target.value))}
|
|
||||||
size="sm"
|
|
||||||
style={{ width: 100 }}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<small className="text-muted">
|
|
||||||
{includedDiners.length > 0 && tipPerPerson > 0
|
|
||||||
? `(${tipPerPerson / 100} Kč / osoba)`
|
|
||||||
: ''}
|
|
||||||
</small>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
{!success && (
|
|
||||||
<>
|
|
||||||
<span className="me-auto text-muted">
|
|
||||||
Příjemci: {includedDiners.length}
|
|
||||||
</span>
|
|
||||||
<Button variant="secondary" onClick={onClose} disabled={loading}>
|
|
||||||
Storno
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="primary"
|
|
||||||
onClick={handleGenerate}
|
|
||||||
disabled={loading || includedDiners.length === 0}
|
|
||||||
>
|
|
||||||
{loading ? 'Generuji...' : 'Vygenerovat QR'}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{success && (
|
|
||||||
<Button variant="secondary" onClick={onClose}>Zavřít</Button>
|
|
||||||
)}
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,222 +0,0 @@
|
|||||||
import { useState, useEffect } from "react";
|
|
||||||
import { Modal, Button, Form, Table, Alert } from "react-bootstrap";
|
|
||||||
import { generateQr, OrderGroup, OrderGroupMember, QrRecipient } from "../../../../types";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
onSuccess?: () => void;
|
|
||||||
group: OrderGroup;
|
|
||||||
payerLogin: string;
|
|
||||||
bankAccount: string;
|
|
||||||
bankAccountHolder: string;
|
|
||||||
groupId?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type DinerEntry = {
|
|
||||||
login: string;
|
|
||||||
member: OrderGroupMember;
|
|
||||||
included: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function PayForGroupModal({ isOpen, onClose, onSuccess, group, payerLogin, bankAccount, bankAccountHolder, groupId }: Readonly<Props>) {
|
|
||||||
const [diners, setDiners] = useState<DinerEntry[]>([]);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [success, setSuccess] = useState(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isOpen) return;
|
|
||||||
const entries: DinerEntry[] = (Object.entries(group.members) as [string, OrderGroupMember][]).map(([login, member]) => ({
|
|
||||||
login,
|
|
||||||
member,
|
|
||||||
included: login !== payerLogin,
|
|
||||||
}));
|
|
||||||
setDiners(entries);
|
|
||||||
setError(null);
|
|
||||||
setSuccess(false);
|
|
||||||
}, [isOpen, group, payerLogin]);
|
|
||||||
|
|
||||||
const memberCount = diners.length;
|
|
||||||
const fees = group.fees ?? 0;
|
|
||||||
const shipping = group.shipping ?? 0;
|
|
||||||
const tip = group.tip ?? 0;
|
|
||||||
const totalFees = fees + shipping + tip;
|
|
||||||
const feeShare = memberCount > 0 ? Math.round(totalFees / memberCount) : 0;
|
|
||||||
|
|
||||||
const getMemberTotal = (entry: DinerEntry): number => {
|
|
||||||
const base = entry.member.amount ?? 0;
|
|
||||||
const surcharge = entry.member.surchargeAmount ?? 0;
|
|
||||||
const discountType = group.discountType;
|
|
||||||
const discountValue = group.discountValue ?? 0;
|
|
||||||
const discount = discountValue > 0
|
|
||||||
? (discountType === 'percent'
|
|
||||||
? Math.round((base + surcharge) * discountValue / 100)
|
|
||||||
: Math.round(discountValue / memberCount))
|
|
||||||
: 0;
|
|
||||||
return base + surcharge + feeShare - discount;
|
|
||||||
};
|
|
||||||
|
|
||||||
const includedNonPayers = diners.filter(d => d.included && d.login !== payerLogin);
|
|
||||||
|
|
||||||
const handleInclude = (login: string, checked: boolean) => {
|
|
||||||
setDiners(prev => prev.map(d => d.login === login ? { ...d, included: checked } : d));
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleGenerate = async () => {
|
|
||||||
setError(null);
|
|
||||||
const recipients: QrRecipient[] = [];
|
|
||||||
|
|
||||||
for (const d of diners) {
|
|
||||||
if (!d.included || d.login === payerLogin) continue;
|
|
||||||
const total = getMemberTotal(d);
|
|
||||||
if (total <= 0) {
|
|
||||||
setError(`Celková částka pro ${d.login} musí být kladná`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const note = d.member.note?.trim();
|
|
||||||
recipients.push({
|
|
||||||
login: d.login,
|
|
||||||
purpose: note ? note.replace(/[^\x00-\xff*]/g, '').replace(/\*/g, '').substring(0, 60) : `Objednávka ${group.name}`.substring(0, 60),
|
|
||||||
amount: total,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (recipients.length === 0) {
|
|
||||||
setError("Nebyl vybrán žádný příjemce");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const response = await generateQr({
|
|
||||||
body: { recipients, bankAccount, bankAccountHolder, ...(groupId ? { groupId } : {}) },
|
|
||||||
});
|
|
||||||
if (response.error) {
|
|
||||||
setError((response.error as any).error || 'Nastala chyba při generování QR kódů');
|
|
||||||
} else {
|
|
||||||
setSuccess(true);
|
|
||||||
onSuccess?.();
|
|
||||||
setTimeout(() => onClose(), 2000);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba při generování QR kódů');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const hasFees = totalFees > 0;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Generovat QR — {group.name}</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{success ? (
|
|
||||||
<Alert variant="success">
|
|
||||||
QR kódy byly úspěšně vygenerovány! Uživatelé je uvidí v sekci „Nevyřízené platby".
|
|
||||||
</Alert>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<p>Zaplatili jste za skupinu. Vyberte, komu vygenerovat QR kód k úhradě.</p>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{hasFees && (
|
|
||||||
<div className="d-flex gap-3 mb-2 text-muted" style={{ fontSize: '0.9em' }}>
|
|
||||||
{fees > 0 && <span>Poplatky: <strong>{fees / 100} Kč</strong></span>}
|
|
||||||
{shipping > 0 && <span>Doprava: <strong>{shipping / 100} Kč</strong></span>}
|
|
||||||
{tip > 0 && <span>Spropitné: <strong>{tip / 100} Kč</strong></span>}
|
|
||||||
<span>→ {feeShare / 100} Kč/os.</span>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{group.discountValue != null && group.discountValue > 0 && (
|
|
||||||
<div className="mb-2 text-success" style={{ fontSize: '0.9em' }}>
|
|
||||||
Sleva: {group.discountType === 'percent' ? `${group.discountValue}%` : `${group.discountValue / 100} Kč`}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Table striped bordered hover responsive size="sm">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th style={{ width: 40 }}></th>
|
|
||||||
<th>Člen</th>
|
|
||||||
<th style={{ width: 90 }} className="text-end">Základ</th>
|
|
||||||
<th style={{ width: 90 }} className="text-end">Příplatek</th>
|
|
||||||
{hasFees && <th style={{ width: 90 }} className="text-end">Poplatek</th>}
|
|
||||||
<th style={{ width: 90 }} className="text-end fw-bold">Celkem</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{diners.map(d => {
|
|
||||||
const isPayer = d.login === payerLogin;
|
|
||||||
const total = getMemberTotal(d);
|
|
||||||
const surcharge = d.member.surchargeAmount ?? 0;
|
|
||||||
return (
|
|
||||||
<tr key={d.login} className={!d.included && !isPayer ? 'text-muted' : ''}>
|
|
||||||
<td className="text-center">
|
|
||||||
{isPayer ? (
|
|
||||||
<small className="text-muted">plátce</small>
|
|
||||||
) : (
|
|
||||||
<Form.Check
|
|
||||||
type="checkbox"
|
|
||||||
checked={d.included}
|
|
||||||
onChange={e => handleInclude(d.login, e.target.checked)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
<strong>{d.login}</strong>
|
|
||||||
{d.member.surchargeText && (
|
|
||||||
<small className="text-muted ms-1">({d.member.surchargeText})</small>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td className="text-end">
|
|
||||||
{(d.member.amount ?? 0) > 0 ? `${d.member.amount! / 100} Kč` : <span className="text-muted">—</span>}
|
|
||||||
</td>
|
|
||||||
<td className="text-end">
|
|
||||||
{surcharge > 0 ? `${surcharge / 100} Kč` : <span className="text-muted">—</span>}
|
|
||||||
</td>
|
|
||||||
{hasFees && (
|
|
||||||
<td className="text-end">
|
|
||||||
{feeShare > 0 ? `${feeShare / 100} Kč` : '—'}
|
|
||||||
</td>
|
|
||||||
)}
|
|
||||||
<td className="text-end fw-bold">
|
|
||||||
{total > 0 ? `${total / 100} Kč` : <span className="text-muted">—</span>}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</tbody>
|
|
||||||
</Table>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
{!success && (
|
|
||||||
<>
|
|
||||||
<span className="me-auto text-muted">Příjemci: {includedNonPayers.length}</span>
|
|
||||||
<Button variant="secondary" onClick={onClose} disabled={loading}>Storno</Button>
|
|
||||||
<Button
|
|
||||||
variant="primary"
|
|
||||||
onClick={handleGenerate}
|
|
||||||
disabled={loading || includedNonPayers.length === 0}
|
|
||||||
>
|
|
||||||
{loading ? 'Generuji...' : 'Vygenerovat QR'}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{success && (
|
|
||||||
<Button variant="secondary" onClick={onClose}>Zavřít</Button>
|
|
||||||
)}
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -15,12 +15,12 @@ export default function PizzaAdditionalFeeModal({ customerName, isOpen, onClose,
|
|||||||
const priceRef = useRef<HTMLInputElement>(null);
|
const priceRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
const doSubmit = () => {
|
const doSubmit = () => {
|
||||||
onSave(customerName, textRef.current?.value, Math.round(Number.parseFloat(priceRef.current?.value ?? "0") * 100));
|
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, Math.round(Number.parseFloat(priceRef.current?.value ?? "0") * 100));
|
onSave(customerName, textRef.current?.value, parseInt(priceRef.current?.value ?? "0"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -36,13 +36,13 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props
|
|||||||
|
|
||||||
// 1. pizza
|
// 1. pizza
|
||||||
if (diameter1Ref.current?.value) {
|
if (diameter1Ref.current?.value) {
|
||||||
const diameter1 = Number.parseInt(diameter1Ref.current?.value);
|
const diameter1 = parseInt(diameter1Ref.current?.value);
|
||||||
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);
|
||||||
if (price1Ref.current?.value) {
|
if (price1Ref.current?.value) {
|
||||||
const price1 = Number.parseInt(price1Ref.current?.value);
|
const price1 = parseInt(price1Ref.current?.value);
|
||||||
if (price1) {
|
if (price1) {
|
||||||
r.pizza1.pricePerM = price1 / r.pizza1.area;
|
r.pizza1.pricePerM = price1 / r.pizza1.area;
|
||||||
} else {
|
} else {
|
||||||
@@ -56,13 +56,13 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props
|
|||||||
|
|
||||||
// 2. pizza
|
// 2. pizza
|
||||||
if (diameter2Ref.current?.value) {
|
if (diameter2Ref.current?.value) {
|
||||||
const diameter2 = Number.parseInt(diameter2Ref.current?.value);
|
const diameter2 = parseInt(diameter2Ref.current?.value);
|
||||||
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);
|
||||||
if (price2Ref.current?.value) {
|
if (price2Ref.current?.value) {
|
||||||
const price2 = Number.parseInt(price2Ref.current?.value);
|
const price2 = parseInt(price2Ref.current?.value);
|
||||||
if (price2) {
|
if (price2) {
|
||||||
r.pizza2.pricePerM = price2 / r.pizza2.area;
|
r.pizza2.pricePerM = price2 / r.pizza2.area;
|
||||||
} else {
|
} else {
|
||||||
@@ -77,8 +77,8 @@ export default function PizzaCalculatorModal({ isOpen, onClose }: Readonly<Props
|
|||||||
// Srovnání
|
// Srovnání
|
||||||
if (r.pizza1?.pricePerM && r.pizza2?.pricePerM && r.pizza1.diameter && r.pizza2.diameter) {
|
if (r.pizza1?.pricePerM && r.pizza2?.pricePerM && r.pizza1.diameter && r.pizza2.diameter) {
|
||||||
r.choice = r.pizza1.pricePerM < r.pizza2.pricePerM ? 1 : 2;
|
r.choice = r.pizza1.pricePerM < r.pizza2.pricePerM ? 1 : 2;
|
||||||
const bigger = Math.max(r.pizza1.pricePerM, r.pizza2.pricePerM);
|
const bigger = r.pizza1.pricePerM > r.pizza2.pricePerM ? r.pizza1.pricePerM : r.pizza2.pricePerM;
|
||||||
const smaller = Math.min(r.pizza1.pricePerM, r.pizza2.pricePerM);
|
const smaller = r.pizza1.pricePerM < r.pizza2.pricePerM ? r.pizza1.pricePerM : r.pizza2.pricePerM;
|
||||||
r.ratio = (bigger / smaller) - 1;
|
r.ratio = (bigger / smaller) - 1;
|
||||||
r.diameterDiff = Math.abs(r.pizza1.diameter - r.pizza2.diameter);
|
r.diameterDiff = Math.abs(r.pizza1.diameter - r.pizza2.diameter);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { useRef, useState } from "react";
|
import { useRef, useState } from "react";
|
||||||
import { Modal, Button, Alert, Form } from "react-bootstrap";
|
import { Modal, Button, Alert } from "react-bootstrap";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@@ -30,6 +30,7 @@ export default function RefreshMenuModal({ isOpen, onClose }: Readonly<Props>) {
|
|||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
setRefreshMessage({ type: 'success', text: 'Uspesny fetch' });
|
setRefreshMessage({ type: 'success', text: 'Uspesny fetch' });
|
||||||
if (refreshPassRef.current) {
|
if (refreshPassRef.current) {
|
||||||
|
// Clean hesla xd
|
||||||
refreshPassRef.current.value = '';
|
refreshPassRef.current.value = '';
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -49,7 +50,7 @@ export default function RefreshMenuModal({ isOpen, onClose }: Readonly<Props>) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal show={isOpen} onHide={handleClose}>
|
<Modal show={isOpen} onHide={handleClose} size="lg">
|
||||||
<Modal.Header closeButton>
|
<Modal.Header closeButton>
|
||||||
<Modal.Title><h2>Přenačtení menu</h2></Modal.Title>
|
<Modal.Title><h2>Přenačtení menu</h2></Modal.Title>
|
||||||
</Modal.Header>
|
</Modal.Header>
|
||||||
@@ -62,29 +63,36 @@ export default function RefreshMenuModal({ isOpen, onClose }: Readonly<Props>) {
|
|||||||
</Alert>
|
</Alert>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
<div className="mb-3">
|
||||||
<Form.Label>Heslo</Form.Label>
|
Heslo: <input
|
||||||
<Form.Control
|
|
||||||
ref={refreshPassRef}
|
ref={refreshPassRef}
|
||||||
type="password"
|
type="password"
|
||||||
placeholder="Zadejte heslo"
|
placeholder="Zadejte heslo"
|
||||||
|
className="form-control d-inline-block"
|
||||||
|
style={{ width: 'auto', marginLeft: '10px' }}
|
||||||
onKeyDown={e => e.stopPropagation()}
|
onKeyDown={e => e.stopPropagation()}
|
||||||
/>
|
/>
|
||||||
</Form.Group>
|
</div>
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
<div className="mb-3">
|
||||||
<Form.Label>Typ refreshe</Form.Label>
|
Typ refreshe: <select
|
||||||
<Form.Select ref={refreshTypeRef} defaultValue="week">
|
ref={refreshTypeRef}
|
||||||
|
className="form-select d-inline-block"
|
||||||
|
style={{ width: 'auto', marginLeft: '10px' }}
|
||||||
|
defaultValue="week"
|
||||||
|
>
|
||||||
<option value="week">Týden</option>
|
<option value="week">Týden</option>
|
||||||
<option value="day">Den</option>
|
<option value="day">Den</option>
|
||||||
</Form.Select>
|
</select>
|
||||||
</Form.Group>
|
</div>
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
|
variant="info"
|
||||||
onClick={handleRefresh}
|
onClick={handleRefresh}
|
||||||
disabled={refreshLoading}
|
disabled={refreshLoading}
|
||||||
|
className="mb-3"
|
||||||
>
|
>
|
||||||
{refreshLoading ? 'Načítám...' : 'Obnovit menu'}
|
{refreshLoading ? 'Refreshing...' : 'Refresh'}
|
||||||
</Button>
|
</Button>
|
||||||
</Modal.Body>
|
</Modal.Body>
|
||||||
<Modal.Footer>
|
<Modal.Footer>
|
||||||
|
|||||||
@@ -1,238 +1,42 @@
|
|||||||
import { useEffect, useRef, useState } from "react";
|
import { useRef } from "react";
|
||||||
import { Modal, Button, Form } from "react-bootstrap"
|
import { Modal, Button } from "react-bootstrap"
|
||||||
import { useSettings, ThemePreference } from "../../context/settings";
|
import { useSettings } from "../../context/settings";
|
||||||
import { NotificationSettings, UdalostEnum, getNotificationSettings, updateNotificationSettings } from "../../../../types";
|
|
||||||
import { useAuth } from "../../context/auth";
|
|
||||||
import { subscribeToPush, unsubscribeFromPush } from "../../hooks/usePushReminder";
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
isOpen: boolean,
|
isOpen: boolean,
|
||||||
onClose: () => void,
|
onClose: () => void,
|
||||||
onSave: (bankAccountNumber?: string, bankAccountHolderName?: string, hideSoupsOption?: boolean, themePreference?: ThemePreference) => void,
|
onSave: (bankAccountNumber?: string, bankAccountHolderName?: string, hideSoupsOption?: boolean) => void,
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 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 }: Readonly<Props>) {
|
||||||
const auth = useAuth();
|
|
||||||
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);
|
||||||
const hideSoupsRef = useRef<HTMLInputElement>(null);
|
const hideSoupsRef = useRef<HTMLInputElement>(null);
|
||||||
const themeRef = useRef<HTMLSelectElement>(null);
|
|
||||||
|
|
||||||
const reminderTimeRef = useRef<HTMLInputElement>(null);
|
return <Modal show={isOpen} onHide={onClose} size="lg">
|
||||||
const ntfyTopicRef = useRef<HTMLInputElement>(null);
|
<Modal.Header closeButton>
|
||||||
const discordWebhookRef = useRef<HTMLInputElement>(null);
|
<Modal.Title><h2>Nastavení</h2></Modal.Title>
|
||||||
const teamsWebhookRef = useRef<HTMLInputElement>(null);
|
</Modal.Header>
|
||||||
const [notifSettings, setNotifSettings] = useState<NotificationSettings>({});
|
<Modal.Body>
|
||||||
const [enabledEvents, setEnabledEvents] = useState<UdalostEnum[]>([]);
|
<h4>Obecné</h4>
|
||||||
|
<span title="V nabídkách nebudou zobrazovány polévky. Tato funkce je experimentální, a zejména u TechTower bývá často problém polévky spolehlivě rozeznat. V případě využití této funkce průběžně nahlašujte stále se zobrazující polévky." style={{ "cursor": "help" }}>
|
||||||
useEffect(() => {
|
<input ref={hideSoupsRef} type="checkbox" defaultChecked={settings?.hideSoups} /> Skrýt polévky
|
||||||
if (isOpen && auth?.login) {
|
</span>
|
||||||
getNotificationSettings().then(response => {
|
<hr />
|
||||||
if (response.data) {
|
<h4>Bankovní účet</h4>
|
||||||
setNotifSettings(response.data);
|
<p>Nastavením čísla účtu umožníte automatické generování QR kódů pro úhradu za vámi provedené objednávky v rámci Pizza day.<br />Pokud vaše číslo účtu neobsahuje předčíslí, je možné ho zcela vynechat.<br /><br />Číslo účtu není ukládáno na serveru, posílá se na něj pouze za účelem vygenerování QR kódů.</p>
|
||||||
setEnabledEvents(response.data.enabledEvents ?? []);
|
Číslo účtu: <input className="mb-3" ref={bankAccountRef} type="text" placeholder="123456-1234567890/1234" defaultValue={settings?.bankAccount} onKeyDown={e => e.stopPropagation()} /> <br />
|
||||||
}
|
Název příjemce (jméno majitele účtu): <input ref={nameRef} type="text" placeholder="Jan Novák" defaultValue={settings?.holderName} onKeyDown={e => e.stopPropagation()} />
|
||||||
}).catch(() => {});
|
</Modal.Body>
|
||||||
}
|
<Modal.Footer>
|
||||||
}, [isOpen, auth?.login]);
|
<Button variant="secondary" onClick={onClose}>
|
||||||
|
Storno
|
||||||
const toggleEvent = (event: UdalostEnum) => {
|
</Button>
|
||||||
setEnabledEvents(prev =>
|
<Button variant="primary" onClick={() => onSave(bankAccountRef.current?.value, nameRef.current?.value, hideSoupsRef.current?.checked)}>
|
||||||
prev.includes(event) ? prev.filter(e => e !== event) : [...prev, event]
|
Uložit
|
||||||
);
|
</Button>
|
||||||
};
|
</Modal.Footer>
|
||||||
|
</Modal>
|
||||||
const handleSave = async () => {
|
|
||||||
const newReminderTime = reminderTimeRef.current?.value || undefined;
|
|
||||||
const oldReminderTime = notifSettings.reminderTime;
|
|
||||||
|
|
||||||
// Uložení notifikačních nastavení na server
|
|
||||||
await updateNotificationSettings({
|
|
||||||
body: {
|
|
||||||
ntfyTopic: ntfyTopicRef.current?.value || undefined,
|
|
||||||
discordWebhookUrl: discordWebhookRef.current?.value || undefined,
|
|
||||||
teamsWebhookUrl: teamsWebhookRef.current?.value || undefined,
|
|
||||||
enabledEvents,
|
|
||||||
reminderTime: newReminderTime,
|
|
||||||
}
|
|
||||||
}).catch(() => {});
|
|
||||||
|
|
||||||
// Správa push subscription pro připomínky
|
|
||||||
if (newReminderTime && newReminderTime !== oldReminderTime) {
|
|
||||||
subscribeToPush(newReminderTime);
|
|
||||||
} else if (!newReminderTime && oldReminderTime) {
|
|
||||||
unsubscribeFromPush();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uložení ostatních nastavení (localStorage)
|
|
||||||
onSave(
|
|
||||||
bankAccountRef.current?.value,
|
|
||||||
nameRef.current?.value,
|
|
||||||
hideSoupsRef.current?.checked,
|
|
||||||
themeRef.current?.value as ThemePreference,
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose} size="lg">
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Nastavení</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
<h4>Vzhled</h4>
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Barevný motiv</Form.Label>
|
|
||||||
<Form.Select ref={themeRef} defaultValue={settings?.themePreference}>
|
|
||||||
<option value="system">Podle systému</option>
|
|
||||||
<option value="light">Světlý</option>
|
|
||||||
<option value="dark">Tmavý</option>
|
|
||||||
</Form.Select>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<hr />
|
|
||||||
|
|
||||||
<h4>Obecné</h4>
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Check
|
|
||||||
id="hideSoupsCheckbox"
|
|
||||||
ref={hideSoupsRef}
|
|
||||||
type="checkbox"
|
|
||||||
label="Skrýt polévky"
|
|
||||||
defaultChecked={settings?.hideSoups}
|
|
||||||
title="V nabídkách nebudou zobrazovány polévky. Tato funkce je experimentální."
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Experimentální funkce - zejména u TechTower bývá problém polévky spolehlivě rozeznat.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<hr />
|
|
||||||
|
|
||||||
<h4>Notifikace</h4>
|
|
||||||
<p>
|
|
||||||
Nastavením notifikací budete dostávat upozornění o událostech (např. "Jdeme na oběd") přímo do vámi zvoleného komunikačního kanálu.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Připomínka výběru oběda</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={reminderTimeRef}
|
|
||||||
type="time"
|
|
||||||
defaultValue={notifSettings.reminderTime ?? ''}
|
|
||||||
key={notifSettings.reminderTime ?? 'reminder-empty'}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
V zadaný čas vám přijde push notifikace, pokud nemáte zvolenou možnost stravování. Nechte prázdné pro vypnutí.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>ntfy téma (topic)</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={ntfyTopicRef}
|
|
||||||
type="text"
|
|
||||||
placeholder="moje-tema"
|
|
||||||
defaultValue={notifSettings.ntfyTopic}
|
|
||||||
key={notifSettings.ntfyTopic ?? 'ntfy-empty'}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Téma pro ntfy push notifikace. Nechte prázdné pro vypnutí.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Discord webhook URL</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={discordWebhookRef}
|
|
||||||
type="text"
|
|
||||||
placeholder="https://discord.com/api/webhooks/..."
|
|
||||||
defaultValue={notifSettings.discordWebhookUrl}
|
|
||||||
key={notifSettings.discordWebhookUrl ?? 'discord-empty'}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
URL webhooku Discord kanálu. Nechte prázdné pro vypnutí.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>MS Teams webhook URL</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={teamsWebhookRef}
|
|
||||||
type="text"
|
|
||||||
placeholder="https://outlook.office.com/webhook/..."
|
|
||||||
defaultValue={notifSettings.teamsWebhookUrl}
|
|
||||||
key={notifSettings.teamsWebhookUrl ?? 'teams-empty'}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
URL webhooku MS Teams kanálu. Nechte prázdné pro vypnutí.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Události k odběru</Form.Label>
|
|
||||||
{Object.values(UdalostEnum).map(event => (
|
|
||||||
<Form.Check
|
|
||||||
key={event}
|
|
||||||
id={`notif-event-${event}`}
|
|
||||||
type="checkbox"
|
|
||||||
label={event}
|
|
||||||
checked={enabledEvents.includes(event)}
|
|
||||||
onChange={() => toggleEvent(event)}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Zvolte události, o kterých chcete být notifikováni. Notifikace jsou odesílány pouze uživatelům se stejnou zvolenou lokalitou.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<hr />
|
|
||||||
|
|
||||||
<h4>Bankovní účet</h4>
|
|
||||||
<p>
|
|
||||||
Nastavením čísla účtu umožníte automatické generování QR kódů pro úhradu za vámi provedené objednávky v rámci Pizza day.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Číslo účtu</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={bankAccountRef}
|
|
||||||
type="text"
|
|
||||||
placeholder="123456-1234567890/1234"
|
|
||||||
defaultValue={settings?.bankAccount}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Pokud vaše číslo účtu neobsahuje předčíslí, je možné ho zcela vynechat.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Název příjemce</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
ref={nameRef}
|
|
||||||
type="text"
|
|
||||||
placeholder="Jan Novák"
|
|
||||||
defaultValue={settings?.holderName}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
<Form.Text className="text-muted">
|
|
||||||
Jméno majitele účtu pro QR platbu.
|
|
||||||
</Form.Text>
|
|
||||||
</Form.Group>
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={onClose}>
|
|
||||||
Storno
|
|
||||||
</Button>
|
|
||||||
<Button onClick={handleSave}>
|
|
||||||
Uložit
|
|
||||||
</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
@@ -1,119 +0,0 @@
|
|||||||
import { useState } from "react";
|
|
||||||
import { Modal, Button, Form, ListGroup, Alert } from "react-bootstrap";
|
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
|
||||||
import { faTrashCan } from "@fortawesome/free-regular-svg-icons";
|
|
||||||
import { addStore, deleteStore } from "../../../../types";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
stores: string[];
|
|
||||||
onStoresChanged: (stores: string[]) => void;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function StoreAdminModal({ isOpen, onClose, stores, onStoresChanged }: Readonly<Props>) {
|
|
||||||
const [newName, setNewName] = useState('');
|
|
||||||
const [heslo, setHeslo] = useState('');
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
const handleAdd = async () => {
|
|
||||||
if (!newName.trim()) return;
|
|
||||||
setError(null);
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const res = await addStore({ body: { name: newName.trim(), heslo } });
|
|
||||||
if (res.error) {
|
|
||||||
setError((res.error as any).error || 'Nastala chyba');
|
|
||||||
} else if (res.data) {
|
|
||||||
onStoresChanged(res.data as string[]);
|
|
||||||
setNewName('');
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleRemove = async (name: string) => {
|
|
||||||
setError(null);
|
|
||||||
setLoading(true);
|
|
||||||
try {
|
|
||||||
const res = await deleteStore({ body: { name, heslo } });
|
|
||||||
if (res.error) {
|
|
||||||
setError((res.error as any).error || 'Nastala chyba');
|
|
||||||
} else if (res.data) {
|
|
||||||
onStoresChanged(res.data as string[]);
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
setError(e.message || 'Nastala chyba');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal show={isOpen} onHide={onClose}>
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title><h2>Správa obchodů</h2></Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
{error && (
|
|
||||||
<Alert variant="danger" onClose={() => setError(null)} dismissible>
|
|
||||||
{error}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Form.Group className="mb-3">
|
|
||||||
<Form.Label>Admin heslo</Form.Label>
|
|
||||||
<Form.Control
|
|
||||||
type="password"
|
|
||||||
placeholder="Heslo"
|
|
||||||
value={heslo}
|
|
||||||
onChange={e => setHeslo(e.target.value)}
|
|
||||||
onKeyDown={e => e.stopPropagation()}
|
|
||||||
/>
|
|
||||||
</Form.Group>
|
|
||||||
|
|
||||||
<hr />
|
|
||||||
<h6>Přidat obchod</h6>
|
|
||||||
<div className="d-flex gap-2 mb-3">
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
placeholder="Název obchodu"
|
|
||||||
value={newName}
|
|
||||||
onChange={e => setNewName(e.target.value)}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleAdd(); }}
|
|
||||||
/>
|
|
||||||
<Button variant="primary" onClick={handleAdd} disabled={loading || !newName.trim() || !heslo}>
|
|
||||||
Přidat
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h6>Aktuální seznam</h6>
|
|
||||||
{stores.length === 0 ? (
|
|
||||||
<p className="text-muted">Žádné obchody v seznamu</p>
|
|
||||||
) : (
|
|
||||||
<ListGroup>
|
|
||||||
{stores.map(s => (
|
|
||||||
<ListGroup.Item key={s} className="d-flex justify-content-between align-items-center">
|
|
||||||
{s}
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faTrashCan}
|
|
||||||
className="action-icon"
|
|
||||||
title="Odebrat"
|
|
||||||
onClick={() => handleRemove(s)}
|
|
||||||
style={{ cursor: 'pointer' }}
|
|
||||||
/>
|
|
||||||
</ListGroup.Item>
|
|
||||||
))}
|
|
||||||
</ListGroup>
|
|
||||||
)}
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={onClose}>Zavřít</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -55,7 +55,7 @@ function useProvideAuth(): AuthContextProps {
|
|||||||
setLoginName(undefined);
|
setLoginName(undefined);
|
||||||
setTrusted(undefined);
|
setTrusted(undefined);
|
||||||
if (trusted && logoutUrl?.length) {
|
if (trusted && logoutUrl?.length) {
|
||||||
globalThis.location.replace(logoutUrl);
|
window.location.replace(logoutUrl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,19 +3,14 @@ import React, { ReactNode, useContext, useEffect, useState } 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';
|
||||||
const HIDE_SOUPS_KEY = 'hide_soups';
|
const HIDE_SOUPS_KEY = 'hide_soups';
|
||||||
const THEME_KEY = 'theme_preference';
|
|
||||||
|
|
||||||
export type ThemePreference = 'system' | 'light' | 'dark';
|
|
||||||
|
|
||||||
export type SettingsContextProps = {
|
export type SettingsContextProps = {
|
||||||
bankAccount?: string,
|
bankAccount?: string,
|
||||||
holderName?: string,
|
holderName?: string,
|
||||||
hideSoups?: boolean,
|
hideSoups?: boolean,
|
||||||
themePreference: ThemePreference,
|
|
||||||
setBankAccountNumber: (accountNumber?: string) => void,
|
setBankAccountNumber: (accountNumber?: string) => void,
|
||||||
setBankAccountHolderName: (holderName?: string) => void,
|
setBankAccountHolderName: (holderName?: string) => void,
|
||||||
setHideSoupsOption: (hideSoups?: boolean) => void,
|
setHideSoupsOption: (hideSoups?: boolean) => void,
|
||||||
setThemePreference: (theme: ThemePreference) => void,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type ContextProps = {
|
type ContextProps = {
|
||||||
@@ -33,23 +28,10 @@ export const useSettings = () => {
|
|||||||
return useContext(settingsContext);
|
return useContext(settingsContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getInitialTheme(): ThemePreference {
|
|
||||||
try {
|
|
||||||
const saved = localStorage.getItem(THEME_KEY) as ThemePreference | null;
|
|
||||||
if (saved && ['system', 'light', 'dark'].includes(saved)) {
|
|
||||||
return saved;
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
// localStorage nedostupný
|
|
||||||
}
|
|
||||||
return 'system';
|
|
||||||
}
|
|
||||||
|
|
||||||
function useProvideSettings(): SettingsContextProps {
|
function useProvideSettings(): SettingsContextProps {
|
||||||
const [bankAccount, setBankAccount] = useState<string | undefined>();
|
const [bankAccount, setBankAccount] = useState<string | undefined>();
|
||||||
const [holderName, setHolderName] = useState<string | undefined>();
|
const [holderName, setHolderName] = useState<string | undefined>();
|
||||||
const [hideSoups, setHideSoups] = useState<boolean | undefined>();
|
const [hideSoups, setHideSoups] = useState<boolean | undefined>();
|
||||||
const [themePreference, setTheme] = useState<ThemePreference>(getInitialTheme);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const accountNumber = localStorage.getItem(BANK_ACCOUNT_NUMBER_KEY);
|
const accountNumber = localStorage.getItem(BANK_ACCOUNT_NUMBER_KEY);
|
||||||
@@ -90,29 +72,6 @@ function useProvideSettings(): SettingsContextProps {
|
|||||||
}
|
}
|
||||||
}, [hideSoups]);
|
}, [hideSoups]);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
localStorage.setItem(THEME_KEY, themePreference);
|
|
||||||
}, [themePreference]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const applyTheme = (theme: 'light' | 'dark') => {
|
|
||||||
document.documentElement.setAttribute('data-bs-theme', theme);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (themePreference === 'system') {
|
|
||||||
const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
|
||||||
applyTheme(mediaQuery.matches ? 'dark' : 'light');
|
|
||||||
|
|
||||||
const handler = (e: MediaQueryListEvent) => {
|
|
||||||
applyTheme(e.matches ? 'dark' : 'light');
|
|
||||||
};
|
|
||||||
mediaQuery.addEventListener('change', handler);
|
|
||||||
return () => mediaQuery.removeEventListener('change', handler);
|
|
||||||
} else {
|
|
||||||
applyTheme(themePreference);
|
|
||||||
}
|
|
||||||
}, [themePreference]);
|
|
||||||
|
|
||||||
function setBankAccountNumber(bankAccount?: string) {
|
function setBankAccountNumber(bankAccount?: string) {
|
||||||
setBankAccount(bankAccount);
|
setBankAccount(bankAccount);
|
||||||
}
|
}
|
||||||
@@ -125,18 +84,12 @@ function useProvideSettings(): SettingsContextProps {
|
|||||||
setHideSoups(hideSoups);
|
setHideSoups(hideSoups);
|
||||||
}
|
}
|
||||||
|
|
||||||
function setThemePreference(theme: ThemePreference) {
|
|
||||||
setTheme(theme);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
bankAccount,
|
bankAccount,
|
||||||
holderName,
|
holderName,
|
||||||
hideSoups,
|
hideSoups,
|
||||||
themePreference,
|
|
||||||
setBankAccountNumber,
|
setBankAccountNumber,
|
||||||
setBankAccountHolderName,
|
setBankAccountHolderName,
|
||||||
setHideSoupsOption,
|
setHideSoupsOption,
|
||||||
setThemePreference,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,8 +7,8 @@ if (process.env.NODE_ENV === 'development') {
|
|||||||
socketUrl = `http://localhost:3001`;
|
socketUrl = `http://localhost:3001`;
|
||||||
socketPath = undefined;
|
socketPath = undefined;
|
||||||
} else {
|
} else {
|
||||||
socketUrl = `${globalThis.location.host}`;
|
socketUrl = `${window.location.host}`;
|
||||||
socketPath = `${globalThis.location.pathname}socket.io`;
|
socketPath = `${window.location.pathname}socket.io`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const socket = socketio.connect(socketUrl, { path: socketPath, transports: ["websocket"] });
|
export const socket = socketio.connect(socketUrl, { path: socketPath, transports: ["websocket"] });
|
||||||
@@ -18,4 +18,3 @@ 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_PENDING_QR = 'pendingQr';
|
|
||||||
|
|||||||
@@ -1,108 +0,0 @@
|
|||||||
import { getToken } from '../Utils';
|
|
||||||
|
|
||||||
/** Převede base64url VAPID klíč na Uint8Array pro PushManager. */
|
|
||||||
function urlBase64ToUint8Array(base64String: string): Uint8Array {
|
|
||||||
const padding = '='.repeat((4 - (base64String.length % 4)) % 4);
|
|
||||||
const base64 = (base64String + padding).replace(/-/g, '+').replace(/_/g, '/');
|
|
||||||
const rawData = atob(base64);
|
|
||||||
const outputArray = new Uint8Array(rawData.length);
|
|
||||||
for (let i = 0; i < rawData.length; ++i) {
|
|
||||||
outputArray[i] = rawData.charCodeAt(i);
|
|
||||||
}
|
|
||||||
return outputArray;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Helper pro autorizované API volání na push endpointy. */
|
|
||||||
async function pushApiFetch(path: string, options: RequestInit = {}): Promise<Response> {
|
|
||||||
const token = getToken();
|
|
||||||
return fetch(`/api/notifications/push${path}`, {
|
|
||||||
...options,
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
...(token ? { Authorization: `Bearer ${token}` } : {}),
|
|
||||||
...options.headers,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Zaregistruje service worker, přihlásí se k push notifikacím
|
|
||||||
* a odešle subscription na server.
|
|
||||||
*/
|
|
||||||
export async function subscribeToPush(reminderTime: string): Promise<boolean> {
|
|
||||||
if (!('serviceWorker' in navigator) || !('PushManager' in window)) {
|
|
||||||
console.warn('Push notifikace nejsou v tomto prohlížeči podporovány');
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Registrace service workeru
|
|
||||||
const registration = await navigator.serviceWorker.register('/sw.js');
|
|
||||||
await navigator.serviceWorker.ready;
|
|
||||||
|
|
||||||
// Vyžádání oprávnění
|
|
||||||
const permission = await Notification.requestPermission();
|
|
||||||
if (permission !== 'granted') {
|
|
||||||
console.warn('Push notifikace: oprávnění zamítnuto');
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Získání VAPID veřejného klíče ze serveru
|
|
||||||
const vapidResponse = await pushApiFetch('/vapidKey');
|
|
||||||
if (!vapidResponse.ok) {
|
|
||||||
console.error('Push notifikace: nepodařilo se získat VAPID klíč');
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const { key: vapidPublicKey } = await vapidResponse.json();
|
|
||||||
|
|
||||||
// Přihlášení k push
|
|
||||||
const subscription = await registration.pushManager.subscribe({
|
|
||||||
userVisibleOnly: true,
|
|
||||||
applicationServerKey: urlBase64ToUint8Array(vapidPublicKey) as BufferSource,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Odeslání subscription na server
|
|
||||||
const response = await pushApiFetch('/subscribe', {
|
|
||||||
method: 'POST',
|
|
||||||
body: JSON.stringify({
|
|
||||||
subscription: subscription.toJSON(),
|
|
||||||
reminderTime,
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error('Push notifikace: nepodařilo se odeslat subscription na server');
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Push notifikace: úspěšně přihlášeno k připomínkám v', reminderTime);
|
|
||||||
return true;
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Push notifikace: chyba při registraci', error);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Odhlásí se z push notifikací a informuje server.
|
|
||||||
*/
|
|
||||||
export async function unsubscribeFromPush(): Promise<void> {
|
|
||||||
if (!('serviceWorker' in navigator)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const registration = await navigator.serviceWorker.getRegistration('/sw.js');
|
|
||||||
if (registration) {
|
|
||||||
const subscription = await registration.pushManager.getSubscription();
|
|
||||||
if (subscription) {
|
|
||||||
await subscription.unsubscribe();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await pushApiFetch('/unsubscribe', { method: 'POST' });
|
|
||||||
console.log('Push notifikace: úspěšně odhlášeno z připomínek');
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Push notifikace: chyba při odhlášení', error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+1
-19
@@ -7,32 +7,14 @@ body,
|
|||||||
|
|
||||||
body {
|
body {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
|
||||||
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
|
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
|
||||||
sans-serif;
|
sans-serif;
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
line-height: 1.5;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
code {
|
code {
|
||||||
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
|
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
|
||||||
monospace;
|
monospace;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Smooth scrolling */
|
|
||||||
html {
|
|
||||||
scroll-behavior: smooth;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Better focus styles */
|
|
||||||
:focus-visible {
|
|
||||||
outline: 2px solid var(--luncher-primary);
|
|
||||||
outline-offset: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Selection color */
|
|
||||||
::selection {
|
|
||||||
background: var(--luncher-primary-light);
|
|
||||||
color: var(--luncher-primary);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ client.setConfig({
|
|||||||
// Interceptor na vyhození toasteru při chybě
|
// Interceptor na vyhození toasteru při chybě
|
||||||
client.interceptors.response.use(async response => {
|
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
|
// 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.includes("/login")) {
|
if (!response.ok && response.url.indexOf("/login") == -1) {
|
||||||
const json = await response.json();
|
const json = await response.json();
|
||||||
toast.error(json.error, { theme: "colored" });
|
toast.error(json.error, { theme: "colored" });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,606 +0,0 @@
|
|||||||
import { useContext, useEffect, useState } from 'react';
|
|
||||||
import { Alert, Badge, Button, Card, Form, Modal, OverlayTrigger, Table, Tooltip } from 'react-bootstrap';
|
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
|
||||||
import { faTrashCan } from '@fortawesome/free-regular-svg-icons';
|
|
||||||
import { faBasketShopping, faCircleCheck, faGear, faLock, faLockOpen, faSearch, faUserPlus } from '@fortawesome/free-solid-svg-icons';
|
|
||||||
import {
|
|
||||||
ClientData, GroupState, MealSlot, OrderGroup, OrderGroupMember,
|
|
||||||
getData, createGroup, deleteGroup, addGroupMember, removeGroupMember, updateGroupMember, setGroupState, updateGroupTimes,
|
|
||||||
} from '../../../types';
|
|
||||||
import { EVENT_MESSAGE, SocketContext } from '../context/socket';
|
|
||||||
import { useAuth } from '../context/auth';
|
|
||||||
import { useSettings } from '../context/settings';
|
|
||||||
import Login from '../Login';
|
|
||||||
import Header from '../components/Header';
|
|
||||||
import Footer from '../components/Footer';
|
|
||||||
import Loader from '../components/Loader';
|
|
||||||
import StoreAdminModal from '../components/modals/StoreAdminModal';
|
|
||||||
import PayForGroupModal from '../components/modals/PayForGroupModal';
|
|
||||||
import EditGroupFeesModal from '../components/modals/EditGroupFeesModal';
|
|
||||||
|
|
||||||
const SLOT = MealSlot.EXTRA;
|
|
||||||
const TIME_REGEX = /^([01]\d|2[0-3]):[0-5]\d$/;
|
|
||||||
|
|
||||||
function stateBadge(state: GroupState) {
|
|
||||||
const map: Record<GroupState, { bg: string; label: string }> = {
|
|
||||||
[GroupState.OPEN]: { bg: 'success', label: 'Otevřeno' },
|
|
||||||
[GroupState.LOCKED]: { bg: 'warning', label: 'Uzamčeno' },
|
|
||||||
[GroupState.ORDERED]: { bg: 'secondary', label: 'Objednáno' },
|
|
||||||
};
|
|
||||||
const { bg, label } = map[state] ?? { bg: 'light', label: state };
|
|
||||||
return <Badge bg={bg}>{label}</Badge>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function OrderGroupsPage() {
|
|
||||||
const auth = useAuth();
|
|
||||||
const settings = useSettings();
|
|
||||||
const socket = useContext(SocketContext);
|
|
||||||
const [data, setData] = useState<ClientData | undefined>();
|
|
||||||
const [failure, setFailure] = useState(false);
|
|
||||||
const [newGroupName, setNewGroupName] = useState('');
|
|
||||||
const [creating, setCreating] = useState(false);
|
|
||||||
const [adminModalOpen, setAdminModalOpen] = useState(false);
|
|
||||||
const [editAmounts, setEditAmounts] = useState<Record<string, string>>({});
|
|
||||||
const [editNotes, setEditNotes] = useState<Record<string, string>>({});
|
|
||||||
const [editSurcharges, setEditSurcharges] = useState<Record<string, { text: string; amount: string }>>({});
|
|
||||||
const [editTimes, setEditTimes] = useState<Record<string, { orderedAt: string; deliveryAt: string }>>({});
|
|
||||||
const [payModal, setPayModal] = useState<OrderGroup | null>(null);
|
|
||||||
const [feesModal, setFeesModal] = useState<OrderGroup | null>(null);
|
|
||||||
const [confirmOrderGroup, setConfirmOrderGroup] = useState<OrderGroup | null>(null);
|
|
||||||
const [pageError, setPageError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
const fetchData = async () => {
|
|
||||||
try {
|
|
||||||
const r = await getData({ query: { slot: SLOT } });
|
|
||||||
if (r.data) setData(r.data);
|
|
||||||
} catch {
|
|
||||||
setFailure(true);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!auth?.login) return;
|
|
||||||
fetchData();
|
|
||||||
}, [auth?.login]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
socket.on(EVENT_MESSAGE, (newData: ClientData) => {
|
|
||||||
if (newData.slot === SLOT) setData(prev => ({
|
|
||||||
...newData,
|
|
||||||
stores: newData.stores ?? prev?.stores,
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
return () => { socket.off(EVENT_MESSAGE); };
|
|
||||||
}, [socket]);
|
|
||||||
|
|
||||||
const refresh = async (fn: () => Promise<any>): Promise<boolean> => {
|
|
||||||
setPageError(null);
|
|
||||||
const result = await fn();
|
|
||||||
if (result?.error) {
|
|
||||||
setPageError((result.error as any).error || 'Nastala chyba');
|
|
||||||
await fetchData();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (result?.data) {
|
|
||||||
setData(result.data);
|
|
||||||
socket.emit?.('message', result.data as ClientData);
|
|
||||||
}
|
|
||||||
await fetchData();
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleCreate = async () => {
|
|
||||||
if (!newGroupName || !auth?.login) return;
|
|
||||||
setCreating(true);
|
|
||||||
const ok = await refresh(() => createGroup({ body: { name: newGroupName } }));
|
|
||||||
if (ok) setNewGroupName('');
|
|
||||||
setCreating(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleJoin = (groupId: string) =>
|
|
||||||
refresh(() => addGroupMember({ body: { id: groupId } }));
|
|
||||||
|
|
||||||
const handleToggleLock = (group: OrderGroup) => {
|
|
||||||
const next = group.state === GroupState.OPEN ? GroupState.LOCKED : GroupState.OPEN;
|
|
||||||
return refresh(() => setGroupState({ body: { id: group.id, state: next } }));
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleConfirmOrdered = async (group: OrderGroup) => {
|
|
||||||
setConfirmOrderGroup(null);
|
|
||||||
await refresh(() => setGroupState({ body: { id: group.id, state: GroupState.ORDERED } }));
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleRevertOrdered = (group: OrderGroup) =>
|
|
||||||
refresh(() => setGroupState({ body: { id: group.id, state: GroupState.LOCKED } }));
|
|
||||||
|
|
||||||
const handleDelete = (groupId: string) =>
|
|
||||||
refresh(() => deleteGroup({ body: { id: groupId } }));
|
|
||||||
|
|
||||||
const handleSaveAmount = async (groupId: string, login: string) => {
|
|
||||||
const key = `${groupId}:${login}`;
|
|
||||||
const raw = editAmounts[key];
|
|
||||||
const n = parseFloat(raw ?? '');
|
|
||||||
if (!raw || isNaN(n) || n < 0) {
|
|
||||||
setPageError('Zadejte platnou kladnou částku');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const ok = await refresh(() => updateGroupMember({ body: { id: groupId, login, amount: Math.round(n * 100) } }));
|
|
||||||
if (ok) setEditAmounts(prev => { const next = { ...prev }; delete next[key]; return next; });
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSaveNote = async (groupId: string, login: string) => {
|
|
||||||
const key = `${groupId}:${login}`;
|
|
||||||
const note = editNotes[key] ?? '';
|
|
||||||
const ok = await refresh(() => updateGroupMember({ body: { id: groupId, login, note } }));
|
|
||||||
if (ok) setEditNotes(prev => { const next = { ...prev }; delete next[key]; return next; });
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSaveSurcharge = async (groupId: string, login: string) => {
|
|
||||||
const key = `${groupId}:${login}`;
|
|
||||||
const surchargeText = editSurcharges[key]?.text ?? '';
|
|
||||||
const rawAmount = editSurcharges[key]?.amount ?? '';
|
|
||||||
const surchargeAmount = rawAmount === '' ? 0 : parseFloat(rawAmount.replace(',', '.'));
|
|
||||||
if (rawAmount !== '' && (isNaN(surchargeAmount) || surchargeAmount < 0)) {
|
|
||||||
setPageError('Zadejte platnou výši příplatku');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const ok = await refresh(() => updateGroupMember({ body: { id: groupId, login, surchargeText, surchargeAmount: rawAmount === '' ? 0 : Math.round(surchargeAmount * 100) } }));
|
|
||||||
if (ok) setEditSurcharges(prev => { const next = { ...prev }; delete next[key]; return next; });
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSaveTimes = async (group: OrderGroup) => {
|
|
||||||
const times = editTimes[group.id];
|
|
||||||
if (!times) return;
|
|
||||||
const { orderedAt, deliveryAt } = times;
|
|
||||||
if (orderedAt && !TIME_REGEX.test(orderedAt)) {
|
|
||||||
setPageError('Čas objednání musí být ve formátu HH:MM');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (deliveryAt && !TIME_REGEX.test(deliveryAt)) {
|
|
||||||
setPageError('Čas doručení musí být ve formátu HH:MM');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const ok = await refresh(() => updateGroupTimes({ body: { id: group.id, orderedAt, deliveryAt } }));
|
|
||||||
if (ok) setEditTimes(prev => { const next = { ...prev }; delete next[group.id]; return next; });
|
|
||||||
};
|
|
||||||
|
|
||||||
const canEditMember = (group: OrderGroup, targetLogin: string) => {
|
|
||||||
if (group.state === GroupState.ORDERED) return false;
|
|
||||||
if (auth?.login === group.creatorLogin) return true;
|
|
||||||
if (auth?.login === targetLogin && group.state === GroupState.OPEN) return true;
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
const canManageMembers = (group: OrderGroup) => {
|
|
||||||
if (group.state === GroupState.ORDERED) return false;
|
|
||||||
if (auth?.login === group.creatorLogin) return true;
|
|
||||||
return group.state === GroupState.OPEN;
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!auth?.login) return <Login />;
|
|
||||||
|
|
||||||
if (failure) return (
|
|
||||||
<Loader icon={faSearch} description="Nepodařilo se načíst data" animation="fa-beat" />
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!data) return (
|
|
||||||
<Loader icon={faSearch} description="Načítám..." animation="fa-bounce" />
|
|
||||||
);
|
|
||||||
|
|
||||||
const stores = data.stores ?? [];
|
|
||||||
const groups = data.groups ?? [];
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="app-container">
|
|
||||||
<Header choices={data.choices} />
|
|
||||||
<div className="wrapper">
|
|
||||||
<div className="d-flex align-items-center justify-content-between mb-1">
|
|
||||||
<h1 className="title mb-0">Objednání</h1>
|
|
||||||
<Button variant="outline-secondary" size="sm" onClick={() => setAdminModalOpen(true)} title="Správa obchodů">
|
|
||||||
<FontAwesomeIcon icon={faGear} />
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
<p style={{ color: 'var(--luncher-text-muted)' }}>Skupinové objednávky z obchodů a restaurací</p>
|
|
||||||
|
|
||||||
{pageError && (
|
|
||||||
<Alert variant="danger" dismissible onClose={() => setPageError(null)} className="mt-2">
|
|
||||||
{pageError}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div className="content-wrapper">
|
|
||||||
<div className="content" style={{ maxWidth: 1200 }}>
|
|
||||||
{/* Vytvoření nové skupiny */}
|
|
||||||
<div className="choice-section fade-in mb-4">
|
|
||||||
<h5>Vytvořit skupinu</h5>
|
|
||||||
{stores.length === 0 ? (
|
|
||||||
<p className="text-muted">
|
|
||||||
Nejsou přidány žádné obchody.{' '}
|
|
||||||
<Button variant="link" size="sm" className="p-0" onClick={() => setAdminModalOpen(true)}>
|
|
||||||
Přidat obchod
|
|
||||||
</Button>
|
|
||||||
</p>
|
|
||||||
) : (
|
|
||||||
<div className="d-flex gap-2 align-items-end flex-wrap">
|
|
||||||
<Form.Select
|
|
||||||
value={newGroupName}
|
|
||||||
onChange={e => setNewGroupName(e.target.value)}
|
|
||||||
style={{ maxWidth: 260 }}
|
|
||||||
>
|
|
||||||
<option value="">— vyberte obchod —</option>
|
|
||||||
{stores.map(s => <option key={s} value={s}>{s}</option>)}
|
|
||||||
</Form.Select>
|
|
||||||
<Button variant="primary" onClick={handleCreate} disabled={creating || !newGroupName}>
|
|
||||||
Vytvořit skupinu
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Seznam skupin */}
|
|
||||||
{groups.length === 0 && (
|
|
||||||
<p className="text-muted fade-in">Zatím žádné skupiny pro dnešní den.</p>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{groups.map(group => {
|
|
||||||
const login = auth!.login ?? '';
|
|
||||||
const isCreator = login === group.creatorLogin;
|
|
||||||
const isMember = login in group.members;
|
|
||||||
const isOrdered = group.state === GroupState.ORDERED;
|
|
||||||
const isLocked = group.state === GroupState.LOCKED;
|
|
||||||
const memberEntries = Object.entries(group.members) as [string, OrderGroupMember][];
|
|
||||||
const memberCount = memberEntries.length;
|
|
||||||
const editingTimes = group.id in editTimes;
|
|
||||||
|
|
||||||
const totalFees = (group.fees ?? 0) + (group.shipping ?? 0) + (group.tip ?? 0);
|
|
||||||
const feeShare = memberCount > 0 ? Math.round(totalFees / memberCount) : 0;
|
|
||||||
const getMemberTotal = (m: OrderGroupMember) => {
|
|
||||||
const base = m.amount ?? 0;
|
|
||||||
const surcharge = m.surchargeAmount ?? 0;
|
|
||||||
const dv = group.discountValue ?? 0;
|
|
||||||
const discount = dv > 0
|
|
||||||
? (group.discountType === 'percent'
|
|
||||||
? Math.round((base + surcharge) * dv / 100)
|
|
||||||
: Math.round(dv / memberCount))
|
|
||||||
: 0;
|
|
||||||
return base + surcharge + feeShare - discount;
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Card key={group.id} className="mb-3 fade-in">
|
|
||||||
<Card.Header className="d-flex justify-content-between align-items-center">
|
|
||||||
<div className="d-flex align-items-center gap-2">
|
|
||||||
<strong>{group.name}</strong>
|
|
||||||
{stateBadge(group.state)}
|
|
||||||
<small className="text-muted">zakladatel: {group.creatorLogin}</small>
|
|
||||||
</div>
|
|
||||||
<div className="d-flex gap-2">
|
|
||||||
{isCreator && !isOrdered && (
|
|
||||||
<>
|
|
||||||
<Button variant="outline-info" size="sm" onClick={() => setFeesModal(group)} title="Upravit poplatky a slevu">
|
|
||||||
Poplatky
|
|
||||||
</Button>
|
|
||||||
<Button variant="outline-secondary" size="sm" onClick={() => handleToggleLock(group)} title={isLocked ? 'Odemknout' : 'Uzamknout'}>
|
|
||||||
<FontAwesomeIcon icon={isLocked ? faLockOpen : faLock} />
|
|
||||||
</Button>
|
|
||||||
{isLocked && (
|
|
||||||
<Button variant="outline-primary" size="sm" onClick={() => setConfirmOrderGroup(group)}>
|
|
||||||
Objednáno
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
<Button variant="outline-danger" size="sm" onClick={() => handleDelete(group.id)} title="Smazat skupinu">
|
|
||||||
<FontAwesomeIcon icon={faTrashCan} />
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{isCreator && isOrdered && (
|
|
||||||
<>
|
|
||||||
{settings?.bankAccount && settings?.holderName && !group.qrGenerated && (
|
|
||||||
<Button variant="primary" size="sm" onClick={() => setPayModal(group)}>
|
|
||||||
<FontAwesomeIcon icon={faBasketShopping} className="me-1" />
|
|
||||||
Generovat QR
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
<Button variant="outline-warning" size="sm" onClick={() => handleRevertOrdered(group)} title="Vrátit na Uzamčeno (smaže QR kódy)">
|
|
||||||
<FontAwesomeIcon icon={faLockOpen} />
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{!isMember && !isOrdered && !isLocked && (
|
|
||||||
<Button variant="outline-success" size="sm" onClick={() => handleJoin(group.id)}>
|
|
||||||
<FontAwesomeIcon icon={faUserPlus} className="me-1" />
|
|
||||||
Přidat se
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</Card.Header>
|
|
||||||
<Card.Body className="p-0">
|
|
||||||
<Table className="mb-0" size="sm">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>Člen</th>
|
|
||||||
<th style={{ width: 180 }}>Částka (bez slev)</th>
|
|
||||||
<th style={{ width: 220 }}>Příplatek</th>
|
|
||||||
<th>Poznámka</th>
|
|
||||||
<th style={{ width: 160 }}>Celkem (s poplatky)</th>
|
|
||||||
<th style={{ width: 40 }}></th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{memberEntries.map(([memberLogin, member]) => {
|
|
||||||
const key = `${group.id}:${memberLogin}`;
|
|
||||||
const editingAmount = key in editAmounts;
|
|
||||||
const editingNote = key in editNotes;
|
|
||||||
const editingSurcharge = key in editSurcharges;
|
|
||||||
const canEdit = canEditMember(group, memberLogin);
|
|
||||||
const memberTotal = getMemberTotal(member);
|
|
||||||
return (
|
|
||||||
<tr key={memberLogin}>
|
|
||||||
<td>
|
|
||||||
<span className="user-info">
|
|
||||||
<strong>{memberLogin}</strong>
|
|
||||||
{memberLogin === group.creatorLogin && (
|
|
||||||
<OverlayTrigger placement="top" overlay={<Tooltip>Zakladatel / objednávající</Tooltip>}>
|
|
||||||
<span className="ms-1"><FontAwesomeIcon icon={faBasketShopping} className="buyer-icon" /></span>
|
|
||||||
</OverlayTrigger>
|
|
||||||
)}
|
|
||||||
{member.paid && (
|
|
||||||
<OverlayTrigger placement="top" overlay={<Tooltip>Zaplaceno</Tooltip>}>
|
|
||||||
<span className="ms-1"><FontAwesomeIcon icon={faCircleCheck} className="text-success" /></span>
|
|
||||||
</OverlayTrigger>
|
|
||||||
)}
|
|
||||||
</span>
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
{canEdit && editingAmount ? (
|
|
||||||
<div className="d-flex gap-1">
|
|
||||||
<Form.Control
|
|
||||||
type="number"
|
|
||||||
size="sm"
|
|
||||||
value={editAmounts[key]}
|
|
||||||
onChange={e => setEditAmounts(prev => ({ ...prev, [key]: e.target.value }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveAmount(group.id, memberLogin); if (e.key === 'Escape') setEditAmounts(prev => { const n = { ...prev }; delete n[key]; return n; }); }}
|
|
||||||
style={{ width: 95 }}
|
|
||||||
autoFocus
|
|
||||||
/>
|
|
||||||
<Button size="sm" variant="outline-success" onClick={() => handleSaveAmount(group.id, memberLogin)}>✓</Button>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<span
|
|
||||||
style={{ cursor: canEdit ? 'pointer' : undefined }}
|
|
||||||
onClick={() => canEdit && setEditAmounts(prev => ({ ...prev, [key]: member.amount != null ? String(member.amount / 100) : '' }))}
|
|
||||||
title={canEdit ? 'Klikněte pro úpravu' : undefined}
|
|
||||||
>
|
|
||||||
{member.amount != null ? `${member.amount / 100} Kč` : <span className="text-muted">—</span>}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
{canEdit && editingSurcharge ? (
|
|
||||||
<div className="d-flex gap-1">
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
size="sm"
|
|
||||||
placeholder="popis"
|
|
||||||
value={editSurcharges[key]?.text ?? ''}
|
|
||||||
onChange={e => setEditSurcharges(prev => ({ ...prev, [key]: { ...prev[key], text: e.target.value } }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveSurcharge(group.id, memberLogin); if (e.key === 'Escape') setEditSurcharges(prev => { const n = { ...prev }; delete n[key]; return n; }); }}
|
|
||||||
style={{ width: 80 }}
|
|
||||||
autoFocus
|
|
||||||
/>
|
|
||||||
<Form.Control
|
|
||||||
type="number"
|
|
||||||
size="sm"
|
|
||||||
placeholder="Kč"
|
|
||||||
value={editSurcharges[key]?.amount ?? ''}
|
|
||||||
onChange={e => setEditSurcharges(prev => ({ ...prev, [key]: { ...prev[key], amount: e.target.value } }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveSurcharge(group.id, memberLogin); if (e.key === 'Escape') setEditSurcharges(prev => { const n = { ...prev }; delete n[key]; return n; }); }}
|
|
||||||
style={{ width: 60 }}
|
|
||||||
/>
|
|
||||||
<Button size="sm" variant="outline-success" onClick={() => handleSaveSurcharge(group.id, memberLogin)}>✓</Button>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<span
|
|
||||||
style={{ cursor: canEdit ? 'pointer' : undefined }}
|
|
||||||
onClick={() => canEdit && setEditSurcharges(prev => ({ ...prev, [key]: { text: member.surchargeText ?? '', amount: member.surchargeAmount != null ? String(member.surchargeAmount / 100) : '' } }))}
|
|
||||||
title={canEdit ? 'Klikněte pro úpravu příplatku' : undefined}
|
|
||||||
>
|
|
||||||
{member.surchargeAmount != null && member.surchargeAmount > 0 ? (
|
|
||||||
<small>{member.surchargeText ? `${member.surchargeText}: ` : ''}<strong>{member.surchargeAmount / 100} Kč</strong></small>
|
|
||||||
) : (
|
|
||||||
<small className="text-muted">—</small>
|
|
||||||
)}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
{canEdit && editingNote ? (
|
|
||||||
<div className="d-flex gap-1">
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
size="sm"
|
|
||||||
value={editNotes[key]}
|
|
||||||
onChange={e => setEditNotes(prev => ({ ...prev, [key]: e.target.value }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveNote(group.id, memberLogin); if (e.key === 'Escape') setEditNotes(prev => { const n = { ...prev }; delete n[key]; return n; }); }}
|
|
||||||
autoFocus
|
|
||||||
/>
|
|
||||||
<Button size="sm" variant="outline-success" onClick={() => handleSaveNote(group.id, memberLogin)}>✓</Button>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<span
|
|
||||||
style={{ cursor: canEdit ? 'pointer' : undefined }}
|
|
||||||
onClick={() => canEdit && setEditNotes(prev => ({ ...prev, [key]: member.note ?? '' }))}
|
|
||||||
title={canEdit ? 'Klikněte pro úpravu poznámky' : undefined}
|
|
||||||
>
|
|
||||||
<small className="text-muted">{member.note || '—'}</small>
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
<td className="text-end">
|
|
||||||
<small className={memberTotal > 0 ? 'fw-bold' : 'text-muted'}>
|
|
||||||
{memberTotal > 0 ? `${memberTotal / 100} Kč` : '—'}
|
|
||||||
</small>
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
<div className="d-flex gap-1 justify-content-end">
|
|
||||||
{canManageMembers(group) && (isCreator || memberLogin === login) && (memberLogin !== group.creatorLogin) && (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faTrashCan}
|
|
||||||
className="action-icon"
|
|
||||||
title={memberLogin === login ? 'Odhlásit se' : 'Odebrat z skupiny'}
|
|
||||||
onClick={() => refresh(() => removeGroupMember({ body: { id: group.id, login: memberLogin } }))}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</tbody>
|
|
||||||
{(() => {
|
|
||||||
const sumBase = memberEntries.reduce((sum, [, m]) => sum + (m.amount ?? 0) + (m.surchargeAmount ?? 0), 0);
|
|
||||||
const dv = group.discountValue ?? 0;
|
|
||||||
const totalDiscount = dv > 0
|
|
||||||
? (group.discountType === 'percent' ? Math.round(sumBase * dv / 100) : dv)
|
|
||||||
: 0;
|
|
||||||
const groupTotal = sumBase + totalFees - totalDiscount;
|
|
||||||
return groupTotal > 0 ? (
|
|
||||||
<tfoot>
|
|
||||||
<tr style={{ fontWeight: 700, borderTop: '2px solid var(--luncher-border)' }}>
|
|
||||||
<td colSpan={4} className="text-end" style={{ fontSize: '0.9em' }}>Celkem za skupinu:</td>
|
|
||||||
<td className="text-end">{groupTotal / 100} Kč</td>
|
|
||||||
<td></td>
|
|
||||||
</tr>
|
|
||||||
</tfoot>
|
|
||||||
) : null;
|
|
||||||
})()}
|
|
||||||
</Table>
|
|
||||||
|
|
||||||
{/* Souhrn poplatků a slevy */}
|
|
||||||
{(totalFees > 0 || (group.discountValue != null && group.discountValue > 0)) && (
|
|
||||||
<div className="px-3 py-2 border-top d-flex gap-3 flex-wrap" style={{ fontSize: '0.85em', color: 'var(--luncher-text-muted)' }}>
|
|
||||||
{group.fees != null && group.fees > 0 && <span>Poplatky: <strong>{group.fees / 100} Kč</strong></span>}
|
|
||||||
{group.shipping != null && group.shipping > 0 && <span>Doprava: <strong>{group.shipping / 100} Kč</strong></span>}
|
|
||||||
{group.tip != null && group.tip > 0 && <span>Spropitné: <strong>{group.tip / 100} Kč</strong></span>}
|
|
||||||
{feeShare > 0 && <span>→ <strong>{feeShare / 100} Kč</strong>/os.</span>}
|
|
||||||
{group.discountValue != null && group.discountValue > 0 && (
|
|
||||||
<span className="text-success">
|
|
||||||
Sleva: <strong>{group.discountType === 'percent' ? `${group.discountValue}%` : `${group.discountValue / 100} Kč`}</strong>
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Časy objednání a doručení */}
|
|
||||||
{isOrdered && (
|
|
||||||
<div className="px-3 py-2 border-top">
|
|
||||||
{isCreator && editingTimes ? (
|
|
||||||
<div className="d-flex align-items-center gap-3 flex-wrap">
|
|
||||||
<div className="d-flex align-items-center gap-1">
|
|
||||||
<small className="text-muted text-nowrap">Objednáno v:</small>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
size="sm"
|
|
||||||
placeholder="HH:MM"
|
|
||||||
value={editTimes[group.id]?.orderedAt ?? ''}
|
|
||||||
onChange={e => setEditTimes(prev => ({ ...prev, [group.id]: { ...prev[group.id], orderedAt: e.target.value } }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveTimes(group); }}
|
|
||||||
style={{ width: 75 }}
|
|
||||||
autoFocus
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="d-flex align-items-center gap-1">
|
|
||||||
<small className="text-muted text-nowrap">Doručení v:</small>
|
|
||||||
<Form.Control
|
|
||||||
type="text"
|
|
||||||
size="sm"
|
|
||||||
placeholder="HH:MM"
|
|
||||||
value={editTimes[group.id]?.deliveryAt ?? ''}
|
|
||||||
onChange={e => setEditTimes(prev => ({ ...prev, [group.id]: { ...prev[group.id], deliveryAt: e.target.value } }))}
|
|
||||||
onKeyDown={e => { e.stopPropagation(); if (e.key === 'Enter') handleSaveTimes(group); }}
|
|
||||||
style={{ width: 75 }}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<Button size="sm" variant="outline-success" onClick={() => handleSaveTimes(group)}>Uložit</Button>
|
|
||||||
<Button size="sm" variant="outline-secondary" onClick={() => setEditTimes(prev => { const n = { ...prev }; delete n[group.id]; return n; })}>Zrušit</Button>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div
|
|
||||||
className="d-flex align-items-center gap-3 flex-wrap"
|
|
||||||
style={{ cursor: isCreator ? 'pointer' : undefined }}
|
|
||||||
onClick={() => isCreator && setEditTimes(prev => ({ ...prev, [group.id]: { orderedAt: group.orderedAt ?? '', deliveryAt: group.deliveryAt ?? '' } }))}
|
|
||||||
title={isCreator ? 'Klikněte pro úpravu časů' : undefined}
|
|
||||||
>
|
|
||||||
<small className="text-muted">
|
|
||||||
Objednáno v: <strong>{group.orderedAt ?? '—'}</strong>
|
|
||||||
</small>
|
|
||||||
<small className="text-muted">
|
|
||||||
Doručení v: <strong>{group.deliveryAt ?? '—'}</strong>
|
|
||||||
</small>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Card.Body>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<Footer />
|
|
||||||
|
|
||||||
{/* Potvrzovací dialog pro přechod do stavu Objednáno */}
|
|
||||||
<Modal show={!!confirmOrderGroup} onHide={() => setConfirmOrderGroup(null)} centered>
|
|
||||||
<Modal.Header closeButton>
|
|
||||||
<Modal.Title>Potvrdit objednání</Modal.Title>
|
|
||||||
</Modal.Header>
|
|
||||||
<Modal.Body>
|
|
||||||
Opravdu chcete označit skupinu <strong>{confirmOrderGroup?.name}</strong> jako objednanou?
|
|
||||||
Tato akce uzavře skupinu a zaznamená čas objednání.
|
|
||||||
</Modal.Body>
|
|
||||||
<Modal.Footer>
|
|
||||||
<Button variant="secondary" onClick={() => setConfirmOrderGroup(null)}>Zrušit</Button>
|
|
||||||
<Button variant="primary" onClick={() => confirmOrderGroup && handleConfirmOrdered(confirmOrderGroup)}>
|
|
||||||
Objednáno
|
|
||||||
</Button>
|
|
||||||
</Modal.Footer>
|
|
||||||
</Modal>
|
|
||||||
|
|
||||||
<StoreAdminModal
|
|
||||||
isOpen={adminModalOpen}
|
|
||||||
onClose={() => setAdminModalOpen(false)}
|
|
||||||
stores={stores}
|
|
||||||
onStoresChanged={updated => setData(prev => prev ? { ...prev, stores: updated } : prev)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
{payModal && settings?.bankAccount && settings?.holderName && (
|
|
||||||
<PayForGroupModal
|
|
||||||
isOpen={!!payModal}
|
|
||||||
onClose={() => setPayModal(null)}
|
|
||||||
onSuccess={fetchData}
|
|
||||||
group={payModal}
|
|
||||||
groupId={payModal.id}
|
|
||||||
payerLogin={auth.login}
|
|
||||||
bankAccount={settings.bankAccount}
|
|
||||||
bankAccountHolder={settings.holderName}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{feesModal && (
|
|
||||||
<EditGroupFeesModal
|
|
||||||
isOpen={!!feesModal}
|
|
||||||
onClose={() => setFeesModal(null)}
|
|
||||||
group={feesModal}
|
|
||||||
onSaved={newData => {
|
|
||||||
if (newData) {
|
|
||||||
setData(newData);
|
|
||||||
socket.emit?.('message', newData as ClientData);
|
|
||||||
}
|
|
||||||
setFeesModal(null);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -2,154 +2,15 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 32px 24px;
|
padding: 20px;
|
||||||
min-height: calc(100vh - 140px);
|
|
||||||
background: var(--luncher-bg);
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
font-size: 2rem;
|
|
||||||
font-weight: 700;
|
|
||||||
color: var(--luncher-text);
|
|
||||||
margin-bottom: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.week-navigator {
|
.week-navigator {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 24px;
|
font-size: xx-large;
|
||||||
margin-bottom: 32px;
|
|
||||||
|
|
||||||
svg {
|
|
||||||
font-size: 1.5rem;
|
|
||||||
color: var(--luncher-text-secondary);
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 12px;
|
|
||||||
border-radius: 50%;
|
|
||||||
background: var(--luncher-bg-card);
|
|
||||||
box-shadow: var(--luncher-shadow-sm);
|
|
||||||
transition: var(--luncher-transition);
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
color: var(--luncher-primary);
|
|
||||||
background: var(--luncher-primary-light);
|
|
||||||
transform: scale(1.05);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.date-range {
|
.date-range {
|
||||||
margin: 0;
|
margin: 5px 20px;
|
||||||
font-size: 1.25rem;
|
|
||||||
font-weight: 600;
|
|
||||||
color: var(--luncher-text);
|
|
||||||
min-width: 280px;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Chart container
|
|
||||||
.recharts-wrapper {
|
|
||||||
background: var(--luncher-bg-card);
|
|
||||||
border-radius: var(--luncher-radius-lg);
|
|
||||||
box-shadow: var(--luncher-shadow);
|
|
||||||
padding: 24px;
|
|
||||||
border: 1px solid var(--luncher-border-light);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Chart text styling
|
|
||||||
.recharts-cartesian-axis-tick-value {
|
|
||||||
fill: var(--luncher-text-secondary);
|
|
||||||
font-size: 0.85rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.recharts-legend-item-text {
|
|
||||||
color: var(--luncher-text) !important;
|
|
||||||
font-weight: 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
.recharts-tooltip-wrapper {
|
|
||||||
.recharts-default-tooltip {
|
|
||||||
background: var(--luncher-bg-card) !important;
|
|
||||||
border: 1px solid var(--luncher-border) !important;
|
|
||||||
border-radius: var(--luncher-radius-sm) !important;
|
|
||||||
box-shadow: var(--luncher-shadow-lg) !important;
|
|
||||||
|
|
||||||
.recharts-tooltip-label {
|
|
||||||
color: var(--luncher-text) !important;
|
|
||||||
font-weight: 600;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.recharts-tooltip-item {
|
|
||||||
color: var(--luncher-text-secondary) !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.recharts-cartesian-grid-horizontal line,
|
|
||||||
.recharts-cartesian-grid-vertical line {
|
|
||||||
stroke: var(--luncher-border);
|
|
||||||
}
|
|
||||||
|
|
||||||
.voting-stats-section {
|
|
||||||
margin-top: 48px;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 800px;
|
|
||||||
|
|
||||||
h2 {
|
|
||||||
font-size: 1.5rem;
|
|
||||||
font-weight: 700;
|
|
||||||
color: var(--luncher-text);
|
|
||||||
margin-bottom: 16px;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.voting-stats-table {
|
|
||||||
width: 100%;
|
|
||||||
background: var(--luncher-bg-card);
|
|
||||||
border-radius: var(--luncher-radius-lg);
|
|
||||||
box-shadow: var(--luncher-shadow);
|
|
||||||
border: 1px solid var(--luncher-border-light);
|
|
||||||
overflow: hidden;
|
|
||||||
border-collapse: collapse;
|
|
||||||
|
|
||||||
th {
|
|
||||||
background: var(--luncher-primary);
|
|
||||||
color: #ffffff;
|
|
||||||
padding: 12px 20px;
|
|
||||||
text-align: left;
|
|
||||||
font-weight: 600;
|
|
||||||
font-size: 0.9rem;
|
|
||||||
|
|
||||||
&:last-child {
|
|
||||||
text-align: center;
|
|
||||||
width: 120px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
td {
|
|
||||||
padding: 12px 20px;
|
|
||||||
border-bottom: 1px solid var(--luncher-border-light);
|
|
||||||
color: var(--luncher-text);
|
|
||||||
font-size: 0.9rem;
|
|
||||||
|
|
||||||
&:last-child {
|
|
||||||
text-align: center;
|
|
||||||
font-weight: 600;
|
|
||||||
color: var(--luncher-primary);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tbody tr {
|
|
||||||
transition: var(--luncher-transition);
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: var(--luncher-bg-hover);
|
|
||||||
}
|
|
||||||
|
|
||||||
&:last-child td {
|
|
||||||
border-bottom: none;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
import { useCallback, useEffect, useState } from "react";
|
||||||
import Footer from "../components/Footer";
|
import Footer from "../components/Footer";
|
||||||
import Header from "../components/Header";
|
import Header from "../components/Header";
|
||||||
import { useAuth } from "../context/auth";
|
import { useAuth } from "../context/auth";
|
||||||
import Login from "../Login";
|
import Login from "../Login";
|
||||||
import { formatDate, getFirstWorkDayOfWeek, getHumanDate, getLastWorkDayOfWeek } from "../Utils";
|
import { formatDate, getFirstWorkDayOfWeek, getHumanDate, getLastWorkDayOfWeek } from "../Utils";
|
||||||
import { WeeklyStats, LunchChoice, VotingStats, FeatureRequest, getStats, getVotingStats } from "../../../types";
|
import { WeeklyStats, LunchChoice, getStats } from "../../../types";
|
||||||
import Loader from "../components/Loader";
|
import Loader from "../components/Loader";
|
||||||
import { faChevronLeft, faChevronRight, faGear } from "@fortawesome/free-solid-svg-icons";
|
import { faChevronLeft, faChevronRight, faGear } from "@fortawesome/free-solid-svg-icons";
|
||||||
import { Legend, Line, LineChart, Tooltip, XAxis, YAxis } from "recharts";
|
import { Legend, Line, LineChart, Tooltip, XAxis, YAxis } from "recharts";
|
||||||
@@ -17,22 +17,22 @@ const CHART_HEIGHT = 700;
|
|||||||
const STROKE_WIDTH = 2.5;
|
const STROKE_WIDTH = 2.5;
|
||||||
|
|
||||||
const COLORS = [
|
const COLORS = [
|
||||||
'#ff1493',
|
// Komentáře jsou kvůli vizualizaci barev ve VS Code
|
||||||
'#1e90ff',
|
'#ff1493', // #ff1493
|
||||||
'#c5a700',
|
'#1e90ff', // #1e90ff
|
||||||
'#006400',
|
'#c5a700', // #c5a700
|
||||||
'#b300ff',
|
'#006400', // #006400
|
||||||
'#ff4500',
|
'#b300ff', // #b300ff
|
||||||
'#bc8f8f',
|
'#ff4500', // #ff4500
|
||||||
'#00ff00',
|
'#bc8f8f', // #bc8f8f
|
||||||
'#7c7c7c',
|
'#00ff00', // #00ff00
|
||||||
|
'#7c7c7c', // #7c7c7c
|
||||||
]
|
]
|
||||||
|
|
||||||
export default function StatsPage() {
|
export default function StatsPage() {
|
||||||
const auth = useAuth();
|
const auth = useAuth();
|
||||||
const [dateRange, setDateRange] = useState<Date[]>();
|
const [dateRange, setDateRange] = useState<Date[]>();
|
||||||
const [data, setData] = useState<WeeklyStats>();
|
const [data, setData] = useState<WeeklyStats>();
|
||||||
const [votingStats, setVotingStats] = useState<VotingStats>();
|
|
||||||
|
|
||||||
// Prvotní nastavení aktuálního týdne
|
// Prvotní nastavení aktuálního týdne
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -49,19 +49,6 @@ export default function StatsPage() {
|
|||||||
}
|
}
|
||||||
}, [dateRange]);
|
}, [dateRange]);
|
||||||
|
|
||||||
// Načtení statistik hlasování
|
|
||||||
useEffect(() => {
|
|
||||||
getVotingStats().then(response => {
|
|
||||||
setVotingStats(response.data);
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const sortedVotingStats = useMemo(() => {
|
|
||||||
if (!votingStats) return [];
|
|
||||||
return Object.entries(votingStats)
|
|
||||||
.sort((a, b) => (b[1] as number) - (a[1] as number));
|
|
||||||
}, [votingStats]);
|
|
||||||
|
|
||||||
const renderLine = (location: LunchChoice) => {
|
const renderLine = (location: LunchChoice) => {
|
||||||
const index = Object.values(LunchChoice).indexOf(location);
|
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} />
|
return <Line key={location} name={getLunchChoiceName(location)} type="monotone" dataKey={data => data.locations[location] ?? 0} stroke={COLORS[index]} strokeWidth={STROKE_WIDTH} />
|
||||||
@@ -87,20 +74,13 @@ export default function StatsPage() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const isCurrentOrFutureWeek = useMemo(() => {
|
|
||||||
if (!dateRange) return true;
|
|
||||||
const currentWeekEnd = getLastWorkDayOfWeek(new Date());
|
|
||||||
currentWeekEnd.setHours(23, 59, 59, 999);
|
|
||||||
return dateRange[1] >= currentWeekEnd;
|
|
||||||
}, [dateRange]);
|
|
||||||
|
|
||||||
const handleKeyDown = useCallback((e: any) => {
|
const handleKeyDown = useCallback((e: any) => {
|
||||||
if (e.keyCode === 37) {
|
if (e.keyCode === 37) {
|
||||||
handlePreviousWeek();
|
handlePreviousWeek();
|
||||||
} else if (e.keyCode === 39 && !isCurrentOrFutureWeek) {
|
} else if (e.keyCode === 39) {
|
||||||
handleNextWeek()
|
handleNextWeek()
|
||||||
}
|
}
|
||||||
}, [dateRange, isCurrentOrFutureWeek]);
|
}, [dateRange]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
document.addEventListener('keydown', handleKeyDown);
|
document.addEventListener('keydown', handleKeyDown);
|
||||||
@@ -132,7 +112,7 @@ export default function StatsPage() {
|
|||||||
</span>
|
</span>
|
||||||
<h2 className="date-range">{getHumanDate(dateRange[0])} - {getHumanDate(dateRange[1])}</h2>
|
<h2 className="date-range">{getHumanDate(dateRange[0])} - {getHumanDate(dateRange[1])}</h2>
|
||||||
<span title="Následující týden">
|
<span title="Následující týden">
|
||||||
<FontAwesomeIcon icon={faChevronRight} style={{ cursor: "pointer", visibility: isCurrentOrFutureWeek ? "hidden" : "visible" }} onClick={handleNextWeek} />
|
<FontAwesomeIcon icon={faChevronRight} style={{ cursor: "pointer" }} onClick={handleNextWeek} />
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<LineChart width={CHART_WIDTH} height={CHART_HEIGHT} data={data}>
|
<LineChart width={CHART_WIDTH} height={CHART_HEIGHT} data={data}>
|
||||||
@@ -142,27 +122,6 @@ export default function StatsPage() {
|
|||||||
<Tooltip />
|
<Tooltip />
|
||||||
<Legend />
|
<Legend />
|
||||||
</LineChart>
|
</LineChart>
|
||||||
{sortedVotingStats.length > 0 && (
|
|
||||||
<div className="voting-stats-section">
|
|
||||||
<h2>Hlasování o funkcích</h2>
|
|
||||||
<table className="voting-stats-table">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>Funkce</th>
|
|
||||||
<th>Počet hlasů</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{sortedVotingStats.map(([feature, count]) => (
|
|
||||||
<tr key={feature}>
|
|
||||||
<td>{FeatureRequest[feature as keyof typeof FeatureRequest] ?? feature}</td>
|
|
||||||
<td>{count as number}</td>
|
|
||||||
</tr>
|
|
||||||
))}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
<Footer />
|
<Footer />
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
/**
|
|
||||||
* Parsuje cenu ve formátu "135 Kč", "135,50 Kč" nebo "135.50 Kč" na číslo.
|
|
||||||
* Vrátí null při selhání.
|
|
||||||
*/
|
|
||||||
export function parsePriceCzk(raw: string | undefined): number | null {
|
|
||||||
if (!raw) return null;
|
|
||||||
const m = raw.replace(',', '.').match(/(\d+(?:\.\d+)?)/);
|
|
||||||
if (!m) return null;
|
|
||||||
const n = parseFloat(m[1]);
|
|
||||||
return Number.isFinite(n) ? n : null;
|
|
||||||
}
|
|
||||||
+603
-608
File diff suppressed because it is too large
Load Diff
@@ -1,3 +0,0 @@
|
|||||||
node_modules/
|
|
||||||
playwright-report/
|
|
||||||
test-results/
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "@luncher/e2e",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"private": true,
|
|
||||||
"scripts": {
|
|
||||||
"test": "playwright test",
|
|
||||||
"test:ui": "playwright test --ui",
|
|
||||||
"test:debug": "playwright test --debug",
|
|
||||||
"report": "playwright show-report"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@playwright/test": "^1.50.0",
|
|
||||||
"@types/node": "^22.0.0",
|
|
||||||
"typescript": "^5.9.3"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
import { defineConfig, devices } from '@playwright/test';
|
|
||||||
import path from 'path';
|
|
||||||
|
|
||||||
// Use 127.0.0.1 explicitly — on Node.js 18+/Windows, `localhost` may resolve to ::1
|
|
||||||
// (IPv6) while the HTTP server only binds to 0.0.0.0 (IPv4), causing the webServer
|
|
||||||
// readiness poll to time out even though the server is listening.
|
|
||||||
// Port 3099 avoids conflicts with locally running Docker containers on 3001-3003.
|
|
||||||
// Override with E2E_PORT env var if needed.
|
|
||||||
const E2E_PORT = process.env.E2E_PORT ?? '3099';
|
|
||||||
const BASE_URL = process.env.E2E_BASE_URL ?? `http://127.0.0.1:${E2E_PORT}`;
|
|
||||||
|
|
||||||
// Server env vars injected for local runs. In CI these are set at the step level.
|
|
||||||
const serverEnv: Record<string, string> = {
|
|
||||||
NODE_ENV: 'test',
|
|
||||||
MOCK_DATA: 'true',
|
|
||||||
STORAGE: process.env.STORAGE ?? 'json',
|
|
||||||
JWT_SECRET: process.env.JWT_SECRET ?? 'e2e-test-secret-min-32-chars-aaaa',
|
|
||||||
HTTP_REMOTE_USER_ENABLED: 'true',
|
|
||||||
HTTP_REMOTE_USER_HEADER_NAME: 'remote-user',
|
|
||||||
HTTP_REMOTE_TRUSTED_IPS: process.env.HTTP_REMOTE_TRUSTED_IPS ?? '127.0.0.1,::1,::ffff:127.0.0.1',
|
|
||||||
PORT: E2E_PORT,
|
|
||||||
};
|
|
||||||
if (process.env.REDIS_HOST) {
|
|
||||||
serverEnv.REDIS_HOST = process.env.REDIS_HOST;
|
|
||||||
serverEnv.REDIS_PORT = process.env.REDIS_PORT ?? '6379';
|
|
||||||
}
|
|
||||||
|
|
||||||
export default defineConfig({
|
|
||||||
testDir: './tests',
|
|
||||||
timeout: 30_000,
|
|
||||||
retries: process.env.CI ? 1 : 0,
|
|
||||||
workers: 1,
|
|
||||||
reporter: [['list'], ['html', { open: 'never' }]],
|
|
||||||
use: {
|
|
||||||
baseURL: BASE_URL,
|
|
||||||
// Default: every test authenticates as e2e-user via trusted header.
|
|
||||||
// Tests that need the real login form should override this in their own context.
|
|
||||||
extraHTTPHeaders: {
|
|
||||||
'remote-user': 'e2e-user',
|
|
||||||
},
|
|
||||||
trace: 'retain-on-failure',
|
|
||||||
video: 'retain-on-failure',
|
|
||||||
},
|
|
||||||
projects: [
|
|
||||||
{ name: 'chromium', use: { ...devices['Desktop Chrome'] } },
|
|
||||||
{ name: 'firefox', use: { ...devices['Desktop Firefox'] } },
|
|
||||||
],
|
|
||||||
// Pre-built server must be started before tests. In CI the step does this
|
|
||||||
// explicitly. Locally: build types+server+client, cp -r client/dist server/public,
|
|
||||||
// then `cd e2e && yarn test` OR let webServer below do it if reuseExistingServer=true
|
|
||||||
// is set and the server is already running.
|
|
||||||
webServer: {
|
|
||||||
command: 'node dist/server/src/index.js',
|
|
||||||
cwd: path.resolve(__dirname, '../server'),
|
|
||||||
// Poll a dedicated health endpoint — polling '/' can stall in Express 5 when
|
|
||||||
// server/public/ doesn't exist in the working directory (no finalhandler match).
|
|
||||||
url: `http://127.0.0.1:${E2E_PORT}/api/health`,
|
|
||||||
timeout: 15_000,
|
|
||||||
reuseExistingServer: !process.env.CI,
|
|
||||||
env: serverEnv,
|
|
||||||
stdout: 'pipe',
|
|
||||||
stderr: 'pipe',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
import { Page, APIRequestContext } from '@playwright/test';
|
|
||||||
|
|
||||||
/** Přihlásí uživatele přes POST /api/login a uloží JWT do localStorage. */
|
|
||||||
export async function loginViaApi(page: Page, login: string): Promise<void> {
|
|
||||||
const response = await page.request.post('/api/login', {
|
|
||||||
headers: { 'Content-Type': 'application/json', 'remote-user': login },
|
|
||||||
data: {},
|
|
||||||
});
|
|
||||||
const token = await response.json() as string;
|
|
||||||
await page.goto('/');
|
|
||||||
await page.evaluate((t) => localStorage.setItem('token', t), token);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Vyčistí stav dne pro zadaný dayIndex (0=pondělí…4=pátek) přes dev API.
|
|
||||||
* /api/dev/* vyžaduje JWT – nejdřív získáme token přes /api/login.
|
|
||||||
*/
|
|
||||||
export async function clearDay(request: APIRequestContext, dayIndex = 4): Promise<void> {
|
|
||||||
const loginResp = await request.post('/api/login', { data: {} });
|
|
||||||
const token = await loginResp.json() as string;
|
|
||||||
await request.post('/api/dev/clear', {
|
|
||||||
headers: { Authorization: `Bearer ${token}` },
|
|
||||||
data: { dayIndex },
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
|
|
||||||
// Tento test záměrně NEPOUŽÍVÁ trusted-header – testuje reálný login formulář.
|
|
||||||
test.use({ extraHTTPHeaders: {} });
|
|
||||||
|
|
||||||
test('uživatel se přihlásí formulářem a uvidí obsah aplikace', async ({ page }) => {
|
|
||||||
// Server běží s HTTP_REMOTE_USER_ENABLED=true, takže POST /api/login vždy vyžaduje
|
|
||||||
// hlavičku remote-user. Zachytíme požadavky z formuláře (mají tělo s polem login)
|
|
||||||
// a přidáme hlavičku; požadavek auto-loginu (bez těla) projde bez hlavičky a selže,
|
|
||||||
// čímž formulář zůstane viditelný.
|
|
||||||
await page.route('**/api/login', async (route) => {
|
|
||||||
const body = route.request().postData();
|
|
||||||
let login: string | undefined;
|
|
||||||
try { login = body ? JSON.parse(body)?.login : undefined; } catch {}
|
|
||||||
await route.continue({
|
|
||||||
headers: login
|
|
||||||
? { ...route.request().headers(), 'remote-user': login }
|
|
||||||
: route.request().headers(),
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
await page.goto('/');
|
|
||||||
|
|
||||||
// Formulář musí být viditelný – auto-login selhal (nepřišla hlavička)
|
|
||||||
const loginInput = page.locator('#login-input');
|
|
||||||
await expect(loginInput).toBeVisible({ timeout: 10_000 });
|
|
||||||
|
|
||||||
// Vyplnění loginu a odeslání Enterem
|
|
||||||
await loginInput.fill('testuser');
|
|
||||||
await loginInput.press('Enter');
|
|
||||||
|
|
||||||
// Po přihlášení musí zmizet login formulář
|
|
||||||
await expect(loginInput).not.toBeVisible({ timeout: 10_000 });
|
|
||||||
|
|
||||||
// JWT musí být uloženo v localStorage jako 3-dílný token
|
|
||||||
const token = await page.evaluate(() => localStorage.getItem('token'));
|
|
||||||
expect(token).toBeTruthy();
|
|
||||||
expect((token as string).split('.')).toHaveLength(3);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('trusted-header přihlášení proběhne automaticky bez formuláře', async ({ page, context }) => {
|
|
||||||
// Obnoví trusted header (přepíše prázdný extraHTTPHeaders z test.use výše)
|
|
||||||
await context.setExtraHTTPHeaders({ 'remote-user': 'e2e-auto-user' });
|
|
||||||
await page.goto('/');
|
|
||||||
|
|
||||||
// Login formulář by se neměl nikdy zobrazit, nebo se ihned schová
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
const loginInput = page.locator('#login-input');
|
|
||||||
await expect(loginInput).not.toBeVisible({ timeout: 5_000 });
|
|
||||||
});
|
|
||||||
@@ -1,68 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
import { clearDay } from './helpers';
|
|
||||||
|
|
||||||
test.beforeEach(async ({ page, request }) => {
|
|
||||||
// Vyčistíme volby dne, aby testy neovlivnily navzájem
|
|
||||||
await clearDay(request);
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
// Počkáme, až se zobrazí volba stravování
|
|
||||||
await expect(page.locator('.choice-section select').first()).toBeVisible({ timeout: 10_000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('výběr restaurace zobrazí seznam jídel', async ({ page }) => {
|
|
||||||
const locationSelect = page.locator('.choice-section select').first();
|
|
||||||
|
|
||||||
// Vybereme Sladovnickou – mock menu existuje
|
|
||||||
await locationSelect.selectOption('SLADOVNICKA');
|
|
||||||
|
|
||||||
// Po výběru restaurace se zobrazí druhý select s jídly
|
|
||||||
const foodSelect = page.locator('.choice-section select').nth(1);
|
|
||||||
await expect(foodSelect).toBeVisible({ timeout: 5_000 });
|
|
||||||
|
|
||||||
// Select musí obsahovat alespoň 2 možnosti (empty + ≥1 jídlo)
|
|
||||||
const options = foodSelect.locator('option');
|
|
||||||
expect(await options.count()).toBeGreaterThan(1);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('výběr jídla se uloží a přetrvá po reload', async ({ page }) => {
|
|
||||||
const locationSelect = page.locator('.choice-section select').first();
|
|
||||||
await locationSelect.selectOption('SLADOVNICKA');
|
|
||||||
|
|
||||||
const foodSelect = page.locator('.choice-section select').nth(1);
|
|
||||||
await expect(foodSelect).toBeVisible({ timeout: 5_000 });
|
|
||||||
|
|
||||||
// Vybereme první nenulovou možnost
|
|
||||||
const options = await foodSelect.locator('option:not([value=""])').all();
|
|
||||||
if (options.length === 0) {
|
|
||||||
test.skip(); // Mock data nejsou dostupná pro tuto restauraci
|
|
||||||
}
|
|
||||||
const firstValue = await options[0].getAttribute('value');
|
|
||||||
await foodSelect.selectOption({ value: firstValue! });
|
|
||||||
|
|
||||||
// Počkáme, až se volba přenese na server
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// Po reload musí volba přetrvat v tabulce choices
|
|
||||||
await page.reload();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
const choicesTable = page.locator('.choices-table');
|
|
||||||
await expect(choicesTable).toBeVisible({ timeout: 5_000 });
|
|
||||||
await expect(choicesTable.locator('text=Sladovnická')).toBeVisible();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('přepnutí na NEOBEDVAM odstraní výběr restaurace', async ({ page }) => {
|
|
||||||
// Nejprve zvolíme restauraci
|
|
||||||
const locationSelect = page.locator('.choice-section select').first();
|
|
||||||
await locationSelect.selectOption('SLADOVNICKA');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// Přepneme na "Neobědvám"
|
|
||||||
await locationSelect.selectOption('NEOBEDVAM');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// Tabulka choices musí zobrazovat "Neobědvám"
|
|
||||||
const choicesTable = page.locator('.choices-table');
|
|
||||||
await expect(choicesTable).toBeVisible({ timeout: 5_000 });
|
|
||||||
await expect(choicesTable.locator('text=Neobědvám')).toBeVisible();
|
|
||||||
});
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
import { clearDay } from './helpers';
|
|
||||||
|
|
||||||
// Pizza day testy musí běžet sekvenčně (sdílejí stav mock dne)
|
|
||||||
test.describe.serial('pizza day životní cyklus', () => {
|
|
||||||
|
|
||||||
test.beforeEach(async ({ request }) => {
|
|
||||||
// Vyčistíme data mock dne před každým testem
|
|
||||||
await clearDay(request);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('zobrazí sekci Pizza Day bez aktivního dne', async ({ page }) => {
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
// Sekce pizza-section se zobrazí jen pokud má uživatel zvolenou možnost "Pizza day"
|
|
||||||
await page.locator('select').selectOption({ label: 'Pizza day' });
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
const pizzaSection = page.locator('.pizza-section');
|
|
||||||
await expect(pizzaSection).toBeVisible({ timeout: 10_000 });
|
|
||||||
await expect(pizzaSection.locator('text=není aktuálně založen')).toBeVisible();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('vytvoří, uzamkne a dokončí pizza day', async ({ page }) => {
|
|
||||||
// Tento test má více kroků a server při MOCK_DATA=true záměrně zpožďuje scraping pizz o 3s
|
|
||||||
test.setTimeout(60_000);
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
// Sekce pizza-section se zobrazí jen pokud má uživatel zvolenou možnost "Pizza day"
|
|
||||||
await page.locator('select').selectOption({ label: 'Pizza day' });
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// Přijmeme všechny window.confirm() dialogy v celém testu (vytvoření i doručení pizza dne)
|
|
||||||
page.on('dialog', dialog => dialog.accept());
|
|
||||||
|
|
||||||
// --- CREATED ---
|
|
||||||
const createBtn = page.locator('.pizza-section button', { hasText: 'Založit Pizza day' });
|
|
||||||
await expect(createBtn).toBeVisible({ timeout: 10_000 });
|
|
||||||
// Čekáme na odpověď API před reloadem – jinak by reload přerušil probíhající request
|
|
||||||
// Server s MOCK_DATA=true záměrně zpožďuje stahování pizz o 3s, proto velkorysý timeout
|
|
||||||
const createResponse = page.waitForResponse(
|
|
||||||
resp => resp.url().includes('/api/pizzaDay/create'),
|
|
||||||
{ timeout: 15_000 },
|
|
||||||
);
|
|
||||||
await createBtn.click();
|
|
||||||
await createResponse;
|
|
||||||
await page.reload();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page.locator('.pizza-section')).toContainText('spravován uživatelem', { timeout: 5_000 });
|
|
||||||
|
|
||||||
// Přidáme pizzu přes API (obejde komplex SelectSearch)
|
|
||||||
const token = await page.evaluate(() => localStorage.getItem('token'));
|
|
||||||
const addResp = await page.request.post('/api/pizzaDay/add', {
|
|
||||||
headers: { 'Authorization': `Bearer ${token}`, 'Content-Type': 'application/json' },
|
|
||||||
data: { pizzaIndex: 0, pizzaSizeIndex: 0 },
|
|
||||||
});
|
|
||||||
expect(addResp.ok()).toBeTruthy();
|
|
||||||
|
|
||||||
// Reload – server aktualizoval data přes WebSocket, ale reload je jistější
|
|
||||||
await page.reload();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// --- LOCK ---
|
|
||||||
const lockBtn = page.locator('.pizza-section button', { hasText: 'Uzamknout' });
|
|
||||||
await expect(lockBtn).toBeEnabled({ timeout: 5_000 });
|
|
||||||
await lockBtn.click();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page.locator('.pizza-section')).toContainText('uzamčeny', { timeout: 5_000 });
|
|
||||||
|
|
||||||
// --- ORDERED ---
|
|
||||||
const orderBtn = page.locator('.pizza-section button', { hasText: 'Objednáno' });
|
|
||||||
await expect(orderBtn).toBeEnabled({ timeout: 5_000 });
|
|
||||||
await orderBtn.click();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page.locator('.pizza-section')).toContainText('objednány', { timeout: 5_000 });
|
|
||||||
|
|
||||||
// --- DELIVERED ---
|
|
||||||
const deliverBtn = page.locator('.pizza-section button', { hasText: 'Doručeno' });
|
|
||||||
await expect(deliverBtn).toBeVisible({ timeout: 5_000 });
|
|
||||||
await deliverBtn.click();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
await expect(page.locator('.pizza-section')).toContainText('doručeny', { timeout: 5_000 });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,79 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
|
|
||||||
test.beforeEach(async ({ page, request }) => {
|
|
||||||
// Naseedujeme 5 uživatelů pro dnešní den – GenerateQrModal pracuje se stávajícími choices
|
|
||||||
await request.post('/api/dev/generate', { data: { dayIndex: 4, count: 5 } });
|
|
||||||
|
|
||||||
// Přednastavíme bankovní účet v localStorage (SettingsContext čte z LS při inicializaci)
|
|
||||||
await page.goto('/');
|
|
||||||
await page.evaluate(() => {
|
|
||||||
localStorage.setItem('bank_account_number', '2400000000/2010');
|
|
||||||
localStorage.setItem('bank_account_holder_name', 'Test User');
|
|
||||||
});
|
|
||||||
// Reload tak, aby SettingsContext načetl nové hodnoty z localStorage
|
|
||||||
await page.reload();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('Nastavení ukládají číslo účtu a jméno do localStorage', async ({ page }) => {
|
|
||||||
// Otevření nastavení
|
|
||||||
await page.locator('#basic-nav-dropdown').click();
|
|
||||||
await page.locator('text=Nastavení').click();
|
|
||||||
|
|
||||||
// Modal musí být viditelný
|
|
||||||
await expect(page.locator('.modal-title')).toContainText('Nastavení', { timeout: 5_000 });
|
|
||||||
|
|
||||||
// Změníme číslo účtu – pressSequentially zajistí spuštění React onChange na každý znak
|
|
||||||
// Číslo 1000000005 je platné (kontrolní součet mod 11 = 0), jinak by validace zamítla uložení
|
|
||||||
const accountInput = page.getByPlaceholder('123456-1234567890/1234');
|
|
||||||
await accountInput.click({ clickCount: 3 });
|
|
||||||
await accountInput.pressSequentially('1000000005/5500');
|
|
||||||
|
|
||||||
// Změníme jméno
|
|
||||||
const nameInput = page.getByPlaceholder('Jan Novák');
|
|
||||||
await nameInput.click({ clickCount: 3 });
|
|
||||||
await nameInput.pressSequentially('Nové Jméno');
|
|
||||||
|
|
||||||
// Uložíme a počkáme na zavření modalu
|
|
||||||
await page.locator('.modal-footer button', { hasText: 'Uložit' }).click();
|
|
||||||
await expect(page.locator('.modal')).not.toBeVisible({ timeout: 5_000 });
|
|
||||||
|
|
||||||
// Ověříme v localStorage
|
|
||||||
const bankAccount = await page.evaluate(() => localStorage.getItem('bank_account_number'));
|
|
||||||
const holderName = await page.evaluate(() => localStorage.getItem('bank_account_holder_name'));
|
|
||||||
expect(bankAccount).toBe('1000000005/5500');
|
|
||||||
expect(holderName).toBe('Nové Jméno');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('otevře modal Generování QR kódů pokud je nastaven účet', async ({ page }) => {
|
|
||||||
// Otevření dropdown menu
|
|
||||||
await page.locator('#basic-nav-dropdown').click();
|
|
||||||
await page.locator('text=Generování QR kódů').click();
|
|
||||||
|
|
||||||
// Modal se otevře
|
|
||||||
await expect(page.locator('.modal')).toBeVisible({ timeout: 5_000 });
|
|
||||||
// Modal musí obsahovat seznam uživatelů nebo prázdný stav
|
|
||||||
await expect(page.locator('.modal-body')).toBeVisible();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('upozorní pokud není nastaven bankovní účet', async ({ page }) => {
|
|
||||||
// Odebereme nastavení účtu
|
|
||||||
await page.evaluate(() => {
|
|
||||||
localStorage.removeItem('bank_account_number');
|
|
||||||
localStorage.removeItem('bank_account_holder_name');
|
|
||||||
});
|
|
||||||
await page.reload();
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
|
|
||||||
// Dialog místo modalu
|
|
||||||
page.on('dialog', async dialog => {
|
|
||||||
expect(dialog.message()).toContain('číslo účtu');
|
|
||||||
await dialog.accept();
|
|
||||||
});
|
|
||||||
|
|
||||||
await page.locator('#basic-nav-dropdown').click();
|
|
||||||
await page.locator('text=Generování QR kódů').click();
|
|
||||||
|
|
||||||
// Modal se NESMÍ otevřít
|
|
||||||
await expect(page.locator('.modal')).not.toBeVisible({ timeout: 3_000 });
|
|
||||||
});
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
|
|
||||||
test.beforeEach(async ({ page }) => {
|
|
||||||
// Trusted-header login runs automatically when Login mounts.
|
|
||||||
// networkidle zaručí, že fetch('/api/data') byl dokončen.
|
|
||||||
await page.goto('/');
|
|
||||||
await page.waitForLoadState('networkidle');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('zobrazí mock datum 10.01.2025', async ({ page }) => {
|
|
||||||
// MOCK_DATA=true pins today to 2025-01-10
|
|
||||||
await expect(page.locator('text=10.01')).toBeVisible({ timeout: 10_000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('zobrazí čtyři restaurační karty z mock dat', async ({ page }) => {
|
|
||||||
// Každá restaurace je obalena v .restaurant-card
|
|
||||||
const cards = page.locator('.restaurant-card');
|
|
||||||
await expect(cards).toHaveCount(4, { timeout: 10_000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('zobrazí alespoň jedno jídlo v menu každé restaurace', async ({ page }) => {
|
|
||||||
await expect(page.locator('.restaurant-card').first()).toBeVisible({ timeout: 10_000 });
|
|
||||||
|
|
||||||
// Každá karta musí mít aspoň jeden řádek v .food-table
|
|
||||||
const cards = page.locator('.restaurant-card');
|
|
||||||
const count = await cards.count();
|
|
||||||
for (let i = 0; i < count; i++) {
|
|
||||||
const card = cards.nth(i);
|
|
||||||
const rows = card.locator('.food-table tr');
|
|
||||||
expect(await rows.count()).toBeGreaterThan(0);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
test('zobrazí volbu stravování před menu', async ({ page }) => {
|
|
||||||
// Sekce .choice-section obsahuje select pro výběr stravování
|
|
||||||
const choiceSection = page.locator('.choice-section');
|
|
||||||
await expect(choiceSection).toBeVisible({ timeout: 10_000 });
|
|
||||||
await expect(choiceSection.locator('select').first()).toBeVisible();
|
|
||||||
});
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"target": "ES2022",
|
|
||||||
"module": "CommonJS",
|
|
||||||
"moduleResolution": "node",
|
|
||||||
"esModuleInterop": true,
|
|
||||||
"strict": true,
|
|
||||||
"skipLibCheck": true
|
|
||||||
},
|
|
||||||
"include": ["**/*.ts"]
|
|
||||||
}
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
|
|
||||||
# yarn lockfile v1
|
|
||||||
|
|
||||||
|
|
||||||
"@playwright/test@^1.50.0":
|
|
||||||
version "1.59.1"
|
|
||||||
resolved "https://registry.yarnpkg.com/@playwright/test/-/test-1.59.1.tgz#5c4d38eac84a61527af466602ae20277685a02d6"
|
|
||||||
integrity sha512-PG6q63nQg5c9rIi4/Z5lR5IVF7yU5MqmKaPOe0HSc0O2cX1fPi96sUQu5j7eo4gKCkB2AnNGoWt7y4/Xx3Kcqg==
|
|
||||||
dependencies:
|
|
||||||
playwright "1.59.1"
|
|
||||||
|
|
||||||
"@types/node@^22.0.0":
|
|
||||||
version "22.19.17"
|
|
||||||
resolved "https://registry.yarnpkg.com/@types/node/-/node-22.19.17.tgz#09c71fb34ba2510f8ac865361b1fcb9552b8a581"
|
|
||||||
integrity sha512-wGdMcf+vPYM6jikpS/qhg6WiqSV/OhG+jeeHT/KlVqxYfD40iYJf9/AE1uQxVWFvU7MipKRkRv8NSHiCGgPr8Q==
|
|
||||||
dependencies:
|
|
||||||
undici-types "~6.21.0"
|
|
||||||
|
|
||||||
fsevents@2.3.2:
|
|
||||||
version "2.3.2"
|
|
||||||
resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a"
|
|
||||||
integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==
|
|
||||||
|
|
||||||
playwright-core@1.59.1:
|
|
||||||
version "1.59.1"
|
|
||||||
resolved "https://registry.yarnpkg.com/playwright-core/-/playwright-core-1.59.1.tgz#d8a2b28bcb8f2bd08ef3df93b02ae83c813244b2"
|
|
||||||
integrity sha512-HBV/RJg81z5BiiZ9yPzIiClYV/QMsDCKUyogwH9p3MCP6IYjUFu/MActgYAvK0oWyV9NlwM3GLBjADyWgydVyg==
|
|
||||||
|
|
||||||
playwright@1.59.1:
|
|
||||||
version "1.59.1"
|
|
||||||
resolved "https://registry.yarnpkg.com/playwright/-/playwright-1.59.1.tgz#f7b0ca61637ae25264cec370df671bbe1f368a4a"
|
|
||||||
integrity sha512-C8oWjPR3F81yljW9o5OxcWzfh6avkVwDD2VYdwIGqTkl+OGFISgypqzfu7dOe4QNLL2aqcWBmI3PMtLIK233lw==
|
|
||||||
dependencies:
|
|
||||||
playwright-core "1.59.1"
|
|
||||||
optionalDependencies:
|
|
||||||
fsevents "2.3.2"
|
|
||||||
|
|
||||||
typescript@^5.9.3:
|
|
||||||
version "5.9.3"
|
|
||||||
resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.9.3.tgz#5b4f59e15310ab17a216f5d6cf53ee476ede670f"
|
|
||||||
integrity sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==
|
|
||||||
|
|
||||||
undici-types@~6.21.0:
|
|
||||||
version "6.21.0"
|
|
||||||
resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.21.0.tgz#691d00af3909be93a7faa13be61b3a5b50ef12cb"
|
|
||||||
integrity sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==
|
|
||||||
-23
@@ -1,23 +0,0 @@
|
|||||||
# Spustí server a klienta v samostatných panelech jednoho okna Windows Terminalu.
|
|
||||||
# Vyžaduje Windows Terminal (wt.exe) — výchozí součást Windows 11.
|
|
||||||
|
|
||||||
$ErrorActionPreference = 'Stop'
|
|
||||||
$ScriptDir = $PSScriptRoot
|
|
||||||
|
|
||||||
Push-Location (Join-Path $ScriptDir 'types')
|
|
||||||
try { yarn openapi-ts } finally { Pop-Location }
|
|
||||||
|
|
||||||
if (-not (Get-Command wt.exe -ErrorAction SilentlyContinue)) {
|
|
||||||
Write-Error "wt.exe (Windows Terminal) nebyl nalezen. Nainstalujte z Microsoft Store nebo použijte run_dev.sh v WSL."
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
|
|
||||||
$serverDir = Join-Path $ScriptDir 'server'
|
|
||||||
$clientDir = Join-Path $ScriptDir 'client'
|
|
||||||
|
|
||||||
# wt splits on ';' before respecting quoting, so encode the compound server command to avoid it
|
|
||||||
$serverCmd = '$env:NODE_ENV = ''development''; yarn startReload'
|
|
||||||
$serverCmdB64 = [Convert]::ToBase64String([Text.Encoding]::Unicode.GetBytes($serverCmd))
|
|
||||||
|
|
||||||
wt -w 0 new-tab --title 'luncher-server' -d $serverDir pwsh -NoExit -EncodedCommand $serverCmdB64 `; `
|
|
||||||
split-pane -H --title 'luncher-client' -d $clientDir pwsh -NoExit -Command "yarn start"
|
|
||||||
@@ -38,17 +38,3 @@
|
|||||||
|
|
||||||
# Název důvěryhodné hlavičky obsahující login uživatele. Výchozí hodnota je 'remote-user'.
|
# Název důvěryhodné hlavičky obsahující login uživatele. Výchozí hodnota je 'remote-user'.
|
||||||
# HTTP_REMOTE_USER_HEADER_NAME=remote-user
|
# HTTP_REMOTE_USER_HEADER_NAME=remote-user
|
||||||
|
|
||||||
# VAPID klíče pro Web Push notifikace (připomínka výběru oběda).
|
|
||||||
# Vygenerovat pomocí: npx web-push generate-vapid-keys
|
|
||||||
# VAPID_PUBLIC_KEY=
|
|
||||||
# VAPID_PRIVATE_KEY=
|
|
||||||
# VAPID_SUBJECT=mailto:admin@example.com
|
|
||||||
|
|
||||||
# Heslo pro bypass rate limitu na endpointu /api/food/refresh (pro skripty/admin).
|
|
||||||
# Bez hesla může refresh volat každý přihlášený uživatel (podléhá rate limitu).
|
|
||||||
# REFRESH_BYPASS_PASSWORD=
|
|
||||||
|
|
||||||
# Admin heslo pro správu seznamu obchodů na stránce /objednani.
|
|
||||||
# Bez hesla nelze přidávat ani odebírat obchody ze seznamu (POST/DELETE na /api/stores vrátí 403).
|
|
||||||
# ADMIN_PASSWORD=
|
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
/dist
|
/dist
|
||||||
/resources/easterEggs
|
/resources/easterEggs
|
||||||
/src/gen
|
/src/gen
|
||||||
/coverage
|
|
||||||
.env.production
|
.env.production
|
||||||
.env.development
|
.env.development
|
||||||
.easter-eggs.json
|
.easter-eggs.json
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
[
|
|
||||||
"Zimní atmosféra",
|
|
||||||
"Skrytí podniku U Motlíků"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Přidání restaurace Zastávka u Michala"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Přidání restaurace Pivovarský šenk Šeříková"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost výběru podniku/jídla kliknutím"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Stránka se statistikami nejoblíbenějších voleb"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Zobrazení počtu osob u každé volby"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Migrace na generované OpenApi"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Odebrání zimní atmosféry"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost ručního přenačtení menu"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Parsování a zobrazení alergenů"
|
|
||||||
]
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
[
|
|
||||||
"Oddělení přenačtení menu do vlastního dialogu",
|
|
||||||
"Podzimní atmosféra"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost převzetí poznámky ostatních uživatelů"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Zimní atmosféra"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost označit se jako objednávající u volby \"Budu objednávat\""
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Podpora dark mode"
|
|
||||||
]
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
[
|
|
||||||
"Redesign aplikace pomocí Claude Code",
|
|
||||||
"Zobrazení uplynulého týdne i o víkendu",
|
|
||||||
"Podpora Discord, ntfy a Teams notifikací (v Nastavení)",
|
|
||||||
"Trvalé zobrazení QR kódů do ručního zavření",
|
|
||||||
"Zobrazení nejvíce požadovaných funkcí (na stránce Statistiky)"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Zobrazení sekce Pizza day pouze při volbě \"Pizza day\""
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost generování obecných QR kódů pro platby i mimo Pizza day (v uživatelském menu)"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Podpora push notifikací pro připomenutí výběru oběda (v nastavení)"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Oprava detekce zastaralého menu"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Automatické zobrazení dialogu s dosud nezobrazenými novinkami"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Automatický výběr výchozího času preferovaného odchodu"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Zobrazení nabídky salátů z Pizza Chefie"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Možnost úhrady celého účtu jednou osobou s rozesláním QR kódů ostatním (na Pizza day)"
|
|
||||||
]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[
|
|
||||||
"Skupinové objednávky s QR platbou — stránka /objednani (více skupin, každá z jiného obchodu, stavový automat open/locked/ordered)"
|
|
||||||
]
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
testEnvironment: 'node',
|
|
||||||
testMatch: ['<rootDir>/src/**/*.test.ts'],
|
|
||||||
testPathIgnorePatterns: ['/node_modules/', '/dist/'],
|
|
||||||
setupFiles: ['<rootDir>/src/tests/setupEnv.ts'],
|
|
||||||
};
|
|
||||||
+1
-5
@@ -19,12 +19,9 @@
|
|||||||
"@types/jsonwebtoken": "^9.0.10",
|
"@types/jsonwebtoken": "^9.0.10",
|
||||||
"@types/node": "^24.10.0",
|
"@types/node": "^24.10.0",
|
||||||
"@types/request-promise": "^4.1.48",
|
"@types/request-promise": "^4.1.48",
|
||||||
"@types/supertest": "^6.0.0",
|
|
||||||
"@types/web-push": "^3.6.4",
|
|
||||||
"babel-jest": "^30.2.0",
|
"babel-jest": "^30.2.0",
|
||||||
"jest": "^30.2.0",
|
"jest": "^30.2.0",
|
||||||
"nodemon": "^3.1.10",
|
"nodemon": "^3.1.10",
|
||||||
"supertest": "^7.0.0",
|
|
||||||
"ts-node": "^10.9.1",
|
"ts-node": "^10.9.1",
|
||||||
"typescript": "^5.9.3"
|
"typescript": "^5.9.3"
|
||||||
},
|
},
|
||||||
@@ -37,7 +34,6 @@
|
|||||||
"jsonwebtoken": "^9.0.0",
|
"jsonwebtoken": "^9.0.0",
|
||||||
"redis": "^5.9.0",
|
"redis": "^5.9.0",
|
||||||
"simple-json-db": "^2.0.0",
|
"simple-json-db": "^2.0.0",
|
||||||
"socket.io": "^4.6.1",
|
"socket.io": "^4.6.1"
|
||||||
"web-push": "^3.6.7"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+8
-8
@@ -9,13 +9,13 @@ import jwt from 'jsonwebtoken';
|
|||||||
*/
|
*/
|
||||||
export function generateToken(login?: string, trusted?: boolean): string {
|
export function generateToken(login?: string, trusted?: boolean): string {
|
||||||
if (!process.env.JWT_SECRET) {
|
if (!process.env.JWT_SECRET) {
|
||||||
throw new Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
throw Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
||||||
}
|
}
|
||||||
if (process.env.JWT_SECRET.length < 32) {
|
if (process.env.JWT_SECRET.length < 32) {
|
||||||
throw new Error("Proměnná prostředí JWT_SECRET musí být minimálně 32 znaků");
|
throw Error("Proměnná prostředí JWT_SECRET musí být minimálně 32 znaků");
|
||||||
}
|
}
|
||||||
if (!login || login.trim().length === 0) {
|
if (!login || login.trim().length === 0) {
|
||||||
throw new Error("Nebyl předán login");
|
throw Error("Nebyl předán login");
|
||||||
}
|
}
|
||||||
const payload = { login, trusted: trusted || false, logoutUrl: process.env.LOGOUT_URL };
|
const payload = { login, trusted: trusted || false, logoutUrl: process.env.LOGOUT_URL };
|
||||||
return jwt.sign(payload, process.env.JWT_SECRET);
|
return jwt.sign(payload, process.env.JWT_SECRET);
|
||||||
@@ -28,7 +28,7 @@ export function generateToken(login?: string, trusted?: boolean): string {
|
|||||||
*/
|
*/
|
||||||
export function verify(token: string): boolean {
|
export function verify(token: string): boolean {
|
||||||
if (!process.env.JWT_SECRET) {
|
if (!process.env.JWT_SECRET) {
|
||||||
throw new Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
throw Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
jwt.verify(token, process.env.JWT_SECRET);
|
jwt.verify(token, process.env.JWT_SECRET);
|
||||||
@@ -45,10 +45,10 @@ export function verify(token: string): boolean {
|
|||||||
*/
|
*/
|
||||||
export function getLogin(token?: string): string {
|
export function getLogin(token?: string): string {
|
||||||
if (!process.env.JWT_SECRET) {
|
if (!process.env.JWT_SECRET) {
|
||||||
throw new Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
throw Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
||||||
}
|
}
|
||||||
if (!token) {
|
if (!token) {
|
||||||
throw new Error("Nebyl předán token");
|
throw Error("Nebyl předán token");
|
||||||
}
|
}
|
||||||
const payload: any = jwt.verify(token, process.env.JWT_SECRET);
|
const payload: any = jwt.verify(token, process.env.JWT_SECRET);
|
||||||
return payload.login;
|
return payload.login;
|
||||||
@@ -61,10 +61,10 @@ export function getLogin(token?: string): string {
|
|||||||
*/
|
*/
|
||||||
export function getTrusted(token?: string): boolean {
|
export function getTrusted(token?: string): boolean {
|
||||||
if (!process.env.JWT_SECRET) {
|
if (!process.env.JWT_SECRET) {
|
||||||
throw new Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
throw Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
||||||
}
|
}
|
||||||
if (!token) {
|
if (!token) {
|
||||||
throw new Error("Nebyl předán token");
|
throw Error("Nebyl předán token");
|
||||||
}
|
}
|
||||||
const payload: any = jwt.verify(token, process.env.JWT_SECRET);
|
const payload: any = jwt.verify(token, process.env.JWT_SECRET);
|
||||||
return payload.trusted || false;
|
return payload.trusted || false;
|
||||||
|
|||||||
+8
-47
@@ -1,7 +1,6 @@
|
|||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { load } from 'cheerio';
|
import { load } from 'cheerio';
|
||||||
import { getPizzaListMock, getSalatListMock } from './mock';
|
import { getPizzaListMock } from './mock';
|
||||||
import { Salat } from '../../types/gen/types.gen';
|
|
||||||
|
|
||||||
// TODO přesunout do types
|
// TODO přesunout do types
|
||||||
type PizzaSize = {
|
type PizzaSize = {
|
||||||
@@ -21,24 +20,20 @@ type Pizza = {
|
|||||||
|
|
||||||
// TODO mělo by být konfigurovatelné proměnnou z prostředí s tímhle jako default
|
// TODO mělo by být konfigurovatelné proměnnou z prostředí s tímhle jako default
|
||||||
const baseUrl = 'https://www.pizzachefie.cz';
|
const baseUrl = 'https://www.pizzachefie.cz';
|
||||||
const pizzyUrl = `${baseUrl}/pizzy.html`;
|
const pizzyUrl = `${baseUrl}/pizzy.html?pobocka=plzen`;
|
||||||
const salayUrl = `${baseUrl}/salaty.html`;
|
|
||||||
|
|
||||||
const buildPizzaUrl = (pizzaUrl: string) => {
|
const buildPizzaUrl = (pizzaUrl: string) => {
|
||||||
return `${baseUrl}/${pizzaUrl}`;
|
return `${baseUrl}/${pizzaUrl}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ceny krabic dle velikosti v haléřích
|
// Ceny krabic dle velikosti
|
||||||
const boxPrices: { [key: string]: number } = {
|
const boxPrices: { [key: string]: number } = {
|
||||||
"30cm": 1300,
|
"30cm": 13,
|
||||||
"35cm": 1500,
|
"35cm": 15,
|
||||||
"40cm": 1800,
|
"40cm": 18,
|
||||||
"50cm": 2500
|
"50cm": 25
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cena obalu pro salát v haléřích
|
|
||||||
const SALAT_BOX_PRICE = 1300;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stáhne a scrapne aktuální pizzy ze stránek Pizza Chefie.
|
* Stáhne a scrapne aktuální pizzy ze stránek Pizza Chefie.
|
||||||
*
|
*
|
||||||
@@ -79,7 +74,7 @@ export async function downloadPizzy(mock: boolean): Promise<Pizza[]> {
|
|||||||
a.each((i, elm) => {
|
a.each((i, elm) => {
|
||||||
const varId = Number.parseInt(elm.attribs.href.split('?varianta=')[1].trim());
|
const varId = Number.parseInt(elm.attribs.href.split('?varianta=')[1].trim());
|
||||||
const size = $($(elm).contents().get(0)).text().trim();
|
const size = $($(elm).contents().get(0)).text().trim();
|
||||||
const price = Number.parseInt($($(elm).contents().get(1)).text().trim().split(" Kč")[0]) * 100;
|
const price = Number.parseInt($($(elm).contents().get(1)).text().trim().split(" Kč")[0]);
|
||||||
sizes.push({ varId, size, pizzaPrice: price, boxPrice: boxPrices[size], price: price + boxPrices[size] });
|
sizes.push({ varId, size, pizzaPrice: price, boxPrice: boxPrices[size], price: price + boxPrices[size] });
|
||||||
})
|
})
|
||||||
result.push({
|
result.push({
|
||||||
@@ -90,37 +85,3 @@ export async function downloadPizzy(mock: boolean): Promise<Pizza[]> {
|
|||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Stáhne a scrapne aktuální saláty ze stránek Pizza Chefie.
|
|
||||||
* Příplatek za obal je pro každý salát pevně 13 Kč.
|
|
||||||
*
|
|
||||||
* @param mock zda vrátit pouze mock data
|
|
||||||
*/
|
|
||||||
export async function downloadSalaty(mock: boolean): Promise<Salat[]> {
|
|
||||||
if (mock) {
|
|
||||||
return new Promise((resolve) => setTimeout(() => resolve(getSalatListMock()), 1000));
|
|
||||||
}
|
|
||||||
const html = await axios.get(salayUrl).then(res => res.data);
|
|
||||||
const $ = load(html);
|
|
||||||
const links = $('.vypisproduktu > div > h4 > a');
|
|
||||||
const urls = [];
|
|
||||||
for (const element of links) {
|
|
||||||
if (element.name === 'a' && element.attribs?.href) {
|
|
||||||
urls.push(buildPizzaUrl(element.attribs.href));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const result: Salat[] = [];
|
|
||||||
for (const url of urls) {
|
|
||||||
const salatHtml = await axios.get(url).then(res => res.data);
|
|
||||||
const name = $('.produkt > h2', salatHtml).first().text().trim();
|
|
||||||
const ingredients: string[] = [];
|
|
||||||
$('.prisady > li', salatHtml).each((i, elm) => {
|
|
||||||
ingredients.push($(elm).text());
|
|
||||||
});
|
|
||||||
const priceText = $('.cena > span', salatHtml).first().text().trim();
|
|
||||||
const price = Number.parseInt(priceText.split(' Kč')[0]) * 100;
|
|
||||||
result.push({ name, ingredients, price: price + SALAT_BOX_PRICE });
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
@@ -1,183 +0,0 @@
|
|||||||
import crypto from "crypto";
|
|
||||||
import getStorage from "./storage";
|
|
||||||
import { getClientData, getToday, initIfNeeded } from "./service";
|
|
||||||
import { getStores } from "./stores";
|
|
||||||
import { removePendingQrsByGroupId } from "./pizza";
|
|
||||||
import { ClientData, GroupState, MealSlot, OrderGroup, OrderGroupMember } from "../../types/gen/types.gen";
|
|
||||||
import { formatDate } from "./utils";
|
|
||||||
|
|
||||||
const storage = getStorage();
|
|
||||||
|
|
||||||
async function getExtraData(date?: Date): Promise<ClientData> {
|
|
||||||
await initIfNeeded(date, MealSlot.EXTRA);
|
|
||||||
const data = await getClientData(date, MealSlot.EXTRA);
|
|
||||||
data.stores = await getStores();
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getExtraKey(date?: Date): string {
|
|
||||||
return `${formatDate(date ?? getToday())}_extra`;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function saveExtraData(data: ClientData, date?: Date): Promise<ClientData> {
|
|
||||||
await storage.setData(getExtraKey(date), data);
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
function findGroup(data: ClientData, id: string): OrderGroup | undefined {
|
|
||||||
return data.groups?.find(g => g.id === id);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createGroup(creatorLogin: string, name: string, date?: Date): Promise<ClientData> {
|
|
||||||
const stores = await getStores();
|
|
||||||
if (!stores.some(s => s.toLowerCase() === name.trim().toLowerCase())) {
|
|
||||||
throw new Error('Obchod není v seznamu povolených obchodů');
|
|
||||||
}
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const canonical = stores.find(s => s.toLowerCase() === name.trim().toLowerCase())!;
|
|
||||||
const group: OrderGroup = {
|
|
||||||
id: crypto.randomUUID(),
|
|
||||||
name: canonical,
|
|
||||||
creatorLogin,
|
|
||||||
state: GroupState.OPEN,
|
|
||||||
members: { [creatorLogin]: {} },
|
|
||||||
};
|
|
||||||
data.groups = [...(data.groups ?? []), group];
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function deleteGroup(login: string, groupId: string, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.creatorLogin !== login) throw new Error('Skupinu může smazat pouze zakladatel');
|
|
||||||
data.groups = (data.groups ?? []).filter(g => g.id !== groupId);
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function addGroupMember(login: string, groupId: string, targetLogin: string, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.state === GroupState.ORDERED) throw new Error('Skupinu ve stavu "objednáno" nelze upravovat');
|
|
||||||
if (login !== group.creatorLogin && login !== targetLogin) {
|
|
||||||
throw new Error('Přidat jiného uživatele může pouze zakladatel');
|
|
||||||
}
|
|
||||||
if (group.state === GroupState.LOCKED && login !== group.creatorLogin) {
|
|
||||||
throw new Error('Skupinu ve stavu "uzamčeno" může upravovat pouze zakladatel');
|
|
||||||
}
|
|
||||||
if (group.members[targetLogin]) throw new Error('Uživatel je již ve skupině');
|
|
||||||
group.members[targetLogin] = {};
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function removeGroupMember(login: string, groupId: string, targetLogin: string, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.state === GroupState.ORDERED) throw new Error('Skupinu ve stavu "objednáno" nelze upravovat');
|
|
||||||
if (login !== group.creatorLogin && login !== targetLogin) {
|
|
||||||
throw new Error('Odebrat jiného uživatele může pouze zakladatel');
|
|
||||||
}
|
|
||||||
if (group.state === GroupState.LOCKED && login !== group.creatorLogin) {
|
|
||||||
throw new Error('Skupinu ve stavu "uzamčeno" může upravovat pouze zakladatel');
|
|
||||||
}
|
|
||||||
if (targetLogin === group.creatorLogin) throw new Error('Zakladatel skupiny nemůže být odebrán');
|
|
||||||
if (!group.members[targetLogin]) throw new Error('Uživatel není ve skupině');
|
|
||||||
delete group.members[targetLogin];
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateGroupMember(login: string, groupId: string, targetLogin: string, patch: Partial<OrderGroupMember>, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.state === GroupState.ORDERED) throw new Error('Skupinu ve stavu "objednáno" nelze upravovat');
|
|
||||||
const isSelf = login === targetLogin;
|
|
||||||
const isCreator = login === group.creatorLogin;
|
|
||||||
if (!isSelf && !isCreator) throw new Error('Upravit jiného uživatele může pouze zakladatel');
|
|
||||||
if (!isCreator && group.state === GroupState.LOCKED) {
|
|
||||||
throw new Error('Skupinu ve stavu "uzamčeno" může upravovat pouze zakladatel');
|
|
||||||
}
|
|
||||||
if (!group.members[targetLogin]) throw new Error('Uživatel není ve skupině');
|
|
||||||
group.members[targetLogin] = { ...group.members[targetLogin], ...patch };
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
const VALID_TRANSITIONS: Record<GroupState, GroupState[]> = {
|
|
||||||
[GroupState.OPEN]: [GroupState.LOCKED],
|
|
||||||
[GroupState.LOCKED]: [GroupState.OPEN, GroupState.ORDERED],
|
|
||||||
[GroupState.ORDERED]: [GroupState.LOCKED],
|
|
||||||
};
|
|
||||||
|
|
||||||
function getCurrentHHMM(): string {
|
|
||||||
const now = new Date();
|
|
||||||
return `${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function setGroupState(login: string, groupId: string, newState: GroupState, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.creatorLogin !== login) throw new Error('Stav může měnit pouze zakladatel');
|
|
||||||
if (!VALID_TRANSITIONS[group.state].includes(newState)) {
|
|
||||||
throw new Error(`Nelze přejít ze stavu "${group.state}" do stavu "${newState}"`);
|
|
||||||
}
|
|
||||||
if (newState === GroupState.ORDERED) {
|
|
||||||
group.orderedAt = getCurrentHHMM();
|
|
||||||
}
|
|
||||||
if (group.state === GroupState.ORDERED && newState === GroupState.LOCKED) {
|
|
||||||
const memberLogins = Object.keys(group.members);
|
|
||||||
await removePendingQrsByGroupId(memberLogins, groupId);
|
|
||||||
group.orderedAt = undefined;
|
|
||||||
group.deliveryAt = undefined;
|
|
||||||
group.qrGenerated = undefined;
|
|
||||||
for (const ml of memberLogins) {
|
|
||||||
group.members[ml] = { ...group.members[ml], paid: undefined };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
group.state = newState;
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function markGroupQrGenerated(login: string, groupId: string, date?: Date): Promise<void> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.creatorLogin !== login) throw new Error('QR kódy může generovat pouze zakladatel');
|
|
||||||
if (group.state !== GroupState.ORDERED) throw new Error('QR kódy lze generovat pouze ve stavu "objednáno"');
|
|
||||||
group.qrGenerated = true;
|
|
||||||
await saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function markGroupMemberPaid(login: string, groupId: string, date?: Date): Promise<ClientData | null> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group || !group.members[login]) return null;
|
|
||||||
group.members[login] = { ...group.members[login], paid: true };
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateGroupFees(login: string, groupId: string, fees?: number, shipping?: number, tip?: number, discountType?: string, discountValue?: number, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.creatorLogin !== login) throw new Error('Poplatky může měnit pouze zakladatel');
|
|
||||||
if (group.state === GroupState.ORDERED) throw new Error('Skupinu ve stavu "objednáno" nelze upravovat');
|
|
||||||
if (fees !== undefined) group.fees = fees > 0 ? fees : undefined;
|
|
||||||
if (shipping !== undefined) group.shipping = shipping > 0 ? shipping : undefined;
|
|
||||||
if (tip !== undefined) group.tip = tip > 0 ? tip : undefined;
|
|
||||||
if (discountType !== undefined) group.discountType = (discountType as any) || undefined;
|
|
||||||
if (discountValue !== undefined) group.discountValue = discountValue > 0 ? discountValue : undefined;
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateGroupTimes(login: string, groupId: string, orderedAt?: string, deliveryAt?: string, date?: Date): Promise<ClientData> {
|
|
||||||
const data = await getExtraData(date);
|
|
||||||
const group = findGroup(data, groupId);
|
|
||||||
if (!group) throw new Error('Skupina nebyla nalezena');
|
|
||||||
if (group.creatorLogin !== login) throw new Error('Časy může měnit pouze zakladatel');
|
|
||||||
if (orderedAt !== undefined) group.orderedAt = orderedAt || undefined;
|
|
||||||
if (deliveryAt !== undefined) group.deliveryAt = deliveryAt || undefined;
|
|
||||||
return saveExtraData(data, date);
|
|
||||||
}
|
|
||||||
+19
-83
@@ -1,35 +1,25 @@
|
|||||||
import express from "express";
|
import express from "express";
|
||||||
import bodyParser from "body-parser";
|
import bodyParser from "body-parser";
|
||||||
import cors from 'cors';
|
import cors from 'cors';
|
||||||
import { getData, addChoice, getDateForWeekIndex, getToday } from "./service";
|
import { getData, getDateForWeekIndex } from "./service";
|
||||||
import { MealSlot } from "../../types/gen/types.gen";
|
|
||||||
import dotenv from 'dotenv';
|
import dotenv from 'dotenv';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import { getQr } from "./qr";
|
import { getQr } from "./qr";
|
||||||
import { generateToken, getLogin, verify } from "./auth";
|
import { generateToken, verify } from "./auth";
|
||||||
import { getIsWeekend, InsufficientPermissions, PizzaDayConflictError, parseToken } from "./utils";
|
import { InsufficientPermissions } from "./utils";
|
||||||
import { getPendingQrs } from "./pizza";
|
import { initWebsocket } from "./websocket";
|
||||||
import { initWebsocket, getWebsocket } from "./websocket";
|
|
||||||
import { startReminderScheduler, verifyQuickChoiceToken } from "./pushReminder";
|
|
||||||
import { storageReady } from "./storage";
|
|
||||||
import pizzaDayRoutes from "./routes/pizzaDayRoutes";
|
import pizzaDayRoutes from "./routes/pizzaDayRoutes";
|
||||||
import foodRoutes, { refreshMetoda } from "./routes/foodRoutes";
|
import foodRoutes, { refreshMetoda } 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";
|
import statsRoutes from "./routes/statsRoutes";
|
||||||
import notificationRoutes from "./routes/notificationRoutes";
|
|
||||||
import qrRoutes from "./routes/qrRoutes";
|
|
||||||
import devRoutes from "./routes/devRoutes";
|
|
||||||
import changelogRoutes from "./routes/changelogRoutes";
|
|
||||||
import groupRoutes from "./routes/groupRoutes";
|
|
||||||
import storeRoutes from "./routes/storeRoutes";
|
|
||||||
|
|
||||||
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) {
|
||||||
throw new Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
throw Error("Není vyplněna proměnná prostředí JWT_SECRET");
|
||||||
}
|
}
|
||||||
|
|
||||||
const app = express();
|
const app = express();
|
||||||
@@ -60,15 +50,11 @@ if (HTTP_REMOTE_USER_ENABLED) {
|
|||||||
|
|
||||||
// ----------- Metody nevyžadující token --------------
|
// ----------- Metody nevyžadující token --------------
|
||||||
|
|
||||||
app.get("/api/health", (_req, res) => {
|
|
||||||
res.status(200).json({ ok: true });
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get("/api/whoami", (req, res) => {
|
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') {
|
if(process.env.ENABLE_HEADERS_LOGGING === 'yes'){
|
||||||
delete req.headers["cookie"]
|
delete req.headers["cookie"]
|
||||||
console.log(req.headers)
|
console.log(req.headers)
|
||||||
}
|
}
|
||||||
@@ -80,30 +66,27 @@ app.post("/api/login", (req, res) => {
|
|||||||
// 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 ) {
|
||||||
res.status(200).json(generateToken(Buffer.from(remoteUser, 'latin1').toString(), true));
|
res.status(200).json(generateToken(Buffer.from(remoteUser, 'latin1').toString(), true));
|
||||||
} else {
|
} else {
|
||||||
throw new Error("Je zapnuto přihlášení přes hlavičky, ale nepřišla hlavička nebo ??");
|
throw Error("Je zapnuto přihlášení přes hlavičky, ale nepřišla hlavička nebo ??");
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Klasická autentizace loginem
|
// Klasická autentizace loginem
|
||||||
if (!req.body?.login || req.body.login.trim().length === 0) {
|
if (!req.body?.login || req.body.login.trim().length === 0) {
|
||||||
throw new Error("Nebyl předán login");
|
throw Error("Nebyl předán login");
|
||||||
}
|
}
|
||||||
// TODO zavést podmínky pro délku loginu (min i max)
|
// TODO zavést podmínky pro délku loginu (min i max)
|
||||||
res.status(200).json(generateToken(req.body.login, false));
|
res.status(200).json(generateToken(req.body.login, false));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// 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", async (req, res) => {
|
app.get("/api/qr", (req, res) => {
|
||||||
if (!req.query?.login) {
|
if (!req.query?.login) {
|
||||||
return res.status(400).json({ error: "Nebyl předán login" });
|
throw Error("Nebyl předán login");
|
||||||
}
|
}
|
||||||
if (!req.query?.id) {
|
const img = getQr(req.query.login as string);
|
||||||
return res.status(400).json({ error: "Nebyl předán identifikátor QR kódu" });
|
|
||||||
}
|
|
||||||
const img = await getQr(req.query.login as string, req.query.id as string);
|
|
||||||
res.writeHead(200, {
|
res.writeHead(200, {
|
||||||
'Content-Type': 'image/png',
|
'Content-Type': 'image/png',
|
||||||
'Content-Length': img.length
|
'Content-Length': img.length
|
||||||
@@ -116,28 +99,12 @@ app.get("/api/qr", async (req, res) => {
|
|||||||
// Přeskočení auth pro refresh dat xd
|
// Přeskočení auth pro refresh dat xd
|
||||||
app.use("/api/food/refresh", refreshMetoda);
|
app.use("/api/food/refresh", refreshMetoda);
|
||||||
|
|
||||||
// Rychlá akce z push notifikace — autentizace pomocí HMAC tokenu z push payloadu (SW nemá přístup k JWT)
|
|
||||||
app.post("/api/notifications/push/quickChoice", async (req, res, next) => {
|
|
||||||
try {
|
|
||||||
const { login, token } = req.body ?? {};
|
|
||||||
if (!login || typeof login !== 'string' || !token || typeof token !== 'string') {
|
|
||||||
return res.status(400).json({ error: 'Chybí login nebo token' });
|
|
||||||
}
|
|
||||||
if (!verifyQuickChoiceToken(login, token)) {
|
|
||||||
return res.status(403).json({ error: 'Neplatný token' });
|
|
||||||
}
|
|
||||||
const updatedData = await addChoice(login, false, 'NEOBEDVAM', undefined, undefined);
|
|
||||||
getWebsocket().emit("message", updatedData);
|
|
||||||
res.status(200).json({});
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
/** 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
|
// Autentizace pomocí trusted headers
|
||||||
const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME);
|
const remoteUser = req.header(HTTP_REMOTE_USER_HEADER_NAME);
|
||||||
if (process.env.ENABLE_HEADERS_LOGGING === 'yes') {
|
if(process.env.ENABLE_HEADERS_LOGGING === 'yes'){
|
||||||
delete req.headers["cookie"]
|
delete req.headers["cookie"]
|
||||||
console.log(req.headers)
|
console.log(req.headers)
|
||||||
}
|
}
|
||||||
@@ -166,26 +133,8 @@ app.get("/api/data", async (req, res) => {
|
|||||||
if (!isNaN(index)) {
|
if (!isNaN(index)) {
|
||||||
date = getDateForWeekIndex(parseInt(req.query.dayIndex));
|
date = getDateForWeekIndex(parseInt(req.query.dayIndex));
|
||||||
}
|
}
|
||||||
} else if (getIsWeekend(getToday())) {
|
|
||||||
// Na víkendu zobrazíme pátek místo hlášky "Užívejte víkend"
|
|
||||||
date = getDateForWeekIndex(4);
|
|
||||||
}
|
}
|
||||||
const slotParam = typeof req.query.slot === 'string' ? req.query.slot as MealSlot : undefined;
|
res.status(200).json(await getData(date));
|
||||||
if (slotParam && slotParam !== MealSlot.OBED && slotParam !== MealSlot.EXTRA) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný slot' });
|
|
||||||
}
|
|
||||||
const data = await getData(date, slotParam);
|
|
||||||
// Připojíme nevyřízené QR kódy pro přihlášeného uživatele
|
|
||||||
try {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const pendingQrs = await getPendingQrs(login);
|
|
||||||
if (pendingQrs.length > 0) {
|
|
||||||
data.pendingQrs = pendingQrs;
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
// Token nemusí být validní, ignorujeme
|
|
||||||
}
|
|
||||||
res.status(200).json(data);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Ostatní routes
|
// Ostatní routes
|
||||||
@@ -194,24 +143,14 @@ 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("/api/stats", statsRoutes);
|
||||||
app.use("/api/notifications", notificationRoutes);
|
|
||||||
app.use("/api/qr", qrRoutes);
|
|
||||||
app.use("/api/dev", devRoutes);
|
|
||||||
app.use("/api/changelogs", changelogRoutes);
|
|
||||||
app.use("/api/groups", groupRoutes);
|
|
||||||
app.use("/api/stores", storeRoutes);
|
|
||||||
|
|
||||||
app.use(express.static(path.join(process.cwd(), 'public')));
|
app.use('/stats', express.static('public'));
|
||||||
app.get('*splat', (_req, res) => {
|
app.use(express.static('public'));
|
||||||
res.sendFile(path.join(process.cwd(), 'public', 'index.html'));
|
|
||||||
});
|
|
||||||
|
|
||||||
// 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) => {
|
||||||
if (err instanceof InsufficientPermissions) {
|
if (err instanceof InsufficientPermissions) {
|
||||||
res.status(403).send({ error: err.message })
|
res.status(403).send({ error: err.message })
|
||||||
} else if (err instanceof PizzaDayConflictError) {
|
|
||||||
res.status(409).send({ error: err.message })
|
|
||||||
} else {
|
} else {
|
||||||
res.status(500).send({ error: err.message })
|
res.status(500).send({ error: err.message })
|
||||||
}
|
}
|
||||||
@@ -221,11 +160,8 @@ app.use((err: any, req: any, res: any, next: any) => {
|
|||||||
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';
|
||||||
|
|
||||||
storageReady.then(() => {
|
server.listen(PORT, () => {
|
||||||
server.listen(PORT, () => {
|
console.log(`Server listening on ${HOST}, port ${PORT}`);
|
||||||
console.log(`Server listening on ${HOST}, port ${PORT}`);
|
|
||||||
startReminderScheduler();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Umožníme vypnutí serveru přes SIGINT, jinak Docker čeká než ho sestřelí
|
// Umožníme vypnutí serveru přes SIGINT, jinak Docker čeká než ho sestřelí
|
||||||
|
|||||||
+236
-255
@@ -661,30 +661,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 1,
|
varId: 1,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 13800,
|
pizzaPrice: 138,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 15100
|
price: 151
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 2,
|
varId: 2,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 16600,
|
pizzaPrice: 166,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 18100
|
price: 181
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 3,
|
varId: 3,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 22300,
|
pizzaPrice: 223,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 24100
|
price: 241
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 4,
|
varId: 4,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 30600,
|
pizzaPrice: 306,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 33100
|
price: 331
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -700,30 +700,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 6,
|
varId: 6,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 14200,
|
pizzaPrice: 142,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 15500
|
price: 155
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 7,
|
varId: 7,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 17200,
|
pizzaPrice: 172,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 18700
|
price: 187
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 8,
|
varId: 8,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 23300,
|
pizzaPrice: 233,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 25100
|
price: 251
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 9,
|
varId: 9,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 31600,
|
pizzaPrice: 316,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 34100
|
price: 341
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -741,30 +741,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 10,
|
varId: 10,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 14200,
|
pizzaPrice: 142,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 15500
|
price: 155
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 11,
|
varId: 11,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 17200,
|
pizzaPrice: 172,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 18700
|
price: 187
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 12,
|
varId: 12,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 23300,
|
pizzaPrice: 233,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 25100
|
price: 251
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 13,
|
varId: 13,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 31600,
|
pizzaPrice: 316,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 34100
|
price: 341
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -780,30 +780,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 14,
|
varId: 14,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 14200,
|
pizzaPrice: 142,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 15500
|
price: 155
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 15,
|
varId: 15,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 17200,
|
pizzaPrice: 172,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 18700
|
price: 187
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 16,
|
varId: 16,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 23300,
|
pizzaPrice: 233,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 25100
|
price: 251
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 17,
|
varId: 17,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 29400,
|
pizzaPrice: 294,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 31900
|
price: 319
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -821,30 +821,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 22,
|
varId: 22,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 16200,
|
pizzaPrice: 162,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 17500
|
price: 175
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 23,
|
varId: 23,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 18600,
|
pizzaPrice: 186,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 24,
|
varId: 24,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 26300,
|
pizzaPrice: 263,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 28100
|
price: 281
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 25,
|
varId: 25,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 34600,
|
pizzaPrice: 346,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 37100
|
price: 371
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -861,30 +861,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 26,
|
varId: 26,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 16200,
|
pizzaPrice: 162,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 17500
|
price: 175
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 27,
|
varId: 27,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 18600,
|
pizzaPrice: 186,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 28,
|
varId: 28,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 26300,
|
pizzaPrice: 263,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 28100
|
price: 281
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 29,
|
varId: 29,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 34600,
|
pizzaPrice: 346,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 37100
|
price: 371
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -902,30 +902,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 30,
|
varId: 30,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 16200,
|
pizzaPrice: 162,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 17500
|
price: 175
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 31,
|
varId: 31,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 18600,
|
pizzaPrice: 186,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 32,
|
varId: 32,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 26300,
|
pizzaPrice: 263,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 28100
|
price: 281
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 33,
|
varId: 33,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 34600,
|
pizzaPrice: 346,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 37100
|
price: 371
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -946,30 +946,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 34,
|
varId: 34,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 16200,
|
pizzaPrice: 162,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 17500
|
price: 175
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 35,
|
varId: 35,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 18600,
|
pizzaPrice: 186,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 36,
|
varId: 36,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 26300,
|
pizzaPrice: 263,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 28100
|
price: 281
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 37,
|
varId: 37,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 34600,
|
pizzaPrice: 346,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 37100
|
price: 371
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -987,30 +987,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 38,
|
varId: 38,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 16200,
|
pizzaPrice: 162,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 17500
|
price: 175
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 39,
|
varId: 39,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 18600,
|
pizzaPrice: 186,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 40,
|
varId: 40,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 26300,
|
pizzaPrice: 263,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 28100
|
price: 281
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 41,
|
varId: 41,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 34600,
|
pizzaPrice: 346,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 37100
|
price: 371
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1028,30 +1028,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 42,
|
varId: 42,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 17200,
|
pizzaPrice: 172,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 18500
|
price: 185
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 43,
|
varId: 43,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 21200,
|
pizzaPrice: 212,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 22700
|
price: 227
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 44,
|
varId: 44,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 29300,
|
pizzaPrice: 293,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 31100
|
price: 311
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 45,
|
varId: 45,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 37600,
|
pizzaPrice: 376,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 40100
|
price: 401
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1069,30 +1069,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 46,
|
varId: 46,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18200,
|
pizzaPrice: 182,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 19500
|
price: 195
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 47,
|
varId: 47,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22200,
|
pizzaPrice: 222,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 23700
|
price: 237
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 48,
|
varId: 48,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 30300,
|
pizzaPrice: 303,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 32100
|
price: 321
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 49,
|
varId: 49,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 38600,
|
pizzaPrice: 386,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 41100
|
price: 411
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1114,30 +1114,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 50,
|
varId: 50,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18200,
|
pizzaPrice: 182,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 19500
|
price: 195
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 51,
|
varId: 51,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22200,
|
pizzaPrice: 222,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 23700
|
price: 237
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 52,
|
varId: 52,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 30300,
|
pizzaPrice: 303,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 32100
|
price: 321
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 53,
|
varId: 53,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 39600,
|
pizzaPrice: 396,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 42100
|
price: 421
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1156,30 +1156,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 54,
|
varId: 54,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18200,
|
pizzaPrice: 182,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 19500
|
price: 195
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 55,
|
varId: 55,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22200,
|
pizzaPrice: 222,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 23700
|
price: 237
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 56,
|
varId: 56,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 30300,
|
pizzaPrice: 303,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 32100
|
price: 321
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 57,
|
varId: 57,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 39600,
|
pizzaPrice: 396,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 42100
|
price: 421
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1199,30 +1199,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 58,
|
varId: 58,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18200,
|
pizzaPrice: 182,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 19500
|
price: 195
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 59,
|
varId: 59,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22200,
|
pizzaPrice: 222,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 23700
|
price: 237
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 60,
|
varId: 60,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 30300,
|
pizzaPrice: 303,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 32100
|
price: 321
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 61,
|
varId: 61,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 39600,
|
pizzaPrice: 396,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 42100
|
price: 421
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1241,30 +1241,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 62,
|
varId: 62,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18800,
|
pizzaPrice: 188,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 63,
|
varId: 63,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22600,
|
pizzaPrice: 226,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 24100
|
price: 241
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 64,
|
varId: 64,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 31300,
|
pizzaPrice: 313,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 33100
|
price: 331
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 65,
|
varId: 65,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 42600,
|
pizzaPrice: 426,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 45100
|
price: 451
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1283,30 +1283,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 66,
|
varId: 66,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18800,
|
pizzaPrice: 188,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 67,
|
varId: 67,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22600,
|
pizzaPrice: 226,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 24100
|
price: 241
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 68,
|
varId: 68,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 31300,
|
pizzaPrice: 313,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 33100
|
price: 331
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 69,
|
varId: 69,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 42600,
|
pizzaPrice: 426,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 45100
|
price: 451
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1327,30 +1327,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 309,
|
varId: 309,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18200,
|
pizzaPrice: 182,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 19500
|
price: 195
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 310,
|
varId: 310,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22200,
|
pizzaPrice: 222,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 23700
|
price: 237
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 311,
|
varId: 311,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 30300,
|
pizzaPrice: 303,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 32100
|
price: 321
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 312,
|
varId: 312,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 39600,
|
pizzaPrice: 396,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 42100
|
price: 421
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -1369,30 +1369,30 @@ const MOCK_PIZZA_LIST = [
|
|||||||
{
|
{
|
||||||
varId: 394,
|
varId: 394,
|
||||||
size: "30cm",
|
size: "30cm",
|
||||||
pizzaPrice: 18800,
|
pizzaPrice: 188,
|
||||||
boxPrice: 1300,
|
boxPrice: 13,
|
||||||
price: 20100
|
price: 201
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 395,
|
varId: 395,
|
||||||
size: "35cm",
|
size: "35cm",
|
||||||
pizzaPrice: 22600,
|
pizzaPrice: 226,
|
||||||
boxPrice: 1500,
|
boxPrice: 15,
|
||||||
price: 24100
|
price: 241
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 396,
|
varId: 396,
|
||||||
size: "40cm",
|
size: "40cm",
|
||||||
pizzaPrice: 31300,
|
pizzaPrice: 313,
|
||||||
boxPrice: 1800,
|
boxPrice: 18,
|
||||||
price: 33100
|
price: 331
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
varId: 397,
|
varId: 397,
|
||||||
size: "50cm",
|
size: "50cm",
|
||||||
pizzaPrice: 42600,
|
pizzaPrice: 426,
|
||||||
boxPrice: 2500,
|
boxPrice: 25,
|
||||||
price: 45100
|
price: 451
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -1429,46 +1429,27 @@ export const getPizzaListMock = () => {
|
|||||||
return MOCK_PIZZA_LIST;
|
return MOCK_PIZZA_LIST;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mockovací data pro saláty
|
|
||||||
const MOCK_SALAT_LIST = [
|
|
||||||
{
|
|
||||||
name: "Greek",
|
|
||||||
ingredients: ["Salát", "Černé olivy", "Paprika mix", "Červená cibule", "Rajčata", "Okurka salátová", "Jogurtový dresing"],
|
|
||||||
price: (174 + 13) * 100,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Caesar",
|
|
||||||
ingredients: ["Salát", "Rajčata", "Kuřecí maso", "Krutony", "Parmazán", "Caesar dresing", "Olivový olej"],
|
|
||||||
price: (184 + 13) * 100,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Šopský salát",
|
|
||||||
ingredients: ["Salátová okurka", "Rajčata", "Paprika mix", "Červená cibule", "Balkánský sýr"],
|
|
||||||
price: (164 + 13) * 100,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Těstovinový salát",
|
|
||||||
ingredients: ["Penne", "Okurka", "Rajčata", "Paprika mix", "Kuřecí maso", "Jogurtový dresing"],
|
|
||||||
price: (184 + 13) * 100,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
export const getSalatListMock = () => {
|
|
||||||
return MOCK_SALAT_LIST;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const getStatsMock = (): WeeklyStats => {
|
export const getStatsMock = (): WeeklyStats => {
|
||||||
const mkDay = (date: string, di: number) => ({
|
|
||||||
date,
|
|
||||||
locations: Object.keys(LunchChoice).reduce((prev, cur, ci) => (
|
|
||||||
{ ...prev, [cur]: (di * 7 + ci * 3) % 10 }
|
|
||||||
), {} as Record<string, number>),
|
|
||||||
});
|
|
||||||
return [
|
return [
|
||||||
mkDay('24.02.', 0),
|
{
|
||||||
mkDay('25.02.', 1),
|
date: '24.02.',
|
||||||
mkDay('26.02.', 2),
|
locations: { ...Object.keys(LunchChoice).reduce((prev, cur) => ({ ...prev, [cur]: Math.floor(Math.random() * 10) }), {}) }
|
||||||
mkDay('27.02.', 3),
|
},
|
||||||
mkDay('28.02.', 4),
|
{
|
||||||
|
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) }), {}) }
|
||||||
|
}
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
+9
-115
@@ -3,56 +3,11 @@ import dotenv from 'dotenv';
|
|||||||
import path from 'path';
|
import path from 'path';
|
||||||
import { getClientData, getToday } from "./service";
|
import { getClientData, getToday } from "./service";
|
||||||
import { getUsersByLocation, getHumanTime } from "./utils";
|
import { getUsersByLocation, getHumanTime } from "./utils";
|
||||||
import { NotifikaceData, NotifikaceInput, NotificationSettings } from '../../types';
|
import { NotifikaceData, NotifikaceInput } from '../../types';
|
||||||
import getStorage from "./storage";
|
|
||||||
|
|
||||||
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 storage = getStorage();
|
|
||||||
const NOTIFICATION_SETTINGS_PREFIX = 'notif';
|
|
||||||
|
|
||||||
/** Vrátí klíč pro uložení notifikačních nastavení uživatele. */
|
|
||||||
function getNotificationSettingsKey(login: string): string {
|
|
||||||
return `${NOTIFICATION_SETTINGS_PREFIX}_${login}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Vrátí nastavení notifikací pro daného uživatele. */
|
|
||||||
export async function getNotificationSettings(login: string): Promise<NotificationSettings> {
|
|
||||||
return await storage.getData<NotificationSettings>(getNotificationSettingsKey(login)) ?? {};
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Uloží nastavení notifikací pro daného uživatele. */
|
|
||||||
export async function saveNotificationSettings(login: string, settings: NotificationSettings): Promise<NotificationSettings> {
|
|
||||||
await storage.setData(getNotificationSettingsKey(login), settings);
|
|
||||||
return settings;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Odešle ntfy notifikaci na dané téma. */
|
|
||||||
async function ntfyCallToTopic(topic: string, message: string) {
|
|
||||||
const url = process.env.NTFY_HOST;
|
|
||||||
const username = process.env.NTFY_USERNAME;
|
|
||||||
const password = process.env.NTFY_PASSWD;
|
|
||||||
if (!url || !username || !password) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const token = Buffer.from(`${username}:${password}`, 'utf8').toString('base64');
|
|
||||||
try {
|
|
||||||
const response = await axios({
|
|
||||||
url: `${url}/${topic}`,
|
|
||||||
method: 'POST',
|
|
||||||
data: message,
|
|
||||||
headers: {
|
|
||||||
'Authorization': `Basic ${token}`,
|
|
||||||
'Tag': 'meat_on_bone'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
console.log(response.data);
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Chyba při odesílání ntfy notifikace na topic ${topic}:`, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const ntfyCall = async (data: NotifikaceInput) => {
|
export const ntfyCall = async (data: NotifikaceInput) => {
|
||||||
const url = process.env.NTFY_HOST
|
const url = process.env.NTFY_HOST
|
||||||
const username = process.env.NTFY_USERNAME;
|
const username = process.env.NTFY_USERNAME;
|
||||||
@@ -132,58 +87,10 @@ export const teamsCall = async (data: NotifikaceInput) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Odešle Teams notifikaci na daný webhook URL. */
|
|
||||||
async function teamsCallToUrl(webhookUrl: string, data: NotifikaceInput) {
|
|
||||||
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",
|
|
||||||
summary: 'Summary description',
|
|
||||||
sections: [
|
|
||||||
{
|
|
||||||
activityTitle: title,
|
|
||||||
text: message,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
try {
|
|
||||||
await axios.post(webhookUrl, card, {
|
|
||||||
headers: {
|
|
||||||
'content-type': 'application/vnd.microsoft.teams.card.o365connector'
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Chyba při odesílání Teams notifikace:`, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Odešle Discord notifikaci na daný webhook URL. */
|
|
||||||
async function discordCall(webhookUrl: string, data: NotifikaceInput) {
|
|
||||||
let time = new Date();
|
|
||||||
time.setTime(time.getTime() + 1000 * 60);
|
|
||||||
const message = `🍖 **${data.udalost}** — ${data.user} (odchod v ${getHumanTime(time)})`;
|
|
||||||
try {
|
|
||||||
await axios.post(webhookUrl, {
|
|
||||||
content: message,
|
|
||||||
}, {
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Chyba při odesílání Discord notifikace:`, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** 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: Promise<any>[] = [];
|
const notifications = [];
|
||||||
|
|
||||||
// Globální notifikace (zpětně kompatibilní)
|
|
||||||
if (ntfy) {
|
if (ntfy) {
|
||||||
const ntfyPromises = await ntfyCall(input);
|
const ntfyPromises = await ntfyCall(input);
|
||||||
if (ntfyPromises) {
|
if (ntfyPromises) {
|
||||||
@@ -193,33 +100,20 @@ export const callNotifikace = async ({ input, teams = true, gotify = false, ntfy
|
|||||||
if (teams) {
|
if (teams) {
|
||||||
const teamsPromises = await teamsCall(input);
|
const teamsPromises = await teamsCall(input);
|
||||||
if (teamsPromises) {
|
if (teamsPromises) {
|
||||||
notifications.push(Promise.resolve(teamsPromises));
|
notifications.push(teamsPromises);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Per-user notifikace: najdeme uživatele se stejnou lokací a odešleme dle jejich nastavení
|
|
||||||
const clientData = await getClientData(getToday());
|
|
||||||
const usersToNotify = getUsersByLocation(clientData.choices, input.user);
|
|
||||||
for (const user of usersToNotify) {
|
|
||||||
if (user === input.user) continue; // Neposíláme notifikaci spouštějícímu uživateli
|
|
||||||
const userSettings = await getNotificationSettings(user);
|
|
||||||
if (!userSettings.enabledEvents?.includes(input.udalost)) continue;
|
|
||||||
|
|
||||||
if (userSettings.ntfyTopic) {
|
|
||||||
notifications.push(ntfyCallToTopic(userSettings.ntfyTopic, `${input.udalost} - spustil: ${input.user}`));
|
|
||||||
}
|
|
||||||
if (userSettings.discordWebhookUrl) {
|
|
||||||
notifications.push(discordCall(userSettings.discordWebhookUrl, input));
|
|
||||||
}
|
|
||||||
if (userSettings.teamsWebhookUrl) {
|
|
||||||
notifications.push(teamsCallToUrl(userSettings.teamsWebhookUrl, input));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// 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);
|
||||||
return results;
|
return results;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error in callNotifikace: ", error);
|
console.error("Error in callNotifikace: ", error);
|
||||||
|
// Handle the error as needed
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
+33
-198
@@ -2,13 +2,11 @@ import { formatDate } from "./utils";
|
|||||||
import { callNotifikace } from "./notifikace";
|
import { callNotifikace } from "./notifikace";
|
||||||
import { generateQr } from "./qr";
|
import { generateQr } from "./qr";
|
||||||
import getStorage from "./storage";
|
import getStorage from "./storage";
|
||||||
import { downloadPizzy, downloadSalaty } from "./chefie";
|
import { downloadPizzy } from "./chefie";
|
||||||
import { getClientData, getToday, initIfNeeded } from "./service";
|
import { getClientData, getToday, initIfNeeded } from "./service";
|
||||||
import { Pizza, Salat, ClientData, PizzaDayState, PizzaSize, PizzaOrder, PizzaVariant, UdalostEnum, PendingQr } from "../../types/gen/types.gen";
|
import { Pizza, ClientData, PizzaDayState, PizzaSize, PizzaOrder, PizzaVariant, UdalostEnum } from "../../types/gen/types.gen";
|
||||||
import crypto from "crypto";
|
|
||||||
|
|
||||||
const storage = getStorage();
|
const storage = getStorage();
|
||||||
const PENDING_QR_PREFIX = 'pending_qr';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vrátí seznam dostupných pizz pro dnešní den.
|
* Vrátí seznam dostupných pizz pro dnešní den.
|
||||||
@@ -39,34 +37,6 @@ export async function savePizzaList(pizzaList: Pizza[]): Promise<ClientData> {
|
|||||||
return clientData;
|
return clientData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Vrátí seznam dostupných salátů pro dnešní den.
|
|
||||||
* Stáhne je, pokud je pro dnešní den nemá.
|
|
||||||
*/
|
|
||||||
export async function getSalatList(): Promise<Salat[] | undefined> {
|
|
||||||
await initIfNeeded();
|
|
||||||
let clientData = await getClientData(getToday());
|
|
||||||
if (!clientData.salatList) {
|
|
||||||
const mock = process.env.MOCK_DATA === 'true';
|
|
||||||
clientData = await saveSalatList(await downloadSalaty(mock));
|
|
||||||
}
|
|
||||||
return Promise.resolve(clientData.salatList);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uloží seznam dostupných salátů pro dnešní den.
|
|
||||||
*
|
|
||||||
* @param salatList seznam dostupných salátů
|
|
||||||
*/
|
|
||||||
export async function saveSalatList(salatList: Salat[]): Promise<ClientData> {
|
|
||||||
await initIfNeeded();
|
|
||||||
const today = formatDate(getToday());
|
|
||||||
const clientData = await getClientData(getToday());
|
|
||||||
clientData.salatList = salatList;
|
|
||||||
await storage.setData(today, clientData);
|
|
||||||
return clientData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vytvoří pizza day pro aktuální den a vrátí data pro klienta.
|
* Vytvoří pizza day pro aktuální den a vrátí data pro klienta.
|
||||||
*/
|
*/
|
||||||
@@ -74,11 +44,11 @@ export async function createPizzaDay(creator: string): Promise<ClientData> {
|
|||||||
await initIfNeeded();
|
await initIfNeeded();
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (clientData.pizzaDay) {
|
if (clientData.pizzaDay) {
|
||||||
throw new 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, salatList] = await Promise.all([getPizzaList(), getSalatList()]);
|
const pizzaList = await getPizzaList();
|
||||||
const data: ClientData = { pizzaDay: { state: PizzaDayState.CREATED, creator, orders: [] }, pizzaList, salatList, ...clientData };
|
const data: ClientData = { pizzaDay: { state: PizzaDayState.CREATED, creator, orders: [] }, pizzaList, ...clientData };
|
||||||
const today = formatDate(getToday());
|
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 } })
|
||||||
@@ -91,10 +61,10 @@ export async function createPizzaDay(creator: string): Promise<ClientData> {
|
|||||||
export async function deletePizzaDay(login: string): Promise<ClientData> {
|
export async function deletePizzaDay(login: string): Promise<ClientData> {
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
throw Error("Pizza day pro dnešní den neexistuje");
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new 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());
|
const today = formatDate(getToday());
|
||||||
@@ -113,10 +83,10 @@ 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 = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new 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 new 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: PizzaOrder | undefined = clientData.pizzaDay?.orders?.find(o => o.customer === login);
|
||||||
if (!order) {
|
if (!order) {
|
||||||
@@ -142,76 +112,6 @@ export async function addPizzaOrder(login: string, pizza: Pizza, size: PizzaSize
|
|||||||
return clientData;
|
return clientData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Přidá objednávku salátu uživateli.
|
|
||||||
*
|
|
||||||
* @param login login uživatele
|
|
||||||
* @param salat zvolený salát
|
|
||||||
*/
|
|
||||||
export async function addSalatOrder(login: string, salat: Salat) {
|
|
||||||
const today = formatDate(getToday());
|
|
||||||
const clientData = await getClientData(getToday());
|
|
||||||
if (!clientData.pizzaDay) {
|
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
|
||||||
}
|
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.CREATED) {
|
|
||||||
throw new Error("Pizza day není ve stavu " + PizzaDayState.CREATED);
|
|
||||||
}
|
|
||||||
let order: PizzaOrder | undefined = clientData.pizzaDay?.orders?.find(o => o.customer === login);
|
|
||||||
if (!order) {
|
|
||||||
order = {
|
|
||||||
customer: login,
|
|
||||||
pizzaList: [],
|
|
||||||
totalPrice: 0,
|
|
||||||
hasQr: false,
|
|
||||||
}
|
|
||||||
clientData.pizzaDay.orders ??= [];
|
|
||||||
clientData.pizzaDay.orders.push(order);
|
|
||||||
}
|
|
||||||
const salatOrder: PizzaVariant = {
|
|
||||||
varId: 0,
|
|
||||||
name: salat.name,
|
|
||||||
size: "1 porce",
|
|
||||||
price: salat.price,
|
|
||||||
category: 'salat',
|
|
||||||
}
|
|
||||||
order.pizzaList ??= [];
|
|
||||||
order.pizzaList.push(salatOrder);
|
|
||||||
order.totalPrice += salatOrder.price;
|
|
||||||
await storage.setData(today, clientData);
|
|
||||||
return clientData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Odstraní všechny pizzy uživatele (celou jeho objednávku).
|
|
||||||
* Pokud Pizza day neexistuje nebo není ve stavu CREATED, neudělá nic.
|
|
||||||
*
|
|
||||||
* @param login login uživatele
|
|
||||||
* @param date datum, pro které se objednávka odstraňuje (výchozí je dnešek)
|
|
||||||
* @returns aktuální data pro klienta
|
|
||||||
*/
|
|
||||||
export async function removeAllUserPizzas(login: string, date?: Date) {
|
|
||||||
const usedDate = date ?? getToday();
|
|
||||||
const today = formatDate(usedDate);
|
|
||||||
const clientData = await getClientData(usedDate);
|
|
||||||
|
|
||||||
if (!clientData.pizzaDay) {
|
|
||||||
return clientData; // Pizza day neexistuje, není co mazat
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.CREATED) {
|
|
||||||
return clientData; // Pizza day není ve stavu CREATED, nelze mazat
|
|
||||||
}
|
|
||||||
|
|
||||||
const orderIndex = clientData.pizzaDay.orders!.findIndex(o => o.customer === login);
|
|
||||||
if (orderIndex >= 0) {
|
|
||||||
clientData.pizzaDay.orders!.splice(orderIndex, 1);
|
|
||||||
await storage.setData(today, clientData);
|
|
||||||
}
|
|
||||||
|
|
||||||
return clientData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Odstraní danou objednávku pizzy.
|
* Odstraní danou objednávku pizzy.
|
||||||
*
|
*
|
||||||
@@ -222,16 +122,16 @@ export async function removePizzaOrder(login: string, pizzaOrder: PizzaVariant)
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new 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 new 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 new 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);
|
||||||
@@ -253,13 +153,13 @@ export async function lockPizzaDay(login: string) {
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
throw Error("Pizza day pro dnešní den neexistuje");
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new Error("Pizza day není spravován uživatelem " + login);
|
throw Error("Pizza day není spravován uživatelem " + login);
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.CREATED && clientData.pizzaDay.state !== PizzaDayState.ORDERED) {
|
if (clientData.pizzaDay.state !== PizzaDayState.CREATED && clientData.pizzaDay.state !== PizzaDayState.ORDERED) {
|
||||||
throw new Error("Pizza day není ve stavu " + PizzaDayState.CREATED + " nebo " + PizzaDayState.ORDERED);
|
throw Error("Pizza day není ve stavu " + PizzaDayState.CREATED + " nebo " + PizzaDayState.ORDERED);
|
||||||
}
|
}
|
||||||
clientData.pizzaDay.state = PizzaDayState.LOCKED;
|
clientData.pizzaDay.state = PizzaDayState.LOCKED;
|
||||||
await storage.setData(today, clientData);
|
await storage.setData(today, clientData);
|
||||||
@@ -276,13 +176,13 @@ export async function unlockPizzaDay(login: string) {
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
throw Error("Pizza day pro dnešní den neexistuje");
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new Error("Pizza day není spravován uživatelem " + login);
|
throw Error("Pizza day není spravován uživatelem " + login);
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.LOCKED) {
|
if (clientData.pizzaDay.state !== PizzaDayState.LOCKED) {
|
||||||
throw new Error("Pizza day není ve stavu " + PizzaDayState.LOCKED);
|
throw Error("Pizza day není ve stavu " + PizzaDayState.LOCKED);
|
||||||
}
|
}
|
||||||
clientData.pizzaDay.state = PizzaDayState.CREATED;
|
clientData.pizzaDay.state = PizzaDayState.CREATED;
|
||||||
await storage.setData(today, clientData);
|
await storage.setData(today, clientData);
|
||||||
@@ -299,13 +199,13 @@ export async function finishPizzaOrder(login: string) {
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
throw Error("Pizza day pro dnešní den neexistuje");
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new Error("Pizza day není spravován uživatelem " + login);
|
throw Error("Pizza day není spravován uživatelem " + login);
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.LOCKED) {
|
if (clientData.pizzaDay.state !== PizzaDayState.LOCKED) {
|
||||||
throw new Error("Pizza day není ve stavu " + PizzaDayState.LOCKED);
|
throw Error("Pizza day není ve stavu " + PizzaDayState.LOCKED);
|
||||||
}
|
}
|
||||||
clientData.pizzaDay.state = PizzaDayState.ORDERED;
|
clientData.pizzaDay.state = PizzaDayState.ORDERED;
|
||||||
await storage.setData(today, clientData);
|
await storage.setData(today, clientData);
|
||||||
@@ -324,13 +224,13 @@ export async function finishPizzaDelivery(login: string, bankAccount?: string, b
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
const clientData = await getClientData(getToday());
|
const clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new Error("Pizza day pro dnešní den neexistuje");
|
throw Error("Pizza day pro dnešní den neexistuje");
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new Error("Pizza day není spravován uživatelem " + login);
|
throw Error("Pizza day není spravován uživatelem " + login);
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.state !== PizzaDayState.ORDERED) {
|
if (clientData.pizzaDay.state !== PizzaDayState.ORDERED) {
|
||||||
throw new Error("Pizza day není ve stavu " + PizzaDayState.ORDERED);
|
throw Error("Pizza day není ve stavu " + PizzaDayState.ORDERED);
|
||||||
}
|
}
|
||||||
clientData.pizzaDay.state = PizzaDayState.DELIVERED;
|
clientData.pizzaDay.state = PizzaDayState.DELIVERED;
|
||||||
|
|
||||||
@@ -338,20 +238,9 @@ export async function finishPizzaDelivery(login: string, bankAccount?: string, b
|
|||||||
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
|
||||||
const id = crypto.randomUUID();
|
let message = order.pizzaList!.map(pizza => `Pizza ${pizza.name} (${pizza.size})`).join(', ');
|
||||||
let message = order.pizzaList!.map(item =>
|
await generateQr(order.customer, bankAccount, bankAccountHolder, order.totalPrice, message);
|
||||||
item.category === 'salat' ? `Salát ${item.name}` : `Pizza ${item.name} (${item.size})`
|
|
||||||
).join(', ');
|
|
||||||
await generateQr(order.customer, bankAccount, bankAccountHolder, order.totalPrice / 100, message, id);
|
|
||||||
order.hasQr = true;
|
order.hasQr = true;
|
||||||
// Uložíme nevyřízený QR kód pro persistentní zobrazení
|
|
||||||
await addPendingQr(order.customer, {
|
|
||||||
id,
|
|
||||||
date: today,
|
|
||||||
creator: login,
|
|
||||||
totalPrice: order.totalPrice,
|
|
||||||
purpose: message,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -370,14 +259,14 @@ export async function updatePizzaDayNote(login: string, note?: string) {
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
let clientData = await getClientData(getToday());
|
let clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new 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 new 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?.pizzaList?.length) {
|
||||||
throw new 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;
|
||||||
await storage.setData(today, clientData);
|
await storage.setData(today, clientData);
|
||||||
@@ -397,17 +286,17 @@ export async function updatePizzaFee(login: string, targetLogin: string, text?:
|
|||||||
const today = formatDate(getToday());
|
const today = formatDate(getToday());
|
||||||
let clientData = await getClientData(getToday());
|
let clientData = await getClientData(getToday());
|
||||||
if (!clientData.pizzaDay) {
|
if (!clientData.pizzaDay) {
|
||||||
throw new 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 new Error(`Pizza day není ve stavu ${PizzaDayState.CREATED}`);
|
throw Error(`Pizza day není ve stavu ${PizzaDayState.CREATED}`);
|
||||||
}
|
}
|
||||||
if (clientData.pizzaDay.creator !== login) {
|
if (clientData.pizzaDay.creator !== login) {
|
||||||
throw new 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?.pizzaList?.length) {
|
||||||
throw new 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) {
|
||||||
delete targetOrder.fee;
|
delete targetOrder.fee;
|
||||||
@@ -419,57 +308,3 @@ export async function updatePizzaFee(login: string, targetLogin: string, text?:
|
|||||||
await storage.setData(today, clientData);
|
await storage.setData(today, clientData);
|
||||||
return clientData;
|
return clientData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Vrátí klíč pro uložení nevyřízených QR kódů uživatele.
|
|
||||||
*/
|
|
||||||
function getPendingQrKey(login: string): string {
|
|
||||||
return `${PENDING_QR_PREFIX}_${login}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Přidá nevyřízený QR kód pro uživatele.
|
|
||||||
*/
|
|
||||||
export async function addPendingQr(login: string, pendingQr: PendingQr): Promise<void> {
|
|
||||||
const key = getPendingQrKey(login);
|
|
||||||
const existing = await storage.getData<PendingQr[]>(key) ?? [];
|
|
||||||
// Deduplikace podle id (ne podle data — jeden den může mít uživatel víc QR kódů)
|
|
||||||
if (!existing.some(qr => qr.id === pendingQr.id)) {
|
|
||||||
existing.push(pendingQr);
|
|
||||||
await storage.setData(key, existing);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Vrátí nevyřízené QR kódy pro uživatele.
|
|
||||||
*/
|
|
||||||
export async function getPendingQrs(login: string): Promise<PendingQr[]> {
|
|
||||||
return await storage.getData<PendingQr[]>(getPendingQrKey(login)) ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Označí QR kód jako uhrazený (odstraní ho ze seznamu nevyřízených).
|
|
||||||
* Vrátí odstraněný QR kód, pokud byl nalezen.
|
|
||||||
*/
|
|
||||||
export async function dismissPendingQr(login: string, id: string): Promise<PendingQr | undefined> {
|
|
||||||
const key = getPendingQrKey(login);
|
|
||||||
const existing = await storage.getData<PendingQr[]>(key) ?? [];
|
|
||||||
const dismissed = existing.find(qr => qr.id === id);
|
|
||||||
const filtered = existing.filter(qr => qr.id !== id);
|
|
||||||
await storage.setData(key, filtered);
|
|
||||||
return dismissed;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Odstraní všechny nevyřízené QR kódy patřící dané skupině objednávky.
|
|
||||||
*/
|
|
||||||
export async function removePendingQrsByGroupId(logins: string[], groupId: string): Promise<void> {
|
|
||||||
for (const login of logins) {
|
|
||||||
const key = getPendingQrKey(login);
|
|
||||||
const existing = await storage.getData<PendingQr[]>(key) ?? [];
|
|
||||||
const filtered = existing.filter(qr => qr.groupId !== groupId);
|
|
||||||
if (filtered.length !== existing.length) {
|
|
||||||
await storage.setData(key, filtered);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,166 +0,0 @@
|
|||||||
import webpush from 'web-push';
|
|
||||||
import crypto from 'crypto';
|
|
||||||
import getStorage from './storage';
|
|
||||||
import { getClientData, getToday } from './service';
|
|
||||||
import { getIsWeekend } from './utils';
|
|
||||||
import { LunchChoices } from '../../types';
|
|
||||||
|
|
||||||
const storage = getStorage();
|
|
||||||
const REGISTRY_KEY = 'push_reminder_registry';
|
|
||||||
|
|
||||||
interface RegistryEntry {
|
|
||||||
time: string;
|
|
||||||
subscription: webpush.PushSubscription;
|
|
||||||
}
|
|
||||||
|
|
||||||
type Registry = Record<string, RegistryEntry>;
|
|
||||||
|
|
||||||
/** Mapa login → timestamp (ms) posledního odeslání připomínky. */
|
|
||||||
const lastReminded = new Map<string, number>();
|
|
||||||
|
|
||||||
const REMINDER_COOLDOWN_MS = 60 * 60 * 1000; // 60 minut mezi připomínkami
|
|
||||||
|
|
||||||
function getCurrentTimeHHMM(): string {
|
|
||||||
const now = new Date();
|
|
||||||
return `${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Zjistí, zda má uživatel zvolenou nějakou možnost stravování. */
|
|
||||||
function userHasChoice(choices: LunchChoices, login: string): boolean {
|
|
||||||
for (const locationKey of Object.keys(choices)) {
|
|
||||||
const locationChoices = choices[locationKey as keyof LunchChoices];
|
|
||||||
if (locationChoices && login in locationChoices) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getRegistry(): Promise<Registry> {
|
|
||||||
return await storage.getData<Registry>(REGISTRY_KEY) ?? {};
|
|
||||||
}
|
|
||||||
|
|
||||||
async function saveRegistry(registry: Registry): Promise<void> {
|
|
||||||
await storage.setData(REGISTRY_KEY, registry);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Přidá nebo aktualizuje push subscription pro uživatele. */
|
|
||||||
export async function subscribePush(login: string, subscription: webpush.PushSubscription, reminderTime: string): Promise<void> {
|
|
||||||
const registry = await getRegistry();
|
|
||||||
registry[login] = { time: reminderTime, subscription };
|
|
||||||
await saveRegistry(registry);
|
|
||||||
console.log(`Push reminder: uživatel ${login} přihlášen k připomínkám v ${reminderTime}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Odebere push subscription pro uživatele. */
|
|
||||||
export async function unsubscribePush(login: string): Promise<void> {
|
|
||||||
const registry = await getRegistry();
|
|
||||||
delete registry[login];
|
|
||||||
await saveRegistry(registry);
|
|
||||||
lastReminded.delete(login);
|
|
||||||
console.log(`Push reminder: uživatel ${login} odhlášen z připomínek`);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Vrátí veřejný VAPID klíč. */
|
|
||||||
export function getVapidPublicKey(): string | undefined {
|
|
||||||
return process.env.VAPID_PUBLIC_KEY;
|
|
||||||
}
|
|
||||||
|
|
||||||
function generateQuickChoiceToken(login: string): string {
|
|
||||||
const today = new Date().toISOString().slice(0, 10);
|
|
||||||
const secret = process.env.JWT_SECRET ?? '';
|
|
||||||
return crypto.createHmac('sha256', secret).update(`${login}:${today}`).digest('hex');
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Ověří jednorázový token z push notifikace. */
|
|
||||||
export function verifyQuickChoiceToken(login: string, token: string): boolean {
|
|
||||||
if (!login || !token || token.length !== 64) return false;
|
|
||||||
const expected = generateQuickChoiceToken(login);
|
|
||||||
return crypto.timingSafeEqual(Buffer.from(expected, 'hex'), Buffer.from(token, 'hex'));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/** Zkontroluje a odešle připomínky uživatelům, kteří si nezvolili oběd. */
|
|
||||||
async function checkAndSendReminders(): Promise<void> {
|
|
||||||
// Přeskočit víkendy
|
|
||||||
if (getIsWeekend(getToday())) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const registry = await getRegistry();
|
|
||||||
const entries = Object.entries(registry);
|
|
||||||
if (entries.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const currentTime = getCurrentTimeHHMM();
|
|
||||||
|
|
||||||
// Získáme data pro dnešek jednou pro všechny uživatele
|
|
||||||
let clientData;
|
|
||||||
try {
|
|
||||||
clientData = await getClientData(getToday());
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Push reminder: chyba při získávání dat', e);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const [login, entry] of entries) {
|
|
||||||
// Ještě nedosáhl čas připomínky
|
|
||||||
if (currentTime < entry.time) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cooldown — nepřipomínat častěji než jednou za hodinu
|
|
||||||
const last = lastReminded.get(login) ?? 0;
|
|
||||||
if (Date.now() - last < REMINDER_COOLDOWN_MS) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Uživatel už má zvolenou možnost
|
|
||||||
if (clientData.choices && userHasChoice(clientData.choices, login)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Odešleme push notifikaci
|
|
||||||
try {
|
|
||||||
await webpush.sendNotification(
|
|
||||||
entry.subscription,
|
|
||||||
JSON.stringify({
|
|
||||||
title: 'Luncher',
|
|
||||||
body: 'Ještě nemáte zvolený oběd!',
|
|
||||||
login,
|
|
||||||
token: generateQuickChoiceToken(login),
|
|
||||||
})
|
|
||||||
);
|
|
||||||
lastReminded.set(login, Date.now());
|
|
||||||
console.log(`Push reminder: odeslána připomínka uživateli ${login}`);
|
|
||||||
} catch (error: any) {
|
|
||||||
if (error.statusCode === 410 || error.statusCode === 404) {
|
|
||||||
// Subscription expirovala nebo je neplatná — odebereme z registry
|
|
||||||
console.log(`Push reminder: subscription uživatele ${login} expirovala, odebírám`);
|
|
||||||
delete registry[login];
|
|
||||||
await saveRegistry(registry);
|
|
||||||
} else {
|
|
||||||
console.error(`Push reminder: chyba při odesílání notifikace uživateli ${login}:`, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Spustí scheduler pro kontrolu a odesílání připomínek každou minutu. */
|
|
||||||
export function startReminderScheduler(): void {
|
|
||||||
const publicKey = process.env.VAPID_PUBLIC_KEY;
|
|
||||||
const privateKey = process.env.VAPID_PRIVATE_KEY;
|
|
||||||
const subject = process.env.VAPID_SUBJECT;
|
|
||||||
|
|
||||||
if (!publicKey || !privateKey || !subject) {
|
|
||||||
console.log('Push reminder: VAPID klíče nejsou nastaveny, scheduler nebude spuštěn');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
webpush.setVapidDetails(subject, publicKey, privateKey);
|
|
||||||
|
|
||||||
// Spustíme kontrolu každou minutu
|
|
||||||
setInterval(checkAndSendReminders, 60_000);
|
|
||||||
console.log('Push reminder: scheduler spuštěn');
|
|
||||||
}
|
|
||||||
+34
-28
@@ -1,20 +1,22 @@
|
|||||||
|
import fs from "fs";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
|
import os from "os";
|
||||||
|
import path from "path";
|
||||||
import crypto from "crypto";
|
import crypto from "crypto";
|
||||||
import getStorage from "./storage";
|
import { formatDate } from "./utils";
|
||||||
|
|
||||||
const QR_GENERATOR_URL = 'https://api.paylibo.com/paylibo/generator/image';
|
const QR_GENERATOR_URL = 'https://api.paylibo.com/paylibo/generator/image';
|
||||||
const COUNTRY_CODE = 'CZ';
|
const COUNTRY_CODE = 'CZ';
|
||||||
const CURRENCY_CODE = 'CZK';
|
const CURRENCY_CODE = 'CZK';
|
||||||
const QR_PIXEL_SIZE = 256;
|
const QR_PIXEL_SIZE = 256;
|
||||||
|
const tmpDir = os.tmpdir();
|
||||||
const storage = getStorage();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Převede číslo účtu z BBAN do IBAN. Automaticky dopočítá kontrolní číslice.
|
* Převede číslo účtu z BBAN do IBAN. Automaticky dopočítá kontrolní číslice.
|
||||||
*
|
*
|
||||||
* @param bankAccountNumber číslo účtu ve formátu BBAN (123456-0123456789/0100)
|
* @param bankAccountNumber číslo účtu ve formátu BBAN (123456-0123456789/0100)
|
||||||
*/
|
*/
|
||||||
export function convertBbanToIban(bankAccountNumber: string): string {
|
function convertBbanToIban(bankAccountNumber: string): string {
|
||||||
// TODO validovat číslo účtu stejně jako na klientovi, pro případ že sem někdo pošle nesmysl
|
// TODO validovat číslo účtu stejně jako na klientovi, pro případ že sem někdo pošle nesmysl
|
||||||
let prefix: string = '';
|
let prefix: string = '';
|
||||||
let accountNumber: string = bankAccountNumber;
|
let accountNumber: string = bankAccountNumber;
|
||||||
@@ -31,33 +33,38 @@ export function convertBbanToIban(bankAccountNumber: string): string {
|
|||||||
// Zatím napevno, nemá smysl řešit nic jiného než CZ
|
// Zatím napevno, nemá smysl řešit nic jiného než CZ
|
||||||
iban = iban.replace('C', '12').replace('Z', '35');
|
iban = iban.replace('C', '12').replace('Z', '35');
|
||||||
const remainder = BigInt(iban) % BigInt(97);
|
const remainder = BigInt(iban) % BigInt(97);
|
||||||
const checkDigits = (BigInt(98) - remainder).toString().padStart(2, '0');
|
const checkDigits = BigInt(98) - remainder;
|
||||||
iban = `${COUNTRY_CODE}${checkDigits}${bankCode}${prefix}${accountNumber}`;
|
iban = `${COUNTRY_CODE}${checkDigits.toString()}${bankCode}${prefix}${accountNumber}`;
|
||||||
if (iban.length !== 24) {
|
if (iban.length !== 24) {
|
||||||
throw new Error("Neplatná délka sestaveného IBAN: " + iban.length + ", očekáváno 24");
|
throw Error("Neplatná délka sestaveného IBAN: " + iban.length + ", očekáváno 24");
|
||||||
}
|
}
|
||||||
return iban;
|
return iban;
|
||||||
}
|
}
|
||||||
|
|
||||||
function createStorageKey(customerName: string, id: string): string {
|
function createNameHash(customerName: string): string {
|
||||||
const nameHash = crypto.createHash('md5').update(customerName).digest('hex');
|
return crypto.createHash('md5').update(customerName).digest('hex');
|
||||||
return `qr_${nameHash}_${id}`;
|
}
|
||||||
|
|
||||||
|
function createFilePath(nameHash: string): string {
|
||||||
|
const fileName = `${formatDate(new Date())}_${nameHash}.png`;
|
||||||
|
return path.join(tmpDir, fileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vygeneruje a uloží obrázek platebního QR kódu do storage (Redis/JSON).
|
* Vygeneruje, uloží a vrátí unikátní ID obrázku platebního QR kódu s danými parametry.
|
||||||
* Data přežijí redeploy — není třeba persistentní filesystém.
|
|
||||||
*
|
*
|
||||||
* @param customerName jméno uživatele, pro kterého je QR kód generován
|
* @param customerName jméno uživatele, pro kterého je QR kód generován
|
||||||
* @param bankAccountNumber číslo cílového bankovního účtu ve formátu BBAN
|
* @param bankAccountNumber číslo cílového bankovního účtu ve formátu BBAN
|
||||||
* @param bankAccountHolder jméno držitele cílového bankovního účtu
|
* @param bankAccountHolder jméno držitele cílového bankovního účtu
|
||||||
* @param amount částka v Kč
|
* @param amount částka v Kč
|
||||||
* @param message zpráva pro příjemce
|
* @param message zpráva pro příjemce
|
||||||
* @param id unikátní identifikátor (UUID) tohoto QR kódu
|
* @returns hash, pomocí kterého lze následně získat vygenerovaný obrázek
|
||||||
*/
|
*/
|
||||||
export async function generateQr(customerName: string, bankAccountNumber: string, bankAccountHolder: string, amount: number, message: string, id: string): Promise<void> {
|
export async function generateQr(customerName: string, bankAccountNumber: string, bankAccountHolder: string, amount: number, message: string): Promise<string> {
|
||||||
// Zpráva nesmí obsahovat '*' a znaky mimo ISO 8859-1; délka max. 60 znaků
|
// Zpráva pro příjemce nesmí dle standardu obsahovat '*' a být delší než 60 znaků
|
||||||
message = message.replace(/[^\x00-\xff]/g, '').replace(/\*/g, '');
|
if (message.indexOf('*') >= 0) {
|
||||||
|
message = message.replace('*', '');
|
||||||
|
}
|
||||||
if (message.length > 60) {
|
if (message.length > 60) {
|
||||||
message = message.substring(0, 60);
|
message = message.substring(0, 60);
|
||||||
}
|
}
|
||||||
@@ -70,23 +77,22 @@ export async function generateQr(customerName: string, bankAccountNumber: string
|
|||||||
branding: false,
|
branding: false,
|
||||||
compress: false,
|
compress: false,
|
||||||
size: QR_PIXEL_SIZE,
|
size: QR_PIXEL_SIZE,
|
||||||
};
|
}
|
||||||
const response = await axios.get(QR_GENERATOR_URL, { responseType: 'arraybuffer', params: { ...payload } });
|
const response = await axios.get(QR_GENERATOR_URL, { responseType: 'stream', params: { ...payload } });
|
||||||
const base64 = Buffer.from(response.data).toString('base64');
|
// Použijeme hash, abychom nemuseli řešit nepovolené znaky ve jménu uživatele
|
||||||
await storage.setData(createStorageKey(customerName, id), base64);
|
const nameHash = createNameHash(customerName);
|
||||||
|
const imgPath = createFilePath(nameHash);
|
||||||
|
response.data.pipe(fs.createWriteStream(imgPath));
|
||||||
|
return nameHash;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vrátí obrázek s QR kódem ze storage.
|
* Vrátí obrázek s QR kódem, pokud existuje.
|
||||||
*
|
*
|
||||||
* @param customerName jméno uživatele
|
* @param customerName jméno uživatele
|
||||||
* @param id unikátní identifikátor QR kódu
|
|
||||||
* @returns data obrázku
|
* @returns data obrázku
|
||||||
*/
|
*/
|
||||||
export async function getQr(customerName: string, id: string): Promise<Buffer> {
|
export function getQr(customerName: string): Buffer {
|
||||||
const base64 = await storage.getData<string>(createStorageKey(customerName, id));
|
const imgPath = createFilePath(createNameHash(customerName));
|
||||||
if (!base64) {
|
return fs.readFileSync(imgPath);
|
||||||
throw new Error("QR kód nebyl nalezen");
|
|
||||||
}
|
|
||||||
return Buffer.from(base64, 'base64');
|
|
||||||
}
|
}
|
||||||
+49
-57
@@ -4,10 +4,6 @@ import { getMenuSladovnickaMock, getMenuTechTowerMock, getMenuUMotlikuMock, getM
|
|||||||
import { formatDate } from "./utils";
|
import { formatDate } from "./utils";
|
||||||
import { Food } from "../../types/gen/types.gen";
|
import { Food } from "../../types/gen/types.gen";
|
||||||
|
|
||||||
export class StaleWeekError extends Error {
|
|
||||||
constructor(public food: Food[][]) { super('Data jsou z jiného týdne'); }
|
|
||||||
}
|
|
||||||
|
|
||||||
// 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 = [
|
||||||
'polévka',
|
'polévka',
|
||||||
@@ -40,7 +36,7 @@ const SENKSERIKOVA_URL = 'https://www.menicka.cz/6561-pivovarsky-senk-serikova.h
|
|||||||
* @param text vstupní text
|
* @param text vstupní text
|
||||||
* @returns true, pokud text představuje polévku
|
* @returns true, pokud text představuje polévku
|
||||||
*/
|
*/
|
||||||
export const isTextSoupName = (text: string): boolean => {
|
const isTextSoupName = (text: string): boolean => {
|
||||||
for (const name of SOUP_NAMES) {
|
for (const name of SOUP_NAMES) {
|
||||||
if (text.toLowerCase().includes(name)) {
|
if (text.toLowerCase().includes(name)) {
|
||||||
return true;
|
return true;
|
||||||
@@ -49,11 +45,11 @@ export const isTextSoupName = (text: string): boolean => {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const capitalize = (word: string): string => {
|
const capitalize = (word: string): string => {
|
||||||
return word.charAt(0).toUpperCase() + word.slice(1);
|
return word.charAt(0).toUpperCase() + word.slice(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
export const sanitizeText = (text: string): string => {
|
const sanitizeText = (text: string): string => {
|
||||||
return text.replace('\t', '').replace(' , ', ', ').trim();
|
return text.replace('\t', '').replace(' , ', ', ').trim();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -64,7 +60,7 @@ export const sanitizeText = (text: string): string => {
|
|||||||
* @param name původní název jídla
|
* @param name původní název jídla
|
||||||
* @returns objekt obsahující vyčištěný název a pole alergenů
|
* @returns objekt obsahující vyčištěný název a pole alergenů
|
||||||
*/
|
*/
|
||||||
export const parseAllergens = (name: string): { cleanName: string, allergens: number[] } => {
|
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
|
// 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 regex = /\s+(\d+(?:\s*,\s*\d+)*)\s*$/;
|
||||||
const match = regex.exec(name);
|
const match = regex.exec(name);
|
||||||
@@ -104,7 +100,7 @@ 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);
|
||||||
|
|
||||||
// Zjistíme, které dny jsou k dispozici z tab elementů
|
// Nejdříve zjistíme, které dny jsou k dispozici z tab elementů
|
||||||
const tabElements = $('#daily-menu-tab-list').children('button[id^="daily-menu-tab-"]');
|
const tabElements = $('#daily-menu-tab-list').children('button[id^="daily-menu-tab-"]');
|
||||||
const availableDays: { [dayIndex: number]: number } = {}; // mapování dayIndex -> contentIndex
|
const availableDays: { [dayIndex: number]: number } = {}; // mapování dayIndex -> contentIndex
|
||||||
|
|
||||||
@@ -116,7 +112,7 @@ export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = f
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const menuContentElements = $('#daily-menu-content-list').children('.daily-menu-content__content').not('.daily-menu-content__content--static');
|
const menuContentElements = $('#daily-menu-content-list').children('[id^="daily-menu-content-"]');
|
||||||
|
|
||||||
const result: Food[][] = [];
|
const result: Food[][] = [];
|
||||||
|
|
||||||
@@ -134,32 +130,59 @@ export const getMenuSladovnicka = async (firstDayOfWeek: Date, mock: boolean = f
|
|||||||
continue; // Přeskočíme, pokud content element neexistuje
|
continue; // Přeskočíme, pokud content element neexistuje
|
||||||
}
|
}
|
||||||
|
|
||||||
const contentElement = $(menuContentElements[contentIndexNum]);
|
const dayChildren = $(menuContentElements[contentIndexNum]).children();
|
||||||
const itemElement = contentElement.find('.daily-menu-content__item');
|
|
||||||
const table = itemElement.find('table.daily-menu-content__table tbody');
|
// Ověříme, že má element očekávanou strukturu
|
||||||
const rows = table.children('tr');
|
if (dayChildren.length < 2) {
|
||||||
|
console.warn(`Neočekávaný počet children v menu Sladovnické pro den ${dayIndexNum}: ${dayChildren.length}, očekávány alespoň 2 (polévka a hlavní jídlo)`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parsování polévky
|
||||||
|
const soupElement = dayChildren.get(0);
|
||||||
|
const soupTable = $(soupElement).find('table tbody tr');
|
||||||
|
const soupCells = soupTable.children('td');
|
||||||
|
if (soupCells.length !== 3) {
|
||||||
|
console.warn(`Neočekávaný počet buněk v tabulce polévky pro den ${dayIndexNum}: ${soupCells.length}, ale očekávány byly 3`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
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[] = [];
|
const currentDayFood: Food[] = [];
|
||||||
|
|
||||||
// Projdeme všechny řádky - první je polévka, zbytek jsou hlavní jídla
|
// Přidáme polévku do seznamu jídel
|
||||||
rows.each((i, row) => {
|
currentDayFood.push({
|
||||||
const cells = $(row).children('td');
|
amount: soupAmount,
|
||||||
if (cells.length !== 3) {
|
name: soupParsed.cleanName,
|
||||||
return; // Přeskočíme řádky s nesprávnou strukturou
|
price: soupPrice,
|
||||||
}
|
isSoup: true,
|
||||||
|
allergens: soupParsed.allergens.length > 0 ? soupParsed.allergens : undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Projdeme všechny řádky hlavních jídel
|
||||||
|
mainCourseRows.each((i, row) => {
|
||||||
|
const cells = $(row).children('td');
|
||||||
const amount = sanitizeText($(cells.get(0)).text());
|
const amount = sanitizeText($(cells.get(0)).text());
|
||||||
const nameRaw = sanitizeText($(cells.get(1)).text());
|
const nameRaw = sanitizeText($(cells.get(1)).text());
|
||||||
const price = sanitizeText($(cells.get(2)).text().replace(' ', '\xA0'));
|
const price = sanitizeText($(cells.get(2)).text().replace(' ', '\xA0'));
|
||||||
const parsed = parseAllergens(nameRaw);
|
const parsed = parseAllergens(nameRaw);
|
||||||
|
|
||||||
// Přeskočíme prázdné řádky
|
// Přeskočíme prázdné řádky (první řádek může být prázdný)
|
||||||
if (parsed.cleanName.trim().length > 0) {
|
if (parsed.cleanName.trim().length > 0) {
|
||||||
currentDayFood.push({
|
currentDayFood.push({
|
||||||
amount,
|
amount,
|
||||||
name: parsed.cleanName,
|
name: parsed.cleanName,
|
||||||
price,
|
price,
|
||||||
isSoup: i === 0, // První řádek je polévka
|
isSoup: false,
|
||||||
allergens: parsed.allergens.length > 0 ? parsed.allergens : undefined,
|
allergens: parsed.allergens.length > 0 ? parsed.allergens : undefined,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -280,7 +303,6 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal
|
|||||||
const $ = load(html);
|
const $ = load(html);
|
||||||
|
|
||||||
let secondTry = false;
|
let secondTry = false;
|
||||||
let thirdTry = false;
|
|
||||||
// První pokus - varianta "Obědy"
|
// První pokus - varianta "Obědy"
|
||||||
let fonts = $('font.wsw-41');
|
let fonts = $('font.wsw-41');
|
||||||
let font = undefined;
|
let font = undefined;
|
||||||
@@ -289,7 +311,7 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal
|
|||||||
font = f;
|
font = f;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
// Druhý pokus - varianta "Jídelní lístek" (starší formát)
|
// Druhý pokus - varianta "Jídelní lístek"
|
||||||
if (!font) {
|
if (!font) {
|
||||||
fonts = $('font.wnd-font-size-90');
|
fonts = $('font.wnd-font-size-90');
|
||||||
fonts.each((i, f) => {
|
fonts.each((i, f) => {
|
||||||
@@ -299,26 +321,13 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
// Třetí pokus - nový formát: font.wsw-41 s textem "Jídelní lístek" (vše v jednom bloku)
|
|
||||||
if (!font) {
|
|
||||||
fonts = $('font.wsw-41');
|
|
||||||
fonts.each((i, f) => {
|
|
||||||
if ($(f).text().trim().startsWith('Jídelní lístek')) {
|
|
||||||
font = f;
|
|
||||||
thirdTry = true;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if (!font) {
|
if (!font) {
|
||||||
throw new Error('Chyba: nenalezen <font> pro obědy v HTML Techtower.');
|
throw new Error('Chyba: nenalezen <font> pro obědy v HTML Techtower.');
|
||||||
}
|
}
|
||||||
|
|
||||||
const result: Food[][] = [];
|
const result: Food[][] = [];
|
||||||
const siblings = thirdTry
|
// TODO validovat, že v textu nalezeného <font> je rozsah, do kterého spadá vstupní datum
|
||||||
? $(font).parent().siblings('p')
|
const siblings = secondTry ? $(font).parent().parent().parent().siblings('p') : $(font).parent().parent().siblings();
|
||||||
: secondTry
|
|
||||||
? $(font).parent().parent().parent().siblings('p')
|
|
||||||
: $(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++) {
|
||||||
@@ -342,13 +351,8 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal
|
|||||||
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('•', '');
|
nameRaw = split[0].replace('•', '');
|
||||||
} else if (text.toLowerCase().endsWith(',-')) {
|
|
||||||
const tmp = text.replace('\xA0', ' ').split(' ');
|
|
||||||
const split = [tmp.slice(0, -1).join(' ')].concat(tmp.slice(-1));
|
|
||||||
price = `${split.slice(1)[0].replace(',-', '')}\xA0Kč`
|
|
||||||
nameRaw = split[0].replace('•', '');
|
|
||||||
}
|
}
|
||||||
if (nameRaw.endsWith('–')|| nameRaw.endsWith('—')) {
|
if (nameRaw.endsWith('–')) {
|
||||||
nameRaw = nameRaw.slice(0, -1).trim();
|
nameRaw = nameRaw.slice(0, -1).trim();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -363,18 +367,6 @@ export const getMenuTechTower = async (firstDayOfWeek: Date, mock: boolean = fal
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validace, zda text hlavičky obsahuje datum odpovídající požadovanému týdnu
|
|
||||||
const headerText = $(font).text().trim();
|
|
||||||
const dateMatch = headerText.match(/(\d{1,2})\.(\d{1,2})\./);
|
|
||||||
if (dateMatch) {
|
|
||||||
const foundDay = parseInt(dateMatch[1]);
|
|
||||||
const foundMonth = parseInt(dateMatch[2]) - 1; // JS months are 0-based
|
|
||||||
if (foundDay !== firstDayOfWeek.getDate() || foundMonth !== firstDayOfWeek.getMonth()) {
|
|
||||||
throw new StaleWeekError(result);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,50 +0,0 @@
|
|||||||
import express, { Request, Response } from "express";
|
|
||||||
import fs from "fs";
|
|
||||||
import path from "path";
|
|
||||||
|
|
||||||
const router = express.Router();
|
|
||||||
|
|
||||||
const CHANGELOGS_DIR = path.resolve(__dirname, "../../changelogs");
|
|
||||||
|
|
||||||
// In-memory cache: datum → seznam změn
|
|
||||||
const cache: Record<string, string[]> = {};
|
|
||||||
|
|
||||||
function loadAllChangelogs(): Record<string, string[]> {
|
|
||||||
let files: string[];
|
|
||||||
try {
|
|
||||||
files = fs.readdirSync(CHANGELOGS_DIR).filter(f => f.endsWith(".json"));
|
|
||||||
} catch {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const file of files) {
|
|
||||||
const date = file.replace(".json", "");
|
|
||||||
if (!cache[date]) {
|
|
||||||
const content = fs.readFileSync(path.join(CHANGELOGS_DIR, file), "utf-8");
|
|
||||||
cache[date] = JSON.parse(content);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return cache;
|
|
||||||
}
|
|
||||||
|
|
||||||
router.get("/", (req: Request, res: Response) => {
|
|
||||||
const all = loadAllChangelogs();
|
|
||||||
const since = typeof req.query.since === "string" ? req.query.since : undefined;
|
|
||||||
|
|
||||||
// Seřazení od nejnovějšího po nejstarší
|
|
||||||
const sortedDates = Object.keys(all).sort((a, b) => b.localeCompare(a));
|
|
||||||
|
|
||||||
const filteredDates = since
|
|
||||||
? sortedDates.filter(date => date > since)
|
|
||||||
: sortedDates;
|
|
||||||
|
|
||||||
const result: Record<string, string[]> = {};
|
|
||||||
for (const date of filteredDates) {
|
|
||||||
result[date] = all[date];
|
|
||||||
}
|
|
||||||
|
|
||||||
res.status(200).json(result);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default router;
|
|
||||||
@@ -1,198 +0,0 @@
|
|||||||
import express, { Request } from "express";
|
|
||||||
import { getDateForWeekIndex, getData, getRestaurantMenu, getToday, initIfNeeded } from "../service";
|
|
||||||
import { formatDate, getDayOfWeekIndex } from "../utils";
|
|
||||||
import getStorage from "../storage";
|
|
||||||
import { getWebsocket } from "../websocket";
|
|
||||||
import { getLogin } from "../auth";
|
|
||||||
import { parseToken } from "../utils";
|
|
||||||
import webpush from 'web-push';
|
|
||||||
|
|
||||||
const router = express.Router();
|
|
||||||
const storage = getStorage();
|
|
||||||
const ENVIRONMENT = process.env.NODE_ENV ?? 'production';
|
|
||||||
|
|
||||||
// Seznam náhodných jmen pro generování mock dat
|
|
||||||
const MOCK_NAMES = [
|
|
||||||
'Alice', 'Bob', 'Charlie', 'David', 'Eva', 'Filip', 'Gita', 'Honza',
|
|
||||||
'Ivana', 'Jakub', 'Kamila', 'Lukáš', 'Markéta', 'Nikola', 'Ondřej',
|
|
||||||
'Petra', 'Quido', 'Radek', 'Simona', 'Tomáš', 'Ursula', 'Viktor',
|
|
||||||
'Wanda', 'Xaver', 'Yvona', 'Zdeněk', 'Aneta', 'Boris', 'Cecílie', 'Daniel'
|
|
||||||
];
|
|
||||||
|
|
||||||
// Volby stravování pro mock data
|
|
||||||
const LUNCH_CHOICES = [
|
|
||||||
'SLADOVNICKA',
|
|
||||||
'TECHTOWER',
|
|
||||||
'ZASTAVKAUMICHALA',
|
|
||||||
'SENKSERIKOVA',
|
|
||||||
'OBJEDNAVAM',
|
|
||||||
'NEOBEDVAM',
|
|
||||||
'ROZHODUJI',
|
|
||||||
];
|
|
||||||
|
|
||||||
// Restaurace s menu
|
|
||||||
const RESTAURANTS_WITH_MENU = [
|
|
||||||
'SLADOVNICKA',
|
|
||||||
'TECHTOWER',
|
|
||||||
'ZASTAVKAUMICHALA',
|
|
||||||
'SENKSERIKOVA',
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware pro kontrolu DEV režimu
|
|
||||||
*/
|
|
||||||
function requireDevMode(req: any, res: any, next: any) {
|
|
||||||
if (ENVIRONMENT !== 'development' && ENVIRONMENT !== 'test') {
|
|
||||||
return res.status(403).json({ error: 'Tento endpoint je dostupný pouze ve vývojovém režimu' });
|
|
||||||
}
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
|
|
||||||
router.use(requireDevMode);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Vygeneruje mock data pro testování.
|
|
||||||
*/
|
|
||||||
router.post("/generate", async (req: Request<{}, any, any>, res, next) => {
|
|
||||||
try {
|
|
||||||
const dayIndex = req.body?.dayIndex ?? getDayOfWeekIndex(getToday());
|
|
||||||
const count = req.body?.count ?? Math.floor(Math.random() * 16) + 5; // 5-20
|
|
||||||
|
|
||||||
if (dayIndex < 0 || dayIndex > 4) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný index dne (0-4)' });
|
|
||||||
}
|
|
||||||
|
|
||||||
const date = getDateForWeekIndex(dayIndex);
|
|
||||||
await initIfNeeded(date);
|
|
||||||
|
|
||||||
const dateKey = formatDate(date);
|
|
||||||
const data = await storage.getData<any>(dateKey);
|
|
||||||
|
|
||||||
// Získání menu restaurací pro vybraný den
|
|
||||||
const menus: { [key: string]: any } = {};
|
|
||||||
for (const restaurant of RESTAURANTS_WITH_MENU) {
|
|
||||||
const menu = await getRestaurantMenu(restaurant as any, date);
|
|
||||||
if (menu?.food?.length) {
|
|
||||||
menus[restaurant] = menu.food;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Vygenerování náhodných uživatelů
|
|
||||||
const usedNames = new Set<string>();
|
|
||||||
for (let i = 0; i < count && usedNames.size < MOCK_NAMES.length; i++) {
|
|
||||||
// Vybereme náhodné jméno, které ještě nebylo použito
|
|
||||||
let name: string;
|
|
||||||
do {
|
|
||||||
name = MOCK_NAMES[Math.floor(Math.random() * MOCK_NAMES.length)];
|
|
||||||
} while (usedNames.has(name));
|
|
||||||
usedNames.add(name);
|
|
||||||
|
|
||||||
// Vybereme náhodnou volbu stravování
|
|
||||||
const choice = LUNCH_CHOICES[Math.floor(Math.random() * LUNCH_CHOICES.length)];
|
|
||||||
|
|
||||||
// Inicializace struktury pro volbu
|
|
||||||
data.choices[choice] ??= {};
|
|
||||||
|
|
||||||
const userChoice: any = {
|
|
||||||
trusted: false,
|
|
||||||
selectedFoods: [],
|
|
||||||
};
|
|
||||||
|
|
||||||
// Pokud má restaurace menu, vybereme náhodné jídlo
|
|
||||||
if (RESTAURANTS_WITH_MENU.includes(choice) && menus[choice]?.length) {
|
|
||||||
const foods = menus[choice];
|
|
||||||
// Vybereme náhodné jídlo (ne polévku)
|
|
||||||
const mainFoods = foods.filter((f: any) => !f.isSoup);
|
|
||||||
if (mainFoods.length > 0) {
|
|
||||||
const randomFoodIndex = foods.indexOf(mainFoods[Math.floor(Math.random() * mainFoods.length)]);
|
|
||||||
userChoice.selectedFoods = [randomFoodIndex];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
data.choices[choice][name] = userChoice;
|
|
||||||
}
|
|
||||||
|
|
||||||
await storage.setData(dateKey, data);
|
|
||||||
|
|
||||||
// Odeslat aktualizovaná data přes WebSocket
|
|
||||||
const clientData = await getData(date);
|
|
||||||
getWebsocket().emit("message", clientData);
|
|
||||||
|
|
||||||
res.status(200).json({ success: true, count: usedNames.size, dayIndex });
|
|
||||||
} catch (e: any) {
|
|
||||||
next(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Smaže všechny volby pro daný den.
|
|
||||||
*/
|
|
||||||
router.post("/clear", async (req: Request<{}, any, any>, res, next) => {
|
|
||||||
try {
|
|
||||||
const dayIndex = req.body?.dayIndex ?? getDayOfWeekIndex(getToday());
|
|
||||||
|
|
||||||
if (dayIndex < 0 || dayIndex > 4) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný index dne (0-4)' });
|
|
||||||
}
|
|
||||||
|
|
||||||
const date = getDateForWeekIndex(dayIndex);
|
|
||||||
await initIfNeeded(date);
|
|
||||||
|
|
||||||
const dateKey = formatDate(date);
|
|
||||||
const data = await storage.getData<any>(dateKey);
|
|
||||||
|
|
||||||
// Vymažeme všechny volby i aktivní pizza day
|
|
||||||
data.choices = {};
|
|
||||||
delete data.pizzaDay;
|
|
||||||
|
|
||||||
await storage.setData(dateKey, data);
|
|
||||||
|
|
||||||
// Odeslat aktualizovaná data přes WebSocket
|
|
||||||
const clientData = await getData(date);
|
|
||||||
getWebsocket().emit("message", clientData);
|
|
||||||
|
|
||||||
res.status(200).json({ success: true, dayIndex });
|
|
||||||
} catch (e: any) {
|
|
||||||
next(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/** Vrátí obsah push reminder registry (pro ladění). */
|
|
||||||
router.get("/pushRegistry", async (_req, res, next) => {
|
|
||||||
try {
|
|
||||||
const registry = await storage.getData<any>('push_reminder_registry') ?? {};
|
|
||||||
const sanitized = Object.fromEntries(
|
|
||||||
Object.entries(registry).map(([login, entry]: [string, any]) => [
|
|
||||||
login,
|
|
||||||
{ time: entry.time, endpoint: entry.subscription?.endpoint?.slice(0, 60) + '…' }
|
|
||||||
])
|
|
||||||
);
|
|
||||||
res.status(200).json(sanitized);
|
|
||||||
} catch (e: any) { next(e) }
|
|
||||||
});
|
|
||||||
|
|
||||||
/** Okamžitě odešle test push notifikaci přihlášenému uživateli (pro ladění). */
|
|
||||||
router.post("/testPush", async (req, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
try {
|
|
||||||
const registry = await storage.getData<any>('push_reminder_registry') ?? {};
|
|
||||||
const entry = registry[login];
|
|
||||||
if (!entry) {
|
|
||||||
return res.status(404).json({ error: `Uživatel ${login} nemá uloženou push subscription. Nastav připomínku v nastavení.` });
|
|
||||||
}
|
|
||||||
const publicKey = process.env.VAPID_PUBLIC_KEY;
|
|
||||||
const privateKey = process.env.VAPID_PRIVATE_KEY;
|
|
||||||
const subject = process.env.VAPID_SUBJECT;
|
|
||||||
if (!publicKey || !privateKey || !subject) {
|
|
||||||
return res.status(503).json({ error: 'VAPID klíče nejsou nastaveny' });
|
|
||||||
}
|
|
||||||
webpush.setVapidDetails(subject, publicKey, privateKey);
|
|
||||||
await webpush.sendNotification(
|
|
||||||
entry.subscription,
|
|
||||||
JSON.stringify({ title: 'Luncher test', body: 'Push notifikace fungují!', login })
|
|
||||||
);
|
|
||||||
res.status(200).json({ ok: true });
|
|
||||||
} catch (e: any) { next(e) }
|
|
||||||
});
|
|
||||||
|
|
||||||
export default router;
|
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
import express, { Request, Response } from "express";
|
import express, { Request, Response } from "express";
|
||||||
import { getLogin, getTrusted } from "../auth";
|
import { getLogin, getTrusted } from "../auth";
|
||||||
import { addChoice, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote, fetchRestaurantWeekMenuData, saveRestaurantWeekMenu, updateBuyer } from "../service";
|
import { addChoice, getDateForWeekIndex, getToday, removeChoice, removeChoices, updateDepartureTime, updateNote, fetchRestaurantWeekMenuData, saveRestaurantWeekMenu } from "../service";
|
||||||
import { getDayOfWeekIndex, parseToken, getFirstWorkDayOfWeek } from "../utils";
|
import { getDayOfWeekIndex, parseToken, getFirstWorkDayOfWeek } from "../utils";
|
||||||
import { getWebsocket } from "../websocket";
|
import { getWebsocket } from "../websocket";
|
||||||
import { callNotifikace } from "../notifikace";
|
import { callNotifikace } from "../notifikace";
|
||||||
import { AddChoiceData, ChangeDepartureTimeData, MealSlot, RemoveChoiceData, RemoveChoicesData, UdalostEnum, UpdateNoteData } from "../../../types/gen/types.gen";
|
import { AddChoiceData, ChangeDepartureTimeData, RemoveChoiceData, RemoveChoicesData, UdalostEnum, UpdateNoteData } from "../../../types/gen/types.gen";
|
||||||
|
|
||||||
|
|
||||||
// RateLimit na refresh endpoint
|
// RateLimit na refresh endpoint
|
||||||
@@ -56,34 +56,24 @@ function checkRateLimit(key: string, limit: number = RATE_LIMIT): boolean {
|
|||||||
*/
|
*/
|
||||||
const parseValidateFutureDayIndex = (req: Request<{}, any, AddChoiceData["body"] | UpdateNoteData["body"]>) => {
|
const parseValidateFutureDayIndex = (req: Request<{}, any, AddChoiceData["body"] | UpdateNoteData["body"]>) => {
|
||||||
if (req.body.dayIndex == null) {
|
if (req.body.dayIndex == null) {
|
||||||
throw new Error(`Nebyl předán index dne v týdnu.`);
|
throw Error(`Nebyl předán index dne v týdnu.`);
|
||||||
}
|
}
|
||||||
const todayDayIndex = getDayOfWeekIndex(getToday());
|
const todayDayIndex = getDayOfWeekIndex(getToday());
|
||||||
const dayIndex = req.body.dayIndex;
|
const dayIndex = req.body.dayIndex;
|
||||||
if (isNaN(dayIndex)) {
|
if (isNaN(dayIndex)) {
|
||||||
throw new Error(`Neplatný index dne v týdnu: ${req.body.dayIndex}`);
|
throw Error(`Neplatný index dne v týdnu: ${req.body.dayIndex}`);
|
||||||
}
|
}
|
||||||
if (dayIndex < todayDayIndex) {
|
if (dayIndex < todayDayIndex) {
|
||||||
throw new Error(`Předaný index dne v týdnu (${dayIndex}) nesmí být nižší než dnešní den (${todayDayIndex})`);
|
throw Error(`Předaný index dne v týdnu (${dayIndex}) nesmí být nižší než dnešní den (${todayDayIndex})`);
|
||||||
}
|
}
|
||||||
return dayIndex;
|
return dayIndex;
|
||||||
}
|
}
|
||||||
|
|
||||||
const parseSlot = (body: Record<string, any>): MealSlot | undefined => {
|
|
||||||
const slot = body?.slot;
|
|
||||||
if (slot != null && slot !== MealSlot.OBED) {
|
|
||||||
throw new Error(`Neplatný slot: ${slot}`);
|
|
||||||
}
|
|
||||||
return slot ?? undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
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, AddChoiceData["body"]>, res, next) => {
|
||||||
const login = getLogin(parseToken(req));
|
const login = getLogin(parseToken(req));
|
||||||
const trusted = getTrusted(parseToken(req));
|
const trusted = getTrusted(parseToken(req));
|
||||||
let slot: MealSlot | undefined;
|
|
||||||
try { slot = parseSlot(req.body); } catch (e: any) { return res.status(400).json({ error: e.message }); }
|
|
||||||
let date = undefined;
|
let date = undefined;
|
||||||
if (req.body.dayIndex != null) {
|
if (req.body.dayIndex != null) {
|
||||||
let dayIndex;
|
let dayIndex;
|
||||||
@@ -95,7 +85,7 @@ router.post("/addChoice", async (req: Request<{}, any, AddChoiceData["body"]>, r
|
|||||||
date = getDateForWeekIndex(dayIndex);
|
date = getDateForWeekIndex(dayIndex);
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const data = await addChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date, slot);
|
const data = await addChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date);
|
||||||
getWebsocket().emit("message", data);
|
getWebsocket().emit("message", data);
|
||||||
return res.status(200).json(data);
|
return res.status(200).json(data);
|
||||||
} catch (e: any) { next(e) }
|
} catch (e: any) { next(e) }
|
||||||
@@ -104,8 +94,6 @@ router.post("/addChoice", async (req: Request<{}, any, AddChoiceData["body"]>, r
|
|||||||
router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["body"]>, res, next) => {
|
router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["body"]>, res, next) => {
|
||||||
const login = getLogin(parseToken(req));
|
const login = getLogin(parseToken(req));
|
||||||
const trusted = getTrusted(parseToken(req));
|
const trusted = getTrusted(parseToken(req));
|
||||||
let slot: MealSlot | undefined;
|
|
||||||
try { slot = parseSlot(req.body); } catch (e: any) { return res.status(400).json({ error: e.message }); }
|
|
||||||
let date = undefined;
|
let date = undefined;
|
||||||
if (req.body.dayIndex != null) {
|
if (req.body.dayIndex != null) {
|
||||||
let dayIndex;
|
let dayIndex;
|
||||||
@@ -117,7 +105,7 @@ router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["bo
|
|||||||
date = getDateForWeekIndex(dayIndex);
|
date = getDateForWeekIndex(dayIndex);
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const data = await removeChoices(login, trusted, req.body.locationKey, date, slot);
|
const data = await removeChoices(login, trusted, req.body.locationKey, date);
|
||||||
getWebsocket().emit("message", data);
|
getWebsocket().emit("message", data);
|
||||||
res.status(200).json(data);
|
res.status(200).json(data);
|
||||||
} catch (e: any) { next(e) }
|
} catch (e: any) { next(e) }
|
||||||
@@ -126,8 +114,6 @@ router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesData["bo
|
|||||||
router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body"]>, res, next) => {
|
router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body"]>, res, next) => {
|
||||||
const login = getLogin(parseToken(req));
|
const login = getLogin(parseToken(req));
|
||||||
const trusted = getTrusted(parseToken(req));
|
const trusted = getTrusted(parseToken(req));
|
||||||
let slot: MealSlot | undefined;
|
|
||||||
try { slot = parseSlot(req.body); } catch (e: any) { return res.status(400).json({ error: e.message }); }
|
|
||||||
let date = undefined;
|
let date = undefined;
|
||||||
if (req.body.dayIndex != null) {
|
if (req.body.dayIndex != null) {
|
||||||
let dayIndex;
|
let dayIndex;
|
||||||
@@ -139,7 +125,7 @@ router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceData["body
|
|||||||
date = getDateForWeekIndex(dayIndex);
|
date = getDateForWeekIndex(dayIndex);
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const data = await removeChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date, slot);
|
const data = await removeChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date);
|
||||||
getWebsocket().emit("message", data);
|
getWebsocket().emit("message", data);
|
||||||
res.status(200).json(data);
|
res.status(200).json(data);
|
||||||
} catch (e: any) { next(e) }
|
} catch (e: any) { next(e) }
|
||||||
@@ -149,11 +135,9 @@ router.post("/updateNote", async (req: Request<{}, any, UpdateNoteData["body"]>,
|
|||||||
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;
|
||||||
let slot: MealSlot | undefined;
|
|
||||||
try { slot = parseSlot(req.body); } catch (e: any) { return res.status(400).json({ error: e.message }); }
|
|
||||||
try {
|
try {
|
||||||
if (note && note.length > 70) {
|
if (note && note.length > 70) {
|
||||||
throw new Error("Poznámka může mít maximálně 70 znaků");
|
throw Error("Poznámka může mít maximálně 70 znaků");
|
||||||
}
|
}
|
||||||
let date = undefined;
|
let date = undefined;
|
||||||
if (req.body.dayIndex != null) {
|
if (req.body.dayIndex != null) {
|
||||||
@@ -165,7 +149,7 @@ router.post("/updateNote", async (req: Request<{}, any, UpdateNoteData["body"]>,
|
|||||||
}
|
}
|
||||||
date = getDateForWeekIndex(dayIndex);
|
date = getDateForWeekIndex(dayIndex);
|
||||||
}
|
}
|
||||||
const data = await updateNote(login, trusted, note, date, slot);
|
const data = await updateNote(login, trusted, note, date);
|
||||||
getWebsocket().emit("message", data);
|
getWebsocket().emit("message", data);
|
||||||
res.status(200).json(data);
|
res.status(200).json(data);
|
||||||
} catch (e: any) { next(e) }
|
} catch (e: any) { next(e) }
|
||||||
@@ -198,31 +182,13 @@ router.post("/jdemeObed", async (req, res, next) => {
|
|||||||
} catch (e: any) { next(e) }
|
} catch (e: any) { next(e) }
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post("/updateBuyer", async (req, res, next) => {
|
// /api/food/refresh?type=week&heslo=docasnyheslo
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
let slot: MealSlot | undefined;
|
|
||||||
try { slot = parseSlot(req.body ?? {}); } catch (e: any) { return res.status(400).json({ error: e.message }); }
|
|
||||||
try {
|
|
||||||
const data = await updateBuyer(login, slot);
|
|
||||||
getWebsocket().emit("message", data);
|
|
||||||
res.status(200).json({});
|
|
||||||
} catch (e: any) { next(e) }
|
|
||||||
});
|
|
||||||
|
|
||||||
// /api/food/refresh?type=week (přihlášený uživatel, nebo ?heslo=... pro bypass rate limitu)
|
|
||||||
export const refreshMetoda = async (req: Request, res: Response) => {
|
export const refreshMetoda = async (req: Request, res: Response) => {
|
||||||
const { type, heslo } = req.query as { type?: string; heslo?: string };
|
const { type, heslo } = req.query as { type?: string; heslo?: string };
|
||||||
const bypassPassword = process.env.REFRESH_BYPASS_PASSWORD;
|
if (heslo !== "docasnyheslo" && heslo !== "tohleheslopavelnesmizjistit123") {
|
||||||
const isBypass = !!bypassPassword && heslo === bypassPassword;
|
return res.status(403).json({ error: "Neplatné heslo" });
|
||||||
|
|
||||||
if (!isBypass) {
|
|
||||||
try {
|
|
||||||
getLogin(parseToken(req));
|
|
||||||
} catch {
|
|
||||||
return res.status(403).json({ error: "Přihlaste se prosím" });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (!checkRateLimit("refresh") && !isBypass) {
|
if (!checkRateLimit("refresh") && heslo !== "tohleheslopavelnesmizjistit123") {
|
||||||
return res.status(429).json({ error: "Refresh už se zavolal, chvíli počkej :))" });
|
return res.status(429).json({ error: "Refresh už se zavolal, chvíli počkej :))" });
|
||||||
}
|
}
|
||||||
if (type !== "week" && type !== "day") {
|
if (type !== "week" && type !== "day") {
|
||||||
|
|||||||
@@ -1,156 +0,0 @@
|
|||||||
import express, { Request } from "express";
|
|
||||||
import { getLogin } from "../auth";
|
|
||||||
import { parseToken } from "../utils";
|
|
||||||
import { getWebsocket } from "../websocket";
|
|
||||||
import { createGroup, deleteGroup, addGroupMember, removeGroupMember, updateGroupMember, setGroupState, updateGroupTimes, updateGroupFees } from "../groups";
|
|
||||||
import { GroupState } from "../../../types/gen/types.gen";
|
|
||||||
|
|
||||||
const router = express.Router();
|
|
||||||
|
|
||||||
function broadcastExtra(data: any) {
|
|
||||||
getWebsocket().emit("message", data);
|
|
||||||
}
|
|
||||||
|
|
||||||
router.post("/create", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { name } = req.body ?? {};
|
|
||||||
if (!name || typeof name !== 'string') {
|
|
||||||
return res.status(400).json({ error: 'Nebyl předán název skupiny' });
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const data = await createGroup(login, name);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/delete", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
try {
|
|
||||||
const data = await deleteGroup(login, id);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/addMember", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, login: targetLogin } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
if (targetLogin !== undefined && (typeof targetLogin !== 'string' || targetLogin.trim() === '')) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný login uživatele' });
|
|
||||||
}
|
|
||||||
const target = targetLogin ?? login;
|
|
||||||
try {
|
|
||||||
const data = await addGroupMember(login, id, target);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/removeMember", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, login: targetLogin } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
if (!targetLogin) return res.status(400).json({ error: 'Nebyl předán login uživatele' });
|
|
||||||
try {
|
|
||||||
const data = await removeGroupMember(login, id, targetLogin);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/updateMember", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, login: targetLogin, amount, note, surchargeText, surchargeAmount } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
if (!targetLogin) return res.status(400).json({ error: 'Nebyl předán login uživatele' });
|
|
||||||
const patch: Record<string, any> = {};
|
|
||||||
if (amount !== undefined) {
|
|
||||||
if (!Number.isInteger(amount) || amount < 0) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná částka' });
|
|
||||||
}
|
|
||||||
patch.amount = amount;
|
|
||||||
}
|
|
||||||
if (note !== undefined) {
|
|
||||||
if (typeof note !== 'string') return res.status(400).json({ error: 'Neplatná poznámka' });
|
|
||||||
patch.note = note;
|
|
||||||
}
|
|
||||||
if (surchargeText !== undefined) {
|
|
||||||
if (typeof surchargeText !== 'string') return res.status(400).json({ error: 'Neplatný text příplatku' });
|
|
||||||
patch.surchargeText = surchargeText;
|
|
||||||
}
|
|
||||||
if (surchargeAmount !== undefined) {
|
|
||||||
if (!Number.isInteger(surchargeAmount) || surchargeAmount < 0) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná výše příplatku' });
|
|
||||||
}
|
|
||||||
patch.surchargeAmount = surchargeAmount;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const data = await updateGroupMember(login, id, targetLogin, patch);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/setState", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, state } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
if (!state || !Object.values(GroupState).includes(state)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný stav skupiny' });
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const data = await setGroupState(login, id, state as GroupState);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/updateFees", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, fees, shipping, tip, discountType, discountValue } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
if (fees !== undefined && (!Number.isInteger(fees) || fees < 0)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná výše poplatků' });
|
|
||||||
}
|
|
||||||
if (shipping !== undefined && (!Number.isInteger(shipping) || shipping < 0)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná výše dopravy' });
|
|
||||||
}
|
|
||||||
if (tip !== undefined && (!Number.isInteger(tip) || tip < 0)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná výše spropitného' });
|
|
||||||
}
|
|
||||||
if (discountType !== undefined && discountType !== '' && !['percent', 'fixed'].includes(discountType)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný typ slevy' });
|
|
||||||
}
|
|
||||||
if (discountValue !== undefined && (!Number.isInteger(discountValue) || discountValue < 0)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatná výše slevy' });
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const data = await updateGroupFees(login, id, fees, shipping, tip, discountType, discountValue);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post("/updateTimes", async (req: Request, res, next) => {
|
|
||||||
const login = getLogin(parseToken(req));
|
|
||||||
const { id, orderedAt, deliveryAt } = req.body ?? {};
|
|
||||||
if (!id) return res.status(400).json({ error: 'Nebylo předáno ID skupiny' });
|
|
||||||
const timeRegex = /^([01]\d|2[0-3]):[0-5]\d$/;
|
|
||||||
if (orderedAt !== undefined && orderedAt !== '' && !timeRegex.test(orderedAt)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný formát času objednání (očekáváno HH:MM)' });
|
|
||||||
}
|
|
||||||
if (deliveryAt !== undefined && deliveryAt !== '' && !timeRegex.test(deliveryAt)) {
|
|
||||||
return res.status(400).json({ error: 'Neplatný formát času doručení (očekáváno HH:MM)' });
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const data = await updateGroupTimes(login, id, orderedAt, deliveryAt);
|
|
||||||
broadcastExtra(data);
|
|
||||||
res.status(200).json(data);
|
|
||||||
} catch (e: any) { next(e); }
|
|
||||||
});
|
|
||||||
|
|
||||||
export default router;
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user