Compare commits
	
		
			376 Commits
		
	
	
		
			0.8.3
			...
			feature/17
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ee91748b3c | |||
| e5241d619b | |||
| d79608edbb | |||
| 4cbd26580e | |||
| fe62ad5539 | |||
| eb13f038a1 | |||
| 9505c2b030 | |||
| 008835c24f | |||
| 7083b3d8d2 | |||
| 754931b2f6 | |||
| 2dc8ffba32 | |||
| d0fe6a2e85 | |||
| 85705b6e68 | |||
| 3ea7a015a9 | |||
| 44329413ed | |||
| 46db68ab22 | |||
| dc9d7f22a2 | |||
| f917018fd9 | |||
| 7b420c430d | |||
| 00359d25c1 | |||
| d8a3063735 | |||
| 6491af19e3 | |||
| 61328d20ed | |||
| 0a6d92a1f3 | |||
| 3a576d1073 | |||
| b30b98b521 | |||
| 43d82a2af0 | |||
| 6a4495b813 | |||
| e8a0ad6647 | |||
| 92b89cc4d8 | |||
| 268b1b1d98 | |||
| 75bc89ca30 | |||
| 0625937068 | |||
| 32a9074963 | |||
| b869b5fd2a | |||
| 3a3e2f7157 | |||
| bea57aa03a | |||
| 30991d5364 | |||
| 5cc8b0811c | |||
| 2c73b9862d | |||
| 732b2f061e | |||
| 3680533eef | |||
| 1307d72c9d | |||
| 405dfa0c34 | |||
| 5c2d154ad1 | |||
| f2bf8d9bac | |||
| f9cfd6bd06 | |||
| 287f63fa52 | |||
| 5fe47634e8 | |||
| a6590910cf | |||
| ad454c386c | |||
| 0b2c296de0 | |||
| 0e85940cba | |||
| 8d479c32f8 | |||
| 549785cf7d | |||
| aafc4c8d62 | |||
| 47dedbdc73 | |||
| 6fe134afc8 | |||
| 63a50f92e7 | |||
| ca6da15ef7 | |||
| 8dfa19fa0f | |||
| 0feee0ae2f | |||
| 2a6a39916a | |||
| f0a2b2859f | |||
| 32ddb66fc8 | |||
| df63c2388d | |||
| 757655ea63 | |||
| 329c1cc037 | |||
| da6dd55d13 | |||
| 0e5490f1c8 | |||
| b82d638de1 | |||
| 224034dcc6 | |||
| 026d3d41c1 | |||
| fd1a06b359 | |||
| 452d010183 | |||
| eb1c17e3ac | |||
| a101873eb0 | |||
| 3d2acb692a | |||
| 0900c2691e | |||
| 1337676e08 | |||
| 2e075eafab | |||
| 14d64b6070 | |||
| 81b8fbf4e3 | |||
| 24d074752f | |||
| 08047a9307 | |||
| 1b0cd5b90b | |||
| 65e8998894 | |||
| 449948050b | |||
| cf97281592 | |||
| 75684efa1a | |||
| 2c4f27a943 | |||
| 53b7dec7cd | |||
| e0cbfb000b | |||
| 3a66f4c862 | |||
| 85ceaa464f | |||
| 976755338b | |||
| 1c980059cf | |||
| 2d8c4c1698 | |||
| 19a333d7bd | |||
| 96c55db63d | |||
| fecb07ee37 | |||
| e10c6480a5 | |||
| f3cc07c009 | |||
| 068076dd47 | |||
| 02158605be | |||
| 674e6a90ec | |||
| f679330466 | |||
| 93fc7c2e83 | |||
| f299617c60 | |||
| 28cbc5b98c | |||
| c28f1ee0bc | |||
| cff112d705 | |||
| 9fc4ad63c4 | |||
| 97054a71c1 | |||
| 2391668a25 | |||
| 717d33547c | |||
| 997be32679 | |||
| 134f00c40e | |||
| 47c898bdfd | |||
| e752ee12d1 | |||
| cc4515ff66 | |||
| f190292171 | |||
| b246f2b349 | |||
| 76b69d851a | |||
| 224f586368 | |||
| 9add6c8ff1 | |||
| 7a63d4eed1 | |||
| e54a4807f7 | |||
| cee04c1d6f | |||
| cbec78589d | |||
| a85db7cb3f | |||
| 2bd3779839 | |||
| 303e33cafb | |||
| b4e689dddf | |||
| 98a0b036c5 | |||
| fb3f30fb10 | |||
| 6213952007 | |||
| 07ac041d69 | |||
| 5c02028841 | |||
| c561b53670 | |||
| dcd0d5a362 | |||
| 18acac83bc | |||
| d7d44470bb | |||
| 0f0aae7ba4 | |||
| 4c0886a5d9 | |||
| 04a3038369 | |||
| bdcf5d3fc0 | |||
| c7a858eed7 | |||
| de5aa9237d | |||
| d015f97395 | |||
| 57618156b4 | |||
| 865254d646 | |||
| 1dbab03fe7 | |||
| a943aaf5fc | |||
| 6e6e8b2617 | |||
| 4c2c24af2c | |||
| 3d3a10aafb | |||
| 000fc97beb | |||
| 5645eeaafa | |||
| 961477d522 | |||
| a5f71015a6 | |||
| e42ea943b7 | |||
| 9c5fc6b61c | |||
| 302caf015f | |||
| e11296071a | |||
| 112eb29f93 | |||
| c6c97516b3 | |||
| 03676b2894 | |||
| 9ca57fac2e | |||
| 18f151c1fb | |||
| e90e56d8b2 | |||
| d241ca5698 | |||
| b512cf8667 | |||
| a24d2923c6 | |||
| 467808abef | |||
| 861f1f2216 | |||
| 509b22bea0 | |||
| 7447b2f4c1 | |||
| fef14b6e4f | |||
| 01d2a7e6aa | |||
| ac586fec5a | |||
| 5476808683 | |||
| 331d737796 | |||
| ef81b8adf9 | |||
| 8a7d635cef | |||
| 4c259c1eef | |||
| 5b4ede5e2f | |||
| d0ab3dda78 | |||
| d9cf51b4bb | |||
| aa17f24220 | |||
| cf60edf7d4 | |||
| ffbc243194 | |||
| b6b07cf30c | |||
| 495a6b22bd | |||
| 0acaffbdfa | |||
| 6043bc4517 | |||
| e6ed066e3f | |||
| ee4e8655b8 | |||
| 37970d2be6 | |||
| 1376788016 | |||
| 4cad86cf85 | |||
| 6304116edb | |||
| 834ff8fa63 | |||
| 1f428a535e | |||
| 0c40966970 | |||
| 9da071fe9b | |||
| 892a04f289 | |||
| 27cc9727f1 | |||
| f0738d451b | |||
| 9e6a8daf2c | |||
| bfacfec765 | |||
| 0bae5bf32b | |||
| 22b09d16d0 | |||
| 9c867e106e | |||
| 304f28a3c1 | |||
| d65d3793de | |||
| 3638d87bd2 | |||
| b97a92860d | |||
| 7c86a5eeb3 | |||
| d23dbaaf69 | |||
| e6ffc371e1 | |||
| 3177c6eaa3 | |||
| acd2f0519d | |||
| 18ec100c33 | |||
| fa55fce76e | |||
| f47d5e347d | |||
| 7488a8b597 | |||
| 2e3ac154be | |||
| 2472640755 | |||
| 7b685d6cad | |||
| 17f6f4e616 | |||
| 48cfc15cfb | |||
| bb9b779cee | |||
| af63ce67ae | |||
| 5cc4871ec4 | |||
| c8cfe669b8 | |||
| 8b74d6d759 | |||
| a9227768de | |||
| d966e1d4de | |||
| ceb2146c1b | |||
| 8d006d8c74 | |||
| 777304f259 | |||
| 12433f7c23 | |||
| 44b53da345 | |||
| ab45fc144e | |||
| e99e9e0708 | |||
| 467404bfc8 | |||
| ce50fa2a62 | |||
| 10a011d842 | |||
| 5352410d0c | |||
| c5d155396a | |||
| 93187099d3 | |||
| aa24b1dce5 | |||
| eb3ede9593 | |||
| d7fecfbd0b | |||
| b065b4ff21 | |||
| 87370d24be | |||
| 8f8b9988ad | |||
| f8ccf4f5d8 | |||
| 25d8b86efd | |||
| 0cd3e937d8 | |||
| 89bb9c215e | |||
| 2d18686ce7 | |||
| 1d999d4910 | |||
| 7dfaa7579a | |||
| 08cb079e97 | |||
| 450aa83592 | |||
| 0614c76e92 | |||
| 97e338f9d4 | |||
| 636f018daa | |||
| c8d639024a | |||
| 6be2ee626a | |||
| f7fc1967a5 | |||
| aedb8a765b | |||
| cf58bd15c3 | |||
| 34f4f68524 | |||
| 09b8144080 | |||
| f1e6fb4ce7 | |||
| 2ca63fd1f6 | |||
| a5d25e7d92 | |||
| 4167819e7a | |||
| 5bd3a463f0 | |||
| 79c447b4c6 | |||
| 540304f947 | |||
| 75d8f7331b | |||
| b2509e9e53 | |||
| 7862f44653 | |||
| 962dd0c1bb | |||
| 5d5f7c7f5c | |||
| 6aaf838451 | |||
| ad3bd312e9 | |||
| 5fa9939696 | |||
| 4956bb0e9c | |||
| c074c12be7 | |||
| ddbc293e9c | |||
| a3921b45c7 | |||
| 38e1c8c5a1 | |||
| c2d29ff233 | |||
| 2316baa898 | |||
| f185d559c0 | |||
| 73d95bc004 | |||
| fcd55f89d7 | |||
| f9fe793573 | |||
| bc36411993 | |||
| 48506236bf | |||
| ded9b589fe | |||
| 67c3732fad | |||
| 2932f4591e | |||
| df53c07450 | |||
| 40899e9d80 | |||
| f794af0950 | |||
| 1665a1a093 | |||
| 4a36fb6d95 | |||
| acf78a8822 | |||
| f5c1ec9939 | |||
| 4b3d38b05b | |||
| 23e0b53107 | |||
| c907486c4d | |||
| 6b5945add8 | |||
| 55693de934 | |||
| d467475b6d | |||
| 44bc14820f | |||
| ec447e2e36 | |||
| 0af2647965 | |||
| 08442154f4 | |||
| 9f7d2234fb | |||
| ddd82a71a7 | |||
| 014ba3bf67 | |||
| c87321f804 | |||
| 8b451b3c67 | |||
| 0cfc87fbe6 | |||
| ae9673070c | |||
| 008027db0e | |||
| aec5e3473e | |||
| 95c8fde72f | |||
| 0f32968fae | |||
| ae79e9fea1 | |||
| 1a52aaf8d1 | |||
| d6c315ab8e | |||
| 983ce56048 | |||
| de2fe0e9f1 | |||
| c3c95bf291 | |||
| d2050b5948 | |||
| 6b92405bae | |||
| 49e87ccb15 | |||
| 50fffef13b | |||
| 82b1811971 | |||
| aeadef60bb | |||
| a1ab65a0e9 | |||
| 17e0805fe6 | |||
| ddd9c396b6 | |||
| ef49e507c1 | |||
| fbe74a5d80 | |||
| 076893981f | |||
| fac059f02c | |||
| 0313f8cc49 | |||
| 7ad6b73574 | |||
| 3cd0468b19 | |||
| f46ccb610e | |||
| 8a32569a3b | |||
| 535b23ae91 | |||
| 4715978f81 | |||
| a516aa7775 | |||
| 77e9c205f9 | |||
| e852305400 | |||
| c6a15264b3 | |||
| 2d0beaaaad | |||
| 5c5ef95d2b | |||
| e838e6f321 | |||
| ca983c72d4 | |||
| 91dd5256e9 | |||
| 3d4dc2d72b | |||
| ba3471068a | |||
| 5a7b2cf886 | |||
| cc926e84fb | |||
| 9aa8e7edff | 
							
								
								
									
										6
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | FROM mcr.microsoft.com/vscode/devcontainers/base:alpine-3.12 | ||||||
|  | RUN apk update | ||||||
|  | RUN apk add --upgrade nodejs-current npm | ||||||
|  | RUN npm i -g yarn rimraf | ||||||
|  | RUN rimraf node_modules | ||||||
|  | RUN yarn set version berry | ||||||
							
								
								
									
										20
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | { | ||||||
|  | 	"name": "Node.js", | ||||||
|  | 	"build": { | ||||||
|  | 		"dockerfile": "Dockerfile" | ||||||
|  | 	}, | ||||||
|  | 	"settings": { | ||||||
|  | 		"terminal.integrated.shell.linux": "/bin/sh" | ||||||
|  | 	}, | ||||||
|  | 	"extensions": [ | ||||||
|  | 		"dbaeumer.vscode-eslint", | ||||||
|  | 		"2gua.rainbow-brackets", | ||||||
|  | 		"christian-kohler.npm-intellisense", | ||||||
|  | 		"remimarsal.prettier-now", | ||||||
|  | 		"svelte.svelte-vscode", | ||||||
|  | 		"lokalise.i18n-ally", | ||||||
|  | 		"fivethree.vscode-svelte-snippets", | ||||||
|  | 		"voorjaar.windicss-intellisense" | ||||||
|  | 	], | ||||||
|  | 	"postCreateCommand": "yarn && yarn dev --open" | ||||||
|  | } | ||||||
| @@ -1,3 +1,2 @@ | |||||||
| public/env.sample.js | public/env.sample.js | ||||||
| public/workbox-*.js | .pnpm-store | ||||||
| public/workbox-*.js.map |  | ||||||
							
								
								
									
										41
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										41
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -19,6 +19,13 @@ get: | |||||||
|   path: odit-git-bot |   path: odit-git-bot | ||||||
|   name: sshkey |   name: sshkey | ||||||
|  |  | ||||||
|  | --- | ||||||
|  | kind: secret | ||||||
|  | name: npm_url | ||||||
|  | get: | ||||||
|  |   path: odit-npm-cache | ||||||
|  |   name: url | ||||||
|  |  | ||||||
| --- | --- | ||||||
| kind: pipeline | kind: pipeline | ||||||
| type: kubernetes | type: kubernetes | ||||||
| @@ -27,10 +34,14 @@ name: build:dev | |||||||
| steps: | steps: | ||||||
|   - name: run full license export |   - name: run full license export | ||||||
|     depends_on: ["clone"] |     depends_on: ["clone"] | ||||||
|     image: node:alpine |     image: registry.odit.services/hub/library/node:19.7.0-alpine3.16 | ||||||
|     commands: |     commands: | ||||||
|       - yarn |       - npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 | ||||||
|       - yarn licenses:export |       - pnpm i | ||||||
|  |       - pnpm licenses:export | ||||||
|  |     environment: | ||||||
|  |       NPM_REGISTRY_URL: | ||||||
|  |         from_secret: npm_url | ||||||
|   - name: push new licenses file to repo |   - name: push new licenses file to repo | ||||||
|     depends_on: ["run full license export"] |     depends_on: ["run full license export"] | ||||||
|     image: appleboy/drone-git-push |     image: appleboy/drone-git-push | ||||||
| @@ -43,18 +54,21 @@ steps: | |||||||
|       ssh_key: |       ssh_key: | ||||||
|         from_secret: git_ssh |         from_secret: git_ssh | ||||||
|   - name: build dev |   - name: build dev | ||||||
|     image: plugins/docker |     depends_on: ["clone"] | ||||||
|     depends_on: [clone] |     image: registry.odit.services/library/drone-kaniko | ||||||
|     settings: |     settings: | ||||||
|       username: |       username: | ||||||
|         from_secret: docker_username |         from_secret: docker_username | ||||||
|       password: |       password: | ||||||
|         from_secret: docker_password |         from_secret: docker_password | ||||||
|       repo: registry.odit.services/lfk/frontend |       build_args: | ||||||
|  |         - NPM_REGISTRY_URL: | ||||||
|  |           from_secret: npm_url | ||||||
|  |       repo: lfk/frontend | ||||||
|       tags: |       tags: | ||||||
|         - dev |         - dev | ||||||
|  |       cache: true | ||||||
|       registry: registry.odit.services |       registry: registry.odit.services | ||||||
|       mtu: 1000 |  | ||||||
| trigger: | trigger: | ||||||
|   branch: |   branch: | ||||||
|     - dev |     - dev | ||||||
| @@ -67,18 +81,21 @@ type: kubernetes | |||||||
| name: build:tags | name: build:tags | ||||||
| steps: | steps: | ||||||
|   - name: build $DRONE_TAG |   - name: build $DRONE_TAG | ||||||
|     image: plugins/docker |     depends_on: ["clone"] | ||||||
|     depends_on: [clone] |     image: registry.odit.services/library/drone-kaniko | ||||||
|     settings: |     settings: | ||||||
|       username: |       username: | ||||||
|         from_secret: docker_username |         from_secret: docker_username | ||||||
|       password: |       password: | ||||||
|         from_secret: docker_password |         from_secret: docker_password | ||||||
|       repo: registry.odit.services/lfk/frontend |       build_args: | ||||||
|  |         - NPM_REGISTRY_URL: | ||||||
|  |           from_secret: npm_url | ||||||
|  |       repo: lfk/frontend | ||||||
|       tags: |       tags: | ||||||
|         - '${DRONE_TAG}' |         - "${DRONE_TAG}" | ||||||
|  |       cache: true | ||||||
|       registry: registry.odit.services |       registry: registry.odit.services | ||||||
|       mtu: 1000 |  | ||||||
| trigger: | trigger: | ||||||
|   event: |   event: | ||||||
|   - tag |   - tag | ||||||
							
								
								
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,6 @@ | |||||||
| node_modules | node_modules | ||||||
| build |  | ||||||
| package-lock.json |  | ||||||
| yarn.lock |  | ||||||
| *.map | *.map | ||||||
| public/env.js | public/env.js | ||||||
| public/sw.js |  | ||||||
| public/index.html |  | ||||||
| public/workbox-*.js |  | ||||||
| svelte.config.js |  | ||||||
| public/index.html | public/index.html | ||||||
|  | /dist | ||||||
|  | .pnpm-store | ||||||
							
								
								
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,7 +5,8 @@ | |||||||
|         "remimarsal.prettier-now", |         "remimarsal.prettier-now", | ||||||
|         "svelte.svelte-vscode", |         "svelte.svelte-vscode", | ||||||
|         "lokalise.i18n-ally", |         "lokalise.i18n-ally", | ||||||
|         "fivethree.vscode-svelte-snippets" |         "fivethree.vscode-svelte-snippets", | ||||||
|  |         "voorjaar.windicss-intellisense" | ||||||
|     ], |     ], | ||||||
|     "unwantedRecommendations": [ |     "unwantedRecommendations": [ | ||||||
|         "antfu.i18n-ally" |         "antfu.i18n-ally" | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| { | { | ||||||
|     "i18n-ally.localesPaths": "src/locales", |     "i18n-ally.localesPaths": "src/locales", | ||||||
|     "i18n-ally.keystyle": "nested" |     "i18n-ally.keystyle": "nested", | ||||||
|  |     "windicss.enableCodeFolding": false, | ||||||
| } | } | ||||||
							
								
								
									
										515
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										515
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,11 +2,524 @@ | |||||||
|  |  | ||||||
| All notable changes to this project will be documented in this file. Dates are displayed in UTC. | All notable changes to this project will be documented in this file. Dates are displayed in UTC. | ||||||
|  |  | ||||||
|  | #### [0.17.3](https://git.odit.services/lfk/frontend/compare/0.17.2...0.17.3) | ||||||
|  |  | ||||||
|  | - dependency fixes [`3ea7a01`](https://git.odit.services/lfk/frontend/commit/3ea7a015a9beba3c2e4d3eb966f24ff6d4ac786e) | ||||||
|  | - set pnpm to @7 [`4432941`](https://git.odit.services/lfk/frontend/commit/44329413ed2ca23f74e86db041b2c25b2b1c2a2b) | ||||||
|  |  | ||||||
|  | #### [0.17.2](https://git.odit.services/lfk/frontend/compare/0.17.1...0.17.2) | ||||||
|  |  | ||||||
|  | > 15 March 2023 | ||||||
|  |  | ||||||
|  | - new license file version [CI SKIP] [`00359d2`](https://git.odit.services/lfk/frontend/commit/00359d25c1bd3efdd6365bf284b3c07634049399) | ||||||
|  | - 🚀RELEASE v0.17.2 [`46db68a`](https://git.odit.services/lfk/frontend/commit/46db68ab229dc740dfff8835ef916f2c2e629b27) | ||||||
|  | - improved ThFilterGroup style [`f917018`](https://git.odit.services/lfk/frontend/commit/f917018fd92a8a5b034f735ac8b6e41995044317) | ||||||
|  |  | ||||||
|  | #### [0.17.1](https://git.odit.services/lfk/frontend/compare/0.17.0...0.17.1) | ||||||
|  |  | ||||||
|  | > 15 March 2023 | ||||||
|  |  | ||||||
|  | - Revert "package dependency fixes, bumps, lockfile update" [`d8a3063`](https://git.odit.services/lfk/frontend/commit/d8a30637351e164599e07a6473d9a1d2b08d245d) | ||||||
|  | - 🚀RELEASE v0.17.1 [`7b420c4`](https://git.odit.services/lfk/frontend/commit/7b420c430d27bf0fc85a4297780164a593fc1be3) | ||||||
|  |  | ||||||
|  | #### [0.17.0](https://git.odit.services/lfk/frontend/compare/0.16.5...0.17.0) | ||||||
|  |  | ||||||
|  | > 15 March 2023 | ||||||
|  |  | ||||||
|  | - new license file version [CI SKIP] [`61328d2`](https://git.odit.services/lfk/frontend/commit/61328d20ed2cfd1df7d3c32767f9c64154879d6d) | ||||||
|  | - wip: pnpm + node version [`732b2f0`](https://git.odit.services/lfk/frontend/commit/732b2f061e465bd08cf34c58d8cd6b021ba25dce) | ||||||
|  | - package dependency fixes, bumps, lockfile update [`2c73b98`](https://git.odit.services/lfk/frontend/commit/2c73b9862d401dac15021eed3f7847d46132a8ed) | ||||||
|  | - Fixed runners not showing up [`75bc89c`](https://git.odit.services/lfk/frontend/commit/75bc89ca3020c48f490c7602374616bd9461e78f) | ||||||
|  | - add ThFilterRunner [`3a3e2f7`](https://git.odit.services/lfk/frontend/commit/3a3e2f71575d3a0e39a5e13b05cff932b8683ac9) | ||||||
|  | - fix styling for table filters th border [`bea57aa`](https://git.odit.services/lfk/frontend/commit/bea57aa03acaaaa4b1860b30228dd5d1298317f8) | ||||||
|  | - You can now create cards from runners by searching via #runnerid [`e8a0ad6`](https://git.odit.services/lfk/frontend/commit/e8a0ad6647ab39252865f62b755f27c34ac2d243) | ||||||
|  | - remodelled for early return [`b869b5f`](https://git.odit.services/lfk/frontend/commit/b869b5fd2a01955fb21f936fa38eb5a9648e7de3) | ||||||
|  | - 🚀RELEASE v0.17.0 [`6491af1`](https://git.odit.services/lfk/frontend/commit/6491af19e375cbeba7ddd94e463b4dfe308a70a8) | ||||||
|  | - Wow this api is fun [`32a9074`](https://git.odit.services/lfk/frontend/commit/32a9074963cce3328f14b1f981ddd5ee49df0008) | ||||||
|  | - Fixed double space in label [`92b89cc`](https://git.odit.services/lfk/frontend/commit/92b89cc4d88c9d5625c2ddf7c81c98494f7f5271) | ||||||
|  | - UsersOverview: drop pfp [`30991d5`](https://git.odit.services/lfk/frontend/commit/30991d5364a09d517b2115a7e9ea3fbf1fe2e57d) | ||||||
|  | - Switched license generation to cache registry and pnpm [`0a6d92a`](https://git.odit.services/lfk/frontend/commit/0a6d92a1f3c5562f11562c433b3a04e3eaae3da4) | ||||||
|  | - Pinned pnpm in dockerfile, thx @philipp [`3a576d1`](https://git.odit.services/lfk/frontend/commit/3a576d1073ee503b68100e01054a1756bab62805) | ||||||
|  | - Pinned ci node version [`b30b98b`](https://git.odit.services/lfk/frontend/commit/b30b98b521eda2bc7fc055097546f716e90d92ef) | ||||||
|  | - Fixed pnpm being called without being installed [`43d82a2`](https://git.odit.services/lfk/frontend/commit/43d82a2af04af49c2169f78a0d0f27ef7e4d7558) | ||||||
|  | - Merge pull request 'bugfix/162-create_card_modal' (#163) from bugfix/162-create_card_modal into dev [`6a4495b`](https://git.odit.services/lfk/frontend/commit/6a4495b8131a31cd48a608c2275e80494d0a0fb4) | ||||||
|  | - Removed unused log [`268b1b1`](https://git.odit.services/lfk/frontend/commit/268b1b1d9830de196d1d95345d7a2467bbf19eb6) | ||||||
|  | - Merge pull request 'filter by runner full names + "#<ID>"' (#160) from feature/159-cardsoverview-filter-for-runner-full-names-and-id into dev [`0625937`](https://git.odit.services/lfk/frontend/commit/0625937068f0786078ffd29b9c8bb54949350b6c) | ||||||
|  | - UsersOverview: change profilepic scaling [`5cc8b08`](https://git.odit.services/lfk/frontend/commit/5cc8b0811cf290f97a4399b23c5ea4d961a5a91c) | ||||||
|  |  | ||||||
|  | #### [0.16.5](https://git.odit.services/lfk/frontend/compare/0.16.4...0.16.5) | ||||||
|  |  | ||||||
|  | > 14 March 2023 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.16.5 [`3680533`](https://git.odit.services/lfk/frontend/commit/3680533eefef042fc77246dd3d374aafe10c428f) | ||||||
|  | - new license file version [CI SKIP] [`405dfa0`](https://git.odit.services/lfk/frontend/commit/405dfa0c34ba87fc450c22e0e9974f92c4cdeffe) | ||||||
|  |  | ||||||
|  | #### [0.16.4](https://git.odit.services/lfk/frontend/compare/0.16.3...0.16.4) | ||||||
|  |  | ||||||
|  | > 14 March 2023 | ||||||
|  |  | ||||||
|  | - fix: OrgDetail: clicking on address will toggle selfservice [`#158`](https://git.odit.services/lfk/frontend/issues/158) | ||||||
|  | - 🚀RELEASE v0.16.4 [`5c2d154`](https://git.odit.services/lfk/frontend/commit/5c2d154ad180ce7916605871c63e2f5ac4428250) | ||||||
|  |  | ||||||
|  | #### [0.16.3](https://git.odit.services/lfk/frontend/compare/0.16.2...0.16.3) | ||||||
|  |  | ||||||
|  | > 23 February 2023 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.16.3 [`f9cfd6b`](https://git.odit.services/lfk/frontend/commit/f9cfd6bd063b01a584774854d8fb5eab96f99528) | ||||||
|  | - Bumped vite build targets [`5fe4763`](https://git.odit.services/lfk/frontend/commit/5fe47634e8980e77b65c05f213c475cf49273609) | ||||||
|  | - new license file version [CI SKIP] [`a659091`](https://git.odit.services/lfk/frontend/commit/a6590910cfdc5e91fba91c1bc9237e407ef15fd2) | ||||||
|  | - Merge pull request 'feature/156-pdf_names' (#157) from feature/156-pdf_names into dev [`ad454c3`](https://git.odit.services/lfk/frontend/commit/ad454c386cbf11abc59d41d269d1a0ef7442c9ed) | ||||||
|  | - Added ids for generated pdfs [`0b2c296`](https://git.odit.services/lfk/frontend/commit/0b2c296de069a4ef302c5535de01bc18236675bc) | ||||||
|  |  | ||||||
|  | #### [0.16.2](https://git.odit.services/lfk/frontend/compare/0.16.1...0.16.2) | ||||||
|  |  | ||||||
|  | > 23 February 2023 | ||||||
|  |  | ||||||
|  | - Fixed scanmodal [`#154`](https://git.odit.services/lfk/frontend/issues/154) | ||||||
|  | - 🚀RELEASE v0.16.2 [`0e85940`](https://git.odit.services/lfk/frontend/commit/0e85940cba292cbccd1ec038aa24f4a719382c19) | ||||||
|  | - Merge pull request 'feature/147-cardoverview_performance' (#153) from feature/147-cardoverview_performance into dev [`8d479c3`](https://git.odit.services/lfk/frontend/commit/8d479c32f82938904aee6a10deb80fea85487e4b) | ||||||
|  | - Merge pull request 'Fixed scanmodal' (#155) from bugfix/154-scan_select_runner_by_id into dev [`549785c`](https://git.odit.services/lfk/frontend/commit/549785cf7d1c9edc1be37dc745b97096232a08ca) | ||||||
|  | - i18n [`ca6da15`](https://git.odit.services/lfk/frontend/commit/ca6da15ef761184a55b18d56f749f660a32cbb83) | ||||||
|  | - Bsic datatable conversion [`757655e`](https://git.odit.services/lfk/frontend/commit/757655ea63b3667bc4612ae1595eb52910b61137) | ||||||
|  | - 1st datatable try with @vincjo/datatables [`81b8fbf`](https://git.odit.services/lfk/frontend/commit/81b8fbf4e341e6f2998a6e9e2053972c5c225021) | ||||||
|  | - Dasboard Cards redesign [`eb1c17e`](https://git.odit.services/lfk/frontend/commit/eb1c17e3ac7e8f5e7310a90421fc9db3ed15c497) | ||||||
|  | - set .phone to null if empty [`da6dd55`](https://git.odit.services/lfk/frontend/commit/da6dd55d139a672fa50204eabdca67d9740614a0) | ||||||
|  | - add group filtering to table [`14d64b6`](https://git.odit.services/lfk/frontend/commit/14d64b6070d98e6368da5709e9ff8221e8a621c7) | ||||||
|  | - formatting [`24d0747`](https://git.odit.services/lfk/frontend/commit/24d074752f1c5dc1a14b075ac14b448d7e129376) | ||||||
|  | - RunnersOverview loading fix [`2e075ea`](https://git.odit.services/lfk/frontend/commit/2e075eafab5c4d78fd9aa9d66834b477b2685bfc) | ||||||
|  | - Added old formatting for runner and status [`df63c23`](https://git.odit.services/lfk/frontend/commit/df63c2388da359dec9ed9968bc9f970be7092a45) | ||||||
|  | - Added custom status filter [`f0a2b28`](https://git.odit.services/lfk/frontend/commit/f0a2b2859fa18426a454b7d9d6dd22dfdfe7ce27) | ||||||
|  | - Basic checkbox fix [`1337676`](https://git.odit.services/lfk/frontend/commit/1337676e0894c46da0b6dcb7553e5ea8f88d0c14) | ||||||
|  | - Fixed all filter [`8dfa19f`](https://git.odit.services/lfk/frontend/commit/8dfa19fa0f9897c61342ece956df88731c7aa861) | ||||||
|  | - improved runner search [`1b0cd5b`](https://git.odit.services/lfk/frontend/commit/1b0cd5b90bcceb92627c6b7cdcdd7792ed81b50f) | ||||||
|  | - set table-layout:fixed + display when loaded [`65e8998`](https://git.odit.services/lfk/frontend/commit/65e89988943807c1606a8b6aea49564b13d52537) | ||||||
|  | - Trigger edit modal [`32ddb66`](https://git.odit.services/lfk/frontend/commit/32ddb66fc8d8cd689f1104759812f4cee4b7a613) | ||||||
|  | - cleaned up table search [`08047a9`](https://git.odit.services/lfk/frontend/commit/08047a93073c32f5dd7a8e958400ae8a5b7f4035) | ||||||
|  | - Fixed edit update bug [`0feee0a`](https://git.odit.services/lfk/frontend/commit/0feee0ae2fb6d8dba0b6fd72cedc0712dc749511) | ||||||
|  | - fix: z-index on action buttons [`224034d`](https://git.odit.services/lfk/frontend/commit/224034dcc6263d3b0a8ea20045e435142d8ed2af) | ||||||
|  | - rename: ThFilterGroup -> ThFilterStatus [`2a6a399`](https://git.odit.services/lfk/frontend/commit/2a6a39916a03c0466e63354e9f5ad7924cb59b6b) | ||||||
|  | - new license file version [CI SKIP] [`0e5490f`](https://git.odit.services/lfk/frontend/commit/0e5490f1c84217a5a6d5c8745c4667b32ca65e1a) | ||||||
|  | - new license file version [CI SKIP] [`026d3d4`](https://git.odit.services/lfk/frontend/commit/026d3d41c1b976a4dc7c733576a6a9e8d4b13b78) | ||||||
|  | - Updated breakpoints [`452d010`](https://git.odit.services/lfk/frontend/commit/452d0101838d72bff7d588a953faae028e2ff819) | ||||||
|  | - Tailwind bump [`a101873`](https://git.odit.services/lfk/frontend/commit/a101873eb0946b284a11a5081642711f5087da14) | ||||||
|  | - Fixed checkbox show [`0900c26`](https://git.odit.services/lfk/frontend/commit/0900c2691e4cfe5046e8ae186c8ac8884c90abd6) | ||||||
|  | - Removed unused console log [`aafc4c8`](https://git.odit.services/lfk/frontend/commit/aafc4c8d62a7a0a493c8bd60149f90c842534bdd) | ||||||
|  | - i18n import [`6fe134a`](https://git.odit.services/lfk/frontend/commit/6fe134afc8bfef4e7470b7e53b9312b172a7322b) | ||||||
|  | - Merge pull request 'fix: RunnerDetail: set .phone to null if empty' (#152) from bugfix/151-runnerdetail--cannot-unset-phone-number into dev [`329c1cc`](https://git.odit.services/lfk/frontend/commit/329c1cc037a43c818ba3b6c72581d29586d76232) | ||||||
|  | - Merge pull request 'feature/146-runner-table-performance-data-table' (#150) from feature/146-runner-table-performance-data-table into dev [`b82d638`](https://git.odit.services/lfk/frontend/commit/b82d638de1aa1f72aada212cf3e4147d808b4fcf) | ||||||
|  | - Merge pull request 'feature/148-dashboard_statscards' (#149) from feature/148-dashboard_statscards into dev [`fd1a06b`](https://git.odit.services/lfk/frontend/commit/fd1a06b3595b3713ad474e623c74105125602d46) | ||||||
|  | - Fixed top checkbox state [`3d2acb6`](https://git.odit.services/lfk/frontend/commit/3d2acb692a28c116790248679e238fb562b24ac5) | ||||||
|  |  | ||||||
|  | #### [0.16.1](https://git.odit.services/lfk/frontend/compare/0.16.0...0.16.1) | ||||||
|  |  | ||||||
|  | > 15 February 2023 | ||||||
|  |  | ||||||
|  | - fix: donor detail: sponsorings: unset middlename will show as "null" [`#145`](https://git.odit.services/lfk/frontend/issues/145) | ||||||
|  | - 🚀RELEASE v0.16.1 [`4499480`](https://git.odit.services/lfk/frontend/commit/449948050b8673d43a8dfbb225c3198e4bbb3c7b) | ||||||
|  |  | ||||||
|  | #### [0.16.0](https://git.odit.services/lfk/frontend/compare/0.15.6...0.16.0) | ||||||
|  |  | ||||||
|  | > 3 February 2023 | ||||||
|  |  | ||||||
|  | - First page for statsclients [`f299617`](https://git.odit.services/lfk/frontend/commit/f299617c600d2bba7b4405c7c3acae9fd93aefa8) | ||||||
|  | - 🚀RELEASE v0.16.0 [`75684ef`](https://git.odit.services/lfk/frontend/commit/75684efa1ae0edb4b4d414757c5acf2a77c572e5) | ||||||
|  | - Basic statsclient detail [`0215860`](https://git.odit.services/lfk/frontend/commit/02158605be824e5ac21a6284731138190988c794) | ||||||
|  | - Updated Add modal [`f679330`](https://git.odit.services/lfk/frontend/commit/f679330466205e6480cd7f2b7c2b4fdc41c51525) | ||||||
|  | - Switched drone to kaniko [`1c98005`](https://git.odit.services/lfk/frontend/commit/1c980059cff5c87c452428b53513507c2339451f) | ||||||
|  | - Re-added copy modal [`fecb07e`](https://git.odit.services/lfk/frontend/commit/fecb07ee373dcaaeaea69fdf8d4c6ee2c257c89c) | ||||||
|  | - Added Statsclients to sidebar [`068076d`](https://git.odit.services/lfk/frontend/commit/068076dd47373c673a25e730cb8a57c686682810) | ||||||
|  | - Fixed imports and naming [`f3cc07c`](https://git.odit.services/lfk/frontend/commit/f3cc07c009ed0a34e61f1aad47a1a31778145439) | ||||||
|  | - new license file version [CI SKIP] [`2c4f27a`](https://git.odit.services/lfk/frontend/commit/2c4f27a943bb35be6728bb49bd5c2263cba78165) | ||||||
|  | - Merge pull request 'feature/143-beamershow_clients' (#144) from feature/143-beamershow_clients into dev [`53b7dec`](https://git.odit.services/lfk/frontend/commit/53b7dec7cd516c908d45591b855f4be09371f9b1) | ||||||
|  | - Updated deletion modal [`93fc7c2`](https://git.odit.services/lfk/frontend/commit/93fc7c2e83f78dd88f15d9246127bb9e69f1a8ee) | ||||||
|  | - Updated mounted variables [`674e6a9`](https://git.odit.services/lfk/frontend/commit/674e6a90ec23dde9377bea64c14a50e41ffa450d) | ||||||
|  | - Removed Key after creation [`e10c648`](https://git.odit.services/lfk/frontend/commit/e10c6480a504338b21e30fdf2577e5b6c3b635db) | ||||||
|  | - Updated docker base images [`9767553`](https://git.odit.services/lfk/frontend/commit/976755338b8621064f9a73147aa600af1f77cd51) | ||||||
|  | - Added translation [`96c55db`](https://git.odit.services/lfk/frontend/commit/96c55db63dbfed92b78ff0e7bdab7a8cce4d76e9) | ||||||
|  | - Pinned versions [`cff112d`](https://git.odit.services/lfk/frontend/commit/cff112d705a74a135286943298f3f344341325ac) | ||||||
|  | - Tailwind bump [`e0cbfb0`](https://git.odit.services/lfk/frontend/commit/e0cbfb000bee59a71e06bd58a9c7ef6a0fc7091d) | ||||||
|  | - Added missing translation [`19a333d`](https://git.odit.services/lfk/frontend/commit/19a333d7bda525fbcb3c68f3cbf85a4f925a9707) | ||||||
|  | - Bumped apiclient [`c28f1ee`](https://git.odit.services/lfk/frontend/commit/c28f1ee0bc4456595c21858f38e52ed6f16871c5) | ||||||
|  | - new license file version [CI SKIP] [`3a66f4c`](https://git.odit.services/lfk/frontend/commit/3a66f4c862db9f35c223cc7007b0560fef4e1d63) | ||||||
|  | - Bumped apiclient [`28cbc5b`](https://git.odit.services/lfk/frontend/commit/28cbc5b98ca09657100e1740b83aa2617243b26b) | ||||||
|  | - Ignore pnpm lock [`2d8c4c1`](https://git.odit.services/lfk/frontend/commit/2d8c4c1698a1675f618e85e678012f310f87c6ee) | ||||||
|  |  | ||||||
|  | #### [0.15.6](https://git.odit.services/lfk/frontend/compare/0.15.5...0.15.6) | ||||||
|  |  | ||||||
|  | > 19 July 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.6 [`9fc4ad6`](https://git.odit.services/lfk/frontend/commit/9fc4ad63c4f77b46d645e83c94b51747b91247b8) | ||||||
|  | - Fixed donations getting reduced to the first one on certificates [`2391668`](https://git.odit.services/lfk/frontend/commit/2391668a25a1e11a1409df572d77ad1635070fbc) | ||||||
|  | - new license file version [CI SKIP] [`97054a7`](https://git.odit.services/lfk/frontend/commit/97054a71c1ab8a045762a55148124965c6994373) | ||||||
|  |  | ||||||
|  | #### [0.15.5](https://git.odit.services/lfk/frontend/compare/0.15.4...0.15.5) | ||||||
|  |  | ||||||
|  | > 5 July 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.5 [`717d335`](https://git.odit.services/lfk/frontend/commit/717d33547c3378424dd720005da9952f8a753f1a) | ||||||
|  | - Merge pull request 'Fixed kilometer conversion' (#142) from bugfix/141-runner_kilometers into dev [`997be32`](https://git.odit.services/lfk/frontend/commit/997be32679dc38c9fb0e92b6ce011057b854d99d) | ||||||
|  | - Fixed kilometer conversion [`134f00c`](https://git.odit.services/lfk/frontend/commit/134f00c40e0c8252e7604a73151e8d6685b2c61d) | ||||||
|  | - new license file version [CI SKIP] [`e752ee1`](https://git.odit.services/lfk/frontend/commit/e752ee12d17a4423f4364f7766eafbe7d4cef2d1) | ||||||
|  |  | ||||||
|  | #### [0.15.4](https://git.odit.services/lfk/frontend/compare/0.15.3...0.15.4) | ||||||
|  |  | ||||||
|  | > 5 July 2021 | ||||||
|  |  | ||||||
|  | - Merge pull request 'fix total donation sum in dashboard' (#140) from bugfix/139-total-donation-sum-is-wrong into dev [`#139`](https://git.odit.services/lfk/frontend/issues/139) | ||||||
|  | - 🚀RELEASE v0.15.4 [`cc4515f`](https://git.odit.services/lfk/frontend/commit/cc4515ff66b1c1de3747d0ee6cc465574accedb7) | ||||||
|  | - divide by 100 + toFixes(2) [`b246f2b`](https://git.odit.services/lfk/frontend/commit/b246f2b349b06d1adea318dfad58f97fb1a249bb) | ||||||
|  |  | ||||||
|  | #### [0.15.3](https://git.odit.services/lfk/frontend/compare/0.15.2...0.15.3) | ||||||
|  |  | ||||||
|  | > 16 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.3 [`76b69d8`](https://git.odit.services/lfk/frontend/commit/76b69d851aa590ecf8caac135b72962a72e83635) | ||||||
|  | - Small bugfix (null got displayed) 🛠 [`224f586`](https://git.odit.services/lfk/frontend/commit/224f5863683ae2543a4a435510ed2c558dc5d307) | ||||||
|  |  | ||||||
|  | #### [0.15.2](https://git.odit.services/lfk/frontend/compare/0.15.1...0.15.2) | ||||||
|  |  | ||||||
|  | > 16 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.2 [`9add6c8`](https://git.odit.services/lfk/frontend/commit/9add6c8ff1fbeed91fe97a7cf262921b716f4e3c) | ||||||
|  | - Footer - noopener link [`cee04c1`](https://git.odit.services/lfk/frontend/commit/cee04c1d6fb6005cefe77fb95855ab6fe2cc448f) | ||||||
|  | - Hotfix: Team change recognition 🐞 [`cbec785`](https://git.odit.services/lfk/frontend/commit/cbec78589d2fa21f12ce87e71bff2b49c3a7d345) | ||||||
|  | - NGINX cache assets [`e54a480`](https://git.odit.services/lfk/frontend/commit/e54a4807f70bc333396885f81d3dcc7ae6c115d9) | ||||||
|  |  | ||||||
|  | #### [0.15.1](https://git.odit.services/lfk/frontend/compare/0.15.0...0.15.1) | ||||||
|  |  | ||||||
|  | > 16 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.1 [`a85db7c`](https://git.odit.services/lfk/frontend/commit/a85db7cb3f89881794e37a66ecd822f8ad5873f1) | ||||||
|  | - Merge pull request '🐞🐳 fix Dockerfile' (#138) from bugfix/136-opacity_reactivity into dev [`2bd3779`](https://git.odit.services/lfk/frontend/commit/2bd3779839de16a89b91a3da93033e2a2b742ab7) | ||||||
|  | - 🚚 move to tailwind [`07ac041`](https://git.odit.services/lfk/frontend/commit/07ac041d69b3b1810e5db538b53fe62084490f7a) | ||||||
|  | - 🐞🐳 fix Dockerfile [`303e33c`](https://git.odit.services/lfk/frontend/commit/303e33cafb4a1be01e4c4b43f46ff0c651cb4620) | ||||||
|  | - Dockerfile now uses selfhosted registry [`b4e689d`](https://git.odit.services/lfk/frontend/commit/b4e689dddf0b93a2794aa30ea83e8c6505d6bbfd) | ||||||
|  | - new license file version [CI SKIP] [`98a0b03`](https://git.odit.services/lfk/frontend/commit/98a0b036c5490b4bc4992e83f3bca02be39927fa) | ||||||
|  | - Merge pull request 'Opacity import fix bugfix/136-opacity_reactivity' (#137) from bugfix/136-opacity_reactivity into dev [`fb3f30f`](https://git.odit.services/lfk/frontend/commit/fb3f30fb1024de61ce1c541dae90374454f6ef96) | ||||||
|  | - Added bs import fix [`6213952`](https://git.odit.services/lfk/frontend/commit/621395200751c2d42b9ad44c77e84bda03b62e83) | ||||||
|  |  | ||||||
|  | #### [0.15.0](https://git.odit.services/lfk/frontend/compare/0.14.0...0.15.0) | ||||||
|  |  | ||||||
|  | > 15 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.15.0 [`5c02028`](https://git.odit.services/lfk/frontend/commit/5c02028841c68d9a284bf6971eec2b6bc2fdf1f3) | ||||||
|  | - Merge pull request 'Mark donations as payed feature/133-donation_payments' (#135) from feature/133-donation_payments into dev [`c561b53`](https://git.odit.services/lfk/frontend/commit/c561b536705a68215d9c0a6b320587d1647bf57f) | ||||||
|  | - Sorted translations [`c7a858e`](https://git.odit.services/lfk/frontend/commit/c7a858eed7962294bc9df3c92ce2e46b0a354796) | ||||||
|  | - Added total donation amount to donor overview [`e42ea94`](https://git.odit.services/lfk/frontend/commit/e42ea943b7821d433fe21599edbd9f76c3128ef2) | ||||||
|  | - Added Add Payment button to donor overview [`6e6e8b2`](https://git.odit.services/lfk/frontend/commit/6e6e8b26171f16542c101520800b4b6ea7c023d3) | ||||||
|  | - You can now open a modal to add a payment to a donation from the donation overview [`a943aaf`](https://git.odit.services/lfk/frontend/commit/a943aaf5fce8f113dd967d3842e2b0d7d50604e9) | ||||||
|  | - You can now add payments from the donation overview [`1dbab03`](https://git.odit.services/lfk/frontend/commit/1dbab03fe73b5e0fc011f9b0af7199bd71bc79c5) | ||||||
|  | - Added payment updating via detail [`bdcf5d3`](https://git.odit.services/lfk/frontend/commit/bdcf5d3fc08d250377226a253642d79b2e82d624) | ||||||
|  | - Added **all** missing toast translations [`de5aa92`](https://git.odit.services/lfk/frontend/commit/de5aa9237d261b5d47a8def35afa7f8e0089aea6) | ||||||
|  | - You can now mark fixed donations as already paid on creation [`3d3a10a`](https://git.odit.services/lfk/frontend/commit/3d3a10aafb16d371be9471eb5172f9251fb2583f) | ||||||
|  | - Added translations 🌎 [`d015f97`](https://git.odit.services/lfk/frontend/commit/d015f9739570c44a7a2fe6ba248c9a45c3047c62) | ||||||
|  | - Changed top info style for donation overview [`4c2c24a`](https://git.odit.services/lfk/frontend/commit/4c2c24af2ca5c2874a583b0fd93bee147a17f449) | ||||||
|  | - Added paid donation amount and status to donation detail [`5645eea`](https://git.odit.services/lfk/frontend/commit/5645eeaafaa4254edf1a81bc597ce0c7a9b03ff0) | ||||||
|  | - Added total donation amount to donation overview [`961477d`](https://git.odit.services/lfk/frontend/commit/961477d5224bc44b552d2fc2851d8514116f4e20) | ||||||
|  | - Fixed chante recognition bug for fixed donation [`0f0aae7`](https://git.odit.services/lfk/frontend/commit/0f0aae7ba4cf5dfab15d56ce48edbdbc7cb7e403) | ||||||
|  | - Added total donation amount to donor detail [`a5f7101`](https://git.odit.services/lfk/frontend/commit/a5f71015a6557d664e9d3f505613352792fc38cb) | ||||||
|  | - Added msiisng runner id conversion [`5761815`](https://git.odit.services/lfk/frontend/commit/57618156b49b2b0f0274f2126fef36a017d90022) | ||||||
|  | - AddDonationModal - vertical alignment for paid status [`18acac8`](https://git.odit.services/lfk/frontend/commit/18acac83bc6532e14d36b3399d867e026d0c88ac) | ||||||
|  | - Added missing updated comparison [`04a3038`](https://git.odit.services/lfk/frontend/commit/04a3038369f2717c43459318b7b5754ebbaa9e45) | ||||||
|  | - DonationsOverview contrast on action [`d7d4447`](https://git.odit.services/lfk/frontend/commit/d7d44470bb08ac06594bc400608c17eeacb0434b) | ||||||
|  | - Fixed typo [`4c0886a`](https://git.odit.services/lfk/frontend/commit/4c0886a5d9b91439967bc8f66b09a57177f967d0) | ||||||
|  | - Fixed styling [`865254d`](https://git.odit.services/lfk/frontend/commit/865254d646b5f7de15720551c67ae649601cbcd2) | ||||||
|  | - Changed top info style for donation detail [`000fc97`](https://git.odit.services/lfk/frontend/commit/000fc97beb14427f69d421ff2c96975dbbdc7a3a) | ||||||
|  |  | ||||||
|  | #### [0.14.0](https://git.odit.services/lfk/frontend/compare/0.13.1...0.14.0) | ||||||
|  |  | ||||||
|  | > 14 April 2021 | ||||||
|  |  | ||||||
|  | - Merge pull request 'added donor receipt list download to DonorsOverview' (#134) from feature/132-export-donors-receipt-list into dev [`#132`](https://git.odit.services/lfk/frontend/issues/132) | ||||||
|  | - Sorted translations 🌎 [`c6c9751`](https://git.odit.services/lfk/frontend/commit/c6c97516b3981ef580d620c0c8a6fcc42f26facd) | ||||||
|  | - Fixed typos in translations [`03676b2`](https://git.odit.services/lfk/frontend/commit/03676b2894892c3559118b93e969c063b53b081e) | ||||||
|  | - added donor receipt list download to DonorsOverview [`d241ca5`](https://git.odit.services/lfk/frontend/commit/d241ca569838abbe9581fbd319f7f3b563cb7dcc) | ||||||
|  | - 🚀RELEASE v0.14.0 [`9c5fc6b`](https://git.odit.services/lfk/frontend/commit/9c5fc6b61c0bb2a6d831d4a23ef8679c6e68c6a1) | ||||||
|  | - ⏫ general version bump [`18f151c`](https://git.odit.services/lfk/frontend/commit/18f151c1fb878a74c3d1a2c2a2debf7913739417) | ||||||
|  | - new license file version [CI SKIP] [`302caf0`](https://git.odit.services/lfk/frontend/commit/302caf015f88f77e2b2ae2b67680e79f987ad81e) | ||||||
|  | - Switched to selfhosted images [`112eb29`](https://git.odit.services/lfk/frontend/commit/112eb29f932cd936f1d6c2308dcaeaf8cb642490) | ||||||
|  | - ⏫ bump @odit/lfk-client-js@0.11.0 [`9ca57fa`](https://git.odit.services/lfk/frontend/commit/9ca57fac2eeabbf25142a507fb9c0fa3c90b4e74) | ||||||
|  | - replace donationAmount with paidDonationAmount [`e90e56d`](https://git.odit.services/lfk/frontend/commit/e90e56d8b26aef23aba2bbb0c3942ba4d7feb224) | ||||||
|  |  | ||||||
|  | #### [0.13.1](https://git.odit.services/lfk/frontend/compare/0.13.0...0.13.1) | ||||||
|  |  | ||||||
|  | > 11 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.13.1 [`b512cf8`](https://git.odit.services/lfk/frontend/commit/b512cf86674f1c60b5ac790985ededdfd6554185) | ||||||
|  | - For await fix [`a24d292`](https://git.odit.services/lfk/frontend/commit/a24d2923c6e6da90d610c05183d29d47eaf2ed30) | ||||||
|  |  | ||||||
|  | #### [0.13.0](https://git.odit.services/lfk/frontend/compare/0.12.5...0.13.0) | ||||||
|  |  | ||||||
|  | > 11 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.13.0 [`467808a`](https://git.odit.services/lfk/frontend/commit/467808abefe127dac66a2837fcce3197dddb140f) | ||||||
|  | - Merge pull request 'Better org pdf generation feature/130-org_doc_splitting' (#131) from feature/130-org_doc_splitting into dev [`861f1f2`](https://git.odit.services/lfk/frontend/commit/861f1f221653283e7586aa2c67b205337fd44398) | ||||||
|  | - Org card generation now runs in sequence [`fef14b6`](https://git.odit.services/lfk/frontend/commit/fef14b6e4fb47ad92da61de91fedce96aea26b2c) | ||||||
|  | - Org certificate generation now runs in sequence [`509b22b`](https://git.odit.services/lfk/frontend/commit/509b22bea0dd3e4446e6ecc37d27644e9bf2ad50) | ||||||
|  | - Org contract generation now runs in sequence [`01d2a7e`](https://git.odit.services/lfk/frontend/commit/01d2a7e6aa709b3f2d71575f705fc962e97e2742) | ||||||
|  | - Emergency document server url change [`5476808`](https://git.odit.services/lfk/frontend/commit/5476808683a919bc34dbaea1f1ed276d49750096) | ||||||
|  | - Fixed const -> let [`7447b2f`](https://git.odit.services/lfk/frontend/commit/7447b2f4c134a585905db6733093eab13e6f7c47) | ||||||
|  | - Hotfix: Org * generation🐞 [`ac586fe`](https://git.odit.services/lfk/frontend/commit/ac586fec5abd324d590ba99cdfe8ddddefbf95e6) | ||||||
|  |  | ||||||
|  | #### [0.12.5](https://git.odit.services/lfk/frontend/compare/0.12.4...0.12.5) | ||||||
|  |  | ||||||
|  | > 8 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.12.5 [`331d737`](https://git.odit.services/lfk/frontend/commit/331d737796c82454b1c19fa1840ccc20e36d2626) | ||||||
|  | - Merge pull request 'Added runner team's parentorg name to runenr overciew' (#129) from feature/128-runner_orgs into dev [`ef81b8a`](https://git.odit.services/lfk/frontend/commit/ef81b8adf9bef685a55936d7544bf645c0d6ecbe) | ||||||
|  | - Switched to html entity [`8a7d635`](https://git.odit.services/lfk/frontend/commit/8a7d635cef2d465e70c84e1f7a7b90b98a8dbab1) | ||||||
|  | - Added runner team's parentorg name to runenr overciew [`4c259c1`](https://git.odit.services/lfk/frontend/commit/4c259c1eef2b0166ce6a8493d0c9e9d5ede11146) | ||||||
|  |  | ||||||
|  | #### [0.12.4](https://git.odit.services/lfk/frontend/compare/0.12.3...0.12.4) | ||||||
|  |  | ||||||
|  | > 8 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.12.4 [`5b4ede5`](https://git.odit.services/lfk/frontend/commit/5b4ede5e2f6a26b475a7a4b430a4146d21fb9671) | ||||||
|  | - 🚑 [HOTFIX] - drop "svelte-infinite-loading" [`d0ab3dd`](https://git.odit.services/lfk/frontend/commit/d0ab3dda78bbad2cea18a2491056530897d56607) | ||||||
|  |  | ||||||
|  | #### [0.12.3](https://git.odit.services/lfk/frontend/compare/0.12.2...0.12.3) | ||||||
|  |  | ||||||
|  | > 8 April 2021 | ||||||
|  |  | ||||||
|  | - Merge pull request 'fix' (#126) from bugfix/125-mobile into dev [`#125`](https://git.odit.services/lfk/frontend/issues/125) | ||||||
|  | - almost fixed... [`495a6b2`](https://git.odit.services/lfk/frontend/commit/495a6b22bd8036593f390bdb862d325524cefbcc) | ||||||
|  | - 🐞 bugfix for svelte x tailwind class names [`b6b07cf`](https://git.odit.services/lfk/frontend/commit/b6b07cf30cc6533bd5dbfec1f813c16fde85634d) | ||||||
|  | - fix [`0acaffb`](https://git.odit.services/lfk/frontend/commit/0acaffbdfa359e52654a5afe2788aa59fe6f9036) | ||||||
|  | - 🚀RELEASE v0.12.3 [`d9cf51b`](https://git.odit.services/lfk/frontend/commit/d9cf51b4bbc2136594a03c5d0eeb8cb3f3440b2a) | ||||||
|  | - custom css fix for collapsed_navigation [`ffbc243`](https://git.odit.services/lfk/frontend/commit/ffbc243194c7faeb4fe61c12711a1c441c3994ef) | ||||||
|  | - new license file version [CI SKIP] [`aa17f24`](https://git.odit.services/lfk/frontend/commit/aa17f242209f7e7cecff774ace7a35b581adec1f) | ||||||
|  |  | ||||||
|  | #### [0.12.2](https://git.odit.services/lfk/frontend/compare/0.12.1...0.12.2) | ||||||
|  |  | ||||||
|  | > 7 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.12.2 [`6043bc4`](https://git.odit.services/lfk/frontend/commit/6043bc45174d51ab110b0ed10a8679d96127ab87) | ||||||
|  | - Merge pull request 'feature/110-virtual_list' (#124) from feature/110-virtual_list into dev [`e6ed066`](https://git.odit.services/lfk/frontend/commit/e6ed066e3ffabba6519f94d801d21a27819d0492) | ||||||
|  | - wip on virtuallist [`6304116`](https://git.odit.services/lfk/frontend/commit/6304116edb7f5e3c7b67c15e0b1740d34c513155) | ||||||
|  | - fixed height table [`4cad86c`](https://git.odit.services/lfk/frontend/commit/4cad86cf852468428d77103d052c6974b17c34c3) | ||||||
|  | - pre-merge fixes [`37970d2`](https://git.odit.services/lfk/frontend/commit/37970d2be6b6502701914e41e5bfe2c418438480) | ||||||
|  | - updated virtual scroll list [`1376788`](https://git.odit.services/lfk/frontend/commit/1376788016e767f006661f8c9e6747781f2dce55) | ||||||
|  |  | ||||||
|  | #### [0.12.1](https://git.odit.services/lfk/frontend/compare/0.12.0...0.12.1) | ||||||
|  |  | ||||||
|  | > 6 April 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.12.1 [`834ff8f`](https://git.odit.services/lfk/frontend/commit/834ff8fa63178f36dcacf931c128ba67a3e7bd1b) | ||||||
|  | - Merge pull request 'ImportRunnerModal Cancel Button feature/122-import_cancel' (#123) from feature/112-import_cancel into dev [`1f428a5`](https://git.odit.services/lfk/frontend/commit/1f428a535e3ae619cbf8db51d04255aac8dd8614) | ||||||
|  | - Added cancel button for the first stage of runner import [`0c40966`](https://git.odit.services/lfk/frontend/commit/0c409669700d3a8096cc04716154b0fdca458fe5) | ||||||
|  | - Escape now triggers foll modal close (including reset) instead of just hiding th modal [`9da071f`](https://git.odit.services/lfk/frontend/commit/9da071fe9ba067160334682bf00163e3630fe919) | ||||||
|  |  | ||||||
|  | #### [0.12.0](https://git.odit.services/lfk/frontend/compare/0.11.0...0.12.0) | ||||||
|  |  | ||||||
|  | > 5 April 2021 | ||||||
|  |  | ||||||
|  | - Merge pull request 'feature/108_vite_migration' (#118) from feature/108_vite_migration into dev [`#108`](https://git.odit.services/lfk/frontend/issues/108) | ||||||
|  | - 🚀RELEASE v0.12.0 [`892a04f`](https://git.odit.services/lfk/frontend/commit/892a04f28930481715eb486b1ef4efeb98a6e999) | ||||||
|  | - Fixed package version [`27cc972`](https://git.odit.services/lfk/frontend/commit/27cc9727f1d02d186c3ccadb06e5b4b1b1d6202d) | ||||||
|  | - Merge pull request 'Implmented certificate generation feature/119-Certificate_generation' (#120) from feature/119-Certificate_generation into dev [`f0738d4`](https://git.odit.services/lfk/frontend/commit/f0738d451b02e4a298b5f9cb8ab0be16aed10a38) | ||||||
|  | - The PFS Prefixes now get translated via i18n [`bfacfec`](https://git.odit.services/lfk/frontend/commit/bfacfec76511cae3015f52698fdcbd80a7a15981) | ||||||
|  | - Sorted translations 🌍 [`9e6a8da`](https://git.odit.services/lfk/frontend/commit/9e6a8daf2c394cf17da532382ec7d049a0f89577) | ||||||
|  | - added missing/ wrong translations + formatting! [`7c86a5e`](https://git.odit.services/lfk/frontend/commit/7c86a5eeb370a43451d180a09a501066b023b9a0) | ||||||
|  | - Added i18n [`17f6f4e`](https://git.odit.services/lfk/frontend/commit/17f6f4e616bf57424ee12ad53b939429c02a0171) | ||||||
|  | - Added basic certificate generation component [`af63ce6`](https://git.odit.services/lfk/frontend/commit/af63ce67ae7d8f8a70706c3bd6755197908996ff) | ||||||
|  | - basic ViteJS migration [`ae79e9f`](https://git.odit.services/lfk/frontend/commit/ae79e9fea1963e977ef468e8e56f87d68916fadd) | ||||||
|  | - Implemented generation for orgs [`2e3ac15`](https://git.odit.services/lfk/frontend/commit/2e3ac154be0bf0776cd00f7d510f41ec676ae690) | ||||||
|  | - Implemented generation for teams [`2472640`](https://git.odit.services/lfk/frontend/commit/2472640755e3e41259a44127a875d00517a25842) | ||||||
|  | - updated default entrypoint [`95c8fde`](https://git.odit.services/lfk/frontend/commit/95c8fde72fca5cd5a644d51a33dc88e0b59fce92) | ||||||
|  | - ⏫📍 version bump + pin [`b065b4f`](https://git.odit.services/lfk/frontend/commit/b065b4ff218d07952fa45989e6e2ee7df13e07c1) | ||||||
|  | - 🧹 reorder + fix package [`12433f7`](https://git.odit.services/lfk/frontend/commit/12433f7c236906fe2b29848a0acaa6be1724da56) | ||||||
|  | - 🔨 re-added VS Code devcontainer config [`9318709`](https://git.odit.services/lfk/frontend/commit/93187099d32c506329b1437642aae985f2850689) | ||||||
|  | - 🐳 new Dockerfiles [`0f32968`](https://git.odit.services/lfk/frontend/commit/0f32968fae8b55a13d387918211983d0e61f85ab) | ||||||
|  | - 📃 added readme [`aa24b1d`](https://git.odit.services/lfk/frontend/commit/aa24b1dce5d6d73c8f42fc57f81b692350bf9665) | ||||||
|  | - Copy-paste fix [`f47d5e3`](https://git.odit.services/lfk/frontend/commit/f47d5e347d97ee127fa0380620138a9672665cd5) | ||||||
|  | - 🔨🔥 alpine based devcontainer with working yarn PnP [`777304f`](https://git.odit.services/lfk/frontend/commit/777304f2593df36f4e89d2ba7680add183ff062f) | ||||||
|  | - Copy-paste fix [`7488a8b`](https://git.odit.services/lfk/frontend/commit/7488a8b597a148c309e1b4499d277fed7f3bf9f4) | ||||||
|  | - You can now generate certificates from the runner overview [`bb9b779`](https://git.odit.services/lfk/frontend/commit/bb9b779cee909ab85ef52f13be0a917f1c0a9e62) | ||||||
|  | - Cleaned up generation strings and added the schem for single runner generations for cards [`9c867e1`](https://git.odit.services/lfk/frontend/commit/9c867e106edd68784e6d19743519c1952a0f0bc7) | ||||||
|  | - Changed the basic nameing generation for runenr certificate files [`d65d379`](https://git.odit.services/lfk/frontend/commit/d65d3793de869bcd6733a1bbdac378d0bc1128b3) | ||||||
|  | - ⏫ version bumps [`d7fecfb`](https://git.odit.services/lfk/frontend/commit/d7fecfbd0bc01f1cd44dea3c3837e0cc44afab12) | ||||||
|  | - Cleaned up generation strings and added the schem for single runner generations for sponsoring contracts [`22b09d1`](https://git.odit.services/lfk/frontend/commit/22b09d16d0acc2883e3448dad95ed0f4ea7c6aeb) | ||||||
|  | - Certificate generation from org overview [`3177c6e`](https://git.odit.services/lfk/frontend/commit/3177c6eaa31636ed4546f4797775a0f0a930f5d1) | ||||||
|  | - certificate pdf names now include their locale [`304f28a`](https://git.odit.services/lfk/frontend/commit/304f28a3c10bc4745aa5b7c80d7ba0e651540706) | ||||||
|  | - Runnercard pdfs now include their locale [`3638d87`](https://git.odit.services/lfk/frontend/commit/3638d87bd2ff83618eefda5af18ba19e38e3c2eb) | ||||||
|  | - 🐞 fix await for esNext [`a922776`](https://git.odit.services/lfk/frontend/commit/a9227768de29305b51d10c8a6e4fa1d39b7d998f) | ||||||
|  | - Certificate generation from team overview [`18ec100`](https://git.odit.services/lfk/frontend/commit/18ec100c33a1fbab526187e769dbae54d9db0867) | ||||||
|  | - Added certificate generation from runner overview and detail [`7b685d6`](https://git.odit.services/lfk/frontend/commit/7b685d6cad97d2f7f48c4b19bfc128e1355b70c4) | ||||||
|  | - package cleanup [`6be2ee6`](https://git.odit.services/lfk/frontend/commit/6be2ee626addaf5113b4b4821bd99a276bf4f329) | ||||||
|  | - sponsoring pdf names now include their locale [`0bae5bf`](https://git.odit.services/lfk/frontend/commit/0bae5bf32b8687057dca50cde21022ea8c3abee8) | ||||||
|  | - ✨ update licenses.json [`e99e9e0`](https://git.odit.services/lfk/frontend/commit/e99e9e07089520d5a48021e10d2af7739656641a) | ||||||
|  | - added windicss settings for VSCode [`008027d`](https://git.odit.services/lfk/frontend/commit/008027db0e2736a9bb9defd67178ab3fe580de04) | ||||||
|  | - Certificate generation from team detail [`acd2f05`](https://git.odit.services/lfk/frontend/commit/acd2f0519d62e55dad8e9c3c41e77b6967212502) | ||||||
|  | - ⚡💾 prevent env.js from being cached [`c5d1553`](https://git.odit.services/lfk/frontend/commit/c5d155396a92dfee6d592fb24a936ab521215f6d) | ||||||
|  | - for await fix - ViteJS [`aec5e34`](https://git.odit.services/lfk/frontend/commit/aec5e3473e687415fbfd69c550d9b012e1b1be43) | ||||||
|  | - Certificate generation from org detail [`e6ffc37`](https://git.odit.services/lfk/frontend/commit/e6ffc371e1ca2d4451e7dd4a3ca3c05564edb5fb) | ||||||
|  | - 🧹 drop unused dependencies [`ce50fa2`](https://git.odit.services/lfk/frontend/commit/ce50fa2a62f8ff98e8be9c66432caeebb3952019) | ||||||
|  | - 🐞 fix NGINX config [`5352410`](https://git.odit.services/lfk/frontend/commit/5352410d0c76fd14575d7beafc6a83f028062efe) | ||||||
|  | - Fixed wrong permissiong getting checked [`b97a928`](https://git.odit.services/lfk/frontend/commit/b97a92860d71eb0384170e245a67fa3ea3fd8e85) | ||||||
|  | - new license file version [CI SKIP] [`5cc4871`](https://git.odit.services/lfk/frontend/commit/5cc4871ec4be9f0af07738f6e3d44bdbe31cd25a) | ||||||
|  | - ⏫ bump @odit/lfk-client-js@0.10.1 [`8b74d6d`](https://git.odit.services/lfk/frontend/commit/8b74d6d759c8481f012c201e2ea0d12b29ddef90) | ||||||
|  | - 🔨 dev container open ⚡ [`ceb2146`](https://git.odit.services/lfk/frontend/commit/ceb2146c1b08bbe9250e4db7846e06bd89526c21) | ||||||
|  | - ⏫ version bump: vite-plugin-windicss@0.12.2 [`8d006d8`](https://git.odit.services/lfk/frontend/commit/8d006d8c74d71c43a9031d58f5a8c7fc55ed95fc) | ||||||
|  | - 🚚 move @svitejs/vite-plugin-svelte to @sveltejs/vite-plugin-svelte [`44b53da`](https://git.odit.services/lfk/frontend/commit/44b53da34516b00671b3e5060ba831e409ac3dd5) | ||||||
|  | - ⏫ upgrade vite-plugin-windicss@0.12.1 [`ab45fc1`](https://git.odit.services/lfk/frontend/commit/ab45fc144eaf14f63d86ee53c1db4eefd88f9c7f) | ||||||
|  | - 🐞 fix main.js linking [`467404b`](https://git.odit.services/lfk/frontend/commit/467404bfc87f3c08c5e227f194d71eea7cc48921) | ||||||
|  | - 🐞 fix vite config for production system [`10a011d`](https://git.odit.services/lfk/frontend/commit/10a011d8426e475105ff5d2d5cf4adca2ef7625c) | ||||||
|  | - fix dev script [`eb3ede9`](https://git.odit.services/lfk/frontend/commit/eb3ede9593e2e527df3e3a2f81c8e179bb555f51) | ||||||
|  | - ⏫ bump vite to 2.1.3 [`0cd3e93`](https://git.odit.services/lfk/frontend/commit/0cd3e937d852eeabe43eb6298bfabe20834240b2) | ||||||
|  | - Removed useless console.log [`d23dbaa`](https://git.odit.services/lfk/frontend/commit/d23dbaaf695a60fe5ebbc9945646a16b5fc45a16) | ||||||
|  | - Removed useless console log [`48cfc15`](https://git.odit.services/lfk/frontend/commit/48cfc15cfb09096db1bd5ddbe9138b1a604d581f) | ||||||
|  |  | ||||||
|  | #### [0.11.0](https://git.odit.services/lfk/frontend/compare/0.10.0...0.11.0) | ||||||
|  |  | ||||||
|  | > 30 March 2021 | ||||||
|  |  | ||||||
|  | - Added button (including translations [`0614c76`](https://git.odit.services/lfk/frontend/commit/0614c76e924b18b512bab59933a26fec07cf483d) | ||||||
|  | - Added button (including translations [`97e338f`](https://git.odit.services/lfk/frontend/commit/97e338f9d4f388596d550990457254c7fa1a3492) | ||||||
|  | - Sorted translations [`89bb9c2`](https://git.odit.services/lfk/frontend/commit/89bb9c215e356e0940678f5cabd9e38bc203040e) | ||||||
|  | - Added function for generating cards with pdf [`c8d6390`](https://git.odit.services/lfk/frontend/commit/c8d639024a5f2f72d6e30d2ce990b08bd71a5471) | ||||||
|  | - 🚀RELEASE v0.11.0 [`f8ccf4f`](https://git.odit.services/lfk/frontend/commit/f8ccf4f5d8f68ecee31430029889b8ab1ecec682) | ||||||
|  | - Fixed button styling [`08cb079`](https://git.odit.services/lfk/frontend/commit/08cb079e9798392e26515d559af2637e74deea97) | ||||||
|  | - Now returning cards on creation with pdf [`1d999d4`](https://git.odit.services/lfk/frontend/commit/1d999d4910acb5efa21b3f9922cdb359babff404) | ||||||
|  | - new license file version [CI SKIP] [`8f8b998`](https://git.odit.services/lfk/frontend/commit/8f8b9988ad94ee9f3729a3fe6fdb4c558828d892) | ||||||
|  | - Merge pull request 'Generate and print bulk blank cards feature/116-download_blanc_cards' (#117) from feature/116-download_blanc_cards into dev [`25d8b86`](https://git.odit.services/lfk/frontend/commit/25d8b86efd89c442d1bf308a8743134820acfd1f) | ||||||
|  | - Added comment [`636f018`](https://git.odit.services/lfk/frontend/commit/636f018daa33b99468a257bfc33477e1e644d081) | ||||||
|  | - Bumped lfk client js version [`2d18686`](https://git.odit.services/lfk/frontend/commit/2d18686ce782a434ca7bd34c07c36a35b9497273) | ||||||
|  | - Bumped lfk-client-js [`7dfaa75`](https://git.odit.services/lfk/frontend/commit/7dfaa7579a22b13194fcdd1c02b4437958261472) | ||||||
|  |  | ||||||
|  | #### [0.10.0](https://git.odit.services/lfk/frontend/compare/0.9.1...0.10.0) | ||||||
|  |  | ||||||
|  | > 26 March 2021 | ||||||
|  |  | ||||||
|  | - Added translations [`79c447b`](https://git.odit.services/lfk/frontend/commit/79c447b4c65e55ebb5af71fb0b09174c36e2cecf) | ||||||
|  | - Sorted translations 🌍 [`5bd3a46`](https://git.odit.services/lfk/frontend/commit/5bd3a463f00abaf2c98ab554f88e5542d01f364a) | ||||||
|  | - Reset can now only be triggered if pw is strong enoug [`75d8f73`](https://git.odit.services/lfk/frontend/commit/75d8f7331b6ae78f3979bb62148188a16f83cb8d) | ||||||
|  | - Module now exports functions that check if a password is strong enough and equal to a potential confirmation field [`b2509e9`](https://git.odit.services/lfk/frontend/commit/b2509e9e53ab6b51dfd55e26712e8928160cd64b) | ||||||
|  | - 🚀RELEASE v0.10.0 [`f7fc196`](https://git.odit.services/lfk/frontend/commit/f7fc1967a50f302af1d8b668628be2f4ab2975a3) | ||||||
|  | - Added more cirteria to the password strength component [`5fa9939`](https://git.odit.services/lfk/frontend/commit/5fa9939696a35d60d762feb0cebef61d31869218) | ||||||
|  | - Now using pw strength component [`6aaf838`](https://git.odit.services/lfk/frontend/commit/6aaf8384512185a3a319ce6b3e2505e910468e64) | ||||||
|  | - Added a password strength verification [`ad3bd31`](https://git.odit.services/lfk/frontend/commit/ad3bd312e9a5785f81029ea2b7e302ea1addd988) | ||||||
|  | - Implemented a custom password strength component [`4956bb0`](https://git.odit.services/lfk/frontend/commit/4956bb0e9c3c1d22d60e849aea5664e35330f897) | ||||||
|  | - User creation can now only be triggered if pw is strong enoug [`540304f`](https://git.odit.services/lfk/frontend/commit/540304f947f60a7072c592ca8088996ce7e95cb4) | ||||||
|  | - Now using pw strength component for user creation [`7862f44`](https://git.odit.services/lfk/frontend/commit/7862f446532903f1a2eac7b21d5c80c3245785e5) | ||||||
|  | - Added missing exports [`962dd0c`](https://git.odit.services/lfk/frontend/commit/962dd0c1bbc0df7f20bcec5b4247188c8935c87e) | ||||||
|  | - new license file version [CI SKIP] [`aedb8a7`](https://git.odit.services/lfk/frontend/commit/aedb8a765ba053545adbba9eb014b3bb0e5aac8c) | ||||||
|  | - Bumped lfk-client version 🔝 [`cf58bd1`](https://git.odit.services/lfk/frontend/commit/cf58bd15c3541c417ab2be83d96135e931a2b6f6) | ||||||
|  | - new license file version [CI SKIP] [`34f4f68`](https://git.odit.services/lfk/frontend/commit/34f4f68524918fd3d1963966a1e259d5b60efaca) | ||||||
|  | - Merge pull request 'Implemented password strength test feature/106-password_strength' (#115) from feature/106-password_strength into dev [`09b8144`](https://git.odit.services/lfk/frontend/commit/09b81440804cf98303fcb723a9717d6d0f432da8) | ||||||
|  | - Formatting🛠 [`4167819`](https://git.odit.services/lfk/frontend/commit/4167819e7a864d3b1dd95ba48ab1525a454f7f30) | ||||||
|  | - Now using pw strength component for reset [`5d5f7c7`](https://git.odit.services/lfk/frontend/commit/5d5f7c7f5c6a69146f41996f4facfeff95791be0) | ||||||
|  |  | ||||||
|  | #### [0.9.1](https://git.odit.services/lfk/frontend/compare/0.9.0...0.9.1) | ||||||
|  |  | ||||||
|  | > 26 March 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.9.1 [`2ca63fd`](https://git.odit.services/lfk/frontend/commit/2ca63fd1f675f0da2b18ba43095074dd4823991d) | ||||||
|  | - Merge pull request 'Org selfservice Link feature/112-org_registration_links' (#114) from feature/112-org_registration_links into dev [`a5d25e7`](https://git.odit.services/lfk/frontend/commit/a5d25e7d92c7c37e90dbb4ba74b787873f920c6b) | ||||||
|  | - Added checkbox to enable registration [`f9fe793`](https://git.odit.services/lfk/frontend/commit/f9fe79357317653b46c09eb95b0db13845cddcf9) | ||||||
|  | - Sorted translations [`c074c12`](https://git.odit.services/lfk/frontend/commit/c074c12be75f285612f7a732c106404d9fb4538a) | ||||||
|  | - You can now copy the selfservice links to your clipboard [`fcd55f8`](https://git.odit.services/lfk/frontend/commit/fcd55f89d72e6ceb9bb2bdd194cc3420145d6d0d) | ||||||
|  | - Formatting [`f185d55`](https://git.odit.services/lfk/frontend/commit/f185d559c0d6476f2f2b9ea74aaad3297411801d) | ||||||
|  | - Copy now 100% worX [`a3921b4`](https://git.odit.services/lfk/frontend/commit/a3921b45c70b410293db593a75d2fdd34c131733) | ||||||
|  | - Fixed changes in wrong file [`73d95bc`](https://git.odit.services/lfk/frontend/commit/73d95bc0042f8f586ba2f2345342e25da1d280c2) | ||||||
|  | - Added check if key exists [`c2d29ff`](https://git.odit.services/lfk/frontend/commit/c2d29ff233f6b3e9dd2555b7e0b845592da2ba35) | ||||||
|  | - Added check if key exists [`2316baa`](https://git.odit.services/lfk/frontend/commit/2316baa8984832382be9f3b4549ca62cf9ccb5a3) | ||||||
|  | - Added translations [`ddbc293`](https://git.odit.services/lfk/frontend/commit/ddbc293e9ca0525910bf3d995de970ee2c35c56a) | ||||||
|  | - new license file version [CI SKIP] [`ded9b58`](https://git.odit.services/lfk/frontend/commit/ded9b589fe087915176c5b54f3c55e412541bc8f) | ||||||
|  | - Merge pull request 'first merge to main 🚀' (#71) from dev into main [`9aa8e7e`](https://git.odit.services/lfk/frontend/commit/9aa8e7edffa7e51b00a5ab7a8f16828b7a469181) | ||||||
|  |  | ||||||
|  | #### [0.9.0](https://git.odit.services/lfk/frontend/compare/0.8.7...0.9.0) | ||||||
|  |  | ||||||
|  | > 26 March 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.9.0 [`67c3732`](https://git.odit.services/lfk/frontend/commit/67c3732fad5a7c64ae11dcbebaaa095e1a2b387c) | ||||||
|  | - Merge pull request 'Runner cards feature/94-runnercard_mgnt' (#111) from feature/94-runnercard_mgnt into dev [`2932f45`](https://git.odit.services/lfk/frontend/commit/2932f4591e62187a4903511051d110e9679c0993) | ||||||
|  | - Sorted translations 🌍 [`1665a1a`](https://git.odit.services/lfk/frontend/commit/1665a1a093862a13be78ec65dcdf64eb7d855593) | ||||||
|  | - Added translations [`6b5945a`](https://git.odit.services/lfk/frontend/commit/6b5945add86a77630c500872545bb91724b2047f) | ||||||
|  | - Sorted translations 🌍 [`d6c315a`](https://git.odit.services/lfk/frontend/commit/d6c315ab8e020bc65b967e2c3f4cd921392d66d5) | ||||||
|  | - Sorted translations [`de2fe0e`](https://git.odit.services/lfk/frontend/commit/de2fe0e9f171efb3deeea8cfe638f60e3ca90423) | ||||||
|  | - Added basic cards page [`5c5ef95`](https://git.odit.services/lfk/frontend/commit/5c5ef95d2be65c0e951dcd472113c8ce0593c9e0) | ||||||
|  | - Moved contract generation to it's own component [`0cfc87f`](https://git.odit.services/lfk/frontend/commit/0cfc87fbe6adfacab5c2fab732866aead3231fbf) | ||||||
|  | - Teams now use the new sponsoring contracts module [`014ba3b`](https://git.odit.services/lfk/frontend/commit/014ba3bf6718ff28f35c67c8f732b53aae50723c) | ||||||
|  | - Basic card generation worX 🎉🎉 [`d467475`](https://git.odit.services/lfk/frontend/commit/d467475b6d61d50bec3a043ea8792533e8593df6) | ||||||
|  | - Orgs now use the new sponsoring contracts module [`8b451b3`](https://git.odit.services/lfk/frontend/commit/8b451b3c6794e7df09898a687533ce8fadd56192) | ||||||
|  | - Added runnercard detail/edit modal [`0313f8c`](https://git.odit.services/lfk/frontend/commit/0313f8cc495088df1237d00e6b9ed1a94f019644) | ||||||
|  | - Implemented Add card modal [`535b23a`](https://git.odit.services/lfk/frontend/commit/535b23ae917de154e08962f5d486c50d6e091fe0) | ||||||
|  | - Added bulk card creation modal [`8a32569`](https://git.odit.services/lfk/frontend/commit/8a32569a3be1ad26ba163f4e2b67a368cfeeb422) | ||||||
|  | - Added basic card overview [`c6a1526`](https://git.odit.services/lfk/frontend/commit/c6a15264b3d13d516f3d97ea4b891ed1c328cead) | ||||||
|  | - Fixed org generation not hiding the generation toast [`c87321f`](https://git.odit.services/lfk/frontend/commit/c87321f804858f84fcccd85a15b9c3fb003c18be) | ||||||
|  | - Working runner runnercard generation [`c907486`](https://git.odit.services/lfk/frontend/commit/c907486c4d1c64114124deb3cd0d0cf11d38a6b1) | ||||||
|  | - Implemented bulk creation [`7ad6b73`](https://git.odit.services/lfk/frontend/commit/7ad6b73574174f24f2d6f23b3caf4823881a85e7) | ||||||
|  | - Now w/ working dialog🎉🎉🎉 [`ae96730`](https://git.odit.services/lfk/frontend/commit/ae9673070c3959ff6190a37123f3fc609b182c5a) | ||||||
|  | - Now w/working editing [`fac059f`](https://git.odit.services/lfk/frontend/commit/fac059f02cae84261443ee95448ec8db06dd755a) | ||||||
|  | - Added runnercard generation for teams [`23e0b53`](https://git.odit.services/lfk/frontend/commit/23e0b53107623c505d07a99a51ce836c9324acce) | ||||||
|  | - Added bulk creation modal to cards view [`f46ccb6`](https://git.odit.services/lfk/frontend/commit/f46ccb610e01654a4ee5e47d78ab500045dd494b) | ||||||
|  | - Added a new runenrcard logo [`acf78a8`](https://git.odit.services/lfk/frontend/commit/acf78a88221d0988f6501ae341e028a4113b578d) | ||||||
|  | - Moved modal import to overview for simplification [`1a52aaf`](https://git.odit.services/lfk/frontend/commit/1a52aaf8d1ad19b03d355aec0e1c48182db024f9) | ||||||
|  | - Added CardsEmptyState + Emtystate graphic [`2d0beaa`](https://git.odit.services/lfk/frontend/commit/2d0beaaaad4efefd036bbef09f10c8c22bdb2760) | ||||||
|  | - Added message for missing runner/blanco card) [`4715978`](https://git.odit.services/lfk/frontend/commit/4715978f810bbb283876f06d147b1ec86d373786) | ||||||
|  | - Fixed counting bug [`f5c1ec9`](https://git.odit.services/lfk/frontend/commit/f5c1ec9939d856804c9ec3ead4b3ed869fc2ea63) | ||||||
|  | - Added card generation/printing from detail [`4a36fb6`](https://git.odit.services/lfk/frontend/commit/4a36fb6d952d9fe4d5edbe1ed0779c7fbcd50ef0) | ||||||
|  | - Formatting [`a516aa7`](https://git.odit.services/lfk/frontend/commit/a516aa7775faa2244862bb2e3c4de623c6405e5b) | ||||||
|  | - Moved the pdf generation related componenets to their own folder [`ddd82a7`](https://git.odit.services/lfk/frontend/commit/ddd82a71a7b67ead892626addfd56ba4cc632750) | ||||||
|  | - Now routing the cards page [`e852305`](https://git.odit.services/lfk/frontend/commit/e852305400a139f8169350077c30012aed556828) | ||||||
|  | - Fuggin snowpack bs [`44bc148`](https://git.odit.services/lfk/frontend/commit/44bc14820fed26d5e0d8b12ecd6b46ca0608ae7b) | ||||||
|  | - Formatting [`9f7d223`](https://git.odit.services/lfk/frontend/commit/9f7d2234fb9603a7391ec9a64253724c2c25c333) | ||||||
|  | - Now with working org runenrcard generation [`4b3d38b`](https://git.odit.services/lfk/frontend/commit/4b3d38b05b3ed74fc3c0d77e00fa2ed245e6325c) | ||||||
|  | - Now importing runner overview [`77e9c20`](https://git.odit.services/lfk/frontend/commit/77e9c205f94cf56c2e3584444899adb1e8bdf3c6) | ||||||
|  | - CardsOverview - move to 'enabled' language key [`df53c07`](https://git.odit.services/lfk/frontend/commit/df53c0745035a220d4c07fdce1b5a5e4d763411d) | ||||||
|  | - ✒ typo - "Geb" -> "Gebe" [`f794af0`](https://git.odit.services/lfk/frontend/commit/f794af0950de59a7a7b468c30abdcb5c145f65fb) | ||||||
|  | - Bumped lfk client lib version [`3cd0468`](https://git.odit.services/lfk/frontend/commit/3cd0468b1921824b131178cb02677540b079f9b0) | ||||||
|  | - drop console log - CardDetailModal [`40899e9`](https://git.odit.services/lfk/frontend/commit/40899e9d80ba07a3fbbcac72782db53d98dc318e) | ||||||
|  | - Removed debug info [`55693de`](https://git.odit.services/lfk/frontend/commit/55693de93420c2d76af296fcacc6bcad644a3cbf) | ||||||
|  |  | ||||||
|  | #### [0.8.7](https://git.odit.services/lfk/frontend/compare/0.8.6...0.8.7) | ||||||
|  |  | ||||||
|  | > 25 March 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.8.7 [`0af2647`](https://git.odit.services/lfk/frontend/commit/0af26479656393b0baea88f6f83c778740a67e62) | ||||||
|  | - Fixed listen on wrong permission🐞 [`0844215`](https://git.odit.services/lfk/frontend/commit/08442154f4bf94fc1101808b4585dc1f95afe8b2) | ||||||
|  |  | ||||||
|  | #### [0.8.6](https://git.odit.services/lfk/frontend/compare/0.8.5...0.8.6) | ||||||
|  |  | ||||||
|  | > 25 March 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.8.6 [`c3c95bf`](https://git.odit.services/lfk/frontend/commit/c3c95bf2916618efe6764a33d9a42d35764d15be) | ||||||
|  | - Merge pull request 'Know Production Bugs 🐞' (#109) from bugfix/107-prod_issues into dev [`d2050b5`](https://git.odit.services/lfk/frontend/commit/d2050b5948890a6077cbb41d82d1a6a1d1106652) | ||||||
|  | - Errors now toast errors❌ [`17e0805`](https://git.odit.services/lfk/frontend/commit/17e0805fe64f6d181f55b81afa502ee6443ebabe) | ||||||
|  | - Sorted translations 👀 [`82b1811`](https://git.odit.services/lfk/frontend/commit/82b1811971b974b686e7618b8a381e1589c168f6) | ||||||
|  | - Fixed missing translations for scanstations🌍 [`aeadef6`](https://git.odit.services/lfk/frontend/commit/aeadef60bbe71da09bb569d20ca7377645beba7f) | ||||||
|  | - Sorted translations🌍 [`a1ab65a`](https://git.odit.services/lfk/frontend/commit/a1ab65a0e975c02c01c603bf6d95a79ada1caa0b) | ||||||
|  | - Fixed runner import getting triggered with invalid information [`ddd9c39`](https://git.odit.services/lfk/frontend/commit/ddd9c396b6bfd39a7b1627d4975151943b367ebf) | ||||||
|  | - Removed middlename search from all files that had it [`6b92405`](https://git.odit.services/lfk/frontend/commit/6b92405bae21e78d694601cbc0b33eed56ef4533) | ||||||
|  | - Fixed mail login bug🐞📧 [`0768939`](https://git.odit.services/lfk/frontend/commit/076893981ff4f7f17330746c561acc570339adac) | ||||||
|  | - Now disabled search by middlename as a quick workaround 🐞 [`49e87cc`](https://git.odit.services/lfk/frontend/commit/49e87ccb15a7ed5edea22a3c3e235f7bee07d3f4) | ||||||
|  | - Fixed conflicting css [`50fffef`](https://git.odit.services/lfk/frontend/commit/50fffef13b8fce885964d8ac277b4ca24d944b2b) | ||||||
|  | - Commented out the buggy runner search to prevent bad UX [`fbe74a5`](https://git.odit.services/lfk/frontend/commit/fbe74a5d8090553a35576a17c97019939cf4f386) | ||||||
|  | - Fixed outsideclick not clearing import modal🛠 [`ef49e50`](https://git.odit.services/lfk/frontend/commit/ef49e507c175510eeb466d33f222755fac8a2a0b) | ||||||
|  |  | ||||||
|  | #### [0.8.5](https://git.odit.services/lfk/frontend/compare/0.8.4...0.8.5) | ||||||
|  |  | ||||||
|  | > 20 March 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.8.5 [`e838e6f`](https://git.odit.services/lfk/frontend/commit/e838e6f321bef1565a7e4316890a3c600b242e5a) | ||||||
|  | - Fixed dupliacate mutation 🐞 [`91dd525`](https://git.odit.services/lfk/frontend/commit/91dd5256e9545f62e4342ae5477c36262d6e3401) | ||||||
|  |  | ||||||
|  | #### [0.8.4](https://git.odit.services/lfk/frontend/compare/0.8.3...0.8.4) | ||||||
|  |  | ||||||
|  | > 20 March 2021 | ||||||
|  |  | ||||||
|  | - CONFIG: default_username + default_password [`cc926e8`](https://git.odit.services/lfk/frontend/commit/cc926e84fb8bd9d6c9fd37349e25eb802e1bb324) | ||||||
|  | - 🚀RELEASE v0.8.4 [`3d4dc2d`](https://git.odit.services/lfk/frontend/commit/3d4dc2d72b129f0134ae9f230810c3301dbd5caa) | ||||||
|  | - CONFIG: add 'demo' as default username/password [`ba34710`](https://git.odit.services/lfk/frontend/commit/ba3471068ab00e2d5dbe21d6d763094e662f8347) | ||||||
|  |  | ||||||
| #### [0.8.3](https://git.odit.services/lfk/frontend/compare/0.8.2...0.8.3) | #### [0.8.3](https://git.odit.services/lfk/frontend/compare/0.8.2...0.8.3) | ||||||
|  |  | ||||||
|  | > 20 March 2021 | ||||||
|  |  | ||||||
| - Sorted translation 🌍 [`d6f6d10`](https://git.odit.services/lfk/frontend/commit/d6f6d10cb6b639a1f988e0da4811355750b0f027) | - Sorted translation 🌍 [`d6f6d10`](https://git.odit.services/lfk/frontend/commit/d6f6d10cb6b639a1f988e0da4811355750b0f027) | ||||||
| - Smaller bugfixes [`8e04377`](https://git.odit.services/lfk/frontend/commit/8e0437728bd04223a23cdf1879c6c739ca8ebef7) | - Smaller bugfixes [`8e04377`](https://git.odit.services/lfk/frontend/commit/8e0437728bd04223a23cdf1879c6c739ca8ebef7) | ||||||
| - More small fixes [`1249904`](https://git.odit.services/lfk/frontend/commit/12499045824c13a3ee35c6cc8c3c3a3130dbec12) | - More small fixes [`1249904`](https://git.odit.services/lfk/frontend/commit/12499045824c13a3ee35c6cc8c3c3a3130dbec12) | ||||||
|  | - 🚀RELEASE v0.8.3 [`fff16e6`](https://git.odit.services/lfk/frontend/commit/fff16e6650cce1231a8d0db43531bf8e3e01f84a) | ||||||
|  |  | ||||||
| #### [0.8.2](https://git.odit.services/lfk/frontend/compare/0.8.1...0.8.2) | #### [0.8.2](https://git.odit.services/lfk/frontend/compare/0.8.1...0.8.2) | ||||||
|  |  | ||||||
| @@ -816,7 +1329,7 @@ All notable changes to this project will be documented in this file. Dates are d | |||||||
| - init [`32357ec`](https://git.odit.services/lfk/frontend/commit/32357ece0a7195ea1135c9c3e4c6c84323f95b4d) | - init [`32357ec`](https://git.odit.services/lfk/frontend/commit/32357ece0a7195ea1135c9c3e4c6c84323f95b4d) | ||||||
| - tmp [`1b7173c`](https://git.odit.services/lfk/frontend/commit/1b7173cda9134ee8058a00bdc030defa80d46bfc) | - tmp [`1b7173c`](https://git.odit.services/lfk/frontend/commit/1b7173cda9134ee8058a00bdc030defa80d46bfc) | ||||||
| - Login - move to env.js import [`8ef0b21`](https://git.odit.services/lfk/frontend/commit/8ef0b21819309752c573d0485f6514152fb684e6) | - Login - move to env.js import [`8ef0b21`](https://git.odit.services/lfk/frontend/commit/8ef0b21819309752c573d0485f6514152fb684e6) | ||||||
| -  initial commit [`4bb3bae`](https://git.odit.services/lfk/frontend/commit/4bb3bae4e6fc89c35a8a2b36b7cd6e6d47958eae) | - initial commit [`4bb3bae`](https://git.odit.services/lfk/frontend/commit/4bb3bae4e6fc89c35a8a2b36b7cd6e6d47958eae) | ||||||
| - Initial license export [`4c96b9a`](https://git.odit.services/lfk/frontend/commit/4c96b9a3e04dbb7c021c71aa8828a29248509fbe) | - Initial license export [`4c96b9a`](https://git.odit.services/lfk/frontend/commit/4c96b9a3e04dbb7c021c71aa8828a29248509fbe) | ||||||
| - 🚚 move to tinro svelte router [`a50ea15`](https://git.odit.services/lfk/frontend/commit/a50ea15b38023b867a9f7757e973184cbcdd2457) | - 🚚 move to tinro svelte router [`a50ea15`](https://git.odit.services/lfk/frontend/commit/a50ea15b38023b867a9f7757e973184cbcdd2457) | ||||||
| - new Dashboard [`7270ce9`](https://git.odit.services/lfk/frontend/commit/7270ce9d32869abd4f6ac65ab7c2c87363633cbe) | - new Dashboard [`7270ce9`](https://git.odit.services/lfk/frontend/commit/7270ce9d32869abd4f6ac65ab7c2c87363633cbe) | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,18 +1,15 @@ | |||||||
| FROM node:15.5.1-alpine3.12 | FROM registry.odit.services/hub/library/node:19.7.0-alpine3.16 as build | ||||||
|  | ARG NPM_REGISTRY_URL=https://registry.npmjs.org | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| RUN npm i -g pnpm |  | ||||||
| COPY package.json ./ | COPY package.json pnpm-lock.yaml *.config.js *.config.cjs index.html ./ | ||||||
| RUN pnpm i | RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 && pnpm i | ||||||
| COPY package.json *.config.js workbox-config.js template-copy.js index.template.html s-config.template.js ./ |  | ||||||
| COPY src ./src | COPY src ./src | ||||||
| COPY public ./public | COPY public ./public | ||||||
| RUN pnpm run build | RUN pnpm build | ||||||
|  |  | ||||||
| # final image | # final image | ||||||
| FROM alpine | FROM registry.odit.services/library/nginx-brotli:3.15 as final | ||||||
| COPY --from=0 /app/build /app | COPY --from=build /app/dist /usr/share/nginx/html | ||||||
| RUN rm -rf /app/build/_dist_/components |  | ||||||
| RUN rm -rf /app/build/_dist_/locales |  | ||||||
| RUN rm -rf /app/build-manifest.json |  | ||||||
| FROM fholzer/nginx-brotli:v1.19.1 |  | ||||||
| COPY --from=1 /app /usr/share/nginx/html |  | ||||||
| COPY ./nginx.conf /etc/nginx/nginx.conf | COPY ./nginx.conf /etc/nginx/nginx.conf | ||||||
							
								
								
									
										22
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | # @odit/lfk-frontend | ||||||
|  |  | ||||||
|  | ## ✒️ Overview | ||||||
|  | This is an API client for [https://git.odit.services/lfk/backend](@lfk/backend) | ||||||
|  | - WebApp built with [Svelte](https://svelte.dev), [WindiCSS](https://windicss.org/) (to compile [TailwindCSS](https://tailwindcss.com/)) and [Vite](https://vitejs.dev). | ||||||
|  |  | ||||||
|  | This application is intended for use by admin users/ members only. | ||||||
|  |  | ||||||
|  | ## 🚀 Getting Started | ||||||
|  | ``` | ||||||
|  | yarn | ||||||
|  | ``` | ||||||
|  | ## Development | ||||||
|  | ``` | ||||||
|  | yarn dev | ||||||
|  | / | ||||||
|  | yarn dev --open | ||||||
|  | ``` | ||||||
|  | ## Build | ||||||
|  | ``` | ||||||
|  | yarn build | ||||||
|  | ``` | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| const config = { |  | ||||||
| 	baseurl: 'https://dev.lauf-fuer-kaya.de', |  | ||||||
| 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe', |  | ||||||
| 	fallback_username: 'admin', |  | ||||||
| 	fallback_password: '72fpTzsev4xUu78QPs2FCbwZ3', |  | ||||||
| 	prefersHashRouting: true |  | ||||||
| }; |  | ||||||
| @@ -10,14 +10,13 @@ | |||||||
|   <meta name="viewport" content="width=device-width, initial-scale=1" /> |   <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||||
|   <meta name="description" content="Lauf Für Kaya! - Admin" /> |   <meta name="description" content="Lauf Für Kaya! - Admin" /> | ||||||
|   <title>Lauf für Kaya! - Admin</title> |   <title>Lauf für Kaya! - Admin</title> | ||||||
|   __TAILWIND_INSERT__ |  | ||||||
| </head> | </head> | ||||||
| 
 | 
 | ||||||
| <body> | <body> | ||||||
|   <span style="display: none;visibility: hidden;" id="buildinfo">RELEASE_INFO-0.8.3-RELEASE_INFO</span> |   <span style="display: none;visibility: hidden;" id="buildinfo">RELEASE_INFO-0.17.3-RELEASE_INFO</span> | ||||||
|   <noscript>You need to enable JavaScript to run this app.</noscript> |   <noscript>You need to enable JavaScript to run this app.</noscript> | ||||||
|   <script src="/env.js"></script> |   <script src="/env.js"></script> | ||||||
|   <script defer type="module" src="/_dist_/index.js"></script> |   <script type="module" src="/src/main.js"></script> | ||||||
| </body> | </body> | ||||||
| 
 | 
 | ||||||
| </html> | </html> | ||||||
							
								
								
									
										14
									
								
								nginx.conf
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								nginx.conf
									
									
									
									
									
								
							| @@ -6,6 +6,20 @@ http { | |||||||
|     server { |     server { | ||||||
|         error_page 404 /index.html; |         error_page 404 /index.html; | ||||||
|         root /usr/share/nginx/html; |         root /usr/share/nginx/html; | ||||||
|  |         location /assets { | ||||||
|  |             expires 1y; | ||||||
|  |             log_not_found off; | ||||||
|  |             access_log off; | ||||||
|  |         } | ||||||
|  |         location = /index.html { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|  |         location = / { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|  |         location = /env.js { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|         location / { |         location / { | ||||||
|             try_files $uri $uri/ /index.html; |             try_files $uri $uri/ /index.html; | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,45 +1,27 @@ | |||||||
| { | { | ||||||
| 	"name": "@odit/lfk-frontend", | 	"name": "@odit/lfk-frontend", | ||||||
| 	"version": "0.8.3", | 	"version": "0.17.3", | ||||||
| 	"scripts": { | 	"scripts": { | ||||||
| 		"i18n-order": "node order.js", | 		"i18n-order": "node order.js", | ||||||
| 		"dev:all": "yarn prebuild && snowpack dev", | 		"dev": "vite", | ||||||
| 		"dev": "cross-env NODE_ENV_ODIT=development_fast node template-copy.js && yarn build:sw && snowpack dev", | 		"build": "vite build", | ||||||
| 		"build": "yarn prebuild && snowpack build", |  | ||||||
| 		"prebuild": "cross-env NODE_ENV_ODIT=production node template-copy.js && yarn build:sw", |  | ||||||
| 		"build:sw": "workbox generateSW workbox-config.js", |  | ||||||
| 		"release": "release-it", | 		"release": "release-it", | ||||||
| 		"licenses:export": "license-exporter --json -o public" | 		"licenses:export": "license-exporter --json -o public" | ||||||
| 	}, | 	}, | ||||||
| 	"license": "CC-BY-NC-SA-4.0", | 	"license": "CC-BY-NC-SA-4.0", | ||||||
| 	"dependencies": { |  | ||||||
| 		"@odit/lfk-client-js": "0.6.4", |  | ||||||
| 		"csvtojson": "^2.0.10", |  | ||||||
| 		"gridjs": "3.3.0", |  | ||||||
| 		"localforage": "1.9.0", |  | ||||||
| 		"marked": "^2.0.1", |  | ||||||
| 		"svelte-focus-trap": "1.0.1", |  | ||||||
| 		"svelte-i18n": "3.3.7", |  | ||||||
| 		"svelte-select": "^3.17.0", |  | ||||||
| 		"tailwindcss": "2.0.3", |  | ||||||
| 		"tinro": "0.6.1", |  | ||||||
| 		"toastify-js": "1.9.3", |  | ||||||
| 		"validator": "13.5.2", |  | ||||||
| 		"xlsx": "^0.16.9" |  | ||||||
| 	}, |  | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@odit/license-exporter": "^0.0.11", | 		"@sveltejs/vite-plugin-svelte": "1.0.0-next.6", | ||||||
| 		"@snowpack/plugin-svelte": "3.5.2", | 		"@odit/license-exporter": "0.0.12", | ||||||
| 		"auto-changelog": "^2.2.1", | 		"@types/html-minifier": "4.0.2", | ||||||
| 		"autoprefixer": "10.2.5", | 		"auto-changelog": "2.4.0", | ||||||
| 		"cross-env": "^7.0.3", | 		"autoprefixer": "10.4.14", | ||||||
| 		"postcss": "8.2.8", | 		"html-minifier": "4.0.0", | ||||||
| 		"postcss-load-config": "3.0.1", | 		"postcss": "8.4.21", | ||||||
| 		"release-it": "^14.4.1", | 		"release-it": "14.6.1", | ||||||
| 		"snowpack": "3.0.13", | 		"svelte-preprocess": "4.7.0", | ||||||
| 		"svelte": "3.35.0", | 		"svelte-select": "3.17.0", | ||||||
| 		"svelte-preprocess": "4.6.9", | 		"tailwindcss": "3.2.7", | ||||||
| 		"workbox-cli": "6.1.2" | 		"vite": "2.1.5" | ||||||
| 	}, | 	}, | ||||||
| 	"release-it": { | 	"release-it": { | ||||||
| 		"git": { | 		"git": { | ||||||
| @@ -55,7 +37,27 @@ | |||||||
| 			"publish": false | 			"publish": false | ||||||
| 		}, | 		}, | ||||||
| 		"hooks": { | 		"hooks": { | ||||||
| 			"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.template.html && node order.js  && git add src/locales" | 			"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.html && node order.js  && git add src/locales" | ||||||
| 		} | 		} | ||||||
|  | 	}, | ||||||
|  | 	"dependencies": { | ||||||
|  | 		"tinro": "0.6.12", | ||||||
|  | 		"toastify-js": "1.12.0", | ||||||
|  | 		"validator": "13.9.0", | ||||||
|  | 		"xlsx": "0.16.9", | ||||||
|  | 		"@odit/lfk-client-js": "0.14.0", | ||||||
|  | 		"@vincjo/datatables": "^1.4.0", | ||||||
|  | 		"check-password-strength": "2.0.7", | ||||||
|  | 		"csvtojson": "2.0.10", | ||||||
|  | 		"gridjs": "3.4.0", | ||||||
|  | 		"localforage": "1.10.0", | ||||||
|  | 		"marked": "2.0.3", | ||||||
|  | 		"svelte": "3.37.0", | ||||||
|  | 		"svelte-focus-trap": "1.2.0", | ||||||
|  | 		"svelte-i18n": "3.3.9", | ||||||
|  | 		"@paralleldrive/cuid2": "^2.2.0" | ||||||
|  | 	}, | ||||||
|  | 	"volta": { | ||||||
|  | 		"node": "19.7.0" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										2996
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										2996
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | module.exports = { | ||||||
|  | 	plugins: { | ||||||
|  | 		tailwindcss: {}, | ||||||
|  | 		autoprefixer: {} | ||||||
|  | 	} | ||||||
|  | }; | ||||||
| @@ -1,6 +1,9 @@ | |||||||
| const config = { | const config = { | ||||||
| 	baseurl: 'http://localhost:4010', | 	baseurl: 'http://localhost:4010', | ||||||
| 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe' | 	baseurl_documentserver: 'http://localhost:4010/documents', | ||||||
|  | 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe', | ||||||
| 	// optional | 	// optional | ||||||
|  | 	default_username: 'demo', | ||||||
|  | 	default_password: 'demo', | ||||||
| 	prefersHashRouting: true | 	prefersHashRouting: true | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| Nostrud tempor dolor aute ea excepteur aute mollit elit eiusmod exercitation. Magna laborum pariatur adipisicing pariatur cupidatat exercitation duis aliquip pariatur sint exercitation deserunt labore. Consectetur id laboris dolore nostrud do velit ipsum. Eu laboris velit do commodo ad ea sint ex cillum. Cillum ipsum qui eiusmod laborum mollit sunt dolore incididunt. Cillum sunt culpa veniam voluptate et qui ut magna anim occaecat ut mollit dolor. Duis irure proident eu incididunt dolore sunt nisi aute dolore amet eu fugiat laboris quis. |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,6 +0,0 @@ | |||||||
| const sveltePreprocess = require('svelte-preprocess'); |  | ||||||
| const preprocess = sveltePreprocess(__insert__); |  | ||||||
|  |  | ||||||
| module.exports = { |  | ||||||
| 	preprocess |  | ||||||
| }; |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| /** @type {import("snowpack").SnowpackUserConfig } */ |  | ||||||
| module.exports = { |  | ||||||
| 	mount: { |  | ||||||
| 		public: '/', |  | ||||||
| 		src: '/_dist_' |  | ||||||
| 	}, |  | ||||||
| 	plugins: [ '@snowpack/plugin-svelte' ], |  | ||||||
| 	routes: [ |  | ||||||
| 		/* Enable an SPA Fallback in development: */ |  | ||||||
| 		{ match: 'routes', src: '.*', dest: '/index.html' } |  | ||||||
| 	], |  | ||||||
| 	packageOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 		sourceMap: false |  | ||||||
| 	}, |  | ||||||
| 	devOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	buildOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	alias: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	optimize: { bundle: true, minify: true } |  | ||||||
| }; |  | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| <script> | <script> | ||||||
|   import "./TailwindStyles.svelte"; |  | ||||||
|   import "toastify-js/src/toastify.css"; |   import "toastify-js/src/toastify.css"; | ||||||
|   import "gridjs/dist/theme/mermaid.css"; |   import "gridjs/dist/theme/mermaid.css"; | ||||||
|   import { Route, router } from "tinro"; |   import { Route, router } from "tinro"; | ||||||
| @@ -53,7 +52,6 @@ | |||||||
|   import { OpenAPI } from "@odit/lfk-client-js"; |   import { OpenAPI } from "@odit/lfk-client-js"; | ||||||
|   import UserDetail from "./components/users/UserDetail.svelte"; |   import UserDetail from "./components/users/UserDetail.svelte"; | ||||||
|   OpenAPI.BASE = config.baseurl; |   OpenAPI.BASE = config.baseurl; | ||||||
|   import { register as registerSW } from "./swmodule"; |  | ||||||
|   import TeamDetail from "./components/teams/TeamDetail.svelte"; |   import TeamDetail from "./components/teams/TeamDetail.svelte"; | ||||||
|   import UserPermissions from "./components/users/UserPermissions.svelte"; |   import UserPermissions from "./components/users/UserPermissions.svelte"; | ||||||
|   import GroupPermissions from "./components/groups/GroupPermissions.svelte"; |   import GroupPermissions from "./components/groups/GroupPermissions.svelte"; | ||||||
| @@ -69,13 +67,14 @@ | |||||||
|   import Donations from "./components/donations/Donations.svelte"; |   import Donations from "./components/donations/Donations.svelte"; | ||||||
|   import DonationDetail from "./components/donations/DonationDetail.svelte"; |   import DonationDetail from "./components/donations/DonationDetail.svelte"; | ||||||
|   import GroupDetail from "./components/groups/GroupDetail.svelte"; |   import GroupDetail from "./components/groups/GroupDetail.svelte"; | ||||||
|   import ScanStationsOverview from "./components/scanstations/ScanStationsOverview.svelte"; |  | ||||||
|   import ScanStations from "./components/scanstations/ScanStations.svelte"; |   import ScanStations from "./components/scanstations/ScanStations.svelte"; | ||||||
|   import ScanStationDetail from "./components/scanstations/ScanStationDetail.svelte"; |   import ScanStationDetail from "./components/scanstations/ScanStationDetail.svelte"; | ||||||
|   import Scans from "./components/scans/Scans.svelte"; |   import Scans from "./components/scans/Scans.svelte"; | ||||||
| import ScanDetail from "./components/scans/ScanDetail.svelte"; |   import ScanDetail from "./components/scans/ScanDetail.svelte"; | ||||||
|  |   import Cards from "./components/cards/Cards.svelte"; | ||||||
|  | 	import StatsClients from "./components/statsclients/StatsClients.svelte"; | ||||||
|  | 	import StatsClientDetail from "./components/statsclients/StatsClientDetail.svelte"; | ||||||
|   store.init(); |   store.init(); | ||||||
|   registerSW(); |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <Route> | <Route> | ||||||
| @@ -185,6 +184,14 @@ import ScanDetail from "./components/scans/ScanDetail.svelte"; | |||||||
|             <DonationDetail {params} /> |             <DonationDetail {params} /> | ||||||
|           </Route> |           </Route> | ||||||
|         </Route> |         </Route> | ||||||
|  |         <Route path="/cards/*"> | ||||||
|  |           <Route path="/"> | ||||||
|  |             <Cards /> | ||||||
|  |           </Route> | ||||||
|  |           <!-- <Route path="/:scanid" let:params> | ||||||
|  |             <ScanDetail {params} /> | ||||||
|  |           </Route> --> | ||||||
|  |         </Route> | ||||||
|         <Route path="/scans/*"> |         <Route path="/scans/*"> | ||||||
|           <Route path="/"> |           <Route path="/"> | ||||||
|             <Scans /> |             <Scans /> | ||||||
| @@ -201,6 +208,14 @@ import ScanDetail from "./components/scans/ScanDetail.svelte"; | |||||||
|             <ScanStationDetail {params} /> |             <ScanStationDetail {params} /> | ||||||
|           </Route> |           </Route> | ||||||
|         </Route> |         </Route> | ||||||
|  |         <Route path="/statsclients/*"> | ||||||
|  |           <Route path="/"> | ||||||
|  |             <StatsClients /> | ||||||
|  |           </Route> | ||||||
|  |           <Route path="/:clientid" let:params> | ||||||
|  |             <StatsClientDetail {params} /> | ||||||
|  |           </Route> | ||||||
|  |         </Route> | ||||||
|         <Route path="/about"> |         <Route path="/about"> | ||||||
|           <About /> |           <About /> | ||||||
|         </Route> |         </Route> | ||||||
|   | |||||||
| @@ -1,6 +0,0 @@ | |||||||
| <style global> |  | ||||||
|   /*! @import */ |  | ||||||
|   @tailwind base; |  | ||||||
|   @tailwind components; |  | ||||||
|   @tailwind utilities; |  | ||||||
| </style> |  | ||||||
| @@ -5,10 +5,11 @@ | |||||||
|   store.init(); |   store.init(); | ||||||
|   import { OpenAPI, AuthService } from "@odit/lfk-client-js"; |   import { OpenAPI, AuthService } from "@odit/lfk-client-js"; | ||||||
|   import Footer from "../general/Footer.svelte"; |   import Footer from "../general/Footer.svelte"; | ||||||
|  |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   // ------ |   // ------ | ||||||
|   let username = "demo"; |   let username = config.default_username || ""; | ||||||
|   let password = "demo"; |   let password = config.default_password || ""; | ||||||
|   let is_blocked_by_autologin = false; |   let is_blocked_by_autologin = false; | ||||||
|   let last_loginclick_processed = true; |   let last_loginclick_processed = true; | ||||||
|  |  | ||||||
| @@ -36,10 +37,19 @@ | |||||||
|         text: $_("login_is_checked"), |         text: $_("login_is_checked"), | ||||||
|         duration: 500, |         duration: 500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       AuthService.authControllerLogin({ |       let postdata = {}; | ||||||
|         username, |       if (isEmail(username)) { | ||||||
|         password, |         postdata = { | ||||||
|       }) |           email: username, | ||||||
|  |           password, | ||||||
|  |         }; | ||||||
|  |       } else { | ||||||
|  |         postdata = { | ||||||
|  |           username, | ||||||
|  |           password, | ||||||
|  |         }; | ||||||
|  |       } | ||||||
|  |       AuthService.authControllerLogin(postdata) | ||||||
|         .then(async (result) => { |         .then(async (result) => { | ||||||
|           await localForage.setItem("logindata", result); |           await localForage.setItem("logindata", result); | ||||||
|           OpenAPI.TOKEN = result.access_token; |           OpenAPI.TOKEN = result.access_token; | ||||||
| @@ -66,7 +76,7 @@ | |||||||
|       // last login was not processed yet |       // last login was not processed yet | ||||||
|     } else { |     } else { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: "chill...", |         text: $_('please-wait-a-moment-your-login-is-still-being-processed'), | ||||||
|         duration: 1500, |         duration: 1500, | ||||||
|         backgroundColor: |         backgroundColor: | ||||||
|           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								src/components/auth/PasswordStrength.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/components/auth/PasswordStrength.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | <script context="module"> | ||||||
|  |     import { passwordStrength } from "check-password-strength"; | ||||||
|  |     export function password_strong_enough(password_change) { | ||||||
|  |         let strength = passwordStrength(password_change); | ||||||
|  |         return ( | ||||||
|  |             strength?.contains.includes("lowercase") && | ||||||
|  |             strength?.contains.includes("uppercase") && | ||||||
|  |             strength?.contains.includes("number") && | ||||||
|  |             strength?.length > 9 | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |     export function password_strong_enough_and_equal( | ||||||
|  |         password_change, | ||||||
|  |         password_confirm | ||||||
|  |     ) { | ||||||
|  |         return ( | ||||||
|  |             password_strong_enough(password_change) && | ||||||
|  |             password_change === password_confirm | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  |     import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |     import { passwordStrength as Strength } from "check-password-strength"; | ||||||
|  |     export let password_change; | ||||||
|  |     export let password_confirm; | ||||||
|  |  | ||||||
|  |     $: strength = Strength(password_change); | ||||||
|  |     $: passwords_match = | ||||||
|  |         !password_confirm || password_confirm === password_change; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="ml-4"> | ||||||
|  |     <ul class="list-disc font-medium tracking-wide text-red-500 text-xs"> | ||||||
|  |         {#if !strength.contains.includes('lowercase')} | ||||||
|  |             <li>{$_('must-contain-a-lowercase-letter')}</li> | ||||||
|  |         {/if} | ||||||
|  |         {#if !strength.contains.includes('uppercase')} | ||||||
|  |             <li>{$_('must-contain-a-uppercase-letter')}</li> | ||||||
|  |         {/if} | ||||||
|  |         {#if !strength.contains.includes('number')} | ||||||
|  |             <li>{$_('must-contain-a-number')}</li> | ||||||
|  |         {/if} | ||||||
|  |         {#if !(strength.length > 9)} | ||||||
|  |             <li>{$_('must-be-at-least-10-characters-long')}</li> | ||||||
|  |         {/if} | ||||||
|  |         {#if !(passwords_match == true)} | ||||||
|  |             <li>{$_('passwords-dont-match')}</li> | ||||||
|  |         {/if} | ||||||
|  |     </ul> | ||||||
|  | </div> | ||||||
| @@ -1,38 +1,43 @@ | |||||||
| <script> | <script> | ||||||
|   import {  AuthService } from "@odit/lfk-client-js"; |   import { AuthService } from "@odit/lfk-client-js"; | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import "toastify-js/src/toastify.css"; |   import "toastify-js/src/toastify.css"; | ||||||
|  |   import PasswordStrength, { | ||||||
|  |     password_strong_enough, | ||||||
|  |   } from "../auth/PasswordStrength.svelte"; | ||||||
|   let state = "reset_in_progress"; |   let state = "reset_in_progress"; | ||||||
|   let password = ""; |   let password = ""; | ||||||
|   export let params; |   export let params; | ||||||
|   function set_new_password() { |   function set_new_password() { | ||||||
|       if(password.trim() !== ""){ |     if (password.trim() !== "") { | ||||||
|         Toastify({ |       Toastify({ | ||||||
|           text: $_('password-reset-in-progress'), |         text: $_("password-reset-in-progress"), | ||||||
|           duration: 3500, |         duration: 3500, | ||||||
|         }).showToast(); |       }).showToast(); | ||||||
|         AuthService.authControllerResetPassword(atob(params.resetkey),{ password }) |       AuthService.authControllerResetPassword(atob(params.resetkey), { | ||||||
|  |         password, | ||||||
|  |       }) | ||||||
|         .then((resp) => { |         .then((resp) => { | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: $_('password-reset-successful'), |             text: $_("password-reset-successful"), | ||||||
|             duration: 3500, |             duration: 3500, | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           state="reset_success"; |           state = "reset_success"; | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           state="reset_error"; |           state = "reset_error"; | ||||||
|         }); |         }); | ||||||
|       } else { |     } else { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: $_('please-provide-a-password'), |         text: $_("please-provide-a-password"), | ||||||
|         duration: 3500, |         duration: 3500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if state==="reset_success"} | {#if state === 'reset_success'} | ||||||
|   <div class="min-h-screen flex items-center justify-center bg-gray-100"> |   <div class="min-h-screen flex items-center justify-center bg-gray-100"> | ||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
| @@ -56,31 +61,31 @@ | |||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| {:else if state==="reset_error"} | {:else if state === 'reset_error'} | ||||||
| <div class="min-h-screen flex items-center justify-center bg-gray-100"> |   <div class="min-h-screen flex items-center justify-center bg-gray-100"> | ||||||
|   <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|     <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|     <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|       {$_('application_name')} |         {$_('application_name')} | ||||||
|     </p> |       </p> | ||||||
|     <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> | ||||||
|       {$_('password-reset-failed')} |         {$_('password-reset-failed')} | ||||||
|     </p> |       </p> | ||||||
|     <p class="mt-2 mb-2 text-sm text-center text-gray-900"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> | ||||||
|       {$_('please-request-a-new-reset-mail')} |         {$_('please-request-a-new-reset-mail')} | ||||||
|     </p> |       </p> | ||||||
|     <div class="mt-6"> |  | ||||||
|       <div class="mt-6"> |       <div class="mt-6"> | ||||||
|         <a |         <div class="mt-6"> | ||||||
|           href="/forgot_password/" |           <a | ||||||
|           class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             href="/forgot_password/" | ||||||
|           {$_('request-a-new-reset-mail')} |             class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> | ||||||
|         </a> |             {$_('request-a-new-reset-mail')} | ||||||
|  |           </a> | ||||||
|  |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </div> | {:else if state === 'reset_in_progress'} | ||||||
| {:else if state==="reset_in_progress"} |  | ||||||
|   <div class="min-h-screen flex items-center justify-center bg-gray-100"> |   <div class="min-h-screen flex items-center justify-center bg-gray-100"> | ||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
| @@ -102,11 +107,14 @@ | |||||||
|               placeholder={$_('new-password')} |               placeholder={$_('new-password')} | ||||||
|               bind:value={password} /> |               bind:value={password} /> | ||||||
|           </div> |           </div> | ||||||
|  |           <PasswordStrength bind:password_change={password} /> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <div class="mt-5"> |         <div class="mt-5"> | ||||||
|           <button |           <button | ||||||
|             on:click={set_new_password} |             on:click={set_new_password} | ||||||
|  |             disabled={!password_strong_enough(password)} | ||||||
|  |             class:opacity-50={!password_strong_enough(password)} | ||||||
|             type="submit" |             type="submit" | ||||||
|             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> | ||||||
|             <span class="absolute left-0 inset-y pl-3"> |             <span class="absolute left-0 inset-y pl-3"> | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								src/components/base/importfixes.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/components/base/importfixes.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | <!-- | ||||||
|  |     Temporary tailwind import fixes for classes that wouldn't be directly used otherwise.  | ||||||
|  |     Or as others may call it: Real big bullshit time. | ||||||
|  |     Issue: https://git.odit.services/lfk/frontend/issues/136 | ||||||
|  |  --> | ||||||
|  | <div class="opacity-50"></div> | ||||||
							
								
								
									
										240
									
								
								src/components/cards/AddCardBulkModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										240
									
								
								src/components/cards/AddCardBulkModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,240 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   export let bulk_modal_open; | ||||||
|  |   export let current_cards; | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: card_count = 0; | ||||||
|  |   $: is_card_count_valid = card_count > 0; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   $: createbtnenabled = is_card_count_valid; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         bulk_modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit_with_print(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit_without_print() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_("creating-blanco-cards"), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, false) | ||||||
|  |         .then((result) => { | ||||||
|  |           bulk_modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("created-blanco-cards"), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function submit_with_print() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_("creating-blanco-cards"), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true) | ||||||
|  |         .then((result) => { | ||||||
|  |           bulk_modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("created-blanco-cards"), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_cards = current_cards.concat(result); | ||||||
|  |           const toast = Toastify({ | ||||||
|  |             text: $_("generating-pdf"), | ||||||
|  |             duration: -1, | ||||||
|  |           }).showToast(); | ||||||
|  |           fetch( | ||||||
|  |             `${config.baseurl_documentserver}/cards?&download=true&key=${config.documentserver_key}`, | ||||||
|  |             { | ||||||
|  |               method: "POST", | ||||||
|  |               headers: { | ||||||
|  |                 "Content-Type": "application/json", | ||||||
|  |               }, | ||||||
|  |               body: JSON.stringify(result), | ||||||
|  |             } | ||||||
|  |           ) | ||||||
|  |             .then((response) => { | ||||||
|  |               if (response.status != "200") { | ||||||
|  |                 toast.hideToast(); | ||||||
|  |                 Toastify({ | ||||||
|  |                   text: $_("pdf-generation-failed"), | ||||||
|  |                   duration: 3500, | ||||||
|  |                   backgroundColor: | ||||||
|  |                     "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |                 }).showToast(); | ||||||
|  |               } else { | ||||||
|  |                 return response.blob(); | ||||||
|  |               } | ||||||
|  |             }) | ||||||
|  |             .then((blob) => { | ||||||
|  |               const url = window.URL.createObjectURL(blob); | ||||||
|  |               let a = document.createElement("a"); | ||||||
|  |               a.href = url; | ||||||
|  |               a.download = "Bulkcards.pdf"; | ||||||
|  |               document.body.appendChild(a); | ||||||
|  |               a.click(); | ||||||
|  |               a.remove(); | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdf-successfully-generated"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |               }).showToast(); | ||||||
|  |             }) | ||||||
|  |             .catch((err) => { | ||||||
|  |               console.error(err); | ||||||
|  |             }); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if bulk_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       bulk_modal_open = false; | ||||||
|  |     }}> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-2xl sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w- rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   fill="currentColor" | ||||||
|  |                   d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" /></svg> | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('create-bulk-blanco-cards')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_('just-enter-how-many-you-want-and-the-system-will-create-them')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="amount" | ||||||
|  |                     class="block text-sm font-medium text-gray-700">{$_('amount')}</label> | ||||||
|  |                   <div class="mt-1 flex rounded-md shadow-sm"> | ||||||
|  |                     <input | ||||||
|  |                       autocomplete="off" | ||||||
|  |                       class:border-red-500={!is_card_count_valid} | ||||||
|  |                       class:focus:border-red-500={!is_card_count_valid} | ||||||
|  |                       class:focus:ring-red-500={!is_card_count_valid} | ||||||
|  |                       bind:value={card_count} | ||||||
|  |                       type="number" | ||||||
|  |                       step="1" | ||||||
|  |                       name="amount" | ||||||
|  |                       class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 p-2" | ||||||
|  |                       placeholder="400" /> | ||||||
|  |                     <span | ||||||
|  |                       class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500 text-sm">{$_('cards')}</span> | ||||||
|  |                   </div> | ||||||
|  |                   {#if !is_card_count_valid} | ||||||
|  |                     <span | ||||||
|  |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |                       {$_('you-must-create-at-least-one-card-or-cancel')} | ||||||
|  |                     </span> | ||||||
|  |                   {/if} | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit_with_print} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('create-and-generate-pdf')} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit_without_print} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-400 text-base font-medium text-white hover:bg-gray-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('create-without-pdf')} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               bulk_modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="mr-auto mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('cancel')} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										204
									
								
								src/components/cards/AddCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								src/components/cards/AddCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,204 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { | ||||||
|  |     RunnerCardService, | ||||||
|  |     RunnerService, | ||||||
|  |     ScanService, | ||||||
|  |   } from "@odit/lfk-client-js"; | ||||||
|  |   import Select from "svelte-select"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let current_cards; | ||||||
|  |  | ||||||
|  |   const getRunnerLabel = (option) => { | ||||||
|  |     if (option.middlename) { | ||||||
|  |       return option.firstname + " " + option.middlename + " " + option.lastname; | ||||||
|  |     } | ||||||
|  |     return option.firstname + " " + option.lastname; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const filterRunners = (label, filterText, option) => { | ||||||
|  |     if (filterText.startsWith("#")) { | ||||||
|  |       return option.value.id == parseInt(filterText.replace("#","")) | ||||||
|  |     } | ||||||
|  |     return ( | ||||||
|  |       label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|  |       option.value.toString().startsWith(filterText.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: runner = 0; | ||||||
|  |   $: enabled = true; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |  | ||||||
|  |   let loading = true; | ||||||
|  |   let runners = []; | ||||||
|  |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|  |     runners = val.map((r) => { | ||||||
|  |       return { label: getRunnerLabel(r), value: r }; | ||||||
|  |     }); | ||||||
|  |     loading = false; | ||||||
|  |   }); | ||||||
|  |   $: createbtnenabled = true; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_("adding-card"), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |       let postdata = { | ||||||
|  |         runner, | ||||||
|  |         enabled, | ||||||
|  |       }; | ||||||
|  |       RunnerCardService.runnerCardControllerPost(postdata) | ||||||
|  |         .then((result) => { | ||||||
|  |           runner = 0; | ||||||
|  |           modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("card-added"), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_cards.push(result); | ||||||
|  |           current_cards = current_cards; | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline" | ||||||
|  |       > | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   fill="currentColor" | ||||||
|  |                   d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("create-a-new-card")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_("you-can-provide-a-runner-but-you-dont-have-to")} | ||||||
|  |                   {$_( | ||||||
|  |                     "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button" | ||||||
|  |                   )} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="donor" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("runner")}</label | ||||||
|  |                   > | ||||||
|  |                   <Select | ||||||
|  |                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |                     itemFilter={(label, filterText, option) => | ||||||
|  |                       filterRunners(label, filterText, option)} | ||||||
|  |                     items={runners} | ||||||
|  |                     bind:loading | ||||||
|  |                     showChevron={!loading} | ||||||
|  |                     placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|  |                     noOptionsMessage={$_("no-runners-found")} | ||||||
|  |                     on:select={(selectedValue) => | ||||||
|  |                       (runner = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (runner = null)} | ||||||
|  |                   /> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm" | ||||||
|  |           > | ||||||
|  |             {$_("create")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										192
									
								
								src/components/cards/CardDetailModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								src/components/cards/CardDetailModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { RunnerCardService, RunnerService } from "@odit/lfk-client-js"; | ||||||
|  |   import Select from "svelte-select"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   export let edit_modal_open; | ||||||
|  |   export let current_cards; | ||||||
|  |   export let runner = {}; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   const getRunnerLabel = (option) => | ||||||
|  |     option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
|  |   const filterRunners = (label, filterText, option) => | ||||||
|  |     label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|  |     option.value.toString().startsWith(filterText.toLowerCase()); | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: runners = []; | ||||||
|  |   $: enabled = true; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   function dataUpdated() { | ||||||
|  | 		dispatch('dataUpdated',); | ||||||
|  | 	} | ||||||
|  |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|  |     runners = val.map((r) => { | ||||||
|  |       return { label: getRunnerLabel(r), value: r }; | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |   $: createbtnenabled = !( | ||||||
|  |     JSON.stringify(editable) === JSON.stringify(original_data) | ||||||
|  |   ); | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         edit_modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_("updating-card"), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |       RunnerCardService.runnerCardControllerPut(original_data.id, editable) | ||||||
|  |         .then((result) => { | ||||||
|  |           let id = original_data.id; | ||||||
|  |           runner = {}; | ||||||
|  |           editable = {}; | ||||||
|  |           original_data = {}; | ||||||
|  |           edit_modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("card-updated"), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_cards[current_cards.findIndex((c) => c.id === id)] = result; | ||||||
|  |           current_cards = current_cards; | ||||||
|  |           dataUpdated(); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if edit_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       edit_modal_open = false; | ||||||
|  |     }}> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   fill="currentColor" | ||||||
|  |                   d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" /></svg> | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('edit-a-card')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_('you-can-provide-a-runner-but-you-dont-have-to')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="runner" | ||||||
|  |                     class="block text-sm font-medium text-gray-700">{$_('runner')}</label> | ||||||
|  |                   <Select | ||||||
|  |                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |                     itemFilter={(label, filterText, option) => filterRunners(label, filterText, option)} | ||||||
|  |                     items={runners} | ||||||
|  |                     showChevron={true} | ||||||
|  |                     placeholder={$_('search-for-runner-by-name-or-id')} | ||||||
|  |                     noOptionsMessage={$_('no-runners-found')} | ||||||
|  |                     bind:selectedValue={runner} | ||||||
|  |                     on:select={(selectedValue) => (editable.runner = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (editable.runner = null)} /> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <p class="text-gray-500"> | ||||||
|  |                     <input | ||||||
|  |                       id="enabled" | ||||||
|  |                       on:change={() => { | ||||||
|  |                         editable.enabled = !editable.enabled; | ||||||
|  |                       }} | ||||||
|  |                       name="enabled" | ||||||
|  |                       type="checkbox" | ||||||
|  |                       checked={editable.enabled} | ||||||
|  |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | ||||||
|  |                     {$_('this-card-is')} | ||||||
|  |                     {#if editable.enabled} | ||||||
|  |                       {$_('enabled')} | ||||||
|  |                     {:else}{$_('disabled')}{/if} | ||||||
|  |                   </p> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('save-changes')} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               edit_modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('cancel')} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										40
									
								
								src/components/cards/Cards.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/components/cards/Cards.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import AddCardBulkModal from "./AddCardBulkModal.svelte"; | ||||||
|  |   import AddCardModal from "./AddCardModal.svelte"; | ||||||
|  |   import CardsOverview from "./CardsOverview.svelte"; | ||||||
|  |   $: current_cards = []; | ||||||
|  |   export let modal_open = false; | ||||||
|  |   export let bulk_modal_open = false; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <section class="container p-5"> | ||||||
|  |   <span class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|  |     {$_('cards')} | ||||||
|  |     {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:CREATE')} | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           modal_open = true; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |         {$_('add-card')} | ||||||
|  |       </button> | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           bulk_modal_open = true; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |         {$_('create-bulk-cards')} | ||||||
|  |       </button> | ||||||
|  |     {/if} | ||||||
|  |   </span> | ||||||
|  |   <CardsOverview bind:current_cards /> | ||||||
|  | </section> | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:CREATE')} | ||||||
|  |   <AddCardModal bind:current_cards bind:modal_open /> | ||||||
|  |   <AddCardBulkModal bind:current_cards bind:bulk_modal_open /> | ||||||
|  | {/if} | ||||||
							
								
								
									
										12
									
								
								src/components/cards/CardsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/components/cards/CardsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import cards_empty from "./cards.svg"; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="text-center items-center justify-center"> | ||||||
|  |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|  |     <img class="m-auto" style="height:15rem" src={cards_empty} alt="" /> | ||||||
|  |     <span class="font-bold">{$_('there-are-no-cards-yet')}</span><br /> | ||||||
|  |     <span>{$_('add-your-first-card')}</span> | ||||||
|  |   </p> | ||||||
|  | </div> | ||||||
							
								
								
									
										233
									
								
								src/components/cards/CardsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										233
									
								
								src/components/cards/CardsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,233 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { DataHandler, Datatable, Th, ThFilter } from "@vincjo/datatables"; | ||||||
|  |   import CardsEmptyState from "./CardsEmptyState.svelte"; | ||||||
|  |   import CardDetailModal from "./CardDetailModal.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import ThFilterStatus from "./ThFilterStatus.svelte"; | ||||||
|  |   import ThFilterRunner from "./ThFilterRunner.svelte"; | ||||||
|  |   export let edit_modal_open = false; | ||||||
|  |   export let runner = {}; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   export let current_cards = []; | ||||||
|  |   const handler = new DataHandler(current_cards, { rowsPerPage: 50 }); | ||||||
|  |   const rows = handler.getRows(); | ||||||
|  |   $: active_deletes = []; | ||||||
|  |   $: cards_show = generate_cards.length > 0; | ||||||
|  |   $: generate_cards = []; | ||||||
|  |   const cards_promise = RunnerCardService.runnerCardControllerGetAll().then( | ||||||
|  |     (val) => { | ||||||
|  |       current_cards = val; | ||||||
|  |       handler.setRows(val); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |   const getRunnerLabel = (option) => | ||||||
|  |     option?.firstname + | ||||||
|  |     " " + | ||||||
|  |     (option?.middlename || "") + | ||||||
|  |     " " + | ||||||
|  |     (option?.lastname || "{$_('non-blanko')}"); | ||||||
|  |   function open_edit_modal(card) { | ||||||
|  |     if (card.runner?.id) { | ||||||
|  |       runner = Object.assign( | ||||||
|  |         { runner }, | ||||||
|  |         { label: getRunnerLabel(card.runner), value: card.runner } | ||||||
|  |       ); | ||||||
|  |       card.runner = card.runner.id; | ||||||
|  |     } else { | ||||||
|  |       card.runner = null; | ||||||
|  |       runner = null; | ||||||
|  |     } | ||||||
|  |     editable = Object.assign(editable, card); | ||||||
|  |     original_data = Object.assign(original_data, card); | ||||||
|  |     edit_modal_open = true; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:UPDATE")} | ||||||
|  |   <CardDetailModal | ||||||
|  |     bind:current_cards | ||||||
|  |     bind:edit_modal_open | ||||||
|  |     bind:runner | ||||||
|  |     bind:editable | ||||||
|  |     bind:original_data | ||||||
|  |     on:dataUpdated={handler.setRows(current_cards)} | ||||||
|  |   /> | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")} | ||||||
|  |   {#await cards_promise} | ||||||
|  |     <div | ||||||
|  |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|  |       role="alert" | ||||||
|  |     > | ||||||
|  |       <p class="font-bold">{$_("loading-cards")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|  |     </div> | ||||||
|  |   {:then} | ||||||
|  |     {#if current_cards.length === 0} | ||||||
|  |       <CardsEmptyState /> | ||||||
|  |     {:else} | ||||||
|  |       <div class="h-12"> | ||||||
|  |         <GenerateRunnerCards bind:cards_show bind:generate_cards /> | ||||||
|  |       </div> | ||||||
|  |       <Datatable {handler}> | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th style="border-bottom: 1px solid #ddd;"> | ||||||
|  |                 <input | ||||||
|  |                   type="checkbox" | ||||||
|  |                   class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                   checked={generate_cards.length == current_cards.length} | ||||||
|  |                   on:click={() => { | ||||||
|  |                     if (generate_cards.length != current_cards.length) { | ||||||
|  |                       generate_cards = current_cards; | ||||||
|  |                     } else { | ||||||
|  |                       generate_cards = []; | ||||||
|  |                     } | ||||||
|  |                   }} | ||||||
|  |                 /> | ||||||
|  |               </th> | ||||||
|  |               <Th {handler} orderBy="code">{$_("code")}</Th> | ||||||
|  |               <Th {handler} orderBy="runner">{$_("runner")}</Th> | ||||||
|  |               <Th {handler} orderBy="status">{$_("status")}</Th> | ||||||
|  |               <th style="border-bottom: 1px solid #ddd;">{$_("action")}</th> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|  |               <ThFilter {handler} filterBy="code" /> | ||||||
|  |               <ThFilterRunner {handler} /> | ||||||
|  |               <ThFilterStatus {handler} /> | ||||||
|  |               <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             {#each $rows as row} | ||||||
|  |               <tr> | ||||||
|  |                 <td> | ||||||
|  |                   <input | ||||||
|  |                     type="checkbox" | ||||||
|  |                     class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                     checked={generate_cards.filter((i) => i.id == row.id) | ||||||
|  |                       .length > 0} | ||||||
|  |                     on:click={() => { | ||||||
|  |                       if ( | ||||||
|  |                         generate_cards.findIndex((i) => i.id == row.id) == -1 | ||||||
|  |                       ) { | ||||||
|  |                         generate_cards.push(row); | ||||||
|  |                         generate_cards = generate_cards; | ||||||
|  |                       } else { | ||||||
|  |                         generate_cards = generate_cards.filter( | ||||||
|  |                           (r) => r.id != row.id | ||||||
|  |                         ); | ||||||
|  |                       } | ||||||
|  |                       console.log(generate_cards); | ||||||
|  |                     }} | ||||||
|  |                   /> | ||||||
|  |                 </td> | ||||||
|  |                 <td>{row.code}</td> | ||||||
|  |                 <td> | ||||||
|  |                   {#if row.runner} | ||||||
|  |                     <a | ||||||
|  |                       href="../runners/{row.runner.id}" | ||||||
|  |                       class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800" | ||||||
|  |                       >{row.runner.firstname} | ||||||
|  |                       {row.runner.middlename || ""} | ||||||
|  |                       {row.runner.lastname}</a | ||||||
|  |                     > | ||||||
|  |                   {:else}{$_("non-blanko")}{/if} | ||||||
|  |                 </td> | ||||||
|  |                 <td> | ||||||
|  |                   {#if row.enabled} | ||||||
|  |                     <span | ||||||
|  |                       class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800" | ||||||
|  |                       >{$_("enabled")}</span | ||||||
|  |                     > | ||||||
|  |                   {:else} | ||||||
|  |                     <span | ||||||
|  |                       class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800" | ||||||
|  |                       >{$_("disabled")}</span | ||||||
|  |                     > | ||||||
|  |                   {/if} | ||||||
|  |                 </td> | ||||||
|  |                 <td> | ||||||
|  |                   {#if active_deletes[row.id] === true} | ||||||
|  |                     <button | ||||||
|  |                       on:click={() => { | ||||||
|  |                         active_deletes[row.id] = false; | ||||||
|  |                       }} | ||||||
|  |                       tabindex="0" | ||||||
|  |                       class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer" | ||||||
|  |                       >{$_("cancel-delete")}</button | ||||||
|  |                     > | ||||||
|  |                     <button | ||||||
|  |                       on:click={() => { | ||||||
|  |                         RunnerCardService.runnerCardControllerRemove( | ||||||
|  |                           row.id, | ||||||
|  |                           true | ||||||
|  |                         ) | ||||||
|  |                           .then((resp) => { | ||||||
|  |                             current_cards = current_cards.filter( | ||||||
|  |                               (obj) => obj.id !== row.id | ||||||
|  |                             ); | ||||||
|  |                           }) | ||||||
|  |                           .catch((err) => {}); | ||||||
|  |                       }} | ||||||
|  |                       tabindex="0" | ||||||
|  |                       class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                       >{$_("confirm-delete")}</button | ||||||
|  |                     > | ||||||
|  |                   {:else} | ||||||
|  |                     <button | ||||||
|  |                       on:click={() => { | ||||||
|  |                         open_edit_modal(row); | ||||||
|  |                       }} | ||||||
|  |                       class="text-indigo-600 hover:text-indigo-900" | ||||||
|  |                       >{$_("details")}</button | ||||||
|  |                     > | ||||||
|  |                     {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:DELETE")} | ||||||
|  |                       <button | ||||||
|  |                         on:click={() => { | ||||||
|  |                           active_deletes[row.id] = true; | ||||||
|  |                         }} | ||||||
|  |                         tabindex="0" | ||||||
|  |                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                         >{$_("delete")}</button | ||||||
|  |                       > | ||||||
|  |                     {/if} | ||||||
|  |                   {/if} | ||||||
|  |                 </td> | ||||||
|  |               </tr> | ||||||
|  |             {/each} | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |       </Datatable> | ||||||
|  |     {/if} | ||||||
|  |   {:catch error} | ||||||
|  |     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | ||||||
|  |       <span class="inline-block align-middle mr-8"> | ||||||
|  |         <b class="capitalize">{$_("general_promise_error")}</b> | ||||||
|  |         {error} | ||||||
|  |       </span> | ||||||
|  |     </div> | ||||||
|  |   {/await} | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   table tbody { | ||||||
|  |     display: block; | ||||||
|  |     overflow-y: scroll; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   table thead, | ||||||
|  |   table tbody tr { | ||||||
|  |     display: table; | ||||||
|  |     width: 100%; | ||||||
|  |     table-layout: fixed; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
							
								
								
									
										57
									
								
								src/components/cards/ThFilterRunner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/components/cards/ThFilterRunner.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | <script> | ||||||
|  |   export let handler; | ||||||
|  |   let filterValue = ""; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th> | ||||||
|  |   <input | ||||||
|  |     on:input={() => { | ||||||
|  |       setTimeout(() => { | ||||||
|  |         const v = filterValue.toLowerCase(); | ||||||
|  |         handler.filter(v, (c) => { | ||||||
|  |           // if (v === "") { | ||||||
|  |           //   return c; | ||||||
|  |           // } | ||||||
|  |  | ||||||
|  |           if (!c.runner && v === "blanko") { | ||||||
|  |             return "blanko"; | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           if (v.startsWith("#")) { | ||||||
|  |             return `#${c.runner?.id}`; | ||||||
|  |           } | ||||||
|  |           if (c.runner) { | ||||||
|  |             let runnerName = `${c.runner.firstname} ${c.runner.lastname}`; | ||||||
|  |             if (c.runner.middlename) { | ||||||
|  |               runnerName = `${c.runner.firstname} ${c.runner.middlename} ${c.runner.lastname}`; | ||||||
|  |             } | ||||||
|  |             runnerName = runnerName.toLowerCase(); | ||||||
|  |             return runnerName; | ||||||
|  |           } | ||||||
|  |           return ""; | ||||||
|  |         }); | ||||||
|  |       }, 150); | ||||||
|  |     }} | ||||||
|  |     bind:value={filterValue} | ||||||
|  |     type="text" | ||||||
|  |     name="runnerfilter" | ||||||
|  |     id="runnerfilter" | ||||||
|  |   /> | ||||||
|  | </th> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   th { | ||||||
|  |     border-bottom: 1px solid #e0e0e0; | ||||||
|  |   } | ||||||
|  |   input { | ||||||
|  |     margin: -1px 0 0 0; | ||||||
|  |     padding: 0; | ||||||
|  |     width: 100%; | ||||||
|  |     height: 24px; | ||||||
|  |     border: none; | ||||||
|  |     text-align: left; | ||||||
|  |     background: inherit; | ||||||
|  |     outline: 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
							
								
								
									
										45
									
								
								src/components/cards/ThFilterStatus.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								src/components/cards/ThFilterStatus.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let handler; | ||||||
|  |   let selected = "all"; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th> | ||||||
|  |   <select | ||||||
|  |     on:input={() => { | ||||||
|  |       setTimeout(() => { | ||||||
|  |         if (`${selected}`.trim()) { | ||||||
|  |           if (selected === "all") { | ||||||
|  |             handler.filter("", "enabled"); | ||||||
|  |           } else { | ||||||
|  |             handler.filter(selected, "enabled"); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, 50); | ||||||
|  |     }} | ||||||
|  |     bind:value={selected} | ||||||
|  |     name="statusfilter" | ||||||
|  |     id="statusfilter" | ||||||
|  |   > | ||||||
|  |     <option value="all">{$_("all")}</option> | ||||||
|  |     <option value="true">{$_("enabled")}</option> | ||||||
|  |     <option value="false">{$_("disabled")}</option> | ||||||
|  |   </select> | ||||||
|  | </th> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   th { | ||||||
|  |     border-bottom: 1px solid #e0e0e0; | ||||||
|  |   } | ||||||
|  |   select { | ||||||
|  |     margin: -1px 0 0 0; | ||||||
|  |     padding: 0; | ||||||
|  |     width: 100%; | ||||||
|  |     height: 24px; | ||||||
|  |     border: none; | ||||||
|  |     text-align: left; | ||||||
|  |     background: inherit; | ||||||
|  |     outline: 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
							
								
								
									
										1
									
								
								src/components/cards/cards.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/components/cards/cards.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <svg data-name="Layer 1" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 653.9 247.6"><path d="M272 211l-53 12s-11-2-1-17l4-4 27-14v-2l-6-41-2-16 17-17 4-3 44 41v1l-19 33z" fill="#ffb7b7"/><path d="M253 198l-54 13a6 6 0 01-5-7 16 16 0 012-5 48 48 0 016-9l28-14-4-24-1-12-2-8-2-16 21-19 22 21 22 20-3 5-3 7-17 30-3 6z" fill="#ffb7b7"/><path d="M346 190s-20-1-28-15a24 24 0 01-3-14l-8-17-11-23-30-4-2 1-21 19-7 6-10 9-49 44a37 37 0 01-7 9 50 50 0 01-9 7c-10 5-24 9-44 7L10 248 0 176l89-29 131-86 89 23 41 58z" fill="#ffb7b7"/><path d="M648 0H275a5 5 0 00-5 5v221a5 5 0 005 6h373a5 5 0 006-6V5a5 5 0 00-6-5z" fill="#fff"/><path d="M648 0H275a5 5 0 00-5 5v221a5 5 0 005 6h373a5 5 0 006-6V5a5 5 0 00-6-5zm4 226a4 4 0 01-4 4H275a4 4 0 01-3-4V5a4 4 0 013-3h373a4 4 0 014 3z" fill="#3f3d56"/><path d="M312 30a9 9 0 119-9 9 9 0 01-9 9zm0-17a8 8 0 107 8 8 8 0 00-7-8z" fill="#6c63ff"/><path d="M297 21a8 8 0 016-8 8 8 0 100 16 8 8 0 01-6-8zM349 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM368 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM386 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM415 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM434 130a7 7 0 01-7-7v-20a7 7 0 0113 0v20a7 7 0 01-6 7zM452 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM481 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM499 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM518 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM546 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM565 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM583 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7z" fill="#6c63ff"/><path d="M396 208h-99a5 5 0 110-10h99a5 5 0 010 10zM364 188h-35a5 5 0 110-10h35a5 5 0 110 10z" fill="#e6e6e6"/><path fill="#3f3d56" d="M271 46h381v2H271z"/><path opacity=".1" d="M228 203l-1-2 33-15 8-27-12-10 1-1 13 10-8 30-34 15zM196 199l-9 4-17 2a50 50 0 01-9 7l-16-1-26-1 88-74 18 4-29 59z"/><path d="M318 175l-8 1-47 4-29 1-38 18-9 4-70 8 95-81 11 2 20 5 22 5 18 1 24 1 20 1a13 13 0 0112 13c0 7-5 14-21 17z" fill="#ffb7b7"/><path d="M325 170s-7-2-9-9c-2-4-1-9 3-15l1 1c-3 6-4 10-3 14 2 5 9 7 9 7zM197 197l34-16v2l-33 16zM218 135l48-19v2l-41 16 35 6v2l-42-7z" opacity=".1"/></svg> | ||||||
| After Width: | Height: | Size: 2.1 KiB | 
| @@ -86,7 +86,7 @@ | |||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       const toast = Toastify({ | ||||||
|         text: "Contact is being added...", |         text: $_('contact-is-being-added'), | ||||||
|         duration: -1, |         duration: -1, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       let address = {}; |       let address = {}; | ||||||
| @@ -123,7 +123,7 @@ | |||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           // |           // | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: "Contact added", |             text: $_('contact-added'), | ||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|   | |||||||
| @@ -12,11 +12,21 @@ | |||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   .collapsed_navigation { | ||||||
|  |     transform: translateX(-100%); | ||||||
|  |   } | ||||||
|  |   @media (min-width: 768px) { | ||||||
|  |     .collapsed_navigation { | ||||||
|  |       transform: translateX(0px); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|  |  | ||||||
| <section class="min-h-screen bg-gray-50"> | <section class="min-h-screen bg-gray-50"> | ||||||
|   <nav |   <div | ||||||
|     class:-translate-x-full={!navOpen} |     class:collapsed_navigation={!navOpen} | ||||||
|     class:translate-x-0={navOpen} |     class="select-none fixed top-0 left-0 z-20 h-full pb-10 overflow-x-hidden overflow-y-auto transition origin-left transform border-r w-60 bg-gray-50"> | ||||||
|     class="select-none fixed top-0 left-0 z-20 h-full pb-10 overflow-x-hidden overflow-y-auto transition origin-left transform border-r w-60 md:translate-x-0 bg-gray-50"> |  | ||||||
|     <a href="/" class="flex items-center px-4 py-5"> |     <a href="/" class="flex items-center px-4 py-5"> | ||||||
|       <img src="/lfk-logo.png" alt="Logo" class="h-10" /> |       <img src="/lfk-logo.png" alt="Logo" class="h-10" /> | ||||||
|       <h3 class="text-lg">Lauf für Kaya! Admin</h3> |       <h3 class="text-lg">Lauf für Kaya! Admin</h3> | ||||||
| @@ -172,6 +182,25 @@ | |||||||
|           <span>{$_('tracks')}</span> |           <span>{$_('tracks')}</span> | ||||||
|         </a> |         </a> | ||||||
|       {/if} |       {/if} | ||||||
|  |       {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:GET')} | ||||||
|  |         <a | ||||||
|  |           class:bg-gray-100={$router.path === '/cards/'} | ||||||
|  |           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | ||||||
|  |           href="/cards/"> | ||||||
|  |           <svg | ||||||
|  |             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | ||||||
|  |             fill="currentColor" | ||||||
|  |             width="24" | ||||||
|  |             height="24" | ||||||
|  |             xmlns="http://www.w3.org/2000/svg" | ||||||
|  |             viewBox="0 0 24 24"> | ||||||
|  |             <path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |             <path | ||||||
|  |               fill="currentColor" | ||||||
|  |               d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" /></svg> | ||||||
|  |           <span>{$_('cards')}</span> | ||||||
|  |         </a> | ||||||
|  |       {/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')} |       {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')} | ||||||
|         <a |         <a | ||||||
|           class:bg-gray-100={$router.path === '/scans/'} |           class:bg-gray-100={$router.path === '/scans/'} | ||||||
| @@ -227,6 +256,26 @@ | |||||||
|           <span>{$_('scanstations')}</span> |           <span>{$_('scanstations')}</span> | ||||||
|         </a> |         </a> | ||||||
|       {/if} |       {/if} | ||||||
|  |       {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:GET')} | ||||||
|  |         <a | ||||||
|  |           class:bg-gray-100={$router.path === '/statsclients/'} | ||||||
|  |           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | ||||||
|  |           href="/statsclients/"> | ||||||
|  |           <svg | ||||||
|  |             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | ||||||
|  |             fill="currentColor" | ||||||
|  |             width="24" | ||||||
|  |             height="24" | ||||||
|  |             viewBox="0 0 24 24" | ||||||
|  |             xmlns="http://www.w3.org/2000/svg"><path | ||||||
|  |               fill="none" | ||||||
|  |               d="M0 0h24v24H0z" /> | ||||||
|  |             <path | ||||||
|  |               fill="currentColor" | ||||||
|  |               d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | ||||||
|  |           <span>{$_('statsclients')}</span> | ||||||
|  |         </a> | ||||||
|  |       {/if} | ||||||
|       <a |       <a | ||||||
|         class:bg-gray-100={$router.path === '/settings/'} |         class:bg-gray-100={$router.path === '/settings/'} | ||||||
|         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" |         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | ||||||
| @@ -278,14 +327,15 @@ | |||||||
|         <span>{$_('logout')}</span> |         <span>{$_('logout')}</span> | ||||||
|       </span> |       </span> | ||||||
|     </nav> |     </nav> | ||||||
|   </nav> |   </div> | ||||||
|   <div class="ml-0 transition md:ml-60"> |   <div class="ml-0 transition md:ml-60"> | ||||||
|     <header |     <header | ||||||
|       on:click={() => { |  | ||||||
|         navOpen = true; |  | ||||||
|       }} |  | ||||||
|       class="flex items-center justify-between w-full px-4 bg-white border-b h-14 md:hidden"> |       class="flex items-center justify-between w-full px-4 bg-white border-b h-14 md:hidden"> | ||||||
|       <button class="block btn btn-light md:hidden"> |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           navOpen = true; | ||||||
|  |         }} | ||||||
|  |         class="block btn btn-light md:hidden"> | ||||||
|         <span class="sr-only">Menu</span><svg |         <span class="sr-only">Menu</span><svg | ||||||
|           class="w-4 h-4" |           class="w-4 h-4" | ||||||
|           xmlns="http://www.w3.org/2000/svg" |           xmlns="http://www.w3.org/2000/svg" | ||||||
| @@ -299,10 +349,13 @@ | |||||||
|       <NoComponentLoaded /> |       <NoComponentLoaded /> | ||||||
|     </slot> |     </slot> | ||||||
|   </div> |   </div> | ||||||
|   <div |   {#if navOpen === true} | ||||||
|     on:click={() => { |     <div | ||||||
|       navOpen = false; |       on:click={() => { | ||||||
|     }} |         navOpen = false; | ||||||
|     class:hidden={!navOpen} |         console.log({ navOpen }); | ||||||
|     class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden" /> |       }} | ||||||
|  |       class:hidden={!navOpen} | ||||||
|  |       class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden" /> | ||||||
|  |   {/if} | ||||||
| </section> | </section> | ||||||
|   | |||||||
| @@ -1,22 +1,157 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import StatCards from "./StatCards.svelte"; |   import { StatsService } from "@odit/lfk-client-js"; | ||||||
|  |   import StatCards from "./StatCard.svelte"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|  |   import StatCard from "./StatCard.svelte"; | ||||||
|   let navOpen = false; |   let navOpen = false; | ||||||
|  |   const stats_promise = StatsService.statsControllerGet(); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <div | <div | ||||||
|   class="p-5 overflow-x-hidden" |   class="p-5 overflow-x-hidden" | ||||||
|   on:click={() => { |   on:click={() => { | ||||||
|     navOpen = false; |     navOpen = false; | ||||||
|   }}> |   }} | ||||||
|  | > | ||||||
|   <h1 class="text-3xl leading-tight"> |   <h1 class="text-3xl leading-tight"> | ||||||
|     <span class="font-extrabold">{$_('dashboard-title')}</span> |     <span class="font-extrabold">{$_("dashboard-title")}</span> | ||||||
|     <span> |     <span> | ||||||
|       - |       - | ||||||
|       {$_('dashboard-greeting')}, |       {$_("dashboard-greeting")}, | ||||||
|       <span |       <span class="text-blue-500" | ||||||
|         class="text-blue-500">{store.state.jwtinfo.userdetails.firstname} {store.state.jwtinfo.userdetails.lastname}</span></span> |         >{store.state.jwtinfo.userdetails.firstname} | ||||||
|  |         {store.state.jwtinfo.userdetails.lastname}</span | ||||||
|  |       ></span | ||||||
|  |     > | ||||||
|   </h1> |   </h1> | ||||||
|   <StatCards /> |   <h1>{$_("general-stats")}</h1> | ||||||
|  |   {#await stats_promise} | ||||||
|  |     <div | ||||||
|  |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|  |       role="alert" | ||||||
|  |     > | ||||||
|  |       <p class="font-bold">{$_("stats-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|  |     </div> | ||||||
|  |   {:then stats} | ||||||
|  |     <div class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-5 2xl:grid-cols-6 gap-4"> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("runners")} | ||||||
|  |         value={stats.total_runners} | ||||||
|  |         href="/runners/" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           height="24" | ||||||
|  |           width="24" | ||||||
|  |           fill="currentColor" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           ><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  |           <path | ||||||
|  |             d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("total-scans")} | ||||||
|  |         value={stats.total_scans} | ||||||
|  |         href="/scans/" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           stroke="currentColor" | ||||||
|  |           fill="currentColor" | ||||||
|  |           stroke-width="2" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           stroke-linecap="round" | ||||||
|  |           stroke-linejoin="round" | ||||||
|  |           size="24" | ||||||
|  |           class="stroke-current text-grey-500" | ||||||
|  |           height="24" | ||||||
|  |           width="24" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           ><polyline points="22 12 18 12 15 21 9 3 6 12 2 12" /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("total-donations")} | ||||||
|  |         value={`${(stats.total_donation / 100).toFixed(2)} €`} | ||||||
|  |         href="/donations/" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           height="24" | ||||||
|  |           fill="currentColor" | ||||||
|  |           width="24" | ||||||
|  |           ><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  |           <path | ||||||
|  |             d="M15 18.5A6.48 6.48 0 019.24 15H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24A6.491 6.491 0 0115 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3A8.955 8.955 0 0015 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06a8.262 8.262 0 000 2H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("total-distance")} | ||||||
|  |         value={`${stats.total_distance / 1000} km`} | ||||||
|  |         href="#" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           fill="currentColor" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           height="24" | ||||||
|  |           width="24" | ||||||
|  |           ><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  |           <path | ||||||
|  |             d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("count_teams")} | ||||||
|  |         value={stats.total_teams} | ||||||
|  |         href="/teams/" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           stroke="currentColor" | ||||||
|  |           fill="none" | ||||||
|  |           stroke-width="2" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           stroke-linecap="round" | ||||||
|  |           stroke-linejoin="round" | ||||||
|  |           size="24" | ||||||
|  |           class="stroke-current text-grey-500" | ||||||
|  |           height="24" | ||||||
|  |           width="24" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           ><path d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2" /> | ||||||
|  |           <circle cx="9" cy="7" r="4" /> | ||||||
|  |           <path d="M23 21v-2a4 4 0 0 0-3-3.87" /> | ||||||
|  |           <path d="M16 3.13a4 4 0 0 1 0 7.75" /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |       <StatCard | ||||||
|  |         title={$_("count_organizations")} | ||||||
|  |         value={stats.total_orgs} | ||||||
|  |         href="/orgs/" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           height="24" | ||||||
|  |           fill="currentColor" | ||||||
|  |           width="24" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |           <path | ||||||
|  |             d="M17 11V3H7v4H3v14h8v-4h2v4h8V11h-4zM7 19H5v-2h2v2zm0-4H5v-2h2v2zm0-4H5V9h2v2zm4 4H9v-2h2v2zm0-4H9V9h2v2zm0-4H9V5h2v2zm4 8h-2v-2h2v2zm0-4h-2V9h2v2zm0-4h-2V5h2v2zm4 12h-2v-2h2v2zm0-4h-2v-2h2v2z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </StatCard> | ||||||
|  |     </div> | ||||||
|  |   {:catch error} | ||||||
|  |     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | ||||||
|  |       <span class="inline-block align-middle mr-8"> | ||||||
|  |         <b class="capitalize">{$_("general_promise_error")}</b> | ||||||
|  |         {error} | ||||||
|  |       </span> | ||||||
|  |     </div> | ||||||
|  |   {/await} | ||||||
| </div> | </div> | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								src/components/dashboard/StatCard.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/components/dashboard/StatCard.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |  | ||||||
|  |   export let href = "#" | ||||||
|  |   export let title = ""; | ||||||
|  |   export let value = ""; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <a href={href}> | ||||||
|  |   <div | ||||||
|  |     class="p-4 rounded-lg bg-white border border-grey-100"> | ||||||
|  |     <div class="flex flex-row items-center justify-between"> | ||||||
|  |       <div class="flex flex-col"> | ||||||
|  |         <div class="text-xs uppercase font-light text-grey-500"> | ||||||
|  |           {title} | ||||||
|  |         </div> | ||||||
|  |         <div class="text-xl font-bold">{value}</div> | ||||||
|  |       </div> | ||||||
|  |       <slot></slot> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </a> | ||||||
| @@ -1,165 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import { StatsService } from "@odit/lfk-client-js"; |  | ||||||
|   import { _ } from "svelte-i18n"; |  | ||||||
|   const stats_promise = StatsService.statsControllerGet(); |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <!--  --> |  | ||||||
| <h1>{$_('general-stats')}</h1> |  | ||||||
| {#await stats_promise} |  | ||||||
|   <div |  | ||||||
|     class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |  | ||||||
|     role="alert"> |  | ||||||
|     <p class="font-bold">{$_('stats-are-being-loaded')}</p> |  | ||||||
|     <p class="text-sm">{$_('this-might-take-a-moment')}</p> |  | ||||||
|   </div> |  | ||||||
| {:then stats} |  | ||||||
|   <div |  | ||||||
|     class="flex flex-col lg:flex-row w-full lg:space-x-2 space-y-2 lg:space-y-0 mb-2 lg:mb-4"> |  | ||||||
|     <a href="/runners/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('runners')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_runners}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 24 24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-scans')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_scans}</div> |  | ||||||
|           </div><svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="currentColor" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             size="24" |  | ||||||
|             class="stroke-current text-grey-500" |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><polyline |  | ||||||
|               points="22 12 18 12 15 21 9 3 6 12 2 12" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-donations')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_donation} €</div> |  | ||||||
|           </div><svg |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             height="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M15 18.5A6.48 6.48 0 019.24 15H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24A6.491 6.491 0 0115 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3A8.955 8.955 0 0015 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06a8.262 8.262 0 000 2H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-distance')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold"> |  | ||||||
|               {stats.total_distance / 1000} |  | ||||||
|               km |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             fill="currentColor" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             height="24" |  | ||||||
|             width="24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <a href="/teams/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('count_teams')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_teams}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="none" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             size="24" |  | ||||||
|             class="stroke-current text-grey-500" |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><path |  | ||||||
|               d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2" /> |  | ||||||
|             <circle cx="9" cy="7" r="4" /> |  | ||||||
|             <path d="M23 21v-2a4 4 0 0 0-3-3.87" /> |  | ||||||
|             <path d="M16 3.13a4 4 0 0 1 0 7.75" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|     <a href="/orgs/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('count_organizations')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_orgs}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             height="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
|             <path |  | ||||||
|               d="M17 11V3H7v4H3v14h8v-4h2v4h8V11h-4zM7 19H5v-2h2v2zm0-4H5v-2h2v2zm0-4H5V9h2v2zm4 4H9v-2h2v2zm0-4H9V9h2v2zm0-4H9V5h2v2zm4 8h-2v-2h2v2zm0-4h-2V9h2v2zm0-4h-2V5h2v2zm4 12h-2v-2h2v2zm0-4h-2v-2h2v2z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|   </div> |  | ||||||
| {:catch error} |  | ||||||
|   <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |  | ||||||
|     <span class="inline-block align-middle mr-8"> |  | ||||||
|       <b class="capitalize">{$_('general_promise_error')}</b> |  | ||||||
|       {error} |  | ||||||
|     </span> |  | ||||||
|   </div> |  | ||||||
| {/await} |  | ||||||
| @@ -9,6 +9,7 @@ | |||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|   import Select from "svelte-select"; |   import Select from "svelte-select"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|  | import { is_promise } from "svelte/internal"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_donations; |   export let current_donations; | ||||||
|   const getDonorLabel = (option) => |   const getDonorLabel = (option) => | ||||||
| @@ -24,6 +25,7 @@ | |||||||
|   $: donors = []; |   $: donors = []; | ||||||
|   $: runners = []; |   $: runners = []; | ||||||
|   $: is_fixed = false; |   $: is_fixed = false; | ||||||
|  |   $: is_paid = false; | ||||||
|   DonorService.donorControllerGetAll().then((val) => { |   DonorService.donorControllerGetAll().then((val) => { | ||||||
|     donors = val.map((r) => { |     donors = val.map((r) => { | ||||||
|       return { label: getDonorLabel(r), value: r }; |       return { label: getDonorLabel(r), value: r }; | ||||||
| @@ -57,14 +59,18 @@ | |||||||
|       let amount_cent = Math.floor(amount_input * 100); |       let amount_cent = Math.floor(amount_input * 100); | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       const toast = Toastify({ | ||||||
|         text: "adding donation", |         text: $_('adding-donation'), | ||||||
|         duration: -1, |         duration: -1, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       if (is_fixed) { |       if (is_fixed) { | ||||||
|         let postdata = { |         let postdata = { | ||||||
|           donor, |           donor, | ||||||
|           amount: amount_cent, |           amount: amount_cent, | ||||||
|  |           paidAmount: 0 | ||||||
|         }; |         }; | ||||||
|  |         if(is_paid){ | ||||||
|  |           postdata.paidAmount = amount_cent; | ||||||
|  |         } | ||||||
|         DonationService.donationControllerPostFixed(postdata) |         DonationService.donationControllerPostFixed(postdata) | ||||||
|           .then((result) => { |           .then((result) => { | ||||||
|             donor = donors[0].id || 0; |             donor = donors[0].id || 0; | ||||||
| @@ -73,7 +79,7 @@ | |||||||
|             modal_open = false; |             modal_open = false; | ||||||
|             // |             // | ||||||
|             Toastify({ |             Toastify({ | ||||||
|               text: "donation_added", |               text: $_('donation_added'), | ||||||
|               duration: 500, |               duration: 500, | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|             }).showToast(); |             }).showToast(); | ||||||
| @@ -102,7 +108,7 @@ | |||||||
|             modal_open = false; |             modal_open = false; | ||||||
|             // |             // | ||||||
|             Toastify({ |             Toastify({ | ||||||
|               text: "donation_added", |               text: $_('donation_added'), | ||||||
|               duration: 500, |               duration: 500, | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|             }).showToast(); |             }).showToast(); | ||||||
| @@ -123,7 +129,7 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <style> | <style> | ||||||
|   input:before { |   .toggle:before { | ||||||
|     content: ""; |     content: ""; | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     width: 1.25rem; |     width: 1.25rem; | ||||||
| @@ -137,12 +143,12 @@ | |||||||
|     transition: 0.2s ease-in-out; |     transition: 0.2s ease-in-out; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   input:checked { |   .toggle:checked { | ||||||
|     /* @apply: bg-indigo-400; */ |     /* @apply: bg-indigo-400; */ | ||||||
|     background-color: #7f9cf5; |     background-color: #7f9cf5; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   input:checked:before { |   .toggle:checked:before { | ||||||
|     left: 1.25rem; |     left: 1.25rem; | ||||||
|   } |   } | ||||||
| </style> | </style> | ||||||
| @@ -195,7 +201,7 @@ | |||||||
|                   class="ml-2 text-base" |                   class="ml-2 text-base" | ||||||
|                   class:text-gray-300={is_fixed}>{$_('distance-donation')}</span> |                   class:text-gray-300={is_fixed}>{$_('distance-donation')}</span> | ||||||
|                 <input |                 <input | ||||||
|                   class="relative w-10 h-5 transition-all duration-200 ease-in-out bg-gray-400 rounded-full shadow-inner outline-none appearance-none align-middle" |                   class="toggle relative w-10 h-5 transition-all duration-200 ease-in-out bg-gray-400 rounded-full shadow-inner outline-none appearance-none align-middle" | ||||||
|                   type="checkbox" |                   type="checkbox" | ||||||
|                   bind:checked={is_fixed} /> |                   bind:checked={is_fixed} /> | ||||||
|                 <span |                 <span | ||||||
| @@ -267,6 +273,29 @@ | |||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|  |                 {#if is_fixed} | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="paid" | ||||||
|  |                     class="block text-sm font-medium text-gray-700">{$_('already-paid')}</label> | ||||||
|  |                     <p class="text-gray-500"> | ||||||
|  |                       <input | ||||||
|  |                         id="paid" | ||||||
|  |                         bind:checked={is_paid} | ||||||
|  |                         name="paid" | ||||||
|  |                         type="checkbox" | ||||||
|  |                         class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" > | ||||||
|  |                         <span class="align-text-bottom"> | ||||||
|  |  | ||||||
|  |                       {#if is_paid} | ||||||
|  |                       {$_('paid')} | ||||||
|  |                       {:else} | ||||||
|  |                       {$_('open')} | ||||||
|  |                       {/if} | ||||||
|  |                         </span> | ||||||
|  |                     </p> | ||||||
|  |                 </div> | ||||||
|  |                 {/if} | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|   | |||||||
							
								
								
									
										202
									
								
								src/components/donations/AddDonationPaymentModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								src/components/donations/AddDonationPaymentModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { DonationService } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   export let payment_modal_open = false; | ||||||
|  |   export let current_donations = []; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   export let paid_amount_input = 0; | ||||||
|  |   $:processed_last_submit=true; | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: createbtnenabled = is_paid_amount_valid && !(paid_amount_input*100 == original_data.paidAmount) | ||||||
|  |   $: is_paid_amount_valid = paid_amount_input > 0; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         payment_modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_('updating-donation'), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |       editable.donor = editable.donor.id; | ||||||
|  |       editable.paidAmount = paid_amount_input*100; | ||||||
|  |       if(editable.responseType == "DISTANCEDONATION" || editable.runner){ | ||||||
|  |         editable.runner = editable.runner.id; | ||||||
|  |         DonationService.donationControllerPutDistance(original_data.id, editable) | ||||||
|  |         .then((result) => { | ||||||
|  |           let id = original_data.id; | ||||||
|  |           editable = {}; | ||||||
|  |           original_data = {}; | ||||||
|  |           payment_modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_('donation-updated'), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_donations[current_donations.findIndex((c) => c.id === id)] = result; | ||||||
|  |           current_donations = current_donations; | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |       else{ | ||||||
|  |           DonationService.donationControllerPutFixed(original_data.id, editable) | ||||||
|  |         .then((result) => { | ||||||
|  |           let id = original_data.id; | ||||||
|  |           editable = {}; | ||||||
|  |           original_data = {}; | ||||||
|  |           payment_modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_('donation-updated'), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_donations[current_donations.findIndex((c) => c.id === id)] = result; | ||||||
|  |           current_donations = current_donations; | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if payment_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       payment_modal_open = false; | ||||||
|  |     }}> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   fill="currentColor" | ||||||
|  |                   d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" /></svg> | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('enter-payment')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_('you-can-enter-the-donations-paid-amount-manually-or-use-the-max-button-to-use-the-donations-exact-amount')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols gap-6"> | ||||||
|  |                 <div class="w-full"> | ||||||
|  |                   <label | ||||||
|  |                   for="token" | ||||||
|  |                   class="block text-sm font-medium text-gray-700">{$_('paid-amount')}</label> | ||||||
|  |                 <div class="inline-flex border-gray-300 border rounded-l-md rounded-r-md bg-gray-50 text-gray-500 w-full"> | ||||||
|  |                   <input | ||||||
|  |                       autocomplete="off" | ||||||
|  |                       class:border-red-500={!is_paid_amount_valid} | ||||||
|  |                       class:focus:border-red-500={!is_paid_amount_valid} | ||||||
|  |                       class:focus:ring-red-500={!is_paid_amount_valid} | ||||||
|  |                       bind:value={paid_amount_input} | ||||||
|  |                       type="number" | ||||||
|  |                       step="0.01" | ||||||
|  |                       name="donation_amount_eur" | ||||||
|  |                       class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm p-2" | ||||||
|  |                       placeholder="2.00" /> | ||||||
|  |                     <button | ||||||
|  |                       on:click={ | ||||||
|  |                         ()=>{ | ||||||
|  |                           paid_amount_input=paid_amount_input = (original_data.amount/100).toFixed(2); | ||||||
|  |                         } | ||||||
|  |                       } | ||||||
|  |                       class="inline-flex items-center p-r-2 text-indigo-300 hover:text-indigo-700 text-sm">MAX</button> | ||||||
|  |                     <span | ||||||
|  |                       class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500 text-sm">€</span> | ||||||
|  |                   </div> | ||||||
|  |                   {#if !is_paid_amount_valid} | ||||||
|  |                     <span | ||||||
|  |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |                       {$_('payment-amount-must-be-greater-than-0-00eur')} | ||||||
|  |                     </span> | ||||||
|  |                   {/if} | ||||||
|  |                 </div> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |           <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |             <button | ||||||
|  |               disabled={!createbtnenabled} | ||||||
|  |               class:opacity-50={!createbtnenabled} | ||||||
|  |               on:click={submit} | ||||||
|  |               type="button" | ||||||
|  |               class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |               {$_('save-changes')} | ||||||
|  |             </button> | ||||||
|  |             <button | ||||||
|  |               on:click={() => { | ||||||
|  |                 payment_modal_open = false; | ||||||
|  |               }} | ||||||
|  |               type="button" | ||||||
|  |               class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |               {$_('cancel')} | ||||||
|  |             </button> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  | {/if} | ||||||
| @@ -20,6 +20,8 @@ | |||||||
|   $: current_runners = []; |   $: current_runners = []; | ||||||
|   $: amount_input = 0; |   $: amount_input = 0; | ||||||
|   $: is_amount_valid = amount_input > 0; |   $: is_amount_valid = amount_input > 0; | ||||||
|  |   $: paid_amount_input = 0; | ||||||
|  |   $: is_paid_amount_valid = paid_amount_input > 0; | ||||||
|   $: is_everything_set = |   $: is_everything_set = | ||||||
|     editable.donor != null && |     editable.donor != null && | ||||||
|     ((original_data.responseType == "DISTANCEDONATION" && |     ((original_data.responseType == "DISTANCEDONATION" && | ||||||
| @@ -30,15 +32,17 @@ | |||||||
|     (original_data.responseType == "DISTANCEDONATION" && |     (original_data.responseType == "DISTANCEDONATION" && | ||||||
|       !(Math.floor(amount_input * 100) === original_data.amountPerDistance)) || |       !(Math.floor(amount_input * 100) === original_data.amountPerDistance)) || | ||||||
|     (original_data.responseType !== "DISTANCEDONATION" && |     (original_data.responseType !== "DISTANCEDONATION" && | ||||||
|       !(Math.floor(amount_input * 100) === original_data.amount)); |       !(Math.floor(amount_input * 100) === original_data.amount)) || | ||||||
|  |       !(Math.floor(paid_amount_input * 100) === original_data.paidAmount); | ||||||
|   $: save_enabled = changes_performed && is_amount_valid && is_everything_set; |   $: save_enabled = changes_performed && is_amount_valid && is_everything_set; | ||||||
|  |  | ||||||
|   const promise = DonationService.donationControllerGetOne( |   const promise = DonationService.donationControllerGetOne( | ||||||
|     params.donationid |     params.donationid | ||||||
|   ).then((data) => { |   ).then((data) => { | ||||||
|     data_loaded = true; |     data_loaded = true; | ||||||
|     original_data = Object.assign(original_data, data); |     original_data = Object.assign({}, data); | ||||||
|     editable = Object.assign(editable, original_data); |     editable = Object.assign({}, original_data); | ||||||
|  |     paid_amount_input = data.paidAmount / 100; | ||||||
|     if (data.responseType == "DISTANCEDONATION") { |     if (data.responseType == "DISTANCEDONATION") { | ||||||
|       amount_input = data.amountPerDistance / 100; |       amount_input = data.amountPerDistance / 100; | ||||||
|       RunnerService.runnerControllerGetAll().then((val) => { |       RunnerService.runnerControllerGetAll().then((val) => { | ||||||
| @@ -66,10 +70,11 @@ | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (data_loaded === true && save_enabled) { |     if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: "Donation is being updated", |         text: $_('updating-donation'), | ||||||
|         duration: 2500, |         duration: 2500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       let postdata = {}; |       let postdata = {}; | ||||||
|  |       editable.paidAmount = paid_amount_input*100; | ||||||
|       if (original_data.responseType === "DISTANCEDONATION") { |       if (original_data.responseType === "DISTANCEDONATION") { | ||||||
|         editable.amountPerDistance = Math.floor(amount_input * 100); |         editable.amountPerDistance = Math.floor(amount_input * 100); | ||||||
|         postdata = Object.assign(postdata, editable); |         postdata = Object.assign(postdata, editable); | ||||||
| @@ -83,7 +88,7 @@ | |||||||
|             Object.assign(original_data, editable); |             Object.assign(original_data, editable); | ||||||
|             original_data = original_data; |             original_data = original_data; | ||||||
|             Toastify({ |             Toastify({ | ||||||
|               text: "updated donation", |               text: $_('donation-updated'), | ||||||
|               duration: 2500, |               duration: 2500, | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|             }).showToast(); |             }).showToast(); | ||||||
| @@ -98,7 +103,7 @@ | |||||||
|             Object.assign(original_data, editable); |             Object.assign(original_data, editable); | ||||||
|             original_data = original_data; |             original_data = original_data; | ||||||
|             Toastify({ |             Toastify({ | ||||||
|               text: "updated donation", |               text: $_('donation-updated'), | ||||||
|               duration: 2500, |               duration: 2500, | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|             }).showToast(); |             }).showToast(); | ||||||
| @@ -112,7 +117,7 @@ | |||||||
|     DonationService.donationControllerRemove(original_data.id, false) |     DonationService.donationControllerRemove(original_data.id, false) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Donation delete", |           text: $_('donation-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
| @@ -219,7 +224,24 @@ | |||||||
|       <span>{(editable.amount / 100) |       <span>{(editable.amount / 100) | ||||||
|           .toFixed(2) |           .toFixed(2) | ||||||
|           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> |           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> | ||||||
|  |       | | ||||||
|  |       <span | ||||||
|  |         class="font-medium text-gray-700">{$_('paid-amount')}:</span> | ||||||
|  |       <span>{(editable.paidAmount / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> | ||||||
|  |       | | ||||||
|  |       <span | ||||||
|  |         class="font-medium text-gray-700">{$_('status')}:</span> | ||||||
|  |         {#if editable.status =="PAID"} | ||||||
|  |           <span | ||||||
|  |             class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('paid')}</span> | ||||||
|  |         {:else} | ||||||
|  |           <span | ||||||
|  |             class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('open')}</span> | ||||||
|  |         {/if} | ||||||
|     </div> |     </div> | ||||||
|  |     <br> | ||||||
|     <div class=" w-full"> |     <div class=" w-full"> | ||||||
|       <label |       <label | ||||||
|         for="donor" |         for="donor" | ||||||
| @@ -232,7 +254,7 @@ | |||||||
|         placeholder={$_('search-for-donor-name-or-id')} |         placeholder={$_('search-for-donor-name-or-id')} | ||||||
|         noOptionsMessage={$_('no-donors-found')} |         noOptionsMessage={$_('no-donors-found')} | ||||||
|         bind:selectedValue={donor} |         bind:selectedValue={donor} | ||||||
|         on:select={(selectedValue) => (editable.donor = selectedValue.detail.value)} |         on:select={(selectedValue) => {editable.donor = selectedValue.detail.value; editable.donor.donationAmount=original_data.donor.donationAmount; editable.donor.paidDonationAmount =original_data.donor.paidDonationAmount}} | ||||||
|         on:clear={() => (editable.donor = null)} /> |         on:clear={() => (editable.donor = null)} /> | ||||||
|     </div> |     </div> | ||||||
|     {#if original_data.responseType == 'DISTANCEDONATION'} |     {#if original_data.responseType == 'DISTANCEDONATION'} | ||||||
| @@ -280,6 +302,39 @@ | |||||||
|         </span> |         </span> | ||||||
|       {/if} |       {/if} | ||||||
|     </div> |     </div> | ||||||
|  |     <div class="w-full"> | ||||||
|  |       <label | ||||||
|  |         for="token" | ||||||
|  |         class="block text-sm font-medium text-gray-700">{$_('paid-amount')}</label> | ||||||
|  |       <div class="inline-flex border-gray-300 border rounded-l-md rounded-r-md bg-gray-50 text-gray-500 w-full"> | ||||||
|  |         <input | ||||||
|  |             autocomplete="off" | ||||||
|  |             class:border-red-500={!is_amount_valid} | ||||||
|  |             class:focus:border-red-500={!is_amount_valid} | ||||||
|  |             class:focus:ring-red-500={!is_amount_valid} | ||||||
|  |             bind:value={paid_amount_input} | ||||||
|  |             type="number" | ||||||
|  |             step="0.01" | ||||||
|  |             name="donation_amount_eur" | ||||||
|  |             class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm p-2" | ||||||
|  |             placeholder="2.00" /> | ||||||
|  |           <button | ||||||
|  |             on:click={ | ||||||
|  |               ()=>{ | ||||||
|  |                 paid_amount_input=paid_amount_input = (original_data.amount/100).toFixed(2); | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |             class="inline-flex items-center p-r-2 text-indigo-300 hover:text-indigo-700 text-sm">MAX</button> | ||||||
|  |           <span | ||||||
|  |             class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500 text-sm">€</span> | ||||||
|  |         </div> | ||||||
|  |         {#if !is_paid_amount_valid} | ||||||
|  |           <span | ||||||
|  |             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |             {$_('payment-amount-must-be-greater-than-0-00eur')} | ||||||
|  |           </span> | ||||||
|  |         {/if} | ||||||
|  |     </div> | ||||||
|   </section> |   </section> | ||||||
| {:catch error} | {:catch error} | ||||||
|   <PromiseError {error} /> |   <PromiseError {error} /> | ||||||
|   | |||||||
| @@ -4,9 +4,14 @@ | |||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import DonationsEmptyState from "./DonationsEmptyState.svelte"; |   import DonationsEmptyState from "./DonationsEmptyState.svelte"; | ||||||
|  |   import AddDonationPaymentModal from "./AddDonationPaymentModal.svelte"; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   export let current_donations = []; |   export let current_donations = []; | ||||||
|  |   export let payment_modal_open = false; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   export let paid_amount_input = 0; | ||||||
|   const donations_promise = DonationService.donationControllerGetAll().then( |   const donations_promise = DonationService.donationControllerGetAll().then( | ||||||
|     (val) => { |     (val) => { | ||||||
|       current_donations = val; |       current_donations = val; | ||||||
| @@ -18,8 +23,15 @@ | |||||||
|     } |     } | ||||||
|     return id.toString() === searchvalue; |     return id.toString() === searchvalue; | ||||||
|   } |   } | ||||||
|  |   function open_payment_modal(donation) { | ||||||
|  |     editable = Object.assign({}, donation); | ||||||
|  |     original_data = Object.assign({}, donation); | ||||||
|  |     paid_amount_input = (donation.paidAmount/100).toFixed(2); | ||||||
|  |     payment_modal_open = true; | ||||||
|  |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | <AddDonationPaymentModal bind:current_donations bind:original_data bind:editable bind:paid_amount_input bind:payment_modal_open /> | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')} | ||||||
|   {#await donations_promise} |   {#await donations_promise} | ||||||
|     <div |     <div | ||||||
| @@ -63,6 +75,16 @@ | |||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|                 {$_('donation-amount')} |                 {$_('donation-amount')} | ||||||
|               </th> |               </th> | ||||||
|  |               <th | ||||||
|  |                 scope="col" | ||||||
|  |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|  |                 {$_('paid-amount')} | ||||||
|  |               </th> | ||||||
|  |               <th | ||||||
|  |                 scope="col" | ||||||
|  |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|  |                 {$_('status')} | ||||||
|  |               </th> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |               <th scope="col" class="relative px-6 py-3"> | ||||||
|                 <span class="sr-only">{$_('action')}</span> |                 <span class="sr-only">{$_('action')}</span> | ||||||
|               </th> |               </th> | ||||||
| @@ -74,20 +96,12 @@ | |||||||
|                 .toLowerCase() |                 .toLowerCase() | ||||||
|                 .includes( |                 .includes( | ||||||
|                   searchvalue.toLowerCase() |                   searchvalue.toLowerCase() | ||||||
|                 ) || donation.donor.middlename |                 ) ||  donation.donor.lastname | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.donor.lastname |  | ||||||
|                   .toLowerCase() |                   .toLowerCase() | ||||||
|                   .includes( |                   .includes( | ||||||
|                     searchvalue.toLowerCase() |                     searchvalue.toLowerCase() | ||||||
|                   ) || donation.runner?.firstname |                   ) || donation.runner?.firstname | ||||||
|                   .toLowerCase() |                   .toLowerCase() | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.runner?.middlename |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |                   .includes( | ||||||
|                     searchvalue.toLowerCase() |                     searchvalue.toLowerCase() | ||||||
|                   ) || donation.runner?.lastname |                   ) || donation.runner?.lastname | ||||||
| @@ -140,6 +154,22 @@ | |||||||
|                         .toLocaleString('de-DE', { valute: 'EUR' })}€ |                         .toLocaleString('de-DE', { valute: 'EUR' })}€ | ||||||
|                     </div> |                     </div> | ||||||
|                   </td> |                   </td> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  |                     <div class="text-sm font-medium text-gray-900"> | ||||||
|  |                       {(donation.paidAmount / 100) | ||||||
|  |                         .toFixed(2) | ||||||
|  |                         .toLocaleString('de-DE', { valute: 'EUR' })}€ | ||||||
|  |                     </div> | ||||||
|  |                   </td> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  |                     {#if donation.status =="PAID"} | ||||||
|  |                       <span | ||||||
|  |                         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('paid')}</span> | ||||||
|  |                     {:else} | ||||||
|  |                       <span | ||||||
|  |                         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('open')}</span> | ||||||
|  |                     {/if} | ||||||
|  |                   </td> | ||||||
|                   {#if active_deletes[donation.id] === true} |                   {#if active_deletes[donation.id] === true} | ||||||
|                     <td |                     <td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
| @@ -157,7 +187,7 @@ | |||||||
|                                 (obj) => obj.id !== donation.id |                                 (obj) => obj.id !== donation.id | ||||||
|                               ); |                               ); | ||||||
|                               Toastify({ |                               Toastify({ | ||||||
|                                 text: 'Donation deleted', |                                 text: $_('donation-deleted'), | ||||||
|                                 duration: 500, |                                 duration: 500, | ||||||
|                                 backgroundColor: |                                 backgroundColor: | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |                                   'linear-gradient(to right, #00b09b, #96c93d)', | ||||||
| @@ -171,6 +201,9 @@ | |||||||
|                   {:else} |                   {:else} | ||||||
|                     <td |                     <td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|  |                       <button | ||||||
|  |                         on:click={() => {open_payment_modal(donation);}} | ||||||
|  |                         class="text-[#025a21] hover:text-green-900 mr-4">{$_('enter-payment')}</button> | ||||||
|                       <a |                       <a | ||||||
|                         href="./{donation.id}" |                         href="./{donation.id}" | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ | |||||||
|     ) |     ) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Donor deleted", |           text: $_('donor-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
|   | |||||||
| @@ -193,6 +193,12 @@ | |||||||
|       <span>{(editable.donationAmount / 100) |       <span>{(editable.donationAmount / 100) | ||||||
|           .toFixed(2) |           .toFixed(2) | ||||||
|           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> |           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> | ||||||
|  |       | | ||||||
|  |       <span | ||||||
|  |         class="font-medium text-gray-700">{$_('total-paid-amount')}:</span> | ||||||
|  |       <span>{(editable.paidDonationAmount / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString('de-DE', { valute: 'EUR' })}€</span> | ||||||
|       <br /> |       <br /> | ||||||
|       <span class="font-medium text-gray-700">{$_('donations')}:</span> |       <span class="font-medium text-gray-700">{$_('donations')}:</span> | ||||||
|       {#if current_donations.filter((d) => d.donor.id == editable.id).length > 0} |       {#if current_donations.filter((d) => d.donor.id == editable.id).length > 0} | ||||||
| @@ -201,7 +207,7 @@ | |||||||
|             <a |             <a | ||||||
|               href="../donations/{d.id}" |               href="../donations/{d.id}" | ||||||
|               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-600 text-white mr-1">{d.runner.firstname} |               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-600 text-white mr-1">{d.runner.firstname} | ||||||
|               {d.runner.middlename} |               {d.runner.middlename || ""} | ||||||
|               {d.runner.lastname}</a> |               {d.runner.lastname}</a> | ||||||
|           {:else} |           {:else} | ||||||
|             <a |             <a | ||||||
|   | |||||||
| @@ -20,6 +20,31 @@ | |||||||
|         {$_('add-donor')} |         {$_('add-donor')} | ||||||
|       </button> |       </button> | ||||||
|     {/if} |     {/if} | ||||||
|  |     {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:GET')} | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           const data = (current_donors.filter(d=>d.receiptNeeded===true)).map(function (d) { | ||||||
|  |             d.address.address2=d.address.address2===""?"":" "+d.address.address2; | ||||||
|  |             const address=`${d.address.address1}${d.address.address2}, ${d.address.postalcode} ${d.address.city}, ${d.address.country}`; | ||||||
|  |             return [d.firstname,d.middlename,d.lastname,d.paidDonationAmount,address]; | ||||||
|  |           }) | ||||||
|  |           let csv = `${$_('csv_import__firstname')};${$_('csv_import__middlename')};${$_('csv_import__lastname')};${$_('total_donation_amount_in_eur')};${$_('address')}\n`; | ||||||
|  |     data.forEach(function(row) { | ||||||
|  |             csv += row.join(';'); | ||||||
|  |             csv += "\n"; | ||||||
|  |     }); | ||||||
|  |     let hiddenElement = document.createElement('a'); | ||||||
|  |     hiddenElement.href = 'data:text/csv;charset=utf-8,' + encodeURI(csv); | ||||||
|  |     hiddenElement.target = '_blank'; | ||||||
|  |     hiddenElement.download = `${$_('filename_sponsoringquittungsliste')}.csv`; | ||||||
|  |     hiddenElement.click(); | ||||||
|  |     hiddenElement.remove(); | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |         {$_('sponsoring-quittungs-liste_herunterladen')} | ||||||
|  |       </button> | ||||||
|  |     {/if} | ||||||
|   </span> |   </span> | ||||||
|   <DonorsOverview bind:current_donors /> |   <DonorsOverview bind:current_donors /> | ||||||
| </section> | </section> | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { DonationService, DonorService } from "@odit/lfk-client-js"; |   import { DonationService, DonorService } from "@odit/lfk-client-js"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import DonorsEmptyState from "./DonorsEmptyState.svelte"; |   import DonorsEmptyState from "./DonorsEmptyState.svelte"; | ||||||
| @@ -77,6 +77,11 @@ | |||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|                 {$_('total-donation-amount')} |                 {$_('total-donation-amount')} | ||||||
|               </th> |               </th> | ||||||
|  |               <th | ||||||
|  |                 scope="col" | ||||||
|  |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|  |                 {$_('total-paid-amount')} | ||||||
|  |               </th> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |               <th scope="col" class="relative px-6 py-3"> | ||||||
|                 <span class="sr-only">{$_('action')}</span> |                 <span class="sr-only">{$_('action')}</span> | ||||||
|               </th> |               </th> | ||||||
| @@ -88,11 +93,7 @@ | |||||||
|                 .toLowerCase() |                 .toLowerCase() | ||||||
|                 .includes( |                 .includes( | ||||||
|                   searchvalue.toLowerCase() |                   searchvalue.toLowerCase() | ||||||
|                 ) || donor.middlename |                 ) ||  donor.lastname | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donor.lastname |  | ||||||
|                   .toLowerCase() |                   .toLowerCase() | ||||||
|                   .includes( |                   .includes( | ||||||
|                     searchvalue.toLowerCase() |                     searchvalue.toLowerCase() | ||||||
| @@ -131,7 +132,7 @@ | |||||||
|                           <a |                           <a | ||||||
|                             href="../donations/{d.id}" |                             href="../donations/{d.id}" | ||||||
|                             class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-600 text-white mr-1">{d.runner.firstname} |                             class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-600 text-white mr-1">{d.runner.firstname} | ||||||
|                             {d.runner.middlename} |                             {d.runner.middlename || ''} | ||||||
|                             {d.runner.lastname}</a> |                             {d.runner.lastname}</a> | ||||||
|                         {:else} |                         {:else} | ||||||
|                           <a |                           <a | ||||||
| @@ -149,6 +150,11 @@ | |||||||
|                       .toFixed(2) |                       .toFixed(2) | ||||||
|                       .toLocaleString('de-DE', { valute: 'EUR' })}€ |                       .toLocaleString('de-DE', { valute: 'EUR' })}€ | ||||||
|                   </td> |                   </td> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  |                     {(donor.paidDonationAmount / 100) | ||||||
|  |                       .toFixed(2) | ||||||
|  |                       .toLocaleString('de-DE', { valute: 'EUR' })}€ | ||||||
|  |                   </td> | ||||||
|                   {#if active_deletes[donor.id] === true} |                   {#if active_deletes[donor.id] === true} | ||||||
|                     <td |                     <td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|   | |||||||
| @@ -32,8 +32,12 @@ | |||||||
|       target="_blank" |       target="_blank" | ||||||
|       rel="noopener, noreferrer" |       rel="noopener, noreferrer" | ||||||
|       href="https://git.odit.services/lfk/frontend/src/tag/{releaseinfo}">{releaseinfo}</a> |       href="https://git.odit.services/lfk/frontend/src/tag/{releaseinfo}">{releaseinfo}</a> | ||||||
|       - |     - | ||||||
|     <a class="underline" href="https://docs.lauf-fuer-kaya.de" target="_blank">{$_('documentation')}</a> |     <a | ||||||
|  |       rel="noopener, noreferrer" | ||||||
|  |       class="underline" | ||||||
|  |       href="https://docs.lauf-fuer-kaya.de" | ||||||
|  |       target="_blank">{$_('documentation')}</a> | ||||||
|     - |     - | ||||||
|     <a class="underline" href="/privacy">{$_('privacy')}</a> |     <a class="underline" href="/privacy">{$_('privacy')}</a> | ||||||
|     - |     - | ||||||
|   | |||||||
| @@ -6,13 +6,15 @@ | |||||||
|   let html = ""; |   let html = ""; | ||||||
|   async function load() { |   async function load() { | ||||||
|     let md = await fetch("/imprint_" + getLocaleFromNavigator() + ".md"); |     let md = await fetch("/imprint_" + getLocaleFromNavigator() + ".md"); | ||||||
|     if((await md.text()).includes("<meta")){ |     let text = (await md.text()).toString(); | ||||||
|  |     if(text.includes("<meta")){ | ||||||
|       md.ok=false |       md.ok=false | ||||||
|     } |     } | ||||||
|     if (!md.ok) { |     if (!md.ok) { | ||||||
|       md = await fetch("/imprint_en.md"); |       md = await fetch("/imprint_en.md"); | ||||||
|  |       text = await md.text(); | ||||||
|     } |     } | ||||||
|     html = marked(await md.text()); |     html = marked(text); | ||||||
|   } |   } | ||||||
|   const promise = load(); |   const promise = load(); | ||||||
| </script> | </script> | ||||||
|   | |||||||
| @@ -6,13 +6,15 @@ | |||||||
|   let html = ""; |   let html = ""; | ||||||
|   async function load() { |   async function load() { | ||||||
|     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); |     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); | ||||||
|     if((await md.text()).includes("<meta")){ |     let text = (await md.text()).toString(); | ||||||
|  |     if(text.includes("<meta")){ | ||||||
|       md.ok=false |       md.ok=false | ||||||
|     } |     } | ||||||
|     if (!md.ok) { |     if (!md.ok) { | ||||||
|       md = await fetch("/privacy_en.md"); |       md = await fetch("/privacy_en.md"); | ||||||
|  |       text = await md.text(); | ||||||
|     } |     } | ||||||
|     html = marked(await md.text()); |     html = marked(text); | ||||||
|   } |   } | ||||||
|   const promise = load(); |   const promise = load(); | ||||||
| </script> | </script> | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ | |||||||
|     ) |     ) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Organization deleted", |           text: $_('organization-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
|   | |||||||
| @@ -10,6 +10,10 @@ | |||||||
|   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; |   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; | ||||||
|   import PromiseError from "../base/PromiseError.svelte"; |   import PromiseError from "../base/PromiseError.svelte"; | ||||||
|   import Select from "svelte-select"; |   import Select from "svelte-select"; | ||||||
|  |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|  |   import { tick } from "svelte"; | ||||||
|   $: delete_triggered = false; |   $: delete_triggered = false; | ||||||
|   $: address_valid_or_none = |   $: address_valid_or_none = | ||||||
|     (isAddress1Valid && iszipcodevalid && iscityvalid) || |     (isAddress1Valid && iszipcodevalid && iscityvalid) || | ||||||
| @@ -18,6 +22,9 @@ | |||||||
|   let original = ""; |   let original = ""; | ||||||
|   let original_object = {}; |   let original_object = {}; | ||||||
|   let contacts = []; |   let contacts = []; | ||||||
|  |   let valueCopy = null; | ||||||
|  |   let areaDom; | ||||||
|  |   let copied = false; | ||||||
|   export let params; |   export let params; | ||||||
|   $: editable = {}; |   $: editable = {}; | ||||||
|   $: contact = {}; |   $: contact = {}; | ||||||
| @@ -26,7 +33,11 @@ | |||||||
|   $: isAddress1Valid = editable.address?.address1?.trim().length !== 0; |   $: isAddress1Valid = editable.address?.address1?.trim().length !== 0; | ||||||
|   $: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; |   $: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; | ||||||
|   $: iscityvalid = editable.address?.city?.trim().length !== 0; |   $: iscityvalid = editable.address?.city?.trim().length !== 0; | ||||||
|   $: sponsoring_contracts_download_open = false; |   $: sponsoring_contracts_show = true; | ||||||
|  |   $: cards_show = true; | ||||||
|  |   $: certificates_show = true; | ||||||
|  |   $: generate_orgs = [original_object]; | ||||||
|  |   $: registrationLink = `${config.baseurl}/selfservice/register/${editable.registrationKey}`; | ||||||
|   const getContactLabel = (option) => |   const getContactLabel = (option) => | ||||||
|     option.firstname + " " + (option.middlename || "") + " " + option.lastname; |     option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
|   const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne( |   const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne( | ||||||
| @@ -60,14 +71,6 @@ | |||||||
|   }); |   }); | ||||||
|   let modal_open = false; |   let modal_open = false; | ||||||
|   let delete_org = {}; |   let delete_org = {}; | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|     if ( |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|     ) { |  | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|   function deleteOrganization() { |   function deleteOrganization() { | ||||||
|     RunnerOrganizationService.runnerOrganizationControllerRemove( |     RunnerOrganizationService.runnerOrganizationControllerRemove( | ||||||
|       original_object.id, |       original_object.id, | ||||||
| @@ -102,6 +105,7 @@ | |||||||
|         postdata |         postdata | ||||||
|       ) |       ) | ||||||
|         .then((resp) => { |         .then((resp) => { | ||||||
|  |           editable.registrationKey = resp.registrationKey; | ||||||
|           original_object = Object.assign({}, editable); |           original_object = Object.assign({}, editable); | ||||||
|           original = JSON.stringify(original_object); |           original = JSON.stringify(original_object); | ||||||
|           Toastify({ |           Toastify({ | ||||||
| @@ -114,58 +118,46 @@ | |||||||
|     } else { |     } else { | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   export let import_modal_open = false; |   async function copy() { | ||||||
|   async function generateSponsoringContract(locale) { |     if(!editable.registrationKey){ | ||||||
|     sponsoring_contracts_download_open = false; |       Toastify({ | ||||||
|     const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |         text: $_('you-have-to-save-your-changes-to-generate-a-link'), | ||||||
|       original_object.id |         duration: 500, | ||||||
|     ); |         backgroundColor: | ||||||
|     const toast = Toastify({ |           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|       text: $_("generating-pdf"), |       }).showToast(); | ||||||
|       duration: -1, |       return; | ||||||
|     }).showToast(); |     } | ||||||
|     fetch( |     valueCopy = registrationLink; | ||||||
|       `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |     await tick(); | ||||||
|       { |     areaDom.focus(); | ||||||
|         method: "POST", |     areaDom.select(); | ||||||
|         headers: { |     try { | ||||||
|           "Content-Type": "application/json", |       const successful = document.execCommand("copy"); | ||||||
|         }, |       if (!successful) { | ||||||
|         body: JSON.stringify(runners), |         throw new Error(); | ||||||
|       } |       } | ||||||
|     ) |       Toastify({ | ||||||
|       .then((response) => { |         text: $_("copied-link-to-clipboard"), | ||||||
|         if (response.status != "200") { |         duration: 500, | ||||||
|           toast.hideToast(); |         backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           Toastify({ |       }).showToast(); | ||||||
|             text: $_("pdf-generation-failed"), |       copied = true; | ||||||
|             duration: 3500, |     } catch (err) { | ||||||
|             backgroundColor: |       Toastify({ | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |         text: $_("error-whyile-copying-to-clipboard"), | ||||||
|           }).showToast(); |         duration: 500, | ||||||
|         } else { |         backgroundColor: | ||||||
|           return response.blob(); |           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|         } |       }).showToast(); | ||||||
|       }) |     } | ||||||
|       .then((blob) => { |     // we can notifi by event or storage about copy status | ||||||
|         const url = window.URL.createObjectURL(blob); |     valueCopy = null; | ||||||
|         let a = document.createElement("a"); |  | ||||||
|         a.href = url; |  | ||||||
|         a.download = "Sponsorings_" + original_object.name + ".pdf"; |  | ||||||
|         document.body.appendChild(a); |  | ||||||
|         a.click(); |  | ||||||
|         a.remove(); |  | ||||||
|         toast.hideToast(); |  | ||||||
|         Toastify({ |  | ||||||
|           text: $_("pdf-successfully-generated"), |  | ||||||
|           duration: 3500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => {}); |  | ||||||
|   } |   } | ||||||
|  |   export let import_modal_open = false; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | {#if valueCopy != null}<textarea bind:this={areaDom}>{valueCopy}</textarea>{/if} | ||||||
| <ImportRunnerModal | <ImportRunnerModal | ||||||
|   on:cancelDelete={(event) => { |   on:cancelDelete={(event) => { | ||||||
|     import_modal_open = false; |     import_modal_open = false; | ||||||
| @@ -182,64 +174,11 @@ | |||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> |     <div class="mb-8 text-3xl font-extrabold leading-tight"> | ||||||
|       {original_object.name} |       {original_object.name} | ||||||
|       <span data-id="org_actions_${editable.id}"> |       <span data-id="org_actions_${editable.id}"> | ||||||
|         <div id="sponsoring:dropdown" class="relative inline-block"> |         <GenerateSponsoringContracts | ||||||
|           <div> |           bind:sponsoring_contracts_show | ||||||
|             <button |           bind:generate_orgs /> | ||||||
|               on:click={() => { |         <GenerateRunnerCards bind:cards_show bind:generate_orgs /> | ||||||
|                 sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |         <GenerateRunnerCertificates bind:certificates_show bind:generate_orgs /> | ||||||
|               }} |  | ||||||
|               type="button" |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |  | ||||||
|               id="options-menu" |  | ||||||
|               aria-haspopup="true" |  | ||||||
|               aria-expanded="true"> |  | ||||||
|               {$_('generate-sponsoring-contracts')} |  | ||||||
|               <svg |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                 width="24" |  | ||||||
|                 height="24" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 class="-mr-1 ml-2 h-5 w-5"><path |  | ||||||
|                   fill="none" |  | ||||||
|                   d="M0 0h24v24H0z" /> |  | ||||||
|                 <path |  | ||||||
|                   fill="currentColor" |  | ||||||
|                   d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|           {#if sponsoring_contracts_download_open} |  | ||||||
|             <div |  | ||||||
|               class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|               id="sponsoring:dropdown:menu"> |  | ||||||
|               <div |  | ||||||
|                 class="py-1" |  | ||||||
|                 role="menu" |  | ||||||
|                 aria-orientation="vertical" |  | ||||||
|                 aria-labelledby="options-menu"> |  | ||||||
|                 <span |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('de'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('german')} |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('en'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('english')} |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           {/if} |  | ||||||
|         </div> |  | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} |         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
| @@ -250,7 +189,7 @@ | |||||||
|             {$_('import-runners')} |             {$_('import-runners')} | ||||||
|           </button> |           </button> | ||||||
|         {/if} |         {/if} | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('USER:DELETE')} |         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:DELETE')} | ||||||
|           {#if delete_triggered} |           {#if delete_triggered} | ||||||
|             <button |             <button | ||||||
|               on:click={deleteOrganization} |               on:click={deleteOrganization} | ||||||
| @@ -382,99 +321,151 @@ | |||||||
|         on:select={(selectedValue) => (editable.contact = selectedValue.detail.value)} |         on:select={(selectedValue) => (editable.contact = selectedValue.detail.value)} | ||||||
|         on:clear={() => (editable.contact = null)} /> |         on:clear={() => (editable.contact = null)} /> | ||||||
|     </div> |     </div> | ||||||
|     <!--  --> |     <div> | ||||||
|     <div class="flex items-start mt-2"> |       <div class="flex items-start mt-2"> | ||||||
|       <div class="flex items-center h-5"> |         <div class="flex items-center h-5"> | ||||||
|         <input |           <input | ||||||
|           bind:checked={editable.address_checked} |             bind:checked={editable.registrationEnabled} | ||||||
|           id="comments" |             id="toggle_selfservice_feature" | ||||||
|           name="comments" |             name="toggle_selfservice_feature" | ||||||
|           type="checkbox" |             type="checkbox" | ||||||
|           class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |             class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | ||||||
|  |         </div> | ||||||
|  |         <div class="ml-3 text-sm"> | ||||||
|  |           <label | ||||||
|  |             for="toggle_selfservice_feature" | ||||||
|  |             class="font-medium text-gray-700">{$_('selfservice-registration')}</label> | ||||||
|  |         </div> | ||||||
|       </div> |       </div> | ||||||
|       <div class="ml-3 text-sm"> |       <div> | ||||||
|         <label |         {#if editable.registrationEnabled} | ||||||
|           for="comments" |           <div class="text-sm w-full"> | ||||||
|           class="font-medium text-gray-700">{$_('address')}</label> |             <div on:click={copy} class="inline-flex w-full"> | ||||||
|  |               <p | ||||||
|  |                 name="token" | ||||||
|  |                 class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 p-2"> | ||||||
|  |                 {#if editable.registrationKey} | ||||||
|  |                 {registrationLink} | ||||||
|  |                 {:else} | ||||||
|  |                 {$_('you-have-to-save-your-changes-to-generate-a-link')} | ||||||
|  |                 {/if} | ||||||
|  |               </p> | ||||||
|  |               <div | ||||||
|  |                 class="bg-gray-200 border-gray-300 border-t border-b border-r text-black rounded-r-md sm:text-sm p-2 mt-1 cursor-pointer"> | ||||||
|  |                 <svg | ||||||
|  |                   xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                   viewBox="0 0 24 24" | ||||||
|  |                   width="24" | ||||||
|  |                   height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                   <path | ||||||
|  |                     fill="currentColor" | ||||||
|  |                     d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" /></svg> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |             {#if editable.registrationKey} | ||||||
|  |             <p class="text-gray-500 text-xs"> | ||||||
|  |               {$_('click-to-copy-the-link-into-your-clipboard')} | ||||||
|  |             </p> | ||||||
|  |             {/if} | ||||||
|  |           </div> | ||||||
|  |         {/if} | ||||||
|  |         <!--  --> | ||||||
|  |         <div> | ||||||
|  |           <div class="flex items-start mt-2"> | ||||||
|  |             <div class="flex items-center h-5"> | ||||||
|  |               <input | ||||||
|  |                 bind:checked={editable.address_checked} | ||||||
|  |                 id="toggle_address_checkbox" | ||||||
|  |                 name="toggle_address_checkbox" | ||||||
|  |                 type="checkbox" | ||||||
|  |                 class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | ||||||
|  |             </div> | ||||||
|  |             <div class="ml-3 text-sm"> | ||||||
|  |               <label | ||||||
|  |                 for="toggle_address_checkbox" | ||||||
|  |                 class="font-medium text-gray-700">{$_('address')}</label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         {#if editable.address_checked === true} | ||||||
|  |           <div class="col-span-6"> | ||||||
|  |             <label | ||||||
|  |               for="address1" | ||||||
|  |               class="block text-sm font-medium text-gray-700">{$_('address')}</label> | ||||||
|  |             <input | ||||||
|  |               autocomplete="off" | ||||||
|  |               placeholder="Address" | ||||||
|  |               class:border-red-500={!isAddress1Valid} | ||||||
|  |               class:focus:border-red-500={!isAddress1Valid} | ||||||
|  |               class:focus:ring-red-500={!isAddress1Valid} | ||||||
|  |               bind:value={editable.address.address1} | ||||||
|  |               type="text" | ||||||
|  |               name="address1" | ||||||
|  |               class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|  |             {#if !isAddress1Valid} | ||||||
|  |               <span | ||||||
|  |                 class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |                 {$_('address-is-required')} | ||||||
|  |               </span> | ||||||
|  |             {/if} | ||||||
|  |           </div> | ||||||
|  |           <div class="col-span-6"> | ||||||
|  |             <label | ||||||
|  |               for="address2" | ||||||
|  |               class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> | ||||||
|  |             <input | ||||||
|  |               autocomplete="off" | ||||||
|  |               placeholder={$_('apartment-suite-etc')} | ||||||
|  |               bind:value={editable.address.address2} | ||||||
|  |               type="text" | ||||||
|  |               name="address2" | ||||||
|  |               class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|  |           </div> | ||||||
|  |           <div class="col-span-6"> | ||||||
|  |             <label | ||||||
|  |               for="zipcode" | ||||||
|  |               class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> | ||||||
|  |             <input | ||||||
|  |               autocomplete="off" | ||||||
|  |               placeholder={$_('zip-postal-code')} | ||||||
|  |               class:border-red-500={!iszipcodevalid} | ||||||
|  |               class:focus:border-red-500={!iszipcodevalid} | ||||||
|  |               class:focus:ring-red-500={!iszipcodevalid} | ||||||
|  |               bind:value={editable.address.postalcode} | ||||||
|  |               type="text" | ||||||
|  |               name="zipcode" | ||||||
|  |               class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|  |             {#if !iszipcodevalid} | ||||||
|  |               <span | ||||||
|  |                 class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |                 {$_('valid-zipcode-postal-code-is-required')} | ||||||
|  |               </span> | ||||||
|  |             {/if} | ||||||
|  |           </div> | ||||||
|  |           <div class="col-span-6"> | ||||||
|  |             <label | ||||||
|  |               for="city" | ||||||
|  |               class="block text-sm font-medium text-gray-700">{$_('city')}</label> | ||||||
|  |             <input | ||||||
|  |               autocomplete="off" | ||||||
|  |               placeholder={$_('city')} | ||||||
|  |               class:border-red-500={!iscityvalid} | ||||||
|  |               class:focus:border-red-500={!iscityvalid} | ||||||
|  |               class:focus:ring-red-500={!iscityvalid} | ||||||
|  |               bind:value={editable.address.city} | ||||||
|  |               type="text" | ||||||
|  |               name="city" | ||||||
|  |               class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|  |             {#if !iscityvalid} | ||||||
|  |               <span | ||||||
|  |                 class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | ||||||
|  |                 {$_('valid-city-is-required')} | ||||||
|  |               </span> | ||||||
|  |             {/if} | ||||||
|  |           </div> | ||||||
|  |         {/if} | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     {#if editable.address_checked === true} |  | ||||||
|       <div class="col-span-6"> |  | ||||||
|         <label |  | ||||||
|           for="address1" |  | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('address')}</label> |  | ||||||
|         <input |  | ||||||
|           autocomplete="off" |  | ||||||
|           placeholder="Address" |  | ||||||
|           class:border-red-500={!isAddress1Valid} |  | ||||||
|           class:focus:border-red-500={!isAddress1Valid} |  | ||||||
|           class:focus:ring-red-500={!isAddress1Valid} |  | ||||||
|           bind:value={editable.address.address1} |  | ||||||
|           type="text" |  | ||||||
|           name="address1" |  | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |  | ||||||
|         {#if !isAddress1Valid} |  | ||||||
|           <span |  | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |  | ||||||
|             {$_('address-is-required')} |  | ||||||
|           </span> |  | ||||||
|         {/if} |  | ||||||
|       </div> |  | ||||||
|       <div class="col-span-6"> |  | ||||||
|         <label |  | ||||||
|           for="address2" |  | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> |  | ||||||
|         <input |  | ||||||
|           autocomplete="off" |  | ||||||
|           placeholder={$_('apartment-suite-etc')} |  | ||||||
|           bind:value={editable.address.address2} |  | ||||||
|           type="text" |  | ||||||
|           name="address2" |  | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |  | ||||||
|       </div> |  | ||||||
|       <div class="col-span-6"> |  | ||||||
|         <label |  | ||||||
|           for="zipcode" |  | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> |  | ||||||
|         <input |  | ||||||
|           autocomplete="off" |  | ||||||
|           placeholder={$_('zip-postal-code')} |  | ||||||
|           class:border-red-500={!iszipcodevalid} |  | ||||||
|           class:focus:border-red-500={!iszipcodevalid} |  | ||||||
|           class:focus:ring-red-500={!iszipcodevalid} |  | ||||||
|           bind:value={editable.address.postalcode} |  | ||||||
|           type="text" |  | ||||||
|           name="zipcode" |  | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |  | ||||||
|         {#if !iszipcodevalid} |  | ||||||
|           <span |  | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |  | ||||||
|             {$_('valid-zipcode-postal-code-is-required')} |  | ||||||
|           </span> |  | ||||||
|         {/if} |  | ||||||
|       </div> |  | ||||||
|       <div class="col-span-6"> |  | ||||||
|         <label |  | ||||||
|           for="city" |  | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('city')}</label> |  | ||||||
|         <input |  | ||||||
|           autocomplete="off" |  | ||||||
|           placeholder={$_('city')} |  | ||||||
|           class:border-red-500={!iscityvalid} |  | ||||||
|           class:focus:border-red-500={!iscityvalid} |  | ||||||
|           class:focus:ring-red-500={!iscityvalid} |  | ||||||
|           bind:value={editable.address.city} |  | ||||||
|           type="text" |  | ||||||
|           name="city" |  | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |  | ||||||
|         {#if !iscityvalid} |  | ||||||
|           <span |  | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |  | ||||||
|             {$_('valid-city-is-required')} |  | ||||||
|           </span> |  | ||||||
|         {/if} |  | ||||||
|       </div> |  | ||||||
|     {/if} |  | ||||||
|   </section> |   </section> | ||||||
| {:else} | {:else} | ||||||
|   {#await promise} |   {#await promise} | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   let modal_open = false; |   let modal_open = false; | ||||||
|   let delete_org = {}; |   let delete_org = {}; | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |   import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
| @@ -7,9 +8,16 @@ | |||||||
|   import OrgsEmptyState from "./OrgsEmptyState.svelte"; |   import OrgsEmptyState from "./OrgsEmptyState.svelte"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import ConfirmOrgDeletion from "./ConfirmOrgDeletion.svelte"; |   import ConfirmOrgDeletion from "./ConfirmOrgDeletion.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   $: sponsoring_contracts_download_open = false; |   $: sponsoring_contracts_show = current_organizations.some((r) => r.is_selected === true); | ||||||
|  |   $: cards_show = current_organizations.some((r) => r.is_selected === true); | ||||||
|  |   $: generate_orgs = current_organizations.filter((r) => r.is_selected === true); | ||||||
|  |   $: certificates_show = current_organizations.some( | ||||||
|  |     (r) => r.is_selected === true | ||||||
|  |   ); | ||||||
|   export let current_organizations = []; |   export let current_organizations = []; | ||||||
|  |  | ||||||
|   const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then( |   const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
| @@ -17,72 +25,6 @@ | |||||||
|       current_organizations = val; |       current_organizations = val; | ||||||
|     } |     } | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|     if ( |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|     ) { |  | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   async function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const orgs = current_organizations.filter((r) => r.is_selected === true); |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdfs"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     let count = 0; |  | ||||||
|     for await (const o of orgs) { |  | ||||||
|       const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |  | ||||||
|         o.id |  | ||||||
|       ); |  | ||||||
|       fetch( |  | ||||||
|         `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|         { |  | ||||||
|           method: "POST", |  | ||||||
|           headers: { |  | ||||||
|             "Content-Type": "application/json", |  | ||||||
|           }, |  | ||||||
|           body: JSON.stringify(runners), |  | ||||||
|         } |  | ||||||
|       ) |  | ||||||
|         .then((response) => { |  | ||||||
|           if (response.status != "200") { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdf-generation-failed"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: |  | ||||||
|                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } else { |  | ||||||
|             return response.blob(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .then((blob) => { |  | ||||||
|           count++; |  | ||||||
|           const url = window.URL.createObjectURL(blob); |  | ||||||
|           let a = document.createElement("a"); |  | ||||||
|           a.href = url; |  | ||||||
|           a.download = "Sponsorings_" + o.name + ".pdf"; |  | ||||||
|           document.body.appendChild(a); |  | ||||||
|           a.click(); |  | ||||||
|           a.remove(); |  | ||||||
|           if (count === orgs.length) { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdfs-successfully-generated"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .catch((err) => {}); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <ConfirmOrgDeletion | <ConfirmOrgDeletion | ||||||
| @@ -111,56 +53,15 @@ | |||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="gridjs-input gridjs-search-input mb-4" /> | ||||||
|       <div class="h-12"> |       <div class="h-12"> | ||||||
|         {#if current_organizations.some((r) => r.is_selected === true)} |         <GenerateSponsoringContracts | ||||||
|         <div id="sponsoring:dropdown" class="relative inline-block"> |             bind:sponsoring_contracts_show | ||||||
|           <div> |             bind:generate_orgs /> | ||||||
|             <button |         <GenerateRunnerCards | ||||||
|               on:click={() => { |             bind:cards_show | ||||||
|                 sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |             bind:generate_orgs /> | ||||||
|               }} |         <GenerateRunnerCertificates | ||||||
|               type="button" |             bind:certificates_show | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |             bind:generate_orgs /> | ||||||
|               id="options-menu" |  | ||||||
|               aria-haspopup="true" |  | ||||||
|               aria-expanded="true"> |  | ||||||
|               {$_('generate-sponsoring-contracts')} |  | ||||||
|               <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" class="-mr-1 ml-2 h-5 w-5"><path fill="none" d="M0 0h24v24H0z"/><path fill="currentColor" d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z"/></svg> |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|           {#if sponsoring_contracts_download_open} |  | ||||||
|             <div |  | ||||||
|               class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|               id="sponsoring:dropdown:menu"> |  | ||||||
|               <div |  | ||||||
|                 class="py-1" |  | ||||||
|                 role="menu" |  | ||||||
|                 aria-orientation="vertical" |  | ||||||
|                 aria-labelledby="options-menu"> |  | ||||||
|                 <span |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('de'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('german')} |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('en'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('english')} |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           {/if} |  | ||||||
|         </div> |  | ||||||
|         {/if} |  | ||||||
|       </div> |       </div> | ||||||
|       <div |       <div | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> | ||||||
|   | |||||||
							
								
								
									
										418
									
								
								src/components/pdf_generation/GenerateRunnerCards.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										418
									
								
								src/components/pdf_generation/GenerateRunnerCards.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,418 @@ | |||||||
|  | <script> | ||||||
|  |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |   import { | ||||||
|  |     RunnerCardService, | ||||||
|  |     RunnerOrganizationService, | ||||||
|  |     RunnerTeamService, | ||||||
|  |   } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { init } from "@paralleldrive/cuid2"; | ||||||
|  |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  |  | ||||||
|  |   export let cards_show = false; | ||||||
|  |   export let generate_cards = []; | ||||||
|  |   export let generate_runners = []; | ||||||
|  |   export let generate_orgs = []; | ||||||
|  |   export let generate_teams = []; | ||||||
|  |   $: cards_dropdown_open = false; | ||||||
|  |   document.addEventListener("click", function (e) { | ||||||
|  |     if ( | ||||||
|  |       e.target.parentNode?.parentNode?.id != "cards:dropdown" && | ||||||
|  |       e.target.parentNode?.parentNode?.id != "cards:dropdown:menu" | ||||||
|  |     ) { | ||||||
|  |       cards_dropdown_open = false; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function generateRunnerCards(locale) { | ||||||
|  |     cards_dropdown_open = false; | ||||||
|  |  | ||||||
|  |     if (generate_orgs.length > 0) { | ||||||
|  |       generateOrgCards(locale); | ||||||
|  |     } else if (generate_teams.length > 0) { | ||||||
|  |       generateTeamCards(locale); | ||||||
|  |     } else if (generate_runners.length > 0) { | ||||||
|  |       generateRunnersCards(locale); | ||||||
|  |     } else { | ||||||
|  |       generateCards(locale); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function generateCards(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdf"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     fetch( | ||||||
|  |       `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify(generate_cards), | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |       .then((response) => { | ||||||
|  |         if (response.status != "200") { | ||||||
|  |           toast.hideToast(); | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("pdf-generation-failed"), | ||||||
|  |             duration: 3500, | ||||||
|  |             backgroundColor: | ||||||
|  |               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |           }).showToast(); | ||||||
|  |         } else { | ||||||
|  |           return response.blob(); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .then((blob) => { | ||||||
|  |         const url = window.URL.createObjectURL(blob); | ||||||
|  |         let a = document.createElement("a"); | ||||||
|  |         a.href = url; | ||||||
|  |         a.download = `${$_("runnercards")}-${locale}-${createId()}.pdf`; | ||||||
|  |         document.body.appendChild(a); | ||||||
|  |         a.click(); | ||||||
|  |         a.remove(); | ||||||
|  |         toast.hideToast(); | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_("pdf-successfully-generated"), | ||||||
|  |           duration: 3500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => { | ||||||
|  |         console.error(err); | ||||||
|  |       }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateRunnersCards(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdf"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  |     let cards = []; | ||||||
|  |     for (let runner of generate_runners) { | ||||||
|  |       let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  |       if (!card) { | ||||||
|  |         card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  |           runner: runner.id, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |       cards.push(card); | ||||||
|  |     } | ||||||
|  |     fetch( | ||||||
|  |       `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify(cards), | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |       .then((response) => { | ||||||
|  |         if (response.status != "200") { | ||||||
|  |           toast.hideToast(); | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("pdf-generation-failed"), | ||||||
|  |             duration: 3500, | ||||||
|  |             backgroundColor: | ||||||
|  |               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |           }).showToast(); | ||||||
|  |         } else { | ||||||
|  |           return response.blob(); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .then((blob) => { | ||||||
|  |         const url = window.URL.createObjectURL(blob); | ||||||
|  |         let a = document.createElement("a"); | ||||||
|  |         a.href = url; | ||||||
|  |         if (generate_runners.length == 1) { | ||||||
|  |           a.download = `${$_("runnercards")}_${generate_runners[0].firstname}_${ | ||||||
|  |             generate_runners[0].lastname | ||||||
|  |           }-${locale}-${createId()}.pdf`; | ||||||
|  |         } else { | ||||||
|  |           a.download = `${$_("runnercards")}-${locale}-${createId()}.pdf`; | ||||||
|  |         } | ||||||
|  |         document.body.appendChild(a); | ||||||
|  |         a.click(); | ||||||
|  |         a.remove(); | ||||||
|  |         toast.hideToast(); | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_("pdf-successfully-generated"), | ||||||
|  |           duration: 3500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => {}); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateTeamCards(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdfs"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     let count = 0; | ||||||
|  |     const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  |     for (const t of generate_teams) { | ||||||
|  |       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |         t.id | ||||||
|  |       ); | ||||||
|  |       let cards = []; | ||||||
|  |       for (let runner of runners) { | ||||||
|  |         let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  |         if (!card) { | ||||||
|  |           card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  |             runner: runner.id, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |         cards.push(card); | ||||||
|  |       } | ||||||
|  |       fetch( | ||||||
|  |         `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(cards), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           count++; | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("runnercards")}_${t.name}-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === generate_teams.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateOrgCards(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdfs"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  |     let count = 0; | ||||||
|  |     let count_orgs = 0; | ||||||
|  |     for (const o of generate_orgs) { | ||||||
|  |       count_orgs++; | ||||||
|  |       let count = 0; | ||||||
|  |       let runners = | ||||||
|  |         await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  |           o.id, | ||||||
|  |           true | ||||||
|  |         ); | ||||||
|  |       let cards = []; | ||||||
|  |       for (let runner of runners) { | ||||||
|  |         let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  |         if (!card) { | ||||||
|  |           card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  |             runner: runner.id, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |         cards.push(card); | ||||||
|  |       } | ||||||
|  |       await fetch( | ||||||
|  |         `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(cards), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("runnercards")}_${o.name}_direct-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === o.teams.length && count_orgs === generate_orgs.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             console.log("here"); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |       for (const t of o.teams) { | ||||||
|  |         count++; | ||||||
|  |         let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |           t.id | ||||||
|  |         ); | ||||||
|  |         let cards = []; | ||||||
|  |         for (let runner of runners) { | ||||||
|  |           let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  |           if (!card) { | ||||||
|  |             card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  |               runner: runner.id, | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |           cards.push(card); | ||||||
|  |         } | ||||||
|  |         await fetch( | ||||||
|  |           `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |           { | ||||||
|  |             method: "POST", | ||||||
|  |             headers: { | ||||||
|  |               "Content-Type": "application/json", | ||||||
|  |             }, | ||||||
|  |             body: JSON.stringify(cards), | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |           .then((response) => { | ||||||
|  |             if (response.status != "200") { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdf-generation-failed"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: | ||||||
|  |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } else { | ||||||
|  |               return response.blob(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .then((blob) => { | ||||||
|  |             const url = window.URL.createObjectURL(blob); | ||||||
|  |             let a = document.createElement("a"); | ||||||
|  |             a.href = url; | ||||||
|  |             a.download = `${$_("runnercards")}_${o.name}_${ | ||||||
|  |               t.name | ||||||
|  |             }-${locale}-${createId()}.pdf`; | ||||||
|  |             document.body.appendChild(a); | ||||||
|  |             a.click(); | ||||||
|  |             a.remove(); | ||||||
|  |             if ( | ||||||
|  |               count === o.teams.length && | ||||||
|  |               count_orgs === generate_orgs.length | ||||||
|  |             ) { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdfs-successfully-generated"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .catch((err) => {}); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if cards_show} | ||||||
|  |   <div id="cards:dropdown" class="relative inline-block"> | ||||||
|  |     <div> | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           cards_dropdown_open = !cards_dropdown_open; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" | ||||||
|  |         id="options-menu" | ||||||
|  |         aria-haspopup="true" | ||||||
|  |         aria-expanded="true" | ||||||
|  |       > | ||||||
|  |         {$_("generate-runnercards")} | ||||||
|  |         <svg | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           width="24" | ||||||
|  |           height="24" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           class="-mr-1 ml-2 h-5 w-5" | ||||||
|  |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |           <path | ||||||
|  |             fill="currentColor" | ||||||
|  |             d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |     {#if cards_dropdown_open} | ||||||
|  |       <div | ||||||
|  |         class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5 z-10" | ||||||
|  |         id="cards:dropdown:menu" | ||||||
|  |       > | ||||||
|  |         <div | ||||||
|  |           class="py-1" | ||||||
|  |           role="menu" | ||||||
|  |           aria-orientation="vertical" | ||||||
|  |           aria-labelledby="options-menu" | ||||||
|  |         > | ||||||
|  |           <span class="block w-full text-left px-4 py-2 text-sm text-gray-700" | ||||||
|  |             >{$_("select-language")}</span | ||||||
|  |           > | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateRunnerCards("de"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("german")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateRunnerCards("en"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("english")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     {/if} | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										355
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										355
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,355 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { | ||||||
|  |     DonationService, | ||||||
|  |     RunnerTeamService, | ||||||
|  |     RunnerOrganizationService, | ||||||
|  |   } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { init } from "@paralleldrive/cuid2"; | ||||||
|  |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  |  | ||||||
|  |   export let certificates_show = false; | ||||||
|  |   export let generate_runners = []; | ||||||
|  |   export let generate_orgs = []; | ||||||
|  |   export let generate_teams = []; | ||||||
|  |   $: certificates_dropdown_open = false; | ||||||
|  |   document.addEventListener("click", function (e) { | ||||||
|  |     if ( | ||||||
|  |       e.target.parentNode?.parentNode?.id != "certificates:dropdown" && | ||||||
|  |       e.target.parentNode?.parentNode?.id != "certificates:dropdown:menu" | ||||||
|  |     ) { | ||||||
|  |       certificates_dropdown_open = false; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function generateCertificates(locale) { | ||||||
|  |     certificates_dropdown_open = false; | ||||||
|  |  | ||||||
|  |     if (generate_orgs.length > 0) { | ||||||
|  |       generateOrgCertificates(locale); | ||||||
|  |     } else if (generate_teams.length > 0) { | ||||||
|  |       generateTeamCertificates(locale); | ||||||
|  |     } else { | ||||||
|  |       generateRunnerCertificates(locale); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateRunnerCertificates(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdf"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     const current_donations = | ||||||
|  |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|  |     let certificateRunners = []; | ||||||
|  |     for (let runner of generate_runners) { | ||||||
|  |       runner.distanceDonations = | ||||||
|  |         current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  |       console.log(runner.distanceDonations); | ||||||
|  |       certificateRunners.push(runner); | ||||||
|  |     } | ||||||
|  |     fetch( | ||||||
|  |       `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify(certificateRunners), | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |       .then((response) => { | ||||||
|  |         if (response.status != "200") { | ||||||
|  |           toast.hideToast(); | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("pdf-generation-failed"), | ||||||
|  |             duration: 3500, | ||||||
|  |             backgroundColor: | ||||||
|  |               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |           }).showToast(); | ||||||
|  |         } else { | ||||||
|  |           return response.blob(); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .then((blob) => { | ||||||
|  |         const url = window.URL.createObjectURL(blob); | ||||||
|  |         let a = document.createElement("a"); | ||||||
|  |         a.href = url; | ||||||
|  |         if (generate_runners.length == 1) { | ||||||
|  |           a.download = `${$_("certificates")}_${ | ||||||
|  |             generate_runners[0].firstname | ||||||
|  |           }_${generate_runners[0].lastname}-${locale}-${createId()}.pdf`; | ||||||
|  |         } else { | ||||||
|  |           a.download = `${$_("certificates")}-${locale}.pdf`; | ||||||
|  |         } | ||||||
|  |         document.body.appendChild(a); | ||||||
|  |         a.click(); | ||||||
|  |         a.remove(); | ||||||
|  |         toast.hideToast(); | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_("pdf-successfully-generated"), | ||||||
|  |           duration: 3500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => {}); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateTeamCertificates(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdfs"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     let count = 0; | ||||||
|  |     const current_donations = | ||||||
|  |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|  |     for (const t of generate_teams) { | ||||||
|  |       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |         t.id | ||||||
|  |       ); | ||||||
|  |       let certificateRunners = []; | ||||||
|  |       for (let runner of runners) { | ||||||
|  |         runner.distanceDonations = | ||||||
|  |           current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  |         certificateRunners.push(runner); | ||||||
|  |       } | ||||||
|  |       fetch( | ||||||
|  |         `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(certificateRunners), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           count++; | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("certificates")}_${t.name}-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === generate_teams.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateOrgCertificates(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdfs"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     const current_donations = | ||||||
|  |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|  |     let count = 0; | ||||||
|  |     let count_orgs = 0; | ||||||
|  |     for (const o of generate_orgs) { | ||||||
|  |       count_orgs++; | ||||||
|  |       let count = 0; | ||||||
|  |       let runners = | ||||||
|  |         await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  |           o.id, | ||||||
|  |           true | ||||||
|  |         ); | ||||||
|  |       let certificateRunners = []; | ||||||
|  |       for (let runner of runners) { | ||||||
|  |         runner.distanceDonations = | ||||||
|  |           current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  |         certificateRunners.push(runner); | ||||||
|  |       } | ||||||
|  |       await fetch( | ||||||
|  |         `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(certificateRunners), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("certificates")}_${o.name}-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === o.teams.length && count_orgs === generate_orgs.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             console.log("here"); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |       for (const t of o.teams) { | ||||||
|  |         count++; | ||||||
|  |         let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |           t.id | ||||||
|  |         ); | ||||||
|  |         let certificateRunners = []; | ||||||
|  |         for (let runner of runners) { | ||||||
|  |           runner.distanceDonations = | ||||||
|  |             current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  |           certificateRunners.push(runner); | ||||||
|  |         } | ||||||
|  |         await fetch( | ||||||
|  |           `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |           { | ||||||
|  |             method: "POST", | ||||||
|  |             headers: { | ||||||
|  |               "Content-Type": "application/json", | ||||||
|  |             }, | ||||||
|  |             body: JSON.stringify(certificateRunners), | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |           .then((response) => { | ||||||
|  |             if (response.status != "200") { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdf-generation-failed"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: | ||||||
|  |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } else { | ||||||
|  |               return response.blob(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .then((blob) => { | ||||||
|  |             const url = window.URL.createObjectURL(blob); | ||||||
|  |             let a = document.createElement("a"); | ||||||
|  |             a.href = url; | ||||||
|  |             a.download = `${$_("certificates")}_${o.name}_${ | ||||||
|  |               t.name | ||||||
|  |             }-${locale}-${createId()}.pdf`; | ||||||
|  |             document.body.appendChild(a); | ||||||
|  |             a.click(); | ||||||
|  |             a.remove(); | ||||||
|  |             if ( | ||||||
|  |               count === o.teams.length && | ||||||
|  |               count_orgs === generate_orgs.length | ||||||
|  |             ) { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdfs-successfully-generated"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .catch((err) => {}); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if certificates_show} | ||||||
|  |   <div id="certificates:dropdown" class="relative inline-block"> | ||||||
|  |     <div> | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           certificates_dropdown_open = !certificates_dropdown_open; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" | ||||||
|  |         id="options-menu" | ||||||
|  |         aria-haspopup="true" | ||||||
|  |         aria-expanded="true" | ||||||
|  |       > | ||||||
|  |         {$_("generate-runner-certificates")} | ||||||
|  |         <svg | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           width="24" | ||||||
|  |           height="24" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           class="-mr-1 ml-2 h-5 w-5" | ||||||
|  |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |           <path | ||||||
|  |             fill="currentColor" | ||||||
|  |             d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |     {#if certificates_dropdown_open} | ||||||
|  |       <div | ||||||
|  |         class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5 z-10" | ||||||
|  |         id="certificates:dropdown:menu" | ||||||
|  |       > | ||||||
|  |         <div | ||||||
|  |           class="py-1" | ||||||
|  |           role="menu" | ||||||
|  |           aria-orientation="vertical" | ||||||
|  |           aria-labelledby="options-menu" | ||||||
|  |         > | ||||||
|  |           <span class="block w-full text-left px-4 py-2 text-sm text-gray-700" | ||||||
|  |             >{$_("select-language")}</span | ||||||
|  |           > | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateCertificates("de"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("german")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateCertificates("en"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("english")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     {/if} | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										324
									
								
								src/components/pdf_generation/GenerateSponsoringContracts.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										324
									
								
								src/components/pdf_generation/GenerateSponsoringContracts.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,324 @@ | |||||||
|  | <script> | ||||||
|  |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |   import { | ||||||
|  |     RunnerOrganizationService, | ||||||
|  |     RunnerTeamService, | ||||||
|  |   } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { init } from "@paralleldrive/cuid2"; | ||||||
|  |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  |  | ||||||
|  |   export let sponsoring_contracts_show = false; | ||||||
|  |   export let generate_runners = []; | ||||||
|  |   export let generate_orgs = []; | ||||||
|  |   export let generate_teams = []; | ||||||
|  |   $: sponsoring_contracts_download_open = false; | ||||||
|  |   document.addEventListener("click", function (e) { | ||||||
|  |     if ( | ||||||
|  |       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && | ||||||
|  |       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" | ||||||
|  |     ) { | ||||||
|  |       sponsoring_contracts_download_open = false; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function generateSponsoringContract(locale) { | ||||||
|  |     sponsoring_contracts_download_open = false; | ||||||
|  |  | ||||||
|  |     if (generate_orgs.length > 0) { | ||||||
|  |       generateOrgContracts(locale); | ||||||
|  |     } else if (generate_teams.length > 0) { | ||||||
|  |       generateTeamContracts(locale); | ||||||
|  |     } else { | ||||||
|  |       generateRunnerContracts(locale); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateTeamContracts(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdfs"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     let count = 0; | ||||||
|  |     for (const t of generate_teams) { | ||||||
|  |       count++; | ||||||
|  |       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |         t.id | ||||||
|  |       ); | ||||||
|  |       fetch( | ||||||
|  |         `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(runners), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("sponsorings")}_${t.name}-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === generate_teams.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function generateOrgContracts(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdf"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     let count_orgs = 0; | ||||||
|  |     for (const o of generate_orgs) { | ||||||
|  |       count_orgs++; | ||||||
|  |       let count = 0; | ||||||
|  |       let runners = | ||||||
|  |         await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  |           o.id, | ||||||
|  |           true | ||||||
|  |         ); | ||||||
|  |       await fetch( | ||||||
|  |         `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |         { | ||||||
|  |           method: "POST", | ||||||
|  |           headers: { | ||||||
|  |             "Content-Type": "application/json", | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify(runners), | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |         .then((response) => { | ||||||
|  |           if (response.status != "200") { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdf-generation-failed"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } else { | ||||||
|  |             return response.blob(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .then((blob) => { | ||||||
|  |           const url = window.URL.createObjectURL(blob); | ||||||
|  |           let a = document.createElement("a"); | ||||||
|  |           a.href = url; | ||||||
|  |           a.download = `${$_("sponsorings")}_${o.name}_direct-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
|  |           a.click(); | ||||||
|  |           a.remove(); | ||||||
|  |           if (count === o.teams.length && count_orgs === generate_orgs.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|  |             console.log("here"); | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("pdfs-successfully-generated"), | ||||||
|  |               duration: 3500, | ||||||
|  |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |             }).showToast(); | ||||||
|  |           } | ||||||
|  |         }) | ||||||
|  |         .catch((err) => {}); | ||||||
|  |       for (const t of o.teams) { | ||||||
|  |         count++; | ||||||
|  |         let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  |           t.id | ||||||
|  |         ); | ||||||
|  |         await fetch( | ||||||
|  |           `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |           { | ||||||
|  |             method: "POST", | ||||||
|  |             headers: { | ||||||
|  |               "Content-Type": "application/json", | ||||||
|  |             }, | ||||||
|  |             body: JSON.stringify(runners), | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |           .then((response) => { | ||||||
|  |             if (response.status != "200") { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdf-generation-failed"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: | ||||||
|  |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } else { | ||||||
|  |               return response.blob(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .then((blob) => { | ||||||
|  |             const url = window.URL.createObjectURL(blob); | ||||||
|  |             let a = document.createElement("a"); | ||||||
|  |             a.href = url; | ||||||
|  |             a.download = `${$_("sponsorings")}_${o.name}_${ | ||||||
|  |               t.name | ||||||
|  |             }-${locale}-${createId()}.pdf`; | ||||||
|  |             document.body.appendChild(a); | ||||||
|  |             a.click(); | ||||||
|  |             a.remove(); | ||||||
|  |             if ( | ||||||
|  |               count === o.teams.length && | ||||||
|  |               count_orgs === generate_orgs.length | ||||||
|  |             ) { | ||||||
|  |               toast.hideToast(); | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("pdfs-successfully-generated"), | ||||||
|  |                 duration: 3500, | ||||||
|  |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |               }).showToast(); | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |           .catch((err) => {}); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function generateRunnerContracts(locale) { | ||||||
|  |     const toast = Toastify({ | ||||||
|  |       text: $_("generating-pdf"), | ||||||
|  |       duration: -1, | ||||||
|  |     }).showToast(); | ||||||
|  |     fetch( | ||||||
|  |       `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify(generate_runners), | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |       .then((response) => { | ||||||
|  |         if (response.status != "200") { | ||||||
|  |           toast.hideToast(); | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("pdf-generation-failed"), | ||||||
|  |             duration: 3500, | ||||||
|  |             backgroundColor: | ||||||
|  |               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |           }).showToast(); | ||||||
|  |         } else { | ||||||
|  |           return response.blob(); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .then((blob) => { | ||||||
|  |         const url = window.URL.createObjectURL(blob); | ||||||
|  |         let a = document.createElement("a"); | ||||||
|  |         a.href = url; | ||||||
|  |         if (generate_runners.length == 1) { | ||||||
|  |           a.download = `${$_("sponsorings")}_${generate_runners[0].firstname}_${ | ||||||
|  |             generate_runners[0].lastname | ||||||
|  |           }-${locale}-${createId()}.pdf`; | ||||||
|  |         } | ||||||
|  |         a.download = `${$_("sponsorings")}-${locale}-${createId()}.pdf`; | ||||||
|  |         document.body.appendChild(a); | ||||||
|  |         a.click(); | ||||||
|  |         a.remove(); | ||||||
|  |         toast.hideToast(); | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_("pdf-successfully-generated"), | ||||||
|  |           duration: 3500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => { | ||||||
|  |         console.error(err); | ||||||
|  |       }); | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if sponsoring_contracts_show} | ||||||
|  |   <div id="sponsoring:dropdown" class="relative inline-block"> | ||||||
|  |     <div> | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           sponsoring_contracts_download_open = | ||||||
|  |             !sponsoring_contracts_download_open; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" | ||||||
|  |         id="options-menu" | ||||||
|  |         aria-haspopup="true" | ||||||
|  |         aria-expanded="true" | ||||||
|  |       > | ||||||
|  |         {$_("generate-sponsoring-contracts")} | ||||||
|  |         <svg | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           width="24" | ||||||
|  |           height="24" | ||||||
|  |           viewBox="0 0 24 24" | ||||||
|  |           class="-mr-1 ml-2 h-5 w-5" | ||||||
|  |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |           <path | ||||||
|  |             fill="currentColor" | ||||||
|  |             d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" | ||||||
|  |           /></svg | ||||||
|  |         > | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |     {#if sponsoring_contracts_download_open} | ||||||
|  |       <div | ||||||
|  |         class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5 z-10" | ||||||
|  |         id="sponsoring:dropdown:menu" | ||||||
|  |       > | ||||||
|  |         <div | ||||||
|  |           class="py-1" | ||||||
|  |           role="menu" | ||||||
|  |           aria-orientation="vertical" | ||||||
|  |           aria-labelledby="options-menu" | ||||||
|  |         > | ||||||
|  |           <span class="block w-full text-left px-4 py-2 text-sm text-gray-700" | ||||||
|  |             >{$_("select-language")}</span | ||||||
|  |           > | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateSponsoringContract("de"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("german")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               generateSponsoringContract("en"); | ||||||
|  |             }} | ||||||
|  |             type="submit" | ||||||
|  |             class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900" | ||||||
|  |             role="menuitem" | ||||||
|  |           > | ||||||
|  |             {$_("english")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     {/if} | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
| @@ -19,6 +19,11 @@ | |||||||
|   export let current_runners; |   export let current_runners; | ||||||
|   export let import_modal_open; |   export let import_modal_open; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|  |   $: importButtonEnabled = | ||||||
|  |     recent_processed && | ||||||
|  |     (!(selected_org_or_team == "" || selected_org_or_team == null) || | ||||||
|  |       !(passed_org?.id == null || passed_org?.id == 0) || | ||||||
|  |       !(passed_team?.id == null || passed_team?.id == 0)); | ||||||
|   const dispatch = createEventDispatcher(); |   const dispatch = createEventDispatcher(); | ||||||
|   function cancelModal() { |   function cancelModal() { | ||||||
|     json_output = []; |     json_output = []; | ||||||
| @@ -29,7 +34,7 @@ | |||||||
|     document.onkeydown = (e) => { |     document.onkeydown = (e) => { | ||||||
|       e = e || window.event; |       e = e || window.event; | ||||||
|       if (e.key === "Escape") { |       if (e.key === "Escape") { | ||||||
|         import_modal_open = false; |         cancelModal(); | ||||||
|       } |       } | ||||||
|       if (e.keyCode === 13) { |       if (e.keyCode === 13) { | ||||||
|         // |         // | ||||||
| @@ -44,7 +49,10 @@ | |||||||
|     groups = groups.concat(orgs); |     groups = groups.concat(orgs); | ||||||
|     RunnerTeamService.runnerTeamControllerGetAll().then((val) => { |     RunnerTeamService.runnerTeamControllerGetAll().then((val) => { | ||||||
|       const teams = val.map((r) => { |       const teams = val.map((r) => { | ||||||
|         return { label: `${r.parentGroup.name} > ${r.name}`, value: `TEAM_${r.id}` }; |         return { | ||||||
|  |           label: `${r.parentGroup.name} > ${r.name}`, | ||||||
|  |           value: `TEAM_${r.id}`, | ||||||
|  |         }; | ||||||
|       }); |       }); | ||||||
|       groups = groups.concat(teams); |       groups = groups.concat(teams); | ||||||
|     }); |     }); | ||||||
| @@ -120,6 +128,13 @@ | |||||||
|           .catch((err) => { |           .catch((err) => { | ||||||
|             toast.hideToast(); |             toast.hideToast(); | ||||||
|             recent_processed = true; |             recent_processed = true; | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("error-during-import"), | ||||||
|  |               duration: 500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |             cancelModal(); | ||||||
|           }); |           }); | ||||||
|       } |       } | ||||||
|       if (opened_from === "TeamDetail") { |       if (opened_from === "TeamDetail") { | ||||||
| @@ -137,6 +152,13 @@ | |||||||
|           .catch((err) => { |           .catch((err) => { | ||||||
|             toast.hideToast(); |             toast.hideToast(); | ||||||
|             recent_processed = true; |             recent_processed = true; | ||||||
|  |             Toastify({ | ||||||
|  |               text: $_("error-during-import"), | ||||||
|  |               duration: 500, | ||||||
|  |               backgroundColor: | ||||||
|  |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |             }).showToast(); | ||||||
|  |             cancelModal(); | ||||||
|           }); |           }); | ||||||
|       } |       } | ||||||
|       if (opened_from === "RunnerOverview") { |       if (opened_from === "RunnerOverview") { | ||||||
| @@ -160,6 +182,13 @@ | |||||||
|             .catch((err) => { |             .catch((err) => { | ||||||
|               toast.hideToast(); |               toast.hideToast(); | ||||||
|               recent_processed = true; |               recent_processed = true; | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("error-during-import"), | ||||||
|  |                 duration: 500, | ||||||
|  |                 backgroundColor: | ||||||
|  |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |               }).showToast(); | ||||||
|  |               cancelModal(); | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|         if (selected_org_or_team.includes("TEAM_")) { |         if (selected_org_or_team.includes("TEAM_")) { | ||||||
| @@ -173,7 +202,7 @@ | |||||||
|               toast.hideToast(); |               toast.hideToast(); | ||||||
|               recent_processed = true; |               recent_processed = true; | ||||||
|               Toastify({ |               Toastify({ | ||||||
|                 text: "Import finished", |                 text: $_('import-finished'), | ||||||
|                 duration: 500, |                 duration: 500, | ||||||
|                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|               }).showToast(); |               }).showToast(); | ||||||
| @@ -182,6 +211,13 @@ | |||||||
|             .catch((err) => { |             .catch((err) => { | ||||||
|               toast.hideToast(); |               toast.hideToast(); | ||||||
|               recent_processed = true; |               recent_processed = true; | ||||||
|  |               Toastify({ | ||||||
|  |                 text: $_("error-during-import"), | ||||||
|  |                 duration: 500, | ||||||
|  |                 backgroundColor: | ||||||
|  |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |               }).showToast(); | ||||||
|  |               cancelModal(); | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
| @@ -195,7 +231,7 @@ | |||||||
|     use:focusTrap |     use:focusTrap | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       import_modal_open = false; |       cancelModal(); | ||||||
|     }}> |     }}> | ||||||
|     <div |     <div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
| @@ -245,6 +281,16 @@ | |||||||
|                   bind:files |                   bind:files | ||||||
|                   type="file" /> |                   type="file" /> | ||||||
|               </div> |               </div> | ||||||
|  |               <div class="overflow-hidden relative mt-4 mb-4"> | ||||||
|  |                 <button | ||||||
|  |                   on:click={() => { | ||||||
|  |                     cancelModal(); | ||||||
|  |                   }} | ||||||
|  |                   type="button" | ||||||
|  |                   class="w-full rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 md:ml-40 mr-0 sm:ml-0 sm:w-auto sm:text-sm"> | ||||||
|  |                   {$_('cancel')} | ||||||
|  |                 </button> | ||||||
|  |               </div> | ||||||
|             {/if} |             {/if} | ||||||
|             {#if json_output.length > 0} |             {#if json_output.length > 0} | ||||||
|               {#if opened_from === 'OrgOverview'} |               {#if opened_from === 'OrgOverview'} | ||||||
| @@ -349,6 +395,8 @@ | |||||||
|                   </table> |                   </table> | ||||||
|                 </div> |                 </div> | ||||||
|                 <button |                 <button | ||||||
|  |                   disabled={!importButtonEnabled} | ||||||
|  |                   class:opacity-50={!importButtonEnabled} | ||||||
|                   on:click={importAction} |                   on:click={importAction} | ||||||
|                   type="button" |                   type="button" | ||||||
|                   class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |                   class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|   | |||||||
| @@ -1,5 +1,8 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import { |   import { | ||||||
|     RunnerService, |     RunnerService, | ||||||
| @@ -14,12 +17,13 @@ | |||||||
|   export let params; |   export let params; | ||||||
|   const runner_promise = RunnerService.runnerControllerGetOne(params.runnerid); |   const runner_promise = RunnerService.runnerControllerGetOne(params.runnerid); | ||||||
|   $: delete_triggered = false; |   $: delete_triggered = false; | ||||||
|   $: sponsoring_contracts_download_open = false; |  | ||||||
|   $: original_data_pdf = {}; |   $: original_data_pdf = {}; | ||||||
|   $: original_data = {}; |   $: original_data = {}; | ||||||
|   $: editable = {}; |   $: editable = {}; | ||||||
|   $: group = {} |   $: group = {}; | ||||||
|   $: changes_performed = !(JSON.stringify(original_data) == JSON.stringify(editable)); |   $: changes_performed = !( | ||||||
|  |     JSON.stringify(original_data) == JSON.stringify(editable) | ||||||
|  |   ); | ||||||
|   $: isEmailValid = |   $: isEmailValid = | ||||||
|     (editable.email || "") === "" || |     (editable.email || "") === "" || | ||||||
|     (editable.email && isEmail(editable.email || "")); |     (editable.email && isEmail(editable.email || "")); | ||||||
| @@ -31,6 +35,10 @@ | |||||||
|     isLastnameValid && |     isLastnameValid && | ||||||
|     isEmailValid && |     isEmailValid && | ||||||
|     editable.group != null; |     editable.group != null; | ||||||
|  |   $: sponsoring_contracts_show = true; | ||||||
|  |   $: cards_show = true; | ||||||
|  |   $: certificates_show = true; | ||||||
|  |   $: generate_runners = [original_data_pdf]; | ||||||
|   runner_promise.then((data) => { |   runner_promise.then((data) => { | ||||||
|     data_loaded = true; |     data_loaded = true; | ||||||
|     original_data_pdf = Object.assign(original_data_pdf, data); |     original_data_pdf = Object.assign(original_data_pdf, data); | ||||||
| @@ -38,27 +46,21 @@ | |||||||
|     original_data = Object.assign(original_data, data); |     original_data = Object.assign(original_data, data); | ||||||
|     editable = Object.assign(editable, original_data); |     editable = Object.assign(editable, original_data); | ||||||
|  |  | ||||||
|     RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { |     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|       const orgs = val.map((r) => { |       (val) => { | ||||||
|         return { label: r.name, value: r }; |         const orgs = val.map((r) => { | ||||||
|       }); |           return { label: r.name, value: r }; | ||||||
|       groups = groups.concat(orgs); |  | ||||||
|       RunnerTeamService.runnerTeamControllerGetAll().then((val) => { |  | ||||||
|         const teams = val.map((r) => { |  | ||||||
|           return { label: `${r.parentGroup.name} > ${r.name}`, value: r }; |  | ||||||
|         }); |         }); | ||||||
|         groups = groups.concat(teams); |         groups = groups.concat(orgs); | ||||||
|         group = groups.find(g => g.value.id == editable.group) |         RunnerTeamService.runnerTeamControllerGetAll().then((val) => { | ||||||
|       }); |           const teams = val.map((r) => { | ||||||
|     }); |             return { label: `${r.parentGroup.name} > ${r.name}`, value: r }; | ||||||
|   }); |           }); | ||||||
|   document.addEventListener("click", function (e) { |           groups = groups.concat(teams); | ||||||
|     if ( |           group = groups.find((g) => g.value.id == editable.group); | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |         }); | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |       } | ||||||
|     ) { |     ); | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |   }); | ||||||
|   let groups = []; |   let groups = []; | ||||||
|   function submit() { |   function submit() { | ||||||
| @@ -69,6 +71,9 @@ | |||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       let postdata = {}; |       let postdata = {}; | ||||||
|       postdata = Object.assign(postdata, editable); |       postdata = Object.assign(postdata, editable); | ||||||
|  |       if (postdata.phone === "") { | ||||||
|  |         postdata.phone = null; | ||||||
|  |       } | ||||||
|       RunnerService.runnerControllerPut(original_data.id, postdata) |       RunnerService.runnerControllerPut(original_data.id, postdata) | ||||||
|         .then((resp) => { |         .then((resp) => { | ||||||
|           Object.assign(original_data, editable); |           Object.assign(original_data, editable); | ||||||
| @@ -90,63 +95,10 @@ | |||||||
|       }) |       }) | ||||||
|       .catch((err) => {}); |       .catch((err) => {}); | ||||||
|   } |   } | ||||||
|   function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdf"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     fetch( |  | ||||||
|       `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|       { |  | ||||||
|         method: "POST", |  | ||||||
|         headers: { |  | ||||||
|           "Content-Type": "application/json", |  | ||||||
|         }, |  | ||||||
|         body: JSON.stringify([original_data_pdf]), |  | ||||||
|       } |  | ||||||
|     ) |  | ||||||
|       .then((response) => { |  | ||||||
|         if (response.status != "200") { |  | ||||||
|           toast.hideToast(); |  | ||||||
|           Toastify({ |  | ||||||
|             text: $_("pdf-generation-failed"), |  | ||||||
|             duration: 3500, |  | ||||||
|             backgroundColor: |  | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|           }).showToast(); |  | ||||||
|         } else { |  | ||||||
|           return response.blob(); |  | ||||||
|         } |  | ||||||
|       }) |  | ||||||
|       .then((blob) => { |  | ||||||
|         const url = window.URL.createObjectURL(blob); |  | ||||||
|         let a = document.createElement("a"); |  | ||||||
|         a.href = url; |  | ||||||
|         a.download = |  | ||||||
|           "Sponsoring_" + |  | ||||||
|           original_data.firstname + |  | ||||||
|           (original_data.middlename || "") + |  | ||||||
|           original_data.lastname + |  | ||||||
|           ".pdf"; |  | ||||||
|         document.body.appendChild(a); |  | ||||||
|         a.click(); |  | ||||||
|         a.remove(); |  | ||||||
|         toast.hideToast(); |  | ||||||
|         Toastify({ |  | ||||||
|           text: $_("pdf-successfully-generated"), |  | ||||||
|           duration: 3500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => { |  | ||||||
|         console.error(err); |  | ||||||
|       }); |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#await runner_promise} | {#await runner_promise} | ||||||
|   {$_('loading-runners')} |   {$_("loading-runners")} | ||||||
| {:then} | {:then} | ||||||
|   <section class="container p-5 select-none"> |   <section class="container p-5 select-none"> | ||||||
|     <div class="flex flex-row mb-4"> |     <div class="flex flex-row mb-4"> | ||||||
| @@ -160,12 +112,15 @@ | |||||||
|                 class="flex-shrink-0 w-5 h-5 mr-2" |                 class="flex-shrink-0 w-5 h-5 mr-2" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   d="M9.83 8.79L8 9.456V13H6V8.05h.015l5.268-1.918c.244-.093.51-.14.782-.131a2.616 2.616 0 0 1 2.427 1.82c.186.583.356.977.51 1.182A4.992 4.992 0 0 0 19 11v2a6.986 6.986 0 0 1-5.402-2.547l-.581 3.297L15 15.67V23h-2v-5.986l-2.05-1.987-.947 4.298-6.894-1.215.348-1.97 4.924.868L9.83 8.79zM13.5 5.5a2 2 0 1 1 0-4 2 2 0 0 1 0 4z" /></svg> |                   d="M9.83 8.79L8 9.456V13H6V8.05h.015l5.268-1.918c.244-.093.51-.14.782-.131a2.616 2.616 0 0 1 2.427 1.82c.186.583.356.977.51 1.182A4.992 4.992 0 0 0 19 11v2a6.986 6.986 0 0 1-5.402-2.547l-.581 3.297L15 15.67V23h-2v-5.986l-2.05-1.987-.947 4.298-6.894-1.215.348-1.97 4.924.868L9.83 8.79zM13.5 5.5a2 2 0 1 1 0-4 2 2 0 0 1 0 4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <a class="mr-2" href="./">{$_('runners')}</a><svg |               <a class="mr-2" href="./">{$_("runners")}</a><svg | ||||||
|                 stroke="currentColor" |                 stroke="currentColor" | ||||||
|                 fill="none" |                 fill="none" | ||||||
|                 stroke-width="2" |                 stroke-width="2" | ||||||
| @@ -175,17 +130,17 @@ | |||||||
|                 class="h-3 w-3 mr-2 stroke-current" |                 class="h-3 w-3 mr-2 stroke-current" | ||||||
|                 height="1em" |                 height="1em" | ||||||
|                 width="1em" |                 width="1em" | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                   x1="5" |                 ><line x1="5" y1="12" x2="19" y2="12" /> | ||||||
|                   y1="12" |                 <polyline points="12 5 19 12 12 19" /></svg | ||||||
|                   x2="19" |               > | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <span class="mr-2">{original_data.firstname} |               <span class="mr-2" | ||||||
|                 {original_data.middlename || ''} |                 >{original_data.firstname} | ||||||
|                 {original_data.lastname}</span> |                 {original_data.middlename || ""} | ||||||
|  |                 {original_data.lastname}</span | ||||||
|  |               > | ||||||
|             </li> |             </li> | ||||||
|           </ol> |           </ol> | ||||||
|         </nav> |         </nav> | ||||||
| @@ -193,85 +148,42 @@ | |||||||
|     </div> |     </div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> |     <div class="mb-8 text-3xl font-extrabold leading-tight"> | ||||||
|       {original_data.firstname} |       {original_data.firstname} | ||||||
|       {original_data.middlename || ''} |       {original_data.middlename || ""} | ||||||
|       {original_data.lastname} |       {original_data.lastname} | ||||||
|       <span data-id="runner_actions_${editable.id}"> |       <span data-id="runner_actions_${editable.id}"> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:DELETE')} |         {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:DELETE")} | ||||||
|           {#if delete_triggered} |           {#if delete_triggered} | ||||||
|             <button |             <button | ||||||
|               on:click={deleteRunner} |               on:click={deleteRunner} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('confirm-deletion')}</button> |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm" | ||||||
|  |               >{$_("confirm-deletion")}</button | ||||||
|  |             > | ||||||
|             <button |             <button | ||||||
|               on:click={() => { |               on:click={() => { | ||||||
|                 delete_triggered = !delete_triggered; |                 delete_triggered = !delete_triggered; | ||||||
|               }} |               }} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm">{$_('cancel')}</button> |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm" | ||||||
|  |               >{$_("cancel")}</button | ||||||
|  |             > | ||||||
|           {/if} |           {/if} | ||||||
|           <div id="sponsoring:dropdown" class="relative inline-block"> |           <GenerateSponsoringContracts | ||||||
|             <div> |             bind:sponsoring_contracts_show | ||||||
|               <button |             bind:generate_runners | ||||||
|                 on:click={() => { |           /> | ||||||
|                   sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |           <GenerateRunnerCards bind:cards_show bind:generate_runners /> | ||||||
|                 }} |           <GenerateRunnerCertificates | ||||||
|                 type="button" |             bind:certificates_show | ||||||
|                 class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |             bind:generate_runners | ||||||
|                 id="options-menu" |           /> | ||||||
|                 aria-haspopup="true" |  | ||||||
|                 aria-expanded="true"> |  | ||||||
|                 {$_('generate-sponsoring-contract')} |  | ||||||
|                 <svg |  | ||||||
|                   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                   width="24" |  | ||||||
|                   height="24" |  | ||||||
|                   viewBox="0 0 24 24" |  | ||||||
|                   class="-mr-1 ml-2 h-5 w-5"><path |  | ||||||
|                     fill="none" |  | ||||||
|                     d="M0 0h24v24H0z" /> |  | ||||||
|                   <path |  | ||||||
|                     fill="currentColor" |  | ||||||
|                     d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |  | ||||||
|               </button> |  | ||||||
|             </div> |  | ||||||
|             {#if sponsoring_contracts_download_open} |  | ||||||
|               <div |  | ||||||
|                 class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|                 id="sponsoring:dropdown:menu"> |  | ||||||
|                 <div |  | ||||||
|                   class="py-1" |  | ||||||
|                   role="menu" |  | ||||||
|                   aria-orientation="vertical" |  | ||||||
|                   aria-labelledby="options-menu"> |  | ||||||
|                   <span |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('de'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('german')} |  | ||||||
|                   </button> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('en'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('english')} |  | ||||||
|                   </button> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             {/if} |  | ||||||
|           </div> |  | ||||||
|           {#if !delete_triggered} |           {#if !delete_triggered} | ||||||
|             <button |             <button | ||||||
|               on:click={() => { |               on:click={() => { | ||||||
|                 delete_triggered = true; |                 delete_triggered = true; | ||||||
|               }} |               }} | ||||||
|               type="button" |               type="button" | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('delete-runner')}</button> |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm" | ||||||
|  |               >{$_("delete-runner")}</button | ||||||
|  |             > | ||||||
|           {/if} |           {/if} | ||||||
|         {/if} |         {/if} | ||||||
|         {#if !delete_triggered} |         {#if !delete_triggered} | ||||||
| @@ -280,117 +192,128 @@ | |||||||
|             class:opacity-50={!save_enabled} |             class:opacity-50={!save_enabled} | ||||||
|             type="button" |             type="button" | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('save-changes')}</button> |             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm" | ||||||
|  |             >{$_("save-changes")}</button | ||||||
|  |           > | ||||||
|         {/if} |         {/if} | ||||||
|       </span> |       </span> | ||||||
|     </div> |     </div> | ||||||
|     <!--  --> |     <!--  --> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label |       <label for="firstname" class="font-medium text-gray-700" | ||||||
|         for="firstname" |         >{$_("first-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('first-name')}</label> |       > | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('first-name')} |         placeholder={$_("first-name")} | ||||||
|         type="text" |         type="text" | ||||||
|         class:border-red-500={!isFirstnameValid} |         class:border-red-500={!isFirstnameValid} | ||||||
|         class:focus:border-red-500={!isFirstnameValid} |         class:focus:border-red-500={!isFirstnameValid} | ||||||
|         class:focus:ring-red-500={!isFirstnameValid} |         class:focus:ring-red-500={!isFirstnameValid} | ||||||
|         bind:value={editable.firstname} |         bind:value={editable.firstname} | ||||||
|         name="firstname" |         name="firstname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |       /> | ||||||
|       {#if !isFirstnameValid} |       {#if !isFirstnameValid} | ||||||
|         <span |         <span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('first-name-is-required')} |         > | ||||||
|  |           {$_("first-name-is-required")} | ||||||
|         </span> |         </span> | ||||||
|       {/if} |       {/if} | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label |       <label for="middlename" class="font-medium text-gray-700" | ||||||
|         for="middlename" |         >{$_("middle-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('middle-name')}</label> |       > | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('middle-name')} |         placeholder={$_("middle-name")} | ||||||
|         type="text" |         type="text" | ||||||
|         bind:value={editable.middlename} |         bind:value={editable.middlename} | ||||||
|         name="middlename" |         name="middlename" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |       /> | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label |       <label for="lastname" class="font-medium text-gray-700" | ||||||
|         for="lastname" |         >{$_("last-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('last-name')}</label> |       > | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('last-name')} |         placeholder={$_("last-name")} | ||||||
|         type="text" |         type="text" | ||||||
|         bind:value={editable.lastname} |         bind:value={editable.lastname} | ||||||
|         class:border-red-500={!isLastnameValid} |         class:border-red-500={!isLastnameValid} | ||||||
|         class:focus:border-red-500={!isLastnameValid} |         class:focus:border-red-500={!isLastnameValid} | ||||||
|         class:focus:ring-red-500={!isLastnameValid} |         class:focus:ring-red-500={!isLastnameValid} | ||||||
|         name="lastname" |         name="lastname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |       /> | ||||||
|       {#if !isLastnameValid} |       {#if !isLastnameValid} | ||||||
|         <span |         <span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('last-name-is-required')} |         > | ||||||
|  |           {$_("last-name-is-required")} | ||||||
|         </span> |         </span> | ||||||
|       {/if} |       {/if} | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label |       <label for="email" class="font-medium text-gray-700" | ||||||
|         for="email" |         >{$_("e-mail-adress")}</label | ||||||
|         class="font-medium text-gray-700">{$_('e-mail-adress')}</label> |       > | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('e-mail-adress')} |         placeholder={$_("e-mail-adress")} | ||||||
|         type="email" |         type="email" | ||||||
|         bind:value={editable.email} |         bind:value={editable.email} | ||||||
|         class:border-red-500={!isEmailValid} |         class:border-red-500={!isEmailValid} | ||||||
|         class:focus:border-red-500={!isEmailValid} |         class:focus:border-red-500={!isEmailValid} | ||||||
|         class:focus:ring-red-500={!isEmailValid} |         class:focus:ring-red-500={!isEmailValid} | ||||||
|         name="email" |         name="email" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |       /> | ||||||
|       {#if !isEmailValid} |       {#if !isEmailValid} | ||||||
|         <span |         <span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('valid-email-is-required')} |         > | ||||||
|  |           {$_("valid-email-is-required")} | ||||||
|         </span> |         </span> | ||||||
|       {/if} |       {/if} | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label for="phone" class="font-medium text-gray-700">{$_('phone')}</label> |       <label for="phone" class="font-medium text-gray-700">{$_("phone")}</label> | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('phone')} |         placeholder={$_("phone")} | ||||||
|         type="tel" |         type="tel" | ||||||
|         bind:value={editable.phone} |         bind:value={editable.phone} | ||||||
|         name="phone" |         name="phone" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|  |       /> | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <span class="font-medium text-gray-700">{$_('group')}</span> |       <span class="font-medium text-gray-700">{$_("group")}</span> | ||||||
|       <Select |       <Select | ||||||
|         containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" |         containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | ||||||
|         itemFilter={(label, filterText, option) => label |         itemFilter={(label, filterText, option) => | ||||||
|             .toLowerCase() |           label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|             .includes( |           option.id.value.toString().startsWith(filterText.toLowerCase())} | ||||||
|               filterText.toLowerCase() |  | ||||||
|             ) || option.id.value.toString().startsWith(filterText.toLowerCase())} |  | ||||||
|         items={groups} |         items={groups} | ||||||
|         showChevron={true} |         showChevron={true} | ||||||
|         placeholder={$_('search-for-an-organization-or-team-by-name-or-id')} |         placeholder={$_("search-for-an-organization-or-team-by-name-or-id")} | ||||||
|         noOptionsMessage={$_('no-organization-or-team-found')} |         noOptionsMessage={$_("no-organization-or-team-found")} | ||||||
|         bind:selectedValue={group} |         bind:selectedValue={group} | ||||||
|         on:select={(selectedValue) => {editable.group = selectedValue.detail.value.id}} |         on:select={(selectedValue) => { | ||||||
|         on:clear={() => (editable.group = null)} /> |           editable.group = selectedValue.detail.value.id; | ||||||
|  |         }} | ||||||
|  |         on:clear={() => (editable.group = null)} | ||||||
|  |       /> | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <span class="font-medium text-gray-700">{$_('distance')}</span> |       <span class="font-medium text-gray-700">{$_("distance")}</span> | ||||||
|       <br /> |       <br /> | ||||||
|       <span class="text-gray-700">{original_data.distance} km</span> |       <span class="text-gray-700">{original_data.distance / 1000} km</span> | ||||||
|     </div> |     </div> | ||||||
|   </section> |   </section> | ||||||
| {:catch error} | {:catch error} | ||||||
|   | |||||||
| @@ -1,26 +1,26 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { |   import { | ||||||
|     RunnerService, |     RunnerService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
|     RunnerOrganizationService, |     RunnerOrganizationService, | ||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|  |   import ThFilterGroup from "./ThFilterGroup.svelte"; | ||||||
|  |   import { DataHandler, Datatable, Th, ThFilter } from "@vincjo/datatables"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import RunnersEmptyState from "./RunnersEmptyState.svelte"; |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   import Select from "svelte-select"; |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|   import Toastify from "toastify-js"; |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: searchvalue = ""; |   import { onMount } from "svelte"; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   export let current_runners = []; |   let dataLoaded = false; | ||||||
|   const runners_promise = RunnerService.runnerControllerGetAll().then((val) => { |   let current_runners = []; | ||||||
|     current_runners = val; |   const handler = new DataHandler(current_runners, { rowsPerPage: 50 }); | ||||||
|   }); |   const rows = handler.getRows(); | ||||||
|   $: selectedFilter_teams = null; |   $: sponsoring_contracts_show = generate_runners.length > 0; | ||||||
|   $: selectedFilter = null; |   $: cards_show = generate_runners.length > 0; | ||||||
|   $: filter__teams = selectedFilter_teams || []; |   $: certificates_show = generate_runners.length > 0; | ||||||
|   $: filter__orgs = selectedFilter || []; |   $: generate_runners = []; //current_runners.filter((r) => r.selected === true); | ||||||
|   $: filterGroupIDs = filter__teams.concat(filter__orgs).map((i) => i.value); |  | ||||||
|   $: sponsoring_contracts_download_open = false; |  | ||||||
|   $: teams = []; |   $: teams = []; | ||||||
|   $: orgs = []; |   $: orgs = []; | ||||||
|   $: mappedteams = teams.map(function (g) { |   $: mappedteams = teams.map(function (g) { | ||||||
| @@ -31,334 +31,193 @@ | |||||||
|       return { value: g.id, label: g.name }; |       return { value: g.id, label: g.name }; | ||||||
|     }) |     }) | ||||||
|     .concat(mappedteams); |     .concat(mappedteams); | ||||||
|   document.addEventListener("click", function (e) { |   onMount(() => { | ||||||
|     if ( |     RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |       current_runners = val; | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |       dataLoaded = true; | ||||||
|     ) { |       handler.setRows(val); | ||||||
|       sponsoring_contracts_download_open = false; |     }); | ||||||
|     } |     RunnerTeamService.runnerTeamControllerGetAll().then((val) => { | ||||||
|   }); |       teams = val; | ||||||
|   RunnerTeamService.runnerTeamControllerGetAll().then((val) => { |     }); | ||||||
|     teams = val; |     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|   }); |       (val) => { | ||||||
|   RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { |         orgs = val; | ||||||
|     orgs = val; |  | ||||||
|   }); |  | ||||||
|   function should_display_based_on_id(id) { |  | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |  | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |  | ||||||
|     } |  | ||||||
|     return id.toString() === searchvalue; |  | ||||||
|   } |  | ||||||
|   function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdf"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     fetch( |  | ||||||
|       `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|       { |  | ||||||
|         method: "POST", |  | ||||||
|         headers: { |  | ||||||
|           "Content-Type": "application/json", |  | ||||||
|         }, |  | ||||||
|         body: JSON.stringify( |  | ||||||
|           current_runners.filter((r) => r.is_selected === true) |  | ||||||
|         ), |  | ||||||
|       } |       } | ||||||
|     ) |     ); | ||||||
|       .then((response) => { |   }); | ||||||
|         if (response.status != "200") { |  | ||||||
|           toast.hideToast(); |  | ||||||
|           Toastify({ |  | ||||||
|             text: $_("pdf-generation-failed"), |  | ||||||
|             duration: 3500, |  | ||||||
|             backgroundColor: |  | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|           }).showToast(); |  | ||||||
|         } else { |  | ||||||
|           return response.blob(); |  | ||||||
|         } |  | ||||||
|       }) |  | ||||||
|       .then((blob) => { |  | ||||||
|         const url = window.URL.createObjectURL(blob); |  | ||||||
|         let a = document.createElement("a"); |  | ||||||
|         a.href = url; |  | ||||||
|         a.download = "Sponsoring.pdf"; |  | ||||||
|         document.body.appendChild(a); |  | ||||||
|         a.click(); |  | ||||||
|         a.remove(); |  | ||||||
|         toast.hideToast(); |  | ||||||
|         Toastify({ |  | ||||||
|           text: $_("pdf-successfully-generated"), |  | ||||||
|           duration: 3500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => { |  | ||||||
|         console.error(err); |  | ||||||
|       }); |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:GET")} | ||||||
|   {#await runners_promise} |   {#if !dataLoaded} | ||||||
|     <div |     <div | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|       role="alert"> |       role="alert" | ||||||
|       <p class="font-bold">{$_('runners-are-being-loaded')}</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("runners-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:else} | ||||||
|     {#if current_runners.length === 0} |     <div class="h-12"> | ||||||
|       <RunnersEmptyState /> |       <GenerateSponsoringContracts | ||||||
|     {:else} |         bind:sponsoring_contracts_show | ||||||
|       <input |         bind:generate_runners | ||||||
|         type="search" |       /> | ||||||
|         bind:value={searchvalue} |       <GenerateRunnerCards bind:cards_show bind:generate_runners /> | ||||||
|         placeholder={$_('datatable.search')} |       <GenerateRunnerCertificates | ||||||
|         aria-label={$_('datatable.search')} |         bind:certificates_show | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         bind:generate_runners | ||||||
|       <div class="block mb-6"> |       /> | ||||||
|         <label |     </div> | ||||||
|           for="country" |     <Datatable {handler}> | ||||||
|           class="text-sm font-medium text-gray-700">{$_('filter-by-organization-team')}</label> |       <table> | ||||||
|         <Select |         <thead> | ||||||
|           on:select={(event) => { |           <tr> | ||||||
|             selectedFilter = event.detail; |             <th style="border-bottom: 1px solid #ddd;"> | ||||||
|           }} |               <input | ||||||
|           selectedValue={selectedFilter} |                 type="checkbox" | ||||||
|           placeholder={$_('filter-by-organization-team')} |                 class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|           containerClasses="mt-1 py-2 px-3 border border-gray-300 bg-white rounded-md shadow-sm focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm" |                 checked={generate_runners.length == current_runners.length} | ||||||
|           items={selectgroups} |  | ||||||
|           isMulti={true} /> |  | ||||||
|       </div> |  | ||||||
|       <div class="h-12"> |  | ||||||
|         {#if current_runners.some((r) => r.is_selected === true)} |  | ||||||
|           <div id="sponsoring:dropdown" class="relative inline-block"> |  | ||||||
|             <div> |  | ||||||
|               <button |  | ||||||
|                 on:click={() => { |                 on:click={() => { | ||||||
|                   sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |                   if (generate_runners.length != current_runners.length) { | ||||||
|  |                     generate_runners = current_runners; | ||||||
|  |                   } else { | ||||||
|  |                     generate_runners = []; | ||||||
|  |                   } | ||||||
|                 }} |                 }} | ||||||
|                 type="button" |               /> | ||||||
|                 class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |             </th> | ||||||
|                 id="options-menu" |             <Th {handler} orderBy="id">ID</Th> | ||||||
|                 aria-haspopup="true" |             <Th {handler} orderBy="firstname">First Name</Th> | ||||||
|                 aria-expanded="true"> |             <Th {handler} orderBy="middlename">Middle Name</Th> | ||||||
|                 {$_('generate-sponsoring-contracts')} |             <Th {handler} orderBy="lastname">Last Name</Th> | ||||||
|                 <svg |             <th style="border-bottom: 1px solid #ddd;">Gruppe</th> | ||||||
|                   xmlns="http://www.w3.org/2000/svg" |             <Th {handler} orderBy="distance">Distanz</Th> | ||||||
|                   width="24" |             <th style="border-bottom: 1px solid #ddd;">{$_("action")}</th> | ||||||
|                   height="24" |           </tr> | ||||||
|                   viewBox="0 0 24 24" |           <tr> | ||||||
|                   class="-mr-1 ml-2 h-5 w-5"><path |             <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|                     fill="none" |             <ThFilter {handler} filterBy="id" /> | ||||||
|                     d="M0 0h24v24H0z" /> |             <ThFilter {handler} filterBy="firstname" /> | ||||||
|                   <path |             <ThFilter {handler} filterBy="middlename" /> | ||||||
|                     fill="currentColor" |             <ThFilter {handler} filterBy="lastname" /> | ||||||
|                     d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |             <ThFilterGroup groups={selectgroups} {handler} /> | ||||||
|               </button> |             <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|             </div> |             <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|             {#if sponsoring_contracts_download_open} |           </tr> | ||||||
|               <div |         </thead> | ||||||
|                 class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |         <tbody> | ||||||
|                 id="sponsoring:dropdown:menu"> |           {#each $rows as row} | ||||||
|                 <div |  | ||||||
|                   class="py-1" |  | ||||||
|                   role="menu" |  | ||||||
|                   aria-orientation="vertical" |  | ||||||
|                   aria-labelledby="options-menu"> |  | ||||||
|                   <span |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('de'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('german')} |  | ||||||
|                   </button> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('en'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('english')} |  | ||||||
|                   </button> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             {/if} |  | ||||||
|           </div> |  | ||||||
|         {/if} |  | ||||||
|       </div> |  | ||||||
|       <div |  | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |  | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |  | ||||||
|           <thead class="bg-gray-50"> |  | ||||||
|             <tr> |             <tr> | ||||||
|               <th |               <td> | ||||||
|                 scope="col" |                 <input | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   type="checkbox" | ||||||
|                 <span |                   class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                   checked={generate_runners.filter((i)=>i.id == row.id).length > 0} | ||||||
|                   on:click={() => { |                   on:click={() => { | ||||||
|                     const newstate = !current_runners.some((r) => r.is_selected === true); |                     if ( | ||||||
|                     current_runners = current_runners.map((r) => { |                       generate_runners.findIndex((i) => i.id == row.id) == -1 | ||||||
|                       r.is_selected = newstate; |                     ) { | ||||||
|                       return r; |                       generate_runners.push(row); | ||||||
|                     }); |                       generate_runners = generate_runners; | ||||||
|  |                     } else { | ||||||
|  |                       generate_runners = generate_runners.filter( | ||||||
|  |                         (r) => r.id != row.id | ||||||
|  |                       ); | ||||||
|  |                     } | ||||||
|  |                     console.log(generate_runners) | ||||||
|                   }} |                   }} | ||||||
|                   class="underline cursor-pointer select-none">{#if current_runners.some((r) => r.is_selected === true)} |                 /> | ||||||
|                     {$_('deselect-all')} |               </td> | ||||||
|                   {:else}{$_('select-all')}{/if} |               <td>{row.id}</td> | ||||||
|                 </span> |               <td>{row.firstname}</td> | ||||||
|               </th> |               <td>{row.middlename || ""}</td> | ||||||
|               <th |               <td>{row.lastname}</td> | ||||||
|                 scope="col" |               <td | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 >{#if row.group.responseType === "RUNNERTEAM"} | ||||||
|                 {$_('name')} |                   <a | ||||||
|               </th> |                     href="../teams/{row.group.id}" | ||||||
|               <th |                     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800" | ||||||
|                 scope="col" |                     >{row.group.parentGroup.name} > {row.group.name}</a | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   > | ||||||
|                 {$_('contact-information')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('group')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('distance-in-km')} |  | ||||||
|               </th> |  | ||||||
|               <th scope="col" class="relative px-6 py-3"> |  | ||||||
|                 <span class="sr-only">{$_('action')}</span> |  | ||||||
|               </th> |  | ||||||
|             </tr> |  | ||||||
|           </thead> |  | ||||||
|           <tbody class="divide-y divide-gray-200"> |  | ||||||
|             {#each current_runners as runner} |  | ||||||
|               {#if runner.firstname |  | ||||||
|                 .toLowerCase() |  | ||||||
|                 .includes( |  | ||||||
|                   searchvalue.toLowerCase() |  | ||||||
|                 ) || runner.middlename |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || runner.lastname |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || should_display_based_on_id(runner.id)} |  | ||||||
|                 {#if filterGroupIDs.includes(runner.group.id) || filterGroupIDs.includes(runner.group.parentGroup?.id) || filterGroupIDs.length === 0} |  | ||||||
|                   <tr |  | ||||||
|                     data-rowid="user_{runner.id}" |  | ||||||
|                     data-groupid={runner.group.id}> |  | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                       <input |  | ||||||
|                         bind:checked={runner.is_selected} |  | ||||||
|                         type="checkbox" |  | ||||||
|                         class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |  | ||||||
|                     </td> |  | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                       <div class="flex items-center"> |  | ||||||
|                         <div class="ml-4"> |  | ||||||
|                           <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                             {runner.firstname} |  | ||||||
|                             {runner.middlename || ''} |  | ||||||
|                             {runner.lastname} |  | ||||||
|                           </div> |  | ||||||
|                         </div> |  | ||||||
|                       </div> |  | ||||||
|                     </td> |  | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                       {#if runner.email} |  | ||||||
|                         <div class="text-sm text-gray-500">{runner.email}</div> |  | ||||||
|                       {/if} |  | ||||||
|                       {#if runner.phone} |  | ||||||
|                         <div class="text-sm text-gray-500">{runner.phone}</div> |  | ||||||
|                       {/if} |  | ||||||
|                       {#if runner.address.address1 !== null} |  | ||||||
|                         {runner.address.address1}<br /> |  | ||||||
|                         {runner.address.address2 || ''}<br /> |  | ||||||
|                         {runner.address.postalcode} |  | ||||||
|                         {runner.address.city} |  | ||||||
|                         {runner.address.country} |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                       {#if runner.group.responseType === 'RUNNERTEAM'} |  | ||||||
|                         <a |  | ||||||
|                           href="../teams/{runner.group.id}" |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{runner.group.name}</a> |  | ||||||
|                       {/if} |  | ||||||
|                       {#if runner.group.responseType === 'RUNNERORGANIZATION'} |  | ||||||
|                         <a |  | ||||||
|                           href="../orgs/{runner.group.id}" |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{runner.group.name}</a> |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                       {runner.distance} |  | ||||||
|                     </td> |  | ||||||
|                     {#if active_deletes[runner.id] === true} |  | ||||||
|                       <td |  | ||||||
|                         class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[runner.id] = false; |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             RunnerService.runnerControllerRemove(runner.id, true) |  | ||||||
|                               .then((resp) => { |  | ||||||
|                                 current_runners = current_runners.filter((obj) => obj.id !== runner.id); |  | ||||||
|                               }) |  | ||||||
|                               .catch((err) => {}); |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |  | ||||||
|                       </td> |  | ||||||
|                     {:else} |  | ||||||
|                       <td |  | ||||||
|                         class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                         <a |  | ||||||
|                           href="./{runner.id}" |  | ||||||
|                           class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |  | ||||||
|                         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:DELETE')} |  | ||||||
|                           <button |  | ||||||
|                             on:click={() => { |  | ||||||
|                               active_deletes[runner.id] = true; |  | ||||||
|                             }} |  | ||||||
|                             tabindex="0" |  | ||||||
|                             class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |  | ||||||
|                         {/if} |  | ||||||
|                       </td> |  | ||||||
|                     {/if} |  | ||||||
|                   </tr> |  | ||||||
|                 {/if} |                 {/if} | ||||||
|               {/if} |                 {#if row.group.responseType === "RUNNERORGANIZATION"} | ||||||
|             {/each} |                   <a | ||||||
|           </tbody> |                     href="../orgs/{row.group.id}" | ||||||
|         </table> |                     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800" | ||||||
|       </div> |                     >{row.group.name}</a | ||||||
|     {/if} |                   > | ||||||
|   {:catch error} |                 {/if}</td | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |               > | ||||||
|       <span class="inline-block align-middle mr-8"> |               <td>{row.distance / 1000} km</td> | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |               <td> | ||||||
|         {error} |                 {#if active_deletes[row.id] === true} | ||||||
|       </span> |                   <button | ||||||
|     </div> |                     on:click={() => { | ||||||
|   {/await} |                       active_deletes[row.id] = false; | ||||||
|  |                     }} | ||||||
|  |                     tabindex="0" | ||||||
|  |                     class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer" | ||||||
|  |                     >{$_("cancel-delete")}</button | ||||||
|  |                   > | ||||||
|  |                   <button | ||||||
|  |                     on:click={() => { | ||||||
|  |                       RunnerService.runnerControllerRemove(row.id, true) | ||||||
|  |                         .then((resp) => { | ||||||
|  |                           current_runners = current_runners.filter( | ||||||
|  |                             (obj) => obj.id !== row.id | ||||||
|  |                           ); | ||||||
|  |                         }) | ||||||
|  |                         .catch((err) => {}); | ||||||
|  |                     }} | ||||||
|  |                     tabindex="0" | ||||||
|  |                     class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                     >{$_("confirm-delete")}</button | ||||||
|  |                   > | ||||||
|  |                 {:else} | ||||||
|  |                   <a | ||||||
|  |                     href="./{row.id}" | ||||||
|  |                     class="text-indigo-600 hover:text-indigo-900" | ||||||
|  |                     >{$_("details")}</a | ||||||
|  |                   > | ||||||
|  |                   {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:DELETE")} | ||||||
|  |                     <button | ||||||
|  |                       on:click={() => { | ||||||
|  |                         active_deletes[row.id] = true; | ||||||
|  |                       }} | ||||||
|  |                       tabindex="0" | ||||||
|  |                       class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                       >{$_("delete")}</button | ||||||
|  |                     > | ||||||
|  |                   {/if} | ||||||
|  |                 {/if} | ||||||
|  |               </td> | ||||||
|  |             </tr> | ||||||
|  |           {/each} | ||||||
|  |         </tbody> | ||||||
|  |       </table> | ||||||
|  |     </Datatable> | ||||||
|  |   {/if} | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   thead { | ||||||
|  |     background: #fff; | ||||||
|  |   } | ||||||
|  |   thead { | ||||||
|  |     position: sticky; | ||||||
|  |     inset-block-start: 0; | ||||||
|  |   } | ||||||
|  |   tbody td { | ||||||
|  |     padding: 4px; | ||||||
|  |   } | ||||||
|  |   tbody tr:nth-child(even) { | ||||||
|  |     background: #fafafa; | ||||||
|  |   } | ||||||
|  |   tbody tr { | ||||||
|  |     transition: all, 0.2s; | ||||||
|  |   } | ||||||
|  |   tbody tr:hover { | ||||||
|  |     background: #f5f5f5; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|   | |||||||
							
								
								
									
										35
									
								
								src/components/runners/ThFilterGroup.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								src/components/runners/ThFilterGroup.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let groups; | ||||||
|  |   export let handler; | ||||||
|  |   let selected = "all"; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th style="border-bottom: 1px solid #ddd;"> | ||||||
|  |   <select | ||||||
|  |     on:input={() => { | ||||||
|  |       setTimeout(() => { | ||||||
|  |         if (`${selected}`.trim()) { | ||||||
|  |           const value = selected; | ||||||
|  |           handler.filter(value, (runner) => { | ||||||
|  |             if ( | ||||||
|  |               runner.group.id === value || | ||||||
|  |               runner?.group?.parentGroup?.id === value || | ||||||
|  |               value === "all" | ||||||
|  |             ) | ||||||
|  |               return runner; | ||||||
|  |             return ""; | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       }, 50); | ||||||
|  |     }} | ||||||
|  |     bind:value={selected} | ||||||
|  |     name="groupfilter" | ||||||
|  |     id="groupfilter" | ||||||
|  |   > | ||||||
|  |     <option value="all">{$_('all')}</option> | ||||||
|  |     {#each groups as g} | ||||||
|  |       <option value={g.value}>{g.label}</option> | ||||||
|  |     {/each} | ||||||
|  |   </select> | ||||||
|  | </th> | ||||||
| @@ -14,7 +14,7 @@ | |||||||
|     option.firstname + " " + (option.middlename || "") + " " + option.lastname; |     option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
|   const filterRunners = (label, filterText, option) => |   const filterRunners = (label, filterText, option) => | ||||||
|     label.toLowerCase().includes(filterText.toLowerCase()) || |     label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|     option.value.toString().startsWith(filterText.toLowerCase()); |     option.value.id.toString().startsWith(filterText.toLowerCase()); | ||||||
|   function focus(el) { |   function focus(el) { | ||||||
|     el.focus(); |     el.focus(); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,129 +1,147 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { |   import { DataHandler, Datatable, Th, ThFilter } from "@vincjo/datatables"; | ||||||
|     ScanService, |   import { ScanService, TrackService } from "@odit/lfk-client-js"; | ||||||
|   } from "@odit/lfk-client-js"; |  | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import ScansEmptyState from "./ScansEmptyState.svelte"; |   import ScansEmptyState from "./ScansEmptyState.svelte"; | ||||||
|   $: searchvalue = ""; |   import ThFilterRunner from "./ThFilterRunner.svelte"; | ||||||
|  |   import ThFilterTrack from "./ThFilterTrack.svelte"; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   export let current_scans = []; |   export let current_scans = []; | ||||||
|  |   const handler = new DataHandler(current_scans, { rowsPerPage: 20 }); | ||||||
|  |   const rows = handler.getRows(); | ||||||
|   const scans_promise = ScanService.scanControllerGetAll().then((val) => { |   const scans_promise = ScanService.scanControllerGetAll().then((val) => { | ||||||
|     current_scans = val; |     current_scans = val; | ||||||
|  |     handler.setRows(val); | ||||||
|   }); |   }); | ||||||
|   function should_display_based_on_id(id) { |   $: allTracks = []; | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |   TrackService.trackControllerGetAll().then((val) => { | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |     allTracks = val; | ||||||
|  |   }); | ||||||
|  |   function format_laptime(laptime) { | ||||||
|  |     if (laptime == 0 || laptime == null) { | ||||||
|  |       return $_("first-scan-of-the-day"); | ||||||
|     } |     } | ||||||
|     return id.toString() === searchvalue; |     if (laptime < 60) { | ||||||
|   } |       return `${laptime}s`; | ||||||
|   function format_laptime(laptime){ |     } | ||||||
|     if(laptime == 0 || laptime == null){return $_('first-scan-of-the-day')} |     if (laptime < 3600) { | ||||||
|     if(laptime < 60){return `${laptime}s`} |       return `${Math.floor(laptime / 60)}min ${ | ||||||
|     if(laptime < 3600){return `${Math.floor(laptime / 60)}min ${laptime - (Math.floor(laptime / 60)*60)}s`} |         laptime - Math.floor(laptime / 60) * 60 | ||||||
|     return `${Math.floor(laptime / 3600)}h ${laptime - (Math.floor(laptime / 3600)*3600)}min ${laptime - (Math.floor(laptime / 3600)*3600) - (Math.floor(laptime / 60)*60)}` |       }s`; | ||||||
|  |     } | ||||||
|  |     return `${Math.floor(laptime / 3600)}h ${ | ||||||
|  |       laptime - Math.floor(laptime / 3600) * 3600 | ||||||
|  |     }min ${ | ||||||
|  |       laptime - | ||||||
|  |       Math.floor(laptime / 3600) * 3600 - | ||||||
|  |       Math.floor(laptime / 60) * 60 | ||||||
|  |     }`; | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes("SCAN:GET")} | ||||||
|   {#await scans_promise} |   {#await scans_promise} | ||||||
|     <div |     <div | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|       role="alert"> |       role="alert" | ||||||
|       <p class="font-bold">{$_('scans-are-being-loaded')}</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("scans-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:then} | ||||||
|     {#if current_scans.length === 0} |     {#if current_scans.length === 0} | ||||||
|       <ScansEmptyState /> |       <ScansEmptyState /> | ||||||
|     {:else} |     {:else} | ||||||
|       <input |  | ||||||
|         type="search" |  | ||||||
|         bind:value={searchvalue} |  | ||||||
|         placeholder={$_('datatable.search')} |  | ||||||
|         aria-label={$_('datatable.search')} |  | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |  | ||||||
|       <div |       <div | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll" | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |       > | ||||||
|           <thead class="bg-gray-50"> |         <Datatable {handler}> | ||||||
|             <tr> |           <table class="divide-y divide-gray-200 w-full"> | ||||||
|               <th |             <thead class="bg-gray-50"> | ||||||
|                 scope="col" |               <tr> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 <Th {handler} orderBy="id">ID</Th> | ||||||
|                 {$_('runner')} |                 <Th {handler}> | ||||||
|               </th> |                   {$_("runner")} | ||||||
|               <th |                 </Th> | ||||||
|                 scope="col" |                 <Th {handler}> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   {$_("distance")} | ||||||
|                 {$_('distance-track')} |                 </Th> | ||||||
|               </th> |                 <Th {handler}> | ||||||
|               <th |                   {$_("track")} | ||||||
|                 scope="col" |                 </Th> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 <Th {handler}> | ||||||
|                 {$_('laptime')} |                   {$_("laptime")} | ||||||
|               </th> |                 </Th> | ||||||
|               <th |                 <Th {handler}> | ||||||
|                 scope="col" |                   {$_("status")} | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                 </Th> | ||||||
|                 {$_('status')} |                 <th | ||||||
|               </th> |                   scope="col" | ||||||
|               <th scope="col" class="relative px-6 py-3"> |                   class="relative px-6 py-3" | ||||||
|                 <span class="sr-only">{$_('action')}</span> |                   style="border-bottom: 1px solid #ddd;" | ||||||
|               </th> |                 > | ||||||
|             </tr> |                   {$_("action")} | ||||||
|           </thead> |                 </th> | ||||||
|           <tbody class="divide-y divide-gray-200"> |               </tr> | ||||||
|             {#each current_scans as scan} |               <tr> | ||||||
|               {#if scan.track?.name |                 <ThFilter {handler} filterBy="id" /> | ||||||
|                 .toLowerCase() |                 <ThFilterRunner {handler} /> | ||||||
|                 .includes( |                 <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|                   searchvalue.toLowerCase() |                 <ThFilterTrack tracks={allTracks} {handler} /> | ||||||
|                 ) || scan.runner?.firstname |                 <!-- <th style="border-bottom: 1px solid #ddd;" /> --> | ||||||
|                   .toLowerCase() |                 <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|                   .includes( |                 <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|                     searchvalue.toLowerCase() |                 <!-- TODO: filter status --> | ||||||
|                   ) || scan.runner?.middlename |                 <th style="border-bottom: 1px solid #ddd;" /> | ||||||
|                   .toLowerCase() |               </tr> | ||||||
|                   .includes( |             </thead> | ||||||
|                     searchvalue.toLowerCase() |             <tbody class="divide-y divide-gray-200"> | ||||||
|                   ) || scan.runner?.lastname |               {#each $rows as scan} | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || should_display_based_on_id(scan.id)} |  | ||||||
|                 <tr data-rowid="scan_{scan.id}"> |                 <tr data-rowid="scan_{scan.id}"> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap text-left"> | ||||||
|  |                     <div class="text-sm font-medium text-gray-900"> | ||||||
|  |                       {scan.id} | ||||||
|  |                     </div> | ||||||
|  |                   </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                     <div class="flex items-center"> |                     <div class="flex items-center"> | ||||||
|                       <a |                       <a | ||||||
|                         href="../runners/{scan.runner.id}" |                         href="../runners/{scan.runner.id}" | ||||||
|                         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{scan.runner.firstname} |                         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800" | ||||||
|                         {scan.runner.middlename || ''} |                         >{scan.runner.firstname} | ||||||
|                         {scan.runner.lastname}</a> |                         {scan.runner.middlename || ""} | ||||||
|  |                         {scan.runner.lastname}</a | ||||||
|  |                       > | ||||||
|                     </div> |                     </div> | ||||||
|                   </td> |                   </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                   <td class="px-6 py-4 whitespace-nowrap text-left"> | ||||||
|                     <div class="text-sm font-medium text-gray-900"> |                     <div class="text-sm font-medium text-gray-900"> | ||||||
|                       {#if scan.distance < 1000} |                       {#if scan.distance < 1000} | ||||||
|                         {scan.distance}m |                         {scan.distance}m | ||||||
|                       {:else}{scan.distance / 1000}km{/if} |                       {:else}{scan.distance / 1000}km{/if} | ||||||
|  |                     </div> | ||||||
|  |                   </td> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap text-left"> | ||||||
|  |                     <div class="text-sm font-medium text-gray-900"> | ||||||
|                       {#if scan.track} |                       {#if scan.track} | ||||||
|                         <a |                         <a | ||||||
|                           href="../tracks" |                           href="../tracks" | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{scan.track.name} |                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800" | ||||||
|  |                           >{scan.track.name} | ||||||
|                         </a> |                         </a> | ||||||
|                       {/if} |                       {/if} | ||||||
|                     </div> |                     </div> | ||||||
|                   </td> |                   </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                   <td class="px-6 py-4 whitespace-nowrap text-left"> | ||||||
|                     {#if scan.responseType === "TRACKSCAN"} |                     {#if scan.responseType === "TRACKSCAN"} | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |                       <div class="text-sm font-medium text-gray-900"> | ||||||
|                         {format_laptime(scan.lapTime)} |                         {format_laptime(scan.lapTime)} | ||||||
|                       </div> |                       </div> | ||||||
|                     {:else} |                     {:else} | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |                       <div class="text-sm font-medium text-gray-900"> | ||||||
|                         {$_('scan-with-fixed-distance')} |                         {$_("scan-with-fixed-distance")} | ||||||
|                       </div> |                       </div> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </td> |                   </td> | ||||||
| @@ -131,23 +149,30 @@ | |||||||
|                     <div class="flex items-center"> |                     <div class="flex items-center"> | ||||||
|                       {#if scan.valid} |                       {#if scan.valid} | ||||||
|                         <span |                         <span | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('valid')}</span> |                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800" | ||||||
|  |                           >{$_("valid")}</span | ||||||
|  |                         > | ||||||
|                       {:else} |                       {:else} | ||||||
|                         <span |                         <span | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('invalid')}</span> |                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800" | ||||||
|  |                           >{$_("invalid")}</span | ||||||
|  |                         > | ||||||
|                       {/if} |                       {/if} | ||||||
|                     </div> |                     </div> | ||||||
|                   </td> |                   </td> | ||||||
|  |  | ||||||
|                   {#if active_deletes[scan.id] === true} |                   {#if active_deletes[scan.id] === true} | ||||||
|                     <td |                     <td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium" | ||||||
|  |                     > | ||||||
|                       <button |                       <button | ||||||
|                         on:click={() => { |                         on:click={() => { | ||||||
|                           active_deletes[scan.id] = false; |                           active_deletes[scan.id] = false; | ||||||
|                         }} |                         }} | ||||||
|                         tabindex="0" |                         tabindex="0" | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer" | ||||||
|  |                         >{$_("cancel-delete")}</button | ||||||
|  |                       > | ||||||
|                       <button |                       <button | ||||||
|                         on:click={() => { |                         on:click={() => { | ||||||
|                           ScanService.scanControllerRemove(scan.id, false).then( |                           ScanService.scanControllerRemove(scan.id, false).then( | ||||||
| @@ -156,44 +181,51 @@ | |||||||
|                                 (obj) => obj.id !== scan.id |                                 (obj) => obj.id !== scan.id | ||||||
|                               ); |                               ); | ||||||
|                               Toastify({ |                               Toastify({ | ||||||
|                                 text: 'Scan deleted', |                                 text: "Scan deleted", | ||||||
|                                 duration: 500, |                                 duration: 500, | ||||||
|                                 backgroundColor: |                                 backgroundColor: | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |                                   "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|                               }).showToast(); |                               }).showToast(); | ||||||
|                             } |                             } | ||||||
|                           ); |                           ); | ||||||
|                         }} |                         }} | ||||||
|                         tabindex="0" |                         tabindex="0" | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                         >{$_("confirm-delete")}</button | ||||||
|  |                       > | ||||||
|                     </td> |                     </td> | ||||||
|                   {:else} |                   {:else} | ||||||
|                     <td |                     <td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium" | ||||||
|  |                     > | ||||||
|                       <a |                       <a | ||||||
|                         href="./{scan.id}" |                         href="./{scan.id}" | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |                         class="text-indigo-600 hover:text-indigo-900" | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:DELETE')} |                         >{$_("details")}</a | ||||||
|  |                       > | ||||||
|  |                       {#if store.state.jwtinfo.userdetails.permissions.includes("SCAN:DELETE")} | ||||||
|                         <button |                         <button | ||||||
|                           on:click={() => { |                           on:click={() => { | ||||||
|                             active_deletes[scan.id] = true; |                             active_deletes[scan.id] = true; | ||||||
|                           }} |                           }} | ||||||
|                           tabindex="0" |                           tabindex="0" | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |                           >{$_("delete")}</button | ||||||
|  |                         > | ||||||
|                       {/if} |                       {/if} | ||||||
|                     </td> |                     </td> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </tr> |                 </tr> | ||||||
|               {/if} |               {/each} | ||||||
|             {/each} |             </tbody> | ||||||
|           </tbody> |           </table> | ||||||
|         </table> |         </Datatable> | ||||||
|       </div> |       </div> | ||||||
|     {/if} |     {/if} | ||||||
|   {:catch error} |   {:catch error} | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | ||||||
|       <span class="inline-block align-middle mr-8"> |       <span class="inline-block align-middle mr-8"> | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |         <b class="capitalize">{$_("general_promise_error")}</b> | ||||||
|         {error} |         {error} | ||||||
|       </span> |       </span> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								src/components/scans/ThFilterRunner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/components/scans/ThFilterRunner.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | <script> | ||||||
|  |   export let handler; | ||||||
|  |   let filterValue = ""; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th> | ||||||
|  |   <input | ||||||
|  |     on:input={() => { | ||||||
|  |       setTimeout(() => { | ||||||
|  |         const v = filterValue.toLowerCase(); | ||||||
|  |         handler.filter(v, (c) => { | ||||||
|  |           if (v.startsWith("#")) { | ||||||
|  |             return `#${c.runner?.id}`; | ||||||
|  |           } | ||||||
|  |           if (c.runner) { | ||||||
|  |             let runnerName = `${c.runner.firstname} ${c.runner.lastname}`; | ||||||
|  |             if (c.runner.middlename) { | ||||||
|  |               runnerName = `${c.runner.firstname} ${c.runner.middlename} ${c.runner.lastname}`; | ||||||
|  |             } | ||||||
|  |             runnerName = runnerName.toLowerCase(); | ||||||
|  |             return runnerName; | ||||||
|  |           } | ||||||
|  |           return ""; | ||||||
|  |         }); | ||||||
|  |       }, 150); | ||||||
|  |     }} | ||||||
|  |     placeholder="Filter" | ||||||
|  |     bind:value={filterValue} | ||||||
|  |     type="text" | ||||||
|  |     name="runnerfilter" | ||||||
|  |     id="runnerfilter" | ||||||
|  |   /> | ||||||
|  | </th> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   th { | ||||||
|  |     border-bottom: 1px solid #e0e0e0; | ||||||
|  |   } | ||||||
|  |   input { | ||||||
|  |     margin: -1px 0 0 0; | ||||||
|  |     padding: 0; | ||||||
|  |     width: 100%; | ||||||
|  |     height: 24px; | ||||||
|  |     border: none; | ||||||
|  |     text-align: left; | ||||||
|  |     background: inherit; | ||||||
|  |     outline: 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
							
								
								
									
										31
									
								
								src/components/scans/ThFilterTrack.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/components/scans/ThFilterTrack.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let tracks; | ||||||
|  |   export let handler; | ||||||
|  |   let selected = "all"; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th style="border-bottom: 1px solid #ddd;"> | ||||||
|  |   <select | ||||||
|  |     on:input={() => { | ||||||
|  |       setTimeout(() => { | ||||||
|  |         if (`${selected}`.trim()) { | ||||||
|  |           const value = selected; | ||||||
|  |           handler.filter(value, (scan) => { | ||||||
|  |             // TODO: fix filter | ||||||
|  |             if (scan.track.id === value || value === "all") return scan.track.id; | ||||||
|  |             return ""; | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       }, 50); | ||||||
|  |     }} | ||||||
|  |     bind:value={selected} | ||||||
|  |     name="trackfilter" | ||||||
|  |     id="trackfilter" | ||||||
|  |   > | ||||||
|  |     <option value="all">{$_("all")}</option> | ||||||
|  |     {#each tracks as track} | ||||||
|  |       <option value={track.id}>{track.name}</option> | ||||||
|  |     {/each} | ||||||
|  |   </select> | ||||||
|  | </th> | ||||||
| @@ -23,14 +23,14 @@ | |||||||
|         throw new Error(); |         throw new Error(); | ||||||
|       } |       } | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: $_('copied-token-to-clipboard'), |         text: $_("copied-token-to-clipboard"), | ||||||
|         duration: 500, |         duration: 500, | ||||||
|         backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |         backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       copied = true; |       copied = true; | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: $_('error-whyile-copying-to-clipboard'), |         text: $_("error-whyile-copying-to-clipboard"), | ||||||
|         duration: 500, |         duration: 500, | ||||||
|         backgroundColor: |         backgroundColor: | ||||||
|           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
| @@ -75,7 +75,9 @@ | |||||||
|                   d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> |                   d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | ||||||
|             </div> |             </div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900">{$_('token')}</h3> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('token')} | ||||||
|  |               </h3> | ||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <p class="text-sm text-gray-500"> |                 <p class="text-sm text-gray-500"> | ||||||
|                   {$_('the-scanstations-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again')} |                   {$_('the-scanstations-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again')} | ||||||
| @@ -86,7 +88,7 @@ | |||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <label |                 <label | ||||||
|                   for="token" |                   for="token" | ||||||
|                   class="block text-sm font-medium text-gray-700">Token</label> |                   class="block text-sm font-medium text-gray-700">{$_('token')}</label> | ||||||
|                 <div on:click={copy} class="inline-flex"> |                 <div on:click={copy} class="inline-flex"> | ||||||
|                   <p |                   <p | ||||||
|                     name="token" |                     name="token" | ||||||
| @@ -106,7 +108,9 @@ | |||||||
|                         d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" /></svg> |                         d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" /></svg> | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|                 <p class="text-gray-500 text-xs">{$_('click-to-copy-token-to-clipboard')}</p> |                 <p class="text-gray-500 text-xs"> | ||||||
|  |                   {$_('click-to-copy-token-to-clipboard')} | ||||||
|  |                 </p> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ | |||||||
|     MeService.meControllerRemove(true) |     MeService.meControllerRemove(true) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Profile deleted!", |           text: $_('profile-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
|   | |||||||
| @@ -4,6 +4,9 @@ | |||||||
|   import { MeService } from "@odit/lfk-client-js"; |   import { MeService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import ConfirmProfileDeletion from "./ConfirmProfileDeletion.svelte"; |   import ConfirmProfileDeletion from "./ConfirmProfileDeletion.svelte"; | ||||||
|  |   import PasswordStrength, { | ||||||
|  |     password_strong_enough_and_equal, | ||||||
|  |   } from "../auth/PasswordStrength.svelte"; | ||||||
|   $: data_loaded = false; |   $: data_loaded = false; | ||||||
|   $: delete_triggered = false; |   $: delete_triggered = false; | ||||||
|   $: original_data = {}; |   $: original_data = {}; | ||||||
| @@ -15,8 +18,10 @@ | |||||||
|     JSON.stringify(editable) === JSON.stringify(original_data) |     JSON.stringify(editable) === JSON.stringify(original_data) | ||||||
|   ); |   ); | ||||||
|   $: save_enabled = changes_performed && isEmail(editable.email); |   $: save_enabled = changes_performed && isEmail(editable.email); | ||||||
|   $: update_password_enabled = |   $: update_password_enabled = password_strong_enough_and_equal( | ||||||
|     password_change.length > 0 && password_change === password_confirm; |     password_change, | ||||||
|  |     password_confirm | ||||||
|  |   ); | ||||||
|   const user_promise = MeService.meControllerGet().then((data) => { |   const user_promise = MeService.meControllerGet().then((data) => { | ||||||
|     data_loaded = true; |     data_loaded = true; | ||||||
|     data.groups = data.groups.map((g) => g.id); |     data.groups = data.groups.map((g) => g.id); | ||||||
| @@ -45,7 +50,7 @@ | |||||||
|   function changePassword() { |   function changePassword() { | ||||||
|     if (data_loaded === true && update_password_enabled) { |     if (data_loaded === true && update_password_enabled) { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: $_('changing-your-password'), |         text: $_("changing-your-password"), | ||||||
|         duration: 2500, |         duration: 2500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       let postdata = Object.assign({}, original_data); |       let postdata = Object.assign({}, original_data); | ||||||
| @@ -56,7 +61,7 @@ | |||||||
|           password_change = ""; |           password_change = ""; | ||||||
|           postdata = {}; |           postdata = {}; | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: $_('password-changed'), |             text: $_("password-changed"), | ||||||
|             duration: 2500, |             duration: 2500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
| @@ -242,10 +247,7 @@ | |||||||
|                     class="border-gray-300 placeholder-gray-500 appearance-none rounded-md relative block w-full px-3 py-2 border focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" |                     class="border-gray-300 placeholder-gray-500 appearance-none rounded-md relative block w-full px-3 py-2 border focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" | ||||||
|                     placeholder={$_('password')} /> |                     placeholder={$_('password')} /> | ||||||
|                 </div> |                 </div> | ||||||
|                 {#if password_change != password_confirm && password_change.length > 0} |                 <PasswordStrength bind:password_change bind:password_confirm /> | ||||||
|                   <span |  | ||||||
|                     class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">{$_('passwords-dont-match')}</span> |  | ||||||
|                 {/if} |  | ||||||
|               </div> |               </div> | ||||||
|               <div class="px-4 py-3 bg-gray-50 text-right sm:px-6"> |               <div class="px-4 py-3 bg-gray-50 text-right sm:px-6"> | ||||||
|                 <button |                 <button | ||||||
| @@ -257,9 +259,9 @@ | |||||||
|                   {$_('update-password')} |                   {$_('update-password')} | ||||||
|                 </button> |                 </button> | ||||||
|                 {#if update_password_enabled} |                 {#if update_password_enabled} | ||||||
|                 <p> |                   <p> | ||||||
|                   {$_('after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that')} |                     {$_('after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that')} | ||||||
|                 </p> |                   </p> | ||||||
|                 {/if} |                 {/if} | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|   | |||||||
							
								
								
									
										151
									
								
								src/components/statsclients/AddStatsClientModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								src/components/statsclients/AddStatsClientModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { StatsClientService } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let new_client; | ||||||
|  |   export let current_clients; | ||||||
|  |   export let copy_modal_open; | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: description = ""; | ||||||
|  |   $: createbtnenabled = description != ""; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       const toast = Toastify({ | ||||||
|  |         text: $_("statsclient-is-being-added"), | ||||||
|  |         duration: -1, | ||||||
|  |       }).showToast(); | ||||||
|  |  | ||||||
|  |       StatsClientService.statsClientControllerPost({description}) | ||||||
|  |         .then((result) => { | ||||||
|  |           description = ""; | ||||||
|  |           modal_open = false; | ||||||
|  |           // | ||||||
|  |           Toastify({ | ||||||
|  |             text: $_("scanstation-added"), | ||||||
|  |             duration: 500, | ||||||
|  |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |           }).showToast(); | ||||||
|  |           current_clients.push(result); | ||||||
|  |           current_clients = current_clients; | ||||||
|  |           new_client = result; | ||||||
|  |           copy_modal_open = true; | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |           // | ||||||
|  |           toast.hideToast(); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }}> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('create-a-new-statsclient')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_('please-provide-the-required-information-to-create-a-new-statsclient')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="description" | ||||||
|  |                     class="block text-sm font-medium text-gray-700">{$_('description')}</label> | ||||||
|  |                   <input | ||||||
|  |                     use:focus | ||||||
|  |                     autocomplete="off" | ||||||
|  |                     placeholder={$_('description')} | ||||||
|  |                     bind:value={description} | ||||||
|  |                     type="text" | ||||||
|  |                     name="description" | ||||||
|  |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('create')} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('cancel')} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
| @@ -0,0 +1,92 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import { ScanStationService } from "@odit/lfk-client-js"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_station; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   function cancelDelete() { | ||||||
|  |     modal_open = false; | ||||||
|  |     dispatch("cancelDelete", { id: delete_station.id }); | ||||||
|  |   } | ||||||
|  |   function deleteClient() { | ||||||
|  |     ScanStationService.donorControllerRemove( | ||||||
|  |       delete_station.id, | ||||||
|  |       true | ||||||
|  |     ) | ||||||
|  |       .then((resp) => { | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_('statsclient-deleted'), | ||||||
|  |           duration: 500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |         location.replace("./"); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => {}); | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     use:focusTrap | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={cancelDelete}> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg class="h-6 w-6 text-blue-600" fill="currentColor" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z"/></svg> | ||||||
|  |             </div> | ||||||
|  |             <!-- <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('attention')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_( | ||||||
|  |                     'do-you-want-to-delete-this-donor-with-all-related-donations' | ||||||
|  |                   )} | ||||||
|  |                   <br />  | ||||||
|  |                   {$_('all-associated-scans-will-get-deleted-as-well')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |             </div> --> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             on:click={deleteClient} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('confirm-delete-statsclient')} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={cancelDelete} | ||||||
|  |             type="button" | ||||||
|  |             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('cancel-keep-statsclient')} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										129
									
								
								src/components/statsclients/CopyStatsClientTokenModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								src/components/statsclients/CopyStatsClientTokenModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { focusTrap } from "svelte-focus-trap"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { tick, createEventDispatcher } from "svelte"; | ||||||
|  |   export let copy_modal_open; | ||||||
|  |   export let new_client; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   let valueCopy = null; | ||||||
|  |   let areaDom; | ||||||
|  |   let copied = false; | ||||||
|  |   function close() { | ||||||
|  |     copy_modal_open = false; | ||||||
|  |   } | ||||||
|  |   async function copy() { | ||||||
|  |     valueCopy = new_client.key; | ||||||
|  |     await tick(); | ||||||
|  |     areaDom.focus(); | ||||||
|  |     areaDom.select(); | ||||||
|  |     try { | ||||||
|  |       const successful = document.execCommand("copy"); | ||||||
|  |       if (!successful) { | ||||||
|  |         throw new Error(); | ||||||
|  |       } | ||||||
|  |       Toastify({ | ||||||
|  |         text: $_("copied-token-to-clipboard"), | ||||||
|  |         duration: 500, | ||||||
|  |         backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |       }).showToast(); | ||||||
|  |       copied = true; | ||||||
|  |     } catch (err) { | ||||||
|  |       Toastify({ | ||||||
|  |         text: $_("error-whyile-copying-to-clipboard"), | ||||||
|  |         duration: 500, | ||||||
|  |         backgroundColor: | ||||||
|  |           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|  |       }).showToast(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // we can notifi by event or storage about copy status | ||||||
|  |     valueCopy = null; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if copy_modal_open} | ||||||
|  |   {#if valueCopy != null} | ||||||
|  |     <textarea bind:this={areaDom}>{valueCopy}</textarea> | ||||||
|  |   {/if} | ||||||
|  |   <div class="fixed z-10 inset-0 overflow-y-auto" use:focusTrap> | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span> | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline"> | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | ||||||
|  |           <div class="sm:flex sm:items-start"> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_('token')} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_('the-statsclient-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again')} | ||||||
|  |                   <br /> | ||||||
|  |                   {$_('please-copy-the-token-and-store-it-somewhere-save')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <label | ||||||
|  |                   for="token" | ||||||
|  |                   class="block text-sm font-medium text-gray-700">{$_('token')}</label> | ||||||
|  |                 <div on:click={copy} class="inline-flex"> | ||||||
|  |                   <p | ||||||
|  |                     name="token" | ||||||
|  |                     class:bg-green-200={copied} | ||||||
|  |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 p-2"> | ||||||
|  |                     {new_client.key} | ||||||
|  |                   </p> | ||||||
|  |                   <div | ||||||
|  |                     class="bg-gray-200 border-gray-300 border-t border-b border-r text-black rounded-r-md sm:text-sm p-2 mt-1 cursor-pointer"> | ||||||
|  |                     <svg | ||||||
|  |                       xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                       viewBox="0 0 24 24" | ||||||
|  |                       width="24" | ||||||
|  |                       height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                       <path | ||||||
|  |                         fill="currentColor" | ||||||
|  |                         d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" /></svg> | ||||||
|  |                   </div> | ||||||
|  |                 </div> | ||||||
|  |                 <p class="text-gray-500 text-xs"> | ||||||
|  |                   {$_('click-to-copy-token-to-clipboard')} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | ||||||
|  |           <button | ||||||
|  |             on:click={close} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-green-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |             {$_('yes-i-copied-the-token')} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										119
									
								
								src/components/statsclients/StatsClientDetail.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										119
									
								
								src/components/statsclients/StatsClientDetail.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,119 @@ | |||||||
|  | <script> | ||||||
|  |   import { t, _ } from "svelte-i18n"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import PromiseError from "../base/PromiseError.svelte"; | ||||||
|  |   import ConfirmStatsClientDeletion from "./ConfirmStatsClientDeletion.svelte"; | ||||||
|  | 	import { StatsClientService } from "@odit/lfk-client-js"; | ||||||
|  |   let data_loaded = false; | ||||||
|  |   let modal_open; | ||||||
|  |   let delete_client; | ||||||
|  |   export let params; | ||||||
|  |   $: delete_triggered = false; | ||||||
|  |   $: original_data = {}; | ||||||
|  |   const promise = StatsClientService.statsClientControllerGetOne( | ||||||
|  |     params.clientid | ||||||
|  |   ).then((data) => { | ||||||
|  |     data_loaded = true; | ||||||
|  |     original_data = Object.assign(original_data, data); | ||||||
|  |   }); | ||||||
|  |   function deleteClient() { | ||||||
|  |     StatsClientService.statsClientControllerRemove(original_data.id, false) | ||||||
|  |       .then((resp) => { | ||||||
|  |         Toastify({ | ||||||
|  |           text: $_("statsclient-deleted"), | ||||||
|  |           duration: 500, | ||||||
|  |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |         }).showToast(); | ||||||
|  |         location.replace("./"); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => { | ||||||
|  |         modal_open = true; | ||||||
|  |         delete_client = original_data; | ||||||
|  |       }); | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <ConfirmStatsClientDeletion bind:modal_open bind:delete_client /> | ||||||
|  | {#await promise} | ||||||
|  |   {$_('loading-statsclient-details')} | ||||||
|  | {:then} | ||||||
|  |   <section class="container p-5 select-none"> | ||||||
|  |     <div class="flex flex-row mb-4"> | ||||||
|  |       <div class="w-full"> | ||||||
|  |         <nav class="w-full flex"> | ||||||
|  |           <ol class="list-none flex flex-row items-center justify-start"> | ||||||
|  |             <li class="flex items-center"> | ||||||
|  |               <svg | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | ||||||
|  |             </li> | ||||||
|  |             <li class="flex items-center ml-2"> | ||||||
|  |               <a class="mr-2" href="./">{$_('statsclient')}</a><svg | ||||||
|  |                 stroke="currentColor" | ||||||
|  |                 fill="none" | ||||||
|  |                 stroke-width="2" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 stroke-linecap="round" | ||||||
|  |                 stroke-linejoin="round" | ||||||
|  |                 class="h-3 w-3 mr-2 stroke-current" | ||||||
|  |                 height="1em" | ||||||
|  |                 width="1em" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg"><line | ||||||
|  |                   x1="5" | ||||||
|  |                   y1="12" | ||||||
|  |                   x2="19" | ||||||
|  |                   y2="12" /> | ||||||
|  |                 <polyline points="12 5 19 12 12 19" /></svg> | ||||||
|  |             </li> | ||||||
|  |             <li class="flex items-center"> | ||||||
|  |               <span class="mr-2">#{original_data.id}</span> | ||||||
|  |             </li> | ||||||
|  |           </ol> | ||||||
|  |         </nav> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |     <div class="mb-8 text-3xl font-extrabold leading-tight"> | ||||||
|  |       #{original_data.id} | ||||||
|  |       <span data-id="stations_actions_${original_data.id}"> | ||||||
|  |         {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:DELETE')} | ||||||
|  |           {#if delete_triggered} | ||||||
|  |             <button | ||||||
|  |               on:click={deleteClient} | ||||||
|  |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('confirm-deletion')}</button> | ||||||
|  |             <button | ||||||
|  |               on:click={() => { | ||||||
|  |                 delete_triggered = !delete_triggered; | ||||||
|  |               }} | ||||||
|  |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm">{$_('cancel')}</button> | ||||||
|  |           {/if} | ||||||
|  |           {#if !delete_triggered} | ||||||
|  |             <button | ||||||
|  |               on:click={() => { | ||||||
|  |                 delete_triggered = true; | ||||||
|  |               }} | ||||||
|  |               type="button" | ||||||
|  |               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('delete-statsclient')}</button> | ||||||
|  |           {/if} | ||||||
|  |         {/if} | ||||||
|  |       </span> | ||||||
|  |     </div> | ||||||
|  |     <!--  --> | ||||||
|  |     <div class="text-sm w-full"> | ||||||
|  |       <label | ||||||
|  |         for="description" | ||||||
|  |         class="font-medium text-gray-700">{$_('description')}</label> | ||||||
|  |       <p | ||||||
|  |         name="description" | ||||||
|  |         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" > | ||||||
|  |         {original_data.description}</p> | ||||||
|  |     </div> | ||||||
|  |   </section> | ||||||
|  | {:catch error} | ||||||
|  |   <PromiseError {error} /> | ||||||
|  | {/await} | ||||||
							
								
								
									
										33
									
								
								src/components/statsclients/StatsClients.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								src/components/statsclients/StatsClients.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import AddStatsClientModal from "./AddStatsClientModal.svelte"; | ||||||
|  | 	import CopyStatsClientTokenModal from "./CopyStatsClientTokenModal.svelte"; | ||||||
|  |   import StatsClientsOverview from "./StatsClientsOverview.svelte"; | ||||||
|  |   export let modal_open = false; | ||||||
|  |   export let copy_modal_open = false; | ||||||
|  |   export let new_client = {}; | ||||||
|  |   let current_clients = []; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <section class="container p-5"> | ||||||
|  |   <span class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|  |     {$_('statsclients')} | ||||||
|  |     {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:CREATE')} | ||||||
|  |       <button | ||||||
|  |         on:click={() => { | ||||||
|  |           modal_open = true; | ||||||
|  |         }} | ||||||
|  |         type="button" | ||||||
|  |         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | ||||||
|  |         {$_('create-a-new-statsclient')} | ||||||
|  |       </button> | ||||||
|  |     {/if} | ||||||
|  |   </span> | ||||||
|  |   <StatsClientsOverview bind:current_clients bind:modal_open bind:new_client bind:copy_modal_open /> | ||||||
|  | </section> | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:CREATE')} | ||||||
|  | <AddStatsClientModal bind:modal_open bind:current_clients bind:new_client bind:copy_modal_open/> | ||||||
|  | <CopyStatsClientTokenModal bind:copy_modal_open bind:new_client /> | ||||||
|  | {/if} | ||||||
							
								
								
									
										21
									
								
								src/components/statsclients/StatsClientsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/components/statsclients/StatsClientsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  | import AddStatsClientModal from "./AddStatsClientModal.svelte"; | ||||||
|  | import CopyScanStationTokenModal from "./CopyStatsClientTokenModal.svelte"; | ||||||
|  |   import scanstations_empty from "./statsclients_empty.svg"; | ||||||
|  |   let modal_open = false; | ||||||
|  |   let copy_modal_open = false; | ||||||
|  |   let new_client = {}; | ||||||
|  |   let current_clients = []; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="text-center items-center justify-center"> | ||||||
|  |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|  |     <img class="w-full h-44" src={scanstations_empty} alt="" /> | ||||||
|  |     <span class="font-bold">{$_('you-dont-have-any-scanclients-yet')}.</span><br /> | ||||||
|  |     <span>{$_('add-the-first-statsclient')}</span> | ||||||
|  |   </p> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <AddStatsClientModal bind:modal_open bind:current_clients bind:new_client bind:copy_modal_open/> | ||||||
|  | <CopyScanStationTokenModal bind:copy_modal_open bind:new_client /> | ||||||
							
								
								
									
										150
									
								
								src/components/statsclients/StatsClientsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								src/components/statsclients/StatsClientsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import { StatsClientService } from "@odit/lfk-client-js"; | ||||||
|  |   const promise = StatsClientService.statsClientControllerGetAll().then( | ||||||
|  |     (result) => { | ||||||
|  |       current_clients = result; | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import StatsClientsEmptyState from "./StatsClientsEmptyState.svelte"; | ||||||
|  |   import ConfirmStatsClientDeletion from "./ConfirmStatsClientDeletion.svelte"; | ||||||
|  |   $: searchvalue = ""; | ||||||
|  |   $: active_deletes = []; | ||||||
|  |   let delete_client = {}; | ||||||
|  |   let modal_open = false; | ||||||
|  |   export let current_clients = []; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <ConfirmStatsClientDeletion | ||||||
|  |   on:cancelDelete={(event) => { | ||||||
|  |     modal_open = false; | ||||||
|  |     active_deletes[event.detail.id] = false; | ||||||
|  |   }} | ||||||
|  |   bind:modal_open | ||||||
|  |   bind:delete_client /> | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:GET')} | ||||||
|  |   {#await promise} | ||||||
|  |     <div | ||||||
|  |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|  |       role="alert"> | ||||||
|  |       <p class="font-bold">{$_('statsclients-are-being-loaded')}</p> | ||||||
|  |       <p class="text-sm">{$_('this-might-take-a-moment')}</p> | ||||||
|  |     </div> | ||||||
|  |   {:then} | ||||||
|  |     {#if current_clients.length === 0} | ||||||
|  |       <StatsClientsEmptyState /> | ||||||
|  |     {:else} | ||||||
|  |       <input | ||||||
|  |         type="search" | ||||||
|  |         bind:value={searchvalue} | ||||||
|  |         placeholder={$_('datatable.search')} | ||||||
|  |         aria-label={$_('datatable.search')} | ||||||
|  |         class="gridjs-input gridjs-search-input mb-4" /> | ||||||
|  |       <div | ||||||
|  |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> | ||||||
|  |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|  |           <thead class="bg-gray-50"> | ||||||
|  |             <tr> | ||||||
|  |               <th | ||||||
|  |               scope="col" | ||||||
|  |               class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|  |               {$_('description')} | ||||||
|  |             </th> | ||||||
|  |             <th | ||||||
|  |               scope="col" | ||||||
|  |               class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|  |               {$_('prefix')} | ||||||
|  |             </th> | ||||||
|  |               <th scope="col" class="relative px-6 py-3"> | ||||||
|  |                 <span class="sr-only">{$_('action')}</span> | ||||||
|  |               </th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody class="divide-y divide-gray-200"> | ||||||
|  |             {#each current_clients as c} | ||||||
|  |               {#if Object.values(c) | ||||||
|  |                 .toString() | ||||||
|  |                 .toLowerCase() | ||||||
|  |                 .includes(searchvalue)} | ||||||
|  |                 <tr data-rowid="station_{c.id}"> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  |                     <div class="flex items-center"> | ||||||
|  |                       <div class="ml-4"> | ||||||
|  |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|  |                             {c.description} | ||||||
|  |                         </div> | ||||||
|  |                       </div> | ||||||
|  |                     </div> | ||||||
|  |                   </td> | ||||||
|  |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  |                     <div class="flex items-center"> | ||||||
|  |                       <div class="ml-4"> | ||||||
|  |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|  |                           {c.prefix} | ||||||
|  |                         </div> | ||||||
|  |                       </div> | ||||||
|  |                     </div> | ||||||
|  |                   </td> | ||||||
|  |                   {#if active_deletes[c.id] === true} | ||||||
|  |                     <td | ||||||
|  |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|  |                       <button | ||||||
|  |                         on:click={() => { | ||||||
|  |                           active_deletes[c.id] = false; | ||||||
|  |                         }} | ||||||
|  |                         tabindex="0" | ||||||
|  |                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> | ||||||
|  |                       <button | ||||||
|  |                         on:click={() => { | ||||||
|  |                           StatsClientService.statsClientControllerRemove(c.id, false) | ||||||
|  |                             .then((resp) => { | ||||||
|  |                               current_clients = current_clients.filter((obj) => obj.id !== c.id); | ||||||
|  |                               Toastify({ | ||||||
|  |                                 text: $_('statsclient-deleted'), | ||||||
|  |                                 duration: 500, | ||||||
|  |                                 backgroundColor: | ||||||
|  |                                   'linear-gradient(to right, #00b09b, #96c93d)', | ||||||
|  |                               }).showToast(); | ||||||
|  |                             }) | ||||||
|  |                             .catch((err) => { | ||||||
|  |                               modal_open = true; | ||||||
|  |                               delete_client = c; | ||||||
|  |                             }); | ||||||
|  |                         }} | ||||||
|  |                         tabindex="0" | ||||||
|  |                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> | ||||||
|  |                     </td> | ||||||
|  |                   {:else} | ||||||
|  |                     <td | ||||||
|  |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|  |                       <a | ||||||
|  |                         href="/statsclients/{c.id}" | ||||||
|  |                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> | ||||||
|  |                       {#if store.state.jwtinfo.userdetails.permissions.includes('STATSCLIENT:DELETE')} | ||||||
|  |                         <button | ||||||
|  |                           on:click={() => { | ||||||
|  |                             active_deletes[c.id] = true; | ||||||
|  |                           }} | ||||||
|  |                           tabindex="0" | ||||||
|  |                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> | ||||||
|  |                       {/if} | ||||||
|  |                     </td> | ||||||
|  |                   {/if} | ||||||
|  |                 </tr> | ||||||
|  |               {/if} | ||||||
|  |             {/each} | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |       </div> | ||||||
|  |     {/if} | ||||||
|  |   {:catch error} | ||||||
|  |     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | ||||||
|  |       <span class="inline-block align-middle mr-8"> | ||||||
|  |         <b class="capitalize">{$_('general_promise_error')}</b> | ||||||
|  |         {error} | ||||||
|  |       </span> | ||||||
|  |     </div> | ||||||
|  |   {/await} | ||||||
|  | {/if} | ||||||
							
								
								
									
										1
									
								
								src/components/statsclients/statsclients_empty.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/components/statsclients/statsclients_empty.svg
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| After Width: | Height: | Size: 5.0 KiB | 
| @@ -43,7 +43,7 @@ | |||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       const toast = Toastify({ | ||||||
|         text: "Team is being added...", |         text: $_('team-is-being-added'), | ||||||
|         duration: -1, |         duration: -1, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       RunnerTeamService.runnerTeamControllerPost({ |       RunnerTeamService.runnerTeamControllerPost({ | ||||||
| @@ -55,7 +55,7 @@ | |||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           // |           // | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: "Team added", |             text: $_('team-added'), | ||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ | |||||||
|     RunnerTeamService.runnerTeamControllerRemove(delete_team.id, true) |     RunnerTeamService.runnerTeamControllerRemove(delete_team.id, true) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Team deleted", |           text: $_('team-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
|   | |||||||
| @@ -11,7 +11,10 @@ | |||||||
|   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; |   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; | ||||||
|   import PromiseError from "../base/PromiseError.svelte"; |   import PromiseError from "../base/PromiseError.svelte"; | ||||||
|   import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte"; |   import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte"; | ||||||
| import Teams from "./Teams.svelte"; |   import Teams from "./Teams.svelte"; | ||||||
|  |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   let [teamdata, original, delete_team, orgs, contacts, modal_open] = [ |   let [teamdata, original, delete_team, orgs, contacts, modal_open] = [ | ||||||
|     {}, |     {}, | ||||||
|     {}, |     {}, | ||||||
| @@ -23,10 +26,13 @@ import Teams from "./Teams.svelte"; | |||||||
|   export let params; |   export let params; | ||||||
|   export let import_modal_open = false; |   export let import_modal_open = false; | ||||||
|   $: delete_triggered = false; |   $: delete_triggered = false; | ||||||
|   $: save_enabled = !data_changed && teamdata.parentGroup != null; |   $: save_enabled = data_changed && teamdata.parentGroup != null; | ||||||
|   $: data_loaded = false; |   $: data_loaded = false; | ||||||
|   $: data_changed = JSON.stringify(teamdata) === JSON.stringify(original); |   $: data_changed = !(JSON.stringify(teamdata) === JSON.stringify(original)); | ||||||
|   $: sponsoring_contracts_download_open = false; |   $: sponsoring_contracts_show = true; | ||||||
|  |   $: cards_show = true; | ||||||
|  |   $: certificates_show = true; | ||||||
|  |   $: generate_teams = [original]; | ||||||
|   $: group = {}; |   $: group = {}; | ||||||
|   $: contact = {}; |   $: contact = {}; | ||||||
|   // |   // | ||||||
| @@ -38,37 +44,32 @@ import Teams from "./Teams.svelte"; | |||||||
|     data_loaded = true; |     data_loaded = true; | ||||||
|     teamdata = Object.assign(teamdata, value); |     teamdata = Object.assign(teamdata, value); | ||||||
|     original = Object.assign(original, value); |     original = Object.assign(original, value); | ||||||
|     RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { |     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|       orgs = val.map((r) => { |       (val) => { | ||||||
|         return { label: r.name, value: r }; |         orgs = val.map((r) => { | ||||||
|       }); |           delete r.contact; | ||||||
|       group = orgs.find((g) => g.value.id == teamdata.parentGroup.id); |           r.teams = []; | ||||||
|     }); |           return { label: r.name, value: r }; | ||||||
|  |         }); | ||||||
|  |         group = orgs.find((g) => g.value.id == teamdata.parentGroup.id); | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|     GroupContactService.groupContactControllerGetAll().then((val) => { |     GroupContactService.groupContactControllerGetAll().then((val) => { | ||||||
|       contacts = val.map((r) => { |       contacts = val.map((r) => { | ||||||
|         return { label: getContactLabel(r), value: r }; |         return { label: getContactLabel(r), value: r }; | ||||||
|       }); |       }); | ||||||
|       if(teamdata.contact){ |       if (teamdata.contact) { | ||||||
|         contact = contacts.find((g) => g.value.id == teamdata.contact.id); |         contact = contacts.find((g) => g.value.id == teamdata.contact.id); | ||||||
|       } |       } else { | ||||||
|       else{ |  | ||||||
|         contact = null; |         contact = null; | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|     if ( |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|     ) { |  | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|   function deleteTeam() { |   function deleteTeam() { | ||||||
|     RunnerTeamService.runnerTeamControllerRemove(original.id, false) |     RunnerTeamService.runnerTeamControllerRemove(original.id, false) | ||||||
|       .then((resp) => { |       .then((resp) => { | ||||||
|         Toastify({ |         Toastify({ | ||||||
|           text: "Organization deleted", |           text: $_('team-deleted'), | ||||||
|           duration: 500, |           duration: 500, | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |         }).showToast(); | ||||||
| @@ -82,7 +83,7 @@ import Teams from "./Teams.svelte"; | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (data_loaded === true && save_enabled) { |     if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: "updating team", |         text: $_('updating-team'), | ||||||
|         duration: 2500, |         duration: 2500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       let postdata = teamdata; |       let postdata = teamdata; | ||||||
| @@ -93,7 +94,7 @@ import Teams from "./Teams.svelte"; | |||||||
|           Object.assign(original, teamdata); |           Object.assign(original, teamdata); | ||||||
|           original = original; |           original = original; | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: "updated team", |             text: $_('updated-team'), | ||||||
|             duration: 2500, |             duration: 2500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
| @@ -101,55 +102,6 @@ import Teams from "./Teams.svelte"; | |||||||
|         .catch((err) => {}); |         .catch((err) => {}); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   async function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const runners = await RunnerTeamService.runnerTeamControllerGetRunners( |  | ||||||
|       teamdata.id |  | ||||||
|     ); |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdf"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     fetch( |  | ||||||
|       `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|       { |  | ||||||
|         method: "POST", |  | ||||||
|         headers: { |  | ||||||
|           "Content-Type": "application/json", |  | ||||||
|         }, |  | ||||||
|         body: JSON.stringify(runners), |  | ||||||
|       } |  | ||||||
|     ) |  | ||||||
|       .then((response) => { |  | ||||||
|         if (response.status != "200") { |  | ||||||
|           toast.hideToast(); |  | ||||||
|           Toastify({ |  | ||||||
|             text: $_("pdf-generation-failed"), |  | ||||||
|             duration: 3500, |  | ||||||
|             backgroundColor: |  | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|           }).showToast(); |  | ||||||
|         } else { |  | ||||||
|           return response.blob(); |  | ||||||
|         } |  | ||||||
|       }) |  | ||||||
|       .then((blob) => { |  | ||||||
|         const url = window.URL.createObjectURL(blob); |  | ||||||
|         let a = document.createElement("a"); |  | ||||||
|         a.href = url; |  | ||||||
|         a.download = "Sponsorings_" + teamdata.name + ".pdf"; |  | ||||||
|         document.body.appendChild(a); |  | ||||||
|         a.click(); |  | ||||||
|         a.remove(); |  | ||||||
|         toast.hideToast(); |  | ||||||
|         Toastify({ |  | ||||||
|           text: $_("pdf-successfully-generated"), |  | ||||||
|           duration: 3500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => {}); |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <ImportRunnerModal | <ImportRunnerModal | ||||||
| @@ -168,64 +120,15 @@ import Teams from "./Teams.svelte"; | |||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> |     <div class="mb-8 text-3xl font-extrabold leading-tight"> | ||||||
|       {original.name} |       {original.name} | ||||||
|       <span data-id="org_actions_${teamdata.id}"> |       <span data-id="org_actions_${teamdata.id}"> | ||||||
|         <div id="sponsoring:dropdown" class="relative inline-block"> |         <GenerateSponsoringContracts | ||||||
|           <div> |           bind:sponsoring_contracts_show | ||||||
|             <button |           bind:generate_teams /> | ||||||
|               on:click={() => { |         <GenerateRunnerCards | ||||||
|                 sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |           bind:cards_show | ||||||
|               }} |           bind:generate_teams /> | ||||||
|               type="button" |         <GenerateRunnerCertificates | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |           bind:certificates_show | ||||||
|               id="options-menu" |           bind:generate_teams /> | ||||||
|               aria-haspopup="true" |  | ||||||
|               aria-expanded="true"> |  | ||||||
|               {$_('generate-sponsoring-contracts')} |  | ||||||
|               <svg |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                 width="24" |  | ||||||
|                 height="24" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 class="-mr-1 ml-2 h-5 w-5"><path |  | ||||||
|                   fill="none" |  | ||||||
|                   d="M0 0h24v24H0z" /> |  | ||||||
|                 <path |  | ||||||
|                   fill="currentColor" |  | ||||||
|                   d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|           {#if sponsoring_contracts_download_open} |  | ||||||
|             <div |  | ||||||
|               class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|               id="sponsoring:dropdown:menu"> |  | ||||||
|               <div |  | ||||||
|                 class="py-1" |  | ||||||
|                 role="menu" |  | ||||||
|                 aria-orientation="vertical" |  | ||||||
|                 aria-labelledby="options-menu"> |  | ||||||
|                 <span |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('de'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('german')} |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('en'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('english')} |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           {/if} |  | ||||||
|         </div> |  | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} |         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
| @@ -367,7 +270,7 @@ import Teams from "./Teams.svelte"; | |||||||
|         placeholder={$_('no-contact-selected')} |         placeholder={$_('no-contact-selected')} | ||||||
|         noOptionsMessage={$_('no-contact-found')} |         noOptionsMessage={$_('no-contact-found')} | ||||||
|         bind:selectedValue={contact} |         bind:selectedValue={contact} | ||||||
|         on:select={(selectedValue)=> teamdata.contact = selectedValue.detail.value} |         on:select={(selectedValue) => (teamdata.contact = selectedValue.detail.value)} | ||||||
|         on:clear={() => (teamdata.contact = null)} /> |         on:clear={() => (teamdata.contact = null)} /> | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
| @@ -380,13 +283,15 @@ import Teams from "./Teams.svelte"; | |||||||
|             .toLowerCase() |             .toLowerCase() | ||||||
|             .includes( |             .includes( | ||||||
|               filterText.toLowerCase() |               filterText.toLowerCase() | ||||||
|             ) || option.id.value.toString().startsWith(filterText.toLowerCase())} |             ) || option.id.value | ||||||
|  |             .toString() | ||||||
|  |             .startsWith(filterText.toLowerCase())} | ||||||
|         items={orgs} |         items={orgs} | ||||||
|         showChevron={true} |         showChevron={true} | ||||||
|         placeholder={$_('search-for-an-organization-by-name-or-id')} |         placeholder={$_('search-for-an-organization-by-name-or-id')} | ||||||
|         noOptionsMessage={$_('no-organizations-found')} |         noOptionsMessage={$_('no-organizations-found')} | ||||||
|         bind:selectedValue={group} |         bind:selectedValue={group} | ||||||
|         on:select={(selectedValue)=> teamdata.parentGroup = selectedValue.detail.value} |         on:select={(selectedValue) => (teamdata.parentGroup = selectedValue.detail.value)} | ||||||
|         on:clear={() => (teamdata.parentGroup = null)} /> |         on:clear={() => (teamdata.parentGroup = null)} /> | ||||||
|     </div> |     </div> | ||||||
|   </section> |   </section> | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ | |||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> |   <span class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('teams')} |     {$_('teams')} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('USER:CREATE')} |     {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:CREATE')} | ||||||
|       <button |       <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
|           modal_open = true; |           modal_open = true; | ||||||
| @@ -27,6 +27,6 @@ | |||||||
|   <TeamsOverview bind:current_teams /> |   <TeamsOverview bind:current_teams /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('USER:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:CREATE')} | ||||||
|   <AddTeamModal bind:current_teams bind:modal_open /> |   <AddTeamModal bind:current_teams bind:modal_open /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -7,9 +7,21 @@ | |||||||
|   import TeamsEmptyState from "./TeamsEmptyState.svelte"; |   import TeamsEmptyState from "./TeamsEmptyState.svelte"; | ||||||
|   import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte"; |   import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   $: sponsoring_contracts_download_open = false; |   $: sponsoring_contracts_show = current_teams.some( | ||||||
|  |     (r) => r.is_selected === true | ||||||
|  |   ); | ||||||
|  |   $: cards_show = current_teams.some( | ||||||
|  |     (r) => r.is_selected === true | ||||||
|  |   ); | ||||||
|  |   $: certificates_show = current_teams.some( | ||||||
|  |     (r) => r.is_selected === true | ||||||
|  |   ); | ||||||
|  |   $: generate_teams = current_teams.filter((r) => r.is_selected === true); | ||||||
|   export let current_teams = []; |   export let current_teams = []; | ||||||
|   let modal_open = false; |   let modal_open = false; | ||||||
|   let delete_team = {}; |   let delete_team = {}; | ||||||
| @@ -19,70 +31,6 @@ | |||||||
|   teams_promise.then((data) => { |   teams_promise.then((data) => { | ||||||
|     usersstore.set(data); |     usersstore.set(data); | ||||||
|   }); |   }); | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|       if ( |  | ||||||
|         e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|         e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|       ) { |  | ||||||
|         sponsoring_contracts_download_open = false; |  | ||||||
|       } |  | ||||||
|   }); |  | ||||||
|   async function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const teams = current_teams.filter((r) => r.is_selected === true); |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdfs"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     let count = 0; |  | ||||||
|     for await (const t of teams) { |  | ||||||
|       count++; |  | ||||||
|       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( |  | ||||||
|         t.id |  | ||||||
|       ); |  | ||||||
|       fetch( |  | ||||||
|         `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|         { |  | ||||||
|           method: "POST", |  | ||||||
|           headers: { |  | ||||||
|             "Content-Type": "application/json", |  | ||||||
|           }, |  | ||||||
|           body: JSON.stringify(runners), |  | ||||||
|         } |  | ||||||
|       ) |  | ||||||
|         .then((response) => { |  | ||||||
|           if (response.status != "200") { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdf-generation-failed"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: |  | ||||||
|                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } else { |  | ||||||
|             return response.blob(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .then((blob) => { |  | ||||||
|           const url = window.URL.createObjectURL(blob); |  | ||||||
|           let a = document.createElement("a"); |  | ||||||
|           a.href = url; |  | ||||||
|           a.download = "Sponsorings_" + t.name + ".pdf"; |  | ||||||
|           document.body.appendChild(a); |  | ||||||
|           a.click(); |  | ||||||
|           a.remove(); |  | ||||||
|           if (count === teams.length) { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdfs-successfully-generated"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .catch((err) => {}); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <ConfirmTeamDeletion | <ConfirmTeamDeletion | ||||||
| @@ -111,69 +59,15 @@ | |||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="gridjs-input gridjs-search-input mb-4" /> | ||||||
|       <div class="h-12"> |       <div class="h-12"> | ||||||
|         {#if current_teams.some((r) => r.is_selected === true)} |         <GenerateSponsoringContracts | ||||||
|           <div id="sponsoring:dropdown" class="relative inline-block"> |           bind:sponsoring_contracts_show | ||||||
|             <div> |           bind:generate_teams /> | ||||||
|               <button |         <GenerateRunnerCards | ||||||
|                 on:click={() => { |           bind:cards_show | ||||||
|                   sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |           bind:generate_teams /> | ||||||
|                 }} |         <GenerateRunnerCertificates | ||||||
|                 type="button" |           bind:certificates_show | ||||||
|                 class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |           bind:generate_teams /> | ||||||
|                 id="options-menu" |  | ||||||
|                 aria-haspopup="true" |  | ||||||
|                 aria-expanded="true"> |  | ||||||
|                 {$_('generate-sponsoring-contracts')} |  | ||||||
|                 <svg |  | ||||||
|                   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                   width="24" |  | ||||||
|                   height="24" |  | ||||||
|                   viewBox="0 0 24 24" |  | ||||||
|                   class="-mr-1 ml-2 h-5 w-5"><path |  | ||||||
|                     fill="none" |  | ||||||
|                     d="M0 0h24v24H0z" /> |  | ||||||
|                   <path |  | ||||||
|                     fill="currentColor" |  | ||||||
|                     d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |  | ||||||
|               </button> |  | ||||||
|             </div> |  | ||||||
|             {#if sponsoring_contracts_download_open} |  | ||||||
|               <div |  | ||||||
|                 class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|                 id="sponsoring:dropdown:menu" |  | ||||||
|                 on:click_outside={() => { |  | ||||||
|                   sponsoring_contracts_download_open = false; |  | ||||||
|                 }}> |  | ||||||
|                 <div |  | ||||||
|                   class="py-1" |  | ||||||
|                   role="menu" |  | ||||||
|                   aria-orientation="vertical" |  | ||||||
|                   aria-labelledby="options-menu"> |  | ||||||
|                   <span |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('de'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('german')} |  | ||||||
|                   </button> |  | ||||||
|                   <button |  | ||||||
|                     on:click={() => { |  | ||||||
|                       generateSponsoringContract('en'); |  | ||||||
|                     }} |  | ||||||
|                     type="submit" |  | ||||||
|                     class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                     role="menuitem"> |  | ||||||
|                     {$_('english')} |  | ||||||
|                   </button> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             {/if} |  | ||||||
|           </div> |  | ||||||
|         {/if} |  | ||||||
|       </div> |       </div> | ||||||
|       <div |       <div | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ | |||||||
|         `[data-id="triggered_table_actions_${trackid}"]` |         `[data-id="triggered_table_actions_${trackid}"]` | ||||||
|       ).parentNode.parentNode.parentNode; |       ).parentNode.parentNode.parentNode; | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: "Track is being updated...", |         text: $_('track-is-being-updated'), | ||||||
|         duration: 500, |         duration: 500, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       TrackService.trackControllerPut(trackid, { |       TrackService.trackControllerPut(trackid, { | ||||||
| @@ -45,7 +45,7 @@ | |||||||
|       }) |       }) | ||||||
|         .then((r) => { |         .then((r) => { | ||||||
|           Toastify({ |           Toastify({ | ||||||
|             text: "Track was updated!", |             text: $_('track-was-updated'), | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|             duration: 1000, |             duration: 1000, | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|   | |||||||
| @@ -5,6 +5,9 @@ | |||||||
|   import { UserService } from "@odit/lfk-client-js"; |   import { UserService } from "@odit/lfk-client-js"; | ||||||
|   import isEmail from "validator/es/lib/isEmail"; |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|  |   import PasswordStrength, { | ||||||
|  |     password_strong_enough, | ||||||
|  |   } from "../auth/PasswordStrength.svelte"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_users; |   export let current_users; | ||||||
|   let firstname_input; |   let firstname_input; | ||||||
| @@ -28,7 +31,10 @@ | |||||||
|   $: isLastnameValid = lastname_input_value.trim().length !== 0; |   $: isLastnameValid = lastname_input_value.trim().length !== 0; | ||||||
|   $: isFirstnameValid = firstname_input_value.trim().length !== 0; |   $: isFirstnameValid = firstname_input_value.trim().length !== 0; | ||||||
|   $: createbtnenabled = |   $: createbtnenabled = | ||||||
|     isFirstnameValid && isLastnameValid && isPasswordValid && isEmailValid; |     isFirstnameValid && | ||||||
|  |     isLastnameValid && | ||||||
|  |     password_strong_enough(password_input_value) && | ||||||
|  |     isEmailValid; | ||||||
|   (function () { |   (function () { | ||||||
|     document.onkeydown = function (e) { |     document.onkeydown = function (e) { | ||||||
|       e = e || window.event; |       e = e || window.event; | ||||||
| @@ -203,12 +209,8 @@ | |||||||
|                     type="password" |                     type="password" | ||||||
|                     name="password" |                     name="password" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | ||||||
|                   {#if !isPasswordValid} |                   <PasswordStrength | ||||||
|                     <span |                     bind:password_change={password_input_value} /> | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |  | ||||||
|                       {$_('password-is-required')} |  | ||||||
|                     </span> |  | ||||||
|                   {/if} |  | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ | |||||||
|   }); |   }); | ||||||
|   function submit() { |   function submit() { | ||||||
|     Toastify({ |     Toastify({ | ||||||
|       text: "updating permissions...", |       text: $_('updating-permissions'), | ||||||
|       duration: 2500, |       duration: 2500, | ||||||
|     }).showToast(); |     }).showToast(); | ||||||
|     to_delete.forEach((d) => { |     to_delete.forEach((d) => { | ||||||
|   | |||||||
| @@ -82,14 +82,6 @@ | |||||||
|                 <tr data-rowid="user_{u.id}"> |                 <tr data-rowid="user_{u.id}"> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                     <div class="flex items-center"> |                     <div class="flex items-center"> | ||||||
|                       {#if u.profilePic} |  | ||||||
|                         <div class="flex-shrink-0 h-10 w-10"> |  | ||||||
|                           <img |  | ||||||
|                             class="h-10 w-10 rounded-full" |  | ||||||
|                             src={u.profilePic} |  | ||||||
|                             alt="" /> |  | ||||||
|                         </div> |  | ||||||
|                       {/if} |  | ||||||
|                       <div class="ml-4"> |                       <div class="ml-4"> | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                           {u.firstname} |                           {u.firstname} | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								src/index.js
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								src/index.js
									
									
									
									
									
								
							| @@ -1,14 +0,0 @@ | |||||||
| import App from './App.svelte'; |  | ||||||
|  |  | ||||||
| const app = new App({ |  | ||||||
| 	target: document.body |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export default app; |  | ||||||
| // HMR |  | ||||||
| if (import.meta.hot) { |  | ||||||
| 	import.meta.hot.accept(); |  | ||||||
| 	import.meta.hot.dispose(() => { |  | ||||||
| 		app.$destroy(); |  | ||||||
| 	}); |  | ||||||
| } |  | ||||||
| @@ -4,11 +4,15 @@ | |||||||
|     "about": "Über", |     "about": "Über", | ||||||
|     "action": "Aktionen", |     "action": "Aktionen", | ||||||
|     "active": "Aktiv", |     "active": "Aktiv", | ||||||
|  |     "add-card": "Karte erstellen", | ||||||
|     "add-donation": "Sponsoring erstellen", |     "add-donation": "Sponsoring erstellen", | ||||||
|     "add-donor": "Sponsor:in erstellen", |     "add-donor": "Sponsor:in erstellen", | ||||||
|  |     "add-or-update-a-payment": "Zahlung hinzufügen oder bearbeiten", | ||||||
|     "add-scan": "Scan erstellen", |     "add-scan": "Scan erstellen", | ||||||
|     "add-the-first-scanstation": "Erstelle deine erste Scannerstation.", |     "add-the-first-scanstation": "Erstelle deine erste Scannerstation.", | ||||||
|  |     "add-the-first-statsclient": "Erstelle deinen ersten Statsclient.", | ||||||
|     "add-user-group": "Neue Gruppe erstellen", |     "add-user-group": "Neue Gruppe erstellen", | ||||||
|  |     "add-your-first-card": "Erstelle deine erste Läuferkarte", | ||||||
|     "add-your-first-contact": "Erstelle den ersten Kontakt", |     "add-your-first-contact": "Erstelle den ersten Kontakt", | ||||||
|     "add-your-first-donor": "Erstelle die erste Sponsor:in", |     "add-your-first-donor": "Erstelle die erste Sponsor:in", | ||||||
|     "add-your-first-group": "Erstelle die erste Gruppe", |     "add-your-first-group": "Erstelle die erste Gruppe", | ||||||
| @@ -19,14 +23,19 @@ | |||||||
|     "add-your-first-user": "Erstelle die erste Benutzer:in", |     "add-your-first-user": "Erstelle die erste Benutzer:in", | ||||||
|     "add-your-fist-donation": "Erstelle dein erstes Sponsoring", |     "add-your-fist-donation": "Erstelle dein erstes Sponsoring", | ||||||
|     "add-your-fist-scan": "Füge deinen ersten Scan hinzu", |     "add-your-fist-scan": "Füge deinen ersten Scan hinzu", | ||||||
|  |     "adding-card": "Karte wird erstellt", | ||||||
|  |     "adding-donation": "Sponsoring wird erstellt...", | ||||||
|     "adding-scan": "Scan wird hinzugefügt", |     "adding-scan": "Scan wird hinzugefügt", | ||||||
|     "address": "Adresse", |     "address": "Adresse", | ||||||
|     "address-is-required": "Du musst eine Adresse angeben", |     "address-is-required": "Du musst eine Adresse angeben", | ||||||
|     "after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!", |     "after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!", | ||||||
|     "after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that": "Nach der Änderung wirst du abgemeldet - bitte melde dich dann mit deinem neuen Passwort an.", |     "after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that": "Nach der Änderung wirst du abgemeldet - bitte melde dich dann mit deinem neuen Passwort an.", | ||||||
|  |     "all": "Alle", | ||||||
|     "all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls gelöscht", |     "all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls gelöscht", | ||||||
|     "all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!", |     "all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!", | ||||||
|     "all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!", |     "all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!", | ||||||
|  |     "already-paid": "Bereits bezahlt", | ||||||
|  |     "amount": "Anzahl", | ||||||
|     "amount-per-kilometer": "Betrag pro Kilometer", |     "amount-per-kilometer": "Betrag pro Kilometer", | ||||||
|     "apartment-suite-etc": "Apartment, Wohnung, etc.", |     "apartment-suite-etc": "Apartment, Wohnung, etc.", | ||||||
|     "application_name": "Lauf für Kaya! - Admin", |     "application_name": "Lauf für Kaya! - Admin", | ||||||
| @@ -40,32 +49,47 @@ | |||||||
|     "cancel-keep-donor": "Abbrechen, Sponsor:in behalten", |     "cancel-keep-donor": "Abbrechen, Sponsor:in behalten", | ||||||
|     "cancel-keep-my-profile": "Abbrechen, mein Profil behalten", |     "cancel-keep-my-profile": "Abbrechen, mein Profil behalten", | ||||||
|     "cancel-keep-organization": "Abbrechen und Organisation bearbeiten", |     "cancel-keep-organization": "Abbrechen und Organisation bearbeiten", | ||||||
|  |     "cancel-keep-statsclient": "Abbrechen und Statsclient behalten", | ||||||
|     "cancel-keep-team": "Abbrechen, Team behalten", |     "cancel-keep-team": "Abbrechen, Team behalten", | ||||||
|     "cannot-reset-your-password-directly": "Schade. \nWir können das Passwort leider nicht direkt zurücksetzen.\nBitte sende uns eine Mail in der du deine Identität bestätigst.", |     "cannot-reset-your-password-directly": "Schade. \nWir können das Passwort leider nicht direkt zurücksetzen.\nBitte sende uns eine Mail in der du deine Identität bestätigst.", | ||||||
|  |     "card-added": "Karte wurde hinzugefügt", | ||||||
|  |     "card-deleted": "Karte gelöscht", | ||||||
|  |     "card-updated": "Karte aktualisiert", | ||||||
|  |     "cards": "Läuferkarten", | ||||||
|  |     "certificates": "Urkunden", | ||||||
|     "change-your-password-here": "Hier kannst du dein Passwort ändern", |     "change-your-password-here": "Hier kannst du dein Passwort ändern", | ||||||
|     "changing-your-password": "Passwort wird geändert", |     "changing-your-password": "Passwort wird geändert", | ||||||
|     "city": "Stadt", |     "city": "Stadt", | ||||||
|  |     "click-to-copy-the-link-into-your-clipboard": "Klicke auf den Link, um ihn in deine Zwischenablage zu kopieren", | ||||||
|  |     "click-to-copy-token-to-clipboard": "Klicke auf den Token, um ihn in deine Zwischenablage zu kopieren", | ||||||
|     "close": "Schließen", |     "close": "Schließen", | ||||||
|  |     "code": "Code", | ||||||
|     "configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit", |     "configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit", | ||||||
|     "confirm": "Bestätigen", |     "confirm": "Bestätigen", | ||||||
|     "confirm-delete": "Löschung Bestätigen", |     "confirm-delete": "Löschung Bestätigen", | ||||||
|     "confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen", |     "confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen", | ||||||
|     "confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen", |     "confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen", | ||||||
|     "confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.", |     "confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.", | ||||||
|  |     "confirm-delete-statsclient": "Bestätigung, Statsclient löschen", | ||||||
|     "confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.", |     "confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.", | ||||||
|     "confirm-deletion": "Löschung Bestätigen", |     "confirm-deletion": "Löschung Bestätigen", | ||||||
|     "confirm-the-new-password": "Neues Passwort bestätigen", |     "confirm-the-new-password": "Neues Passwort bestätigen", | ||||||
|     "contact": "Kontakt", |     "contact": "Kontakt", | ||||||
|  |     "contact-added": "Kontakt wurde hinzugefügt", | ||||||
|     "contact-deleted": "Kontakt gelöscht", |     "contact-deleted": "Kontakt gelöscht", | ||||||
|     "contact-information": "Kontaktinformation", |     "contact-information": "Kontaktinformation", | ||||||
|  |     "contact-is-being-added": "Kontakt wird erstellt...", | ||||||
|     "contact-is-being-updated": "Kontakt wird aktualisiert ...", |     "contact-is-being-updated": "Kontakt wird aktualisiert ...", | ||||||
|     "contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe", |     "contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe", | ||||||
|     "contacts": "Kontakte", |     "contacts": "Kontakte", | ||||||
|     "contacts-are-being-loaded": "Kontakte werden geladen ...", |     "contacts-are-being-loaded": "Kontakte werden geladen ...", | ||||||
|  |     "copied-link-to-clipboard": "Link wurde in die Zwischenablage kopiert", | ||||||
|  |     "copied-token-to-clipboard": "Token wurde in die Zwischenablage kopiert", | ||||||
|     "count_organizations": "Organisationen (Anzahl)", |     "count_organizations": "Organisationen (Anzahl)", | ||||||
|     "count_teams": "Teams (Anzahl)", |     "count_teams": "Teams (Anzahl)", | ||||||
|     "create": "Erstellen", |     "create": "Erstellen", | ||||||
|     "create-a-new": "Erstelle eine neue", |     "create-a-new": "Erstelle eine neue", | ||||||
|  |     "create-a-new-card": "Neue Läuferkarte erstellen", | ||||||
|     "create-a-new-contact": "Kontakt erstellen", |     "create-a-new-contact": "Kontakt erstellen", | ||||||
|     "create-a-new-distance-donation": "Erstelle ein neues Sponsoring", |     "create-a-new-distance-donation": "Erstelle ein neues Sponsoring", | ||||||
|     "create-a-new-donor": "Neue Sponsor:in erstellen", |     "create-a-new-donor": "Neue Sponsor:in erstellen", | ||||||
| @@ -74,14 +98,21 @@ | |||||||
|     "create-a-new-runner": "Neue Läufer:in erstellen", |     "create-a-new-runner": "Neue Läufer:in erstellen", | ||||||
|     "create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)", |     "create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)", | ||||||
|     "create-a-new-scanstation": "Neue Station erstellen", |     "create-a-new-scanstation": "Neue Station erstellen", | ||||||
|  |     "create-a-new-statsclient": "Neuen Statsclient erstellen", | ||||||
|     "create-a-new-team": "Erstelle ein neues Team", |     "create-a-new-team": "Erstelle ein neues Team", | ||||||
|     "create-a-new-track": "Neuen Track erstellen", |     "create-a-new-track": "Neuen Track erstellen", | ||||||
|     "create-a-new-user": "Neue Benutzer:in anlegen", |     "create-a-new-user": "Neue Benutzer:in anlegen", | ||||||
|     "create-a-new-user-group": "Erstelle eine neue Gruppe", |     "create-a-new-user-group": "Erstelle eine neue Gruppe", | ||||||
|  |     "create-and-generate-pdf": "Erstellen und PDF herunterladen", | ||||||
|  |     "create-bulk-blanco-cards": "Blankokarten erstellen", | ||||||
|  |     "create-bulk-cards": "Blankokarten erstellen", | ||||||
|     "create-organization": "Organisation erstellen", |     "create-organization": "Organisation erstellen", | ||||||
|     "create-team": "Team erstellen", |     "create-team": "Team erstellen", | ||||||
|     "create-track": "Track erstellen", |     "create-track": "Track erstellen", | ||||||
|     "create-user": "Benutzer anlegen", |     "create-user": "Benutzer anlegen", | ||||||
|  |     "create-without-pdf": "Ohne PDF erstellen", | ||||||
|  |     "created-blanco-cards": "Blankokarten wurden erstellt", | ||||||
|  |     "creating-blanco-cards": "Erstelle Blankokarten", | ||||||
|     "credits": "Credits", |     "credits": "Credits", | ||||||
|     "csv_import__class": "Klasse", |     "csv_import__class": "Klasse", | ||||||
|     "csv_import__firstname": "Vorname", |     "csv_import__firstname": "Vorname", | ||||||
| @@ -108,7 +139,7 @@ | |||||||
|     }, |     }, | ||||||
|     "delete": "Löschen", |     "delete": "Löschen", | ||||||
|     "delete-contact": "Kontakt löschen", |     "delete-contact": "Kontakt löschen", | ||||||
|     "delete-donation": "Sponsporing löschen", |     "delete-donation": "Sponsoring löschen", | ||||||
|     "delete-donor": "Sponsor:in löschen", |     "delete-donor": "Sponsor:in löschen", | ||||||
|     "delete-group": "Gruppe löschen", |     "delete-group": "Gruppe löschen", | ||||||
|     "delete-organization": "Organisation löschen", |     "delete-organization": "Organisation löschen", | ||||||
| @@ -116,6 +147,7 @@ | |||||||
|     "delete-runner": "Läufer:in löschen", |     "delete-runner": "Läufer:in löschen", | ||||||
|     "delete-scan": "Scan löschen", |     "delete-scan": "Scan löschen", | ||||||
|     "delete-station": "Station löschen", |     "delete-station": "Station löschen", | ||||||
|  |     "delete-statsclient": "Statsclient löschen", | ||||||
|     "delete-team": "Team Löschen", |     "delete-team": "Team Löschen", | ||||||
|     "delete-user": "Benutzer:in löschen", |     "delete-user": "Benutzer:in löschen", | ||||||
|     "deleted-scan": "Scan wurde gelöscht", |     "deleted-scan": "Scan wurde gelöscht", | ||||||
| @@ -124,6 +156,7 @@ | |||||||
|     "description-optional": "Beschreibung (optional)", |     "description-optional": "Beschreibung (optional)", | ||||||
|     "deselect-all": "Alle abwählen", |     "deselect-all": "Alle abwählen", | ||||||
|     "details": "Details", |     "details": "Details", | ||||||
|  |     "disabled": "deaktiviert", | ||||||
|     "distance": "Distanz", |     "distance": "Distanz", | ||||||
|     "distance-donation": "Sponsoring", |     "distance-donation": "Sponsoring", | ||||||
|     "distance-in-km": "Distanz (in KM)", |     "distance-in-km": "Distanz (in KM)", | ||||||
| @@ -135,6 +168,9 @@ | |||||||
|     "documentation": "Dokumentation", |     "documentation": "Dokumentation", | ||||||
|     "donation-amount": "Sponsoringbetrag", |     "donation-amount": "Sponsoringbetrag", | ||||||
|     "donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.", |     "donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.", | ||||||
|  |     "donation-deleted": "Sponsoring gelöscht", | ||||||
|  |     "donation-updated": "Sponsoring wurde aktualisiert", | ||||||
|  |     "donation_added": "Sponsoring hinzugefügt", | ||||||
|     "donations": "Sponsorings", |     "donations": "Sponsorings", | ||||||
|     "donor": "Sponsor:in", |     "donor": "Sponsor:in", | ||||||
|     "donor-added": "Sponsor:in hinzugefügt", |     "donor-added": "Sponsor:in hinzugefügt", | ||||||
| @@ -148,15 +184,21 @@ | |||||||
|     "dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌", |     "dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌", | ||||||
|     "e-mail-adress": "E-Mail-Adresse", |     "e-mail-adress": "E-Mail-Adresse", | ||||||
|     "edit": "Bearbeiten", |     "edit": "Bearbeiten", | ||||||
|  |     "edit-a-card": "Läuferkarte bearbeiten", | ||||||
|     "edit-permissions": "Berechtigungen bearbeiten", |     "edit-permissions": "Berechtigungen bearbeiten", | ||||||
|     "email_address_or_username": "E-Mail-Adresse/ Benutzername", |     "email_address_or_username": "E-Mail-Adresse/ Benutzername", | ||||||
|     "enabled": "aktiviert", |     "enabled": "aktiviert", | ||||||
|  |     "enabled_large": "Aktiviert", | ||||||
|     "english": "Englisch", |     "english": "Englisch", | ||||||
|  |     "enter-payment": "Zahlung eingeben", | ||||||
|  |     "error-during-import": "Fehler beim Importieren", | ||||||
|  |     "error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage", | ||||||
|     "error_on_login": "😢Fehler beim Login", |     "error_on_login": "😢Fehler beim Login", | ||||||
|     "erteilte": "Direkt erteilte", |     "erteilte": "Direkt erteilte", | ||||||
|     "everything-concerning-your-profile": "Alles zu deinem Profil", |     "everything-concerning-your-profile": "Alles zu deinem Profil", | ||||||
|     "everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃♂️🏃♀️🏃♂️", |     "everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃♂️🏃♀️🏃♂️", | ||||||
|     "faq": "FAQ", |     "faq": "FAQ", | ||||||
|  |     "filename_sponsoringquittungsliste": "SponsoringQuittungsListe", | ||||||
|     "filter-by-organization-team": "Filtern nach Organisation / Team", |     "filter-by-organization-team": "Filtern nach Organisation / Team", | ||||||
|     "first-name": "Vorname", |     "first-name": "Vorname", | ||||||
|     "first-name-is-required": "Vorname muss angegeben werden", |     "first-name-is-required": "Vorname muss angegeben werden", | ||||||
| @@ -166,9 +208,12 @@ | |||||||
|     "geerbte": "geerbte", |     "geerbte": "geerbte", | ||||||
|     "general-stats": "Allgemeine Statistiken", |     "general-stats": "Allgemeine Statistiken", | ||||||
|     "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten", |     "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten", | ||||||
|  |     "generate-runner-certificate": "Urkunde generieren", | ||||||
|  |     "generate-runner-certificates": "Urkunden generieren", | ||||||
|  |     "generate-runnercards": "Läuferkarten generieren", | ||||||
|     "generate-sponsoring-contract": "Sponsoringvertrag generieren", |     "generate-sponsoring-contract": "Sponsoringvertrag generieren", | ||||||
|     "generate-sponsoring-contracts": "Sponsoringverträge generieren", |     "generate-sponsoring-contracts": "Sponsoringverträge generieren", | ||||||
|     "generating-pdf": "Pdf wird generiert...", |     "generating-pdf": "PDF wird generiert...", | ||||||
|     "generating-pdfs": "PDFs werden generiert...", |     "generating-pdfs": "PDFs werden generiert...", | ||||||
|     "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.", |     "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.", | ||||||
|     "german": "Deutsch", |     "german": "Deutsch", | ||||||
| @@ -184,6 +229,7 @@ | |||||||
|     "groups-are-being-loaded": "Gruppen werden geladen", |     "groups-are-being-loaded": "Gruppen werden geladen", | ||||||
|     "home": "Start", |     "home": "Start", | ||||||
|     "icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:", |     "icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:", | ||||||
|  |     "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button": "Wenn du mehrere Blankokarten erstellen willst, nutze doch den \"Blankokarten erstellen\" Knopf.", | ||||||
|     "import-finished": "Import abgeschlossen", |     "import-finished": "Import abgeschlossen", | ||||||
|     "import-runners": "Läufer:innen importieren", |     "import-runners": "Läufer:innen importieren", | ||||||
|     "import__target-organization": "Ziel Organisation", |     "import__target-organization": "Ziel Organisation", | ||||||
| @@ -194,6 +240,8 @@ | |||||||
|     "internal-error": "Interner Fehler", |     "internal-error": "Interner Fehler", | ||||||
|     "invalid": "Ungültig", |     "invalid": "Ungültig", | ||||||
|     "invalid-mail-reset": "Das ist keine gültige E-Mail", |     "invalid-mail-reset": "Das ist keine gültige E-Mail", | ||||||
|  |     "just-enter-how-many-you-want-and-the-system-will-create-them": "Gebe einfach ein, wie viele Blankokarten das System erstellen soll.", | ||||||
|  |     "key": "Schlüssel", | ||||||
|     "laeufer-hinzufuegen": "Läufer:in hinzufügen", |     "laeufer-hinzufuegen": "Läufer:in hinzufügen", | ||||||
|     "laeufer-importieren": "Läufer:innen importieren", |     "laeufer-importieren": "Läufer:innen importieren", | ||||||
|     "laptime": "Rundenzeit", |     "laptime": "Rundenzeit", | ||||||
| @@ -202,6 +250,7 @@ | |||||||
|     "lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.", |     "lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.", | ||||||
|     "license": "Lizenz", |     "license": "Lizenz", | ||||||
|     "licenses-are-being-loaded": "Lizenzen werden geladen...", |     "licenses-are-being-loaded": "Lizenzen werden geladen...", | ||||||
|  |     "loading-cards": "Läuferkarten werden geladen", | ||||||
|     "loading-contact-details": "Kontaktdaten werden geladen ...", |     "loading-contact-details": "Kontaktdaten werden geladen ...", | ||||||
|     "loading-donation-details": "Lade Sponsoringdetails", |     "loading-donation-details": "Lade Sponsoringdetails", | ||||||
|     "loading-donor-details": "Lade Details", |     "loading-donor-details": "Lade Details", | ||||||
| @@ -218,6 +267,10 @@ | |||||||
|     "middle-name": "Mittelname", |     "middle-name": "Mittelname", | ||||||
|     "minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)", |     "minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)", | ||||||
|     "minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein", |     "minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein", | ||||||
|  |     "must-be-at-least-10-characters-long": "Passwort muss mindestens 10 Zeichen lang sein!", | ||||||
|  |     "must-contain-a-lowercase-letter": "Passwort muss einen Großbuchstaben enthalten!", | ||||||
|  |     "must-contain-a-number": "Passwort muss eine Zahl enthalten!", | ||||||
|  |     "must-contain-a-uppercase-letter": "Passwort muss einen Kleinbuchstaben enthalten!", | ||||||
|     "name": "Name", |     "name": "Name", | ||||||
|     "name-is-required": "Der Gruppenname muss angegeben werden", |     "name-is-required": "Der Gruppenname muss angegeben werden", | ||||||
|     "new-password": "Neues Passwort", |     "new-password": "Neues Passwort", | ||||||
| @@ -231,6 +284,8 @@ | |||||||
|     "no-organizations-found": "Keine Organisationen gefunden", |     "no-organizations-found": "Keine Organisationen gefunden", | ||||||
|     "no-runners-found": "Keine Läufer:innen gefunden", |     "no-runners-found": "Keine Läufer:innen gefunden", | ||||||
|     "no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.", |     "no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.", | ||||||
|  |     "non-blanko": "Keine/Blankokarte", | ||||||
|  |     "open": "OFFEN", | ||||||
|     "organization": "Organisation", |     "organization": "Organisation", | ||||||
|     "organization-added": "Organisation hinzugefügt", |     "organization-added": "Organisation hinzugefügt", | ||||||
|     "organization-deleted": "Organisation gelöscht", |     "organization-deleted": "Organisation gelöscht", | ||||||
| @@ -241,6 +296,8 @@ | |||||||
|     "organizations-are-being-loaded": "Organisationen werden geladen ...", |     "organizations-are-being-loaded": "Organisationen werden geladen ...", | ||||||
|     "orgs": "Organisationen", |     "orgs": "Organisationen", | ||||||
|     "oss_credit_description": "Wir verwenden eine Menge Open Source-Software bei diesen Projekten und möchten uns bei den folgenden Projekten und Mitwirkenden bedanken, die dazu beitragen, Open Source großartig zu machen!", |     "oss_credit_description": "Wir verwenden eine Menge Open Source-Software bei diesen Projekten und möchten uns bei den folgenden Projekten und Mitwirkenden bedanken, die dazu beitragen, Open Source großartig zu machen!", | ||||||
|  |     "paid": "BEZAHLT", | ||||||
|  |     "paid-amount": "Gezahlter Betrag", | ||||||
|     "password": "Passwort", |     "password": "Passwort", | ||||||
|     "password-changed": "Passwort wurde aktualisiert!", |     "password-changed": "Passwort wurde aktualisiert!", | ||||||
|     "password-is-required": "Passwort muss angegeben werden", |     "password-is-required": "Passwort muss angegeben werden", | ||||||
| @@ -248,7 +305,8 @@ | |||||||
|     "password-reset-in-progress": "Passwort wird zurückgesetzt...", |     "password-reset-in-progress": "Passwort wird zurückgesetzt...", | ||||||
|     "password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.", |     "password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.", | ||||||
|     "password-reset-successful": "Passwort erfolgreich zurückgesetzt!", |     "password-reset-successful": "Passwort erfolgreich zurückgesetzt!", | ||||||
|     "passwords-dont-match": "Die Passwörter stimmen nicht überein.", |     "passwords-dont-match": "Die Passwörter stimmen nicht überein!", | ||||||
|  |     "payment-amount-must-be-greater-than-0-00eur": "Der Zahlungsbetrag muss größer als 0.00€ sein!", | ||||||
|     "pdf-generation-failed": "PDF Generierung fehlgeschlagen!", |     "pdf-generation-failed": "PDF Generierung fehlgeschlagen!", | ||||||
|     "pdf-successfully-generated": "PDF wurde erfolgreich generiert!", |     "pdf-successfully-generated": "PDF wurde erfolgreich generiert!", | ||||||
|     "pdfs-successfully-generated": "Alle PDFs wurden generiert!", |     "pdfs-successfully-generated": "Alle PDFs wurden generiert!", | ||||||
| @@ -256,6 +314,7 @@ | |||||||
|     "permissions": "Berechtigungen", |     "permissions": "Berechtigungen", | ||||||
|     "permissions-updated": "Berechtigungen aktualisiert!", |     "permissions-updated": "Berechtigungen aktualisiert!", | ||||||
|     "phone": "Telefon", |     "phone": "Telefon", | ||||||
|  |     "please-copy-the-token-and-store-it-somewhere-save": "Bitte kopiere dir den Token und bewahre ihn gut auf.", | ||||||
|     "please-provide-a-password": "Bitte gebe ein Passwort an...", |     "please-provide-a-password": "Bitte gebe ein Passwort an...", | ||||||
|     "please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen", |     "please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen", | ||||||
|     "please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.", |     "please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.", | ||||||
| @@ -268,10 +327,15 @@ | |||||||
|     "please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.", |     "please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.", | ||||||
|     "please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.", |     "please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.", | ||||||
|     "please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.", |     "please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.", | ||||||
|  |     "please-provide-the-required-information-to-create-a-new-scanstation": "Bitte gebe alle für eine Scannerstation notwendigen Informationen an", | ||||||
|  |     "please-provide-the-required-information-to-create-a-new-statsclient": "Bitte gebe alle für einen Statsclient notwendigen Informationen an", | ||||||
|     "please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...", |     "please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...", | ||||||
|  |     "please-wait-a-moment-your-login-is-still-being-processed": "Bitte warte einen Moment, deine Anmeldung wird verarbeitet", | ||||||
|  |     "prefix": "Prefix", | ||||||
|     "privacy": "Datenschutz", |     "privacy": "Datenschutz", | ||||||
|     "privacy-loading": "Datenschutzerklärung lädt...", |     "privacy-loading": "Datenschutzerklärung lädt...", | ||||||
|     "profile": "Profil", |     "profile": "Profil", | ||||||
|  |     "profile-deleted": "Profil gelöscht!", | ||||||
|     "profile-picture": "Profilbild", |     "profile-picture": "Profilbild", | ||||||
|     "profile-updated": "Profil wurde aktualisiert!", |     "profile-updated": "Profil wurde aktualisiert!", | ||||||
|     "read-license": "Lizenz-Text lesen", |     "read-license": "Lizenz-Text lesen", | ||||||
| @@ -285,6 +349,7 @@ | |||||||
|     "runner-import": "Läufer:innen Import", |     "runner-import": "Läufer:innen Import", | ||||||
|     "runner-is-being-added": "Läufer:in wird hinzugefügt...", |     "runner-is-being-added": "Läufer:in wird hinzugefügt...", | ||||||
|     "runner-updated": "Läufer:in aktualisiert!", |     "runner-updated": "Läufer:in aktualisiert!", | ||||||
|  |     "runnercards": "Laeuferkarten", | ||||||
|     "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen", |     "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen", | ||||||
|     "runners": "Läufer", |     "runners": "Läufer", | ||||||
|     "runners-are-being-imported": "Läufer:innen werden importiert ...", |     "runners-are-being-imported": "Läufer:innen werden importiert ...", | ||||||
| @@ -297,6 +362,8 @@ | |||||||
|     "scans": "Scans", |     "scans": "Scans", | ||||||
|     "scans-are-being-loaded": "Scans werden geladen", |     "scans-are-being-loaded": "Scans werden geladen", | ||||||
|     "scanstation": "Scanner Station", |     "scanstation": "Scanner Station", | ||||||
|  |     "scanstation-added": "Station wurde erstellt", | ||||||
|  |     "scanstation-is-being-added": "Scannerstation wird angelegt...", | ||||||
|     "scanstations": "Scanner Stationen", |     "scanstations": "Scanner Stationen", | ||||||
|     "scanstations-are-being-loaded": "Scannerstationen werden geladen...", |     "scanstations-are-being-loaded": "Scannerstationen werden geladen...", | ||||||
|     "search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)", |     "search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)", | ||||||
| @@ -306,23 +373,36 @@ | |||||||
|     "search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)", |     "search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)", | ||||||
|     "select-all": "Alle auswählen", |     "select-all": "Alle auswählen", | ||||||
|     "select-language": "Sprache auswählen", |     "select-language": "Sprache auswählen", | ||||||
|  |     "selfservice-registration": "Selfservice Registrierung", | ||||||
|     "send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services", |     "send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services", | ||||||
|     "set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen", |     "set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen", | ||||||
|     "settings": "Einstellungen", |     "settings": "Einstellungen", | ||||||
|     "settings-for-your-profile": "Die Einstellungen deines Accounts", |     "settings-for-your-profile": "Die Einstellungen deines Accounts", | ||||||
|     "something-about-the-group": "Infos zur Gruppe", |     "something-about-the-group": "Infos zur Gruppe", | ||||||
|  |     "sponsoring-quittungs-liste_herunterladen": "Sponsoring-Quittungs-Liste herunterladen", | ||||||
|  |     "sponsorings": "Sponsoringerklaerungen", | ||||||
|     "stats-are-being-loaded": "Die Statistiken werden geladen...", |     "stats-are-being-loaded": "Die Statistiken werden geladen...", | ||||||
|  |     "statsclient-deleted": "Statsclient wurde gelöscht", | ||||||
|  |     "statsclient-is-being-added": "Statsclient wird angelegt...", | ||||||
|  |     "statsclients": "Statsclient (aka Beamershow)", | ||||||
|  |     "statsclients-are-being-loaded": "Statsclients werden geladen", | ||||||
|     "status": "Status", |     "status": "Status", | ||||||
|     "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können", |     "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können", | ||||||
|     "successful-password-reset": "Passwort erfolgreich zurückgesetzt!", |     "successful-password-reset": "Passwort erfolgreich zurückgesetzt!", | ||||||
|     "team": "Team", |     "team": "Team", | ||||||
|  |     "team-added": "Team wurde hinzugefügt", | ||||||
|  |     "team-deleted": "Team gelöscht", | ||||||
|     "team-detail-is-being-loaded": "Team wird geladen...", |     "team-detail-is-being-loaded": "Team wird geladen...", | ||||||
|  |     "team-is-being-added": "Team wird erstellt...", | ||||||
|     "team-name": "Teamname", |     "team-name": "Teamname", | ||||||
|     "team-name-is-required": "Teamname ist erforderlich", |     "team-name-is-required": "Teamname ist erforderlich", | ||||||
|     "teams": "Teams", |     "teams": "Teams", | ||||||
|     "teams-are-being-loaded": "Teams werden geladen ...", |     "teams-are-being-loaded": "Teams werden geladen ...", | ||||||
|     "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number": "Die angegebene Telefonnummer ist nicht korrekt. <br /> Bitte gebe eine Telefonnummer im internationalen Format an...", |     "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number": "Die angegebene Telefonnummer ist nicht korrekt. <br /> Bitte gebe eine Telefonnummer im internationalen Format an...", | ||||||
|     "the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.", |     "the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.", | ||||||
|  |     "the-scanstations-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again": "Der Scannerstation Token wird nur einmal angezeigt - du kannst ihn nicht ändern oder ihn dir nochmal anzeigen lassen!", | ||||||
|  |     "the-statsclient-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again": "Der Statsclient Token wird nur einmal angezeigt - du kannst ihn nicht ändern oder ihn dir nochmal anzeigen lassen!", | ||||||
|  |     "there-are-no-cards-yet": "Es gibt noch keine Läuferkarten.", | ||||||
|     "there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.", |     "there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.", | ||||||
|     "there-are-no-donations-yet": "Es gibt noch keine Sponsorings", |     "there-are-no-donations-yet": "Es gibt noch keine Sponsorings", | ||||||
|     "there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen", |     "there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen", | ||||||
| @@ -332,30 +412,42 @@ | |||||||
|     "there-are-no-scans-yet": "Es gibt noch keine Scans", |     "there-are-no-scans-yet": "Es gibt noch keine Scans", | ||||||
|     "there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.", |     "there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.", | ||||||
|     "there-are-no-users-added-yet": "Es wurden noch keine Benutzer hinzugefügt.", |     "there-are-no-users-added-yet": "Es wurden noch keine Benutzer hinzugefügt.", | ||||||
|  |     "this-card-is": "Diese Karte ist", | ||||||
|     "this-might-take-a-moment": "Das könnte einen kleinen Moment dauern", |     "this-might-take-a-moment": "Das könnte einen kleinen Moment dauern", | ||||||
|     "this-scanstation-is": "Diese Station ist", |     "this-scanstation-is": "Diese Station ist", | ||||||
|  |     "token": "Token", | ||||||
|     "total-distance": "gelaufene Strecke", |     "total-distance": "gelaufene Strecke", | ||||||
|     "total-donation-amount": "Gesamtbetrag", |     "total-donation-amount": "Gesamtbetrag", | ||||||
|     "total-donations": "Spendensumme", |     "total-donations": "Spendensumme", | ||||||
|  |     "total-paid-amount": "Gezahlter Gesamtbetrag", | ||||||
|     "total-scans": "gesamte Scans", |     "total-scans": "gesamte Scans", | ||||||
|  |     "total_donation_amount_in_eur": "Gesamtbetrag in €", | ||||||
|     "track": "Track", |     "track": "Track", | ||||||
|     "track-added": "Track hinzugefügt", |     "track-added": "Track hinzugefügt", | ||||||
|     "track-data-is-being-loaded": "Trackdaten werden geladen", |     "track-data-is-being-loaded": "Trackdaten werden geladen", | ||||||
|     "track-is-being-added": "Track wird hinzugefügt...", |     "track-is-being-added": "Track wird hinzugefügt...", | ||||||
|  |     "track-is-being-updated": "Track wird aktualisiert...", | ||||||
|     "track-length-in-m": "Tracklänge (in Metern)", |     "track-length-in-m": "Tracklänge (in Metern)", | ||||||
|     "track-length-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein", |     "track-length-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein", | ||||||
|     "track-name": "Trackname", |     "track-name": "Trackname", | ||||||
|     "track-name-must-not-be-empty": "Der Name muss angegeben werden", |     "track-name-must-not-be-empty": "Der Name muss angegeben werden", | ||||||
|  |     "track-was-updated": "Track wurde aktualisiert", | ||||||
|     "tracks": "Tracks", |     "tracks": "Tracks", | ||||||
|  |     "unpaid": "Offen", | ||||||
|  |     "update-card": "Karte aktualisieren", | ||||||
|     "update-password": "Passwort ändern", |     "update-password": "Passwort ändern", | ||||||
|     "updated-contact": "Kontakt aktualisiert!", |     "updated-contact": "Kontakt aktualisiert!", | ||||||
|     "updated-donor": "Sponsor:in wurde aktualisiert", |     "updated-donor": "Sponsor:in wurde aktualisiert", | ||||||
|     "updated-organization": "Organisation wurde aktualisiert", |     "updated-organization": "Organisation wurde aktualisiert", | ||||||
|     "updated-scan": "Scan wurde aktualisiert", |     "updated-scan": "Scan wurde aktualisiert", | ||||||
|  |     "updated-team": "Team wurde aktualisiert", | ||||||
|     "updateing-group": "Gruppe wird aktualisiert...", |     "updateing-group": "Gruppe wird aktualisiert...", | ||||||
|  |     "updating-card": "Karte wird aktualisiert", | ||||||
|  |     "updating-donation": "Sponsoring wird aktualisiert", | ||||||
|     "updating-organization": "Organisation wird aktualisiert", |     "updating-organization": "Organisation wird aktualisiert", | ||||||
|     "updating-permissions": "Berechtigungen werden aktualisiert...", |     "updating-permissions": "Berechtigungen werden aktualisiert...", | ||||||
|     "updating-runner": "Läufer:in wird aktualisiert.", |     "updating-runner": "Läufer:in wird aktualisiert.", | ||||||
|  |     "updating-team": "Team wird aktualisiert", | ||||||
|     "updating-user": "Benutzer:in wird aktualisiert...", |     "updating-user": "Benutzer:in wird aktualisiert...", | ||||||
|     "updating-your-profile": "Profil wird aktualisiert...", |     "updating-your-profile": "Profil wird aktualisiert...", | ||||||
|     "user-added": "Benutzer hinzugefügt", |     "user-added": "Benutzer hinzugefügt", | ||||||
| @@ -373,8 +465,13 @@ | |||||||
|     "welcome_wavinghand": "Willkommen 👋", |     "welcome_wavinghand": "Willkommen 👋", | ||||||
|     "yes-i-copied-the-token": "Ja, ich habe den Token kopiert", |     "yes-i-copied-the-token": "Ja, ich habe den Token kopiert", | ||||||
|     "you-are-going-to-loose-all-permissions-and-access-to-the-runner-system": "Du wirst all deine Berechtigungen und den Zugriff aufs Läufersystem verlieren!", |     "you-are-going-to-loose-all-permissions-and-access-to-the-runner-system": "Du wirst all deine Berechtigungen und den Zugriff aufs Läufersystem verlieren!", | ||||||
|  |     "you-can-enter-the-donations-paid-amount-manually-or-use-the-max-button-to-use-the-donations-exact-amount": "Du kannst den Betrag der Zahlung entweder manuell eingeben oder über den MAX Button auf den Spendenbetrag setzen", | ||||||
|     "you-can-now-use-your-new-password-to-log-in-to-your-account": "Du kannst dich jetzt mit deinem neuen Passwort anmelden! 🎉", |     "you-can-now-use-your-new-password-to-log-in-to-your-account": "Du kannst dich jetzt mit deinem neuen Passwort anmelden! 🎉", | ||||||
|  |     "you-can-provide-a-runner-but-you-dont-have-to": "Du kannst eine Läufer:in angeben, musst aber nicht.", | ||||||
|  |     "you-dont-have-any-scanclients-yet": "Es gibt noch keine Statsclients", | ||||||
|     "you-dont-have-any-scanstations-yet": "Es gibt noch keine Scannerstationen", |     "you-dont-have-any-scanstations-yet": "Es gibt noch keine Scannerstationen", | ||||||
|     "you-have-to-provide-an-organization": "Du musst eine Organisation angeben", |     "you-have-to-provide-an-organization": "Du musst eine Organisation angeben", | ||||||
|  |     "you-have-to-save-your-changes-to-generate-a-link": "Du musst deine Änderungen speichern, um einen Link zu generieren.", | ||||||
|  |     "you-must-create-at-least-one-card-or-cancel": "Du musst mindestens eine Blankokarte erstellen (oder abbrechen).", | ||||||
|     "zip-postal-code": "Postleitzahl" |     "zip-postal-code": "Postleitzahl" | ||||||
| } | } | ||||||
| @@ -4,11 +4,15 @@ | |||||||
|     "about": "About", |     "about": "About", | ||||||
|     "action": "Action", |     "action": "Action", | ||||||
|     "active": "Active", |     "active": "Active", | ||||||
|  |     "add-card": "Add Card", | ||||||
|     "add-donation": "Add donation", |     "add-donation": "Add donation", | ||||||
|     "add-donor": "add donor", |     "add-donor": "Add donor", | ||||||
|  |     "add-or-update-a-payment": "Add or update a payment", | ||||||
|     "add-scan": "Add scan", |     "add-scan": "Add scan", | ||||||
|     "add-the-first-scanstation": "Add your first scanstation.", |     "add-the-first-scanstation": "Add your first scanstation.", | ||||||
|  |     "add-the-first-statsclient": "Add your first statsclient.", | ||||||
|     "add-user-group": "Add User Group", |     "add-user-group": "Add User Group", | ||||||
|  |     "add-your-first-card": "Add your first card", | ||||||
|     "add-your-first-contact": "Add your first contact", |     "add-your-first-contact": "Add your first contact", | ||||||
|     "add-your-first-donor": "add your first donor", |     "add-your-first-donor": "add your first donor", | ||||||
|     "add-your-first-group": "Add your first group", |     "add-your-first-group": "Add your first group", | ||||||
| @@ -19,14 +23,19 @@ | |||||||
|     "add-your-first-user": "Add your first user", |     "add-your-first-user": "Add your first user", | ||||||
|     "add-your-fist-donation": "Add your fist donation", |     "add-your-fist-donation": "Add your fist donation", | ||||||
|     "add-your-fist-scan": "Add your fist scan", |     "add-your-fist-scan": "Add your fist scan", | ||||||
|  |     "adding-card": "Adding Card", | ||||||
|  |     "adding-donation": "Adding donation...", | ||||||
|     "adding-scan": "Adding Scan", |     "adding-scan": "Adding Scan", | ||||||
|     "address": "Address", |     "address": "Address", | ||||||
|     "address-is-required": "Address is required", |     "address-is-required": "Address is required", | ||||||
|     "after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!", |     "after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!", | ||||||
|     "after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that": "After the update you'll get logged out - Please login with your new password after that.", |     "after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that": "After the update you'll get logged out - Please login with your new password after that.", | ||||||
|  |     "all": "all", | ||||||
|     "all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well", |     "all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well", | ||||||
|     "all-associated-runners-will-be-deleted-too": "All associated runners will be deleted too!", |     "all-associated-runners-will-be-deleted-too": "All associated runners will be deleted too!", | ||||||
|     "all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!", |     "all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!", | ||||||
|  |     "already-paid": "Already paid", | ||||||
|  |     "amount": "Amount", | ||||||
|     "amount-per-kilometer": "Amount per kilometer", |     "amount-per-kilometer": "Amount per kilometer", | ||||||
|     "apartment-suite-etc": "Apartment, suite, etc.", |     "apartment-suite-etc": "Apartment, suite, etc.", | ||||||
|     "application_name": "Lauf für Kaya! - Admin", |     "application_name": "Lauf für Kaya! - Admin", | ||||||
| @@ -40,32 +49,47 @@ | |||||||
|     "cancel-keep-donor": "Cancel, keep donor", |     "cancel-keep-donor": "Cancel, keep donor", | ||||||
|     "cancel-keep-my-profile": "Cancel, keep my profile", |     "cancel-keep-my-profile": "Cancel, keep my profile", | ||||||
|     "cancel-keep-organization": "Cancel, keep organization", |     "cancel-keep-organization": "Cancel, keep organization", | ||||||
|  |     "cancel-keep-statsclient": "Cancel and keep statsclient", | ||||||
|     "cancel-keep-team": "Cancel, keep team", |     "cancel-keep-team": "Cancel, keep team", | ||||||
|     "cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity", |     "cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity", | ||||||
|  |     "card-added": "Card added", | ||||||
|  |     "card-deleted": "Card deleted", | ||||||
|  |     "card-updated": "Card updated", | ||||||
|  |     "cards": "Cards", | ||||||
|  |     "certificates": "Certificates", | ||||||
|     "change-your-password-here": "Change your password here", |     "change-your-password-here": "Change your password here", | ||||||
|     "changing-your-password": "Changing your password", |     "changing-your-password": "Changing your password", | ||||||
|     "city": "City", |     "city": "City", | ||||||
|  |     "click-to-copy-the-link-into-your-clipboard": "Click to copy the link into your clipboard", | ||||||
|  |     "click-to-copy-token-to-clipboard": "Click to copy the token to your clipboard", | ||||||
|     "close": "Close", |     "close": "Close", | ||||||
|  |     "code": "Code", | ||||||
|     "configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times", |     "configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times", | ||||||
|     "confirm": "Confirm", |     "confirm": "Confirm", | ||||||
|     "confirm-delete": "Confirm Delete", |     "confirm-delete": "Confirm Delete", | ||||||
|     "confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations", |     "confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations", | ||||||
|     "confirm-delete-my-user-profile": "Confirm, delete my user profile", |     "confirm-delete-my-user-profile": "Confirm, delete my user profile", | ||||||
|     "confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.", |     "confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.", | ||||||
|  |     "confirm-delete-statsclient": "Confirm, delete statsclient", | ||||||
|     "confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.", |     "confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.", | ||||||
|     "confirm-deletion": "Confirm Deletion", |     "confirm-deletion": "Confirm Deletion", | ||||||
|     "confirm-the-new-password": "Confirm the new password", |     "confirm-the-new-password": "Confirm the new password", | ||||||
|     "contact": "Contact", |     "contact": "Contact", | ||||||
|  |     "contact-added": "Contact added", | ||||||
|     "contact-deleted": "Contact deleted", |     "contact-deleted": "Contact deleted", | ||||||
|     "contact-information": "Contact Information", |     "contact-information": "Contact Information", | ||||||
|  |     "contact-is-being-added": "Contact is being added...", | ||||||
|     "contact-is-being-updated": "Contact is being updated...", |     "contact-is-being-updated": "Contact is being updated...", | ||||||
|     "contact-is-not-a-member-in-any-group": "Contact is not a member in any group", |     "contact-is-not-a-member-in-any-group": "Contact is not a member in any group", | ||||||
|     "contacts": "Contacts", |     "contacts": "Contacts", | ||||||
|     "contacts-are-being-loaded": "contacts are being loaded...", |     "contacts-are-being-loaded": "contacts are being loaded...", | ||||||
|  |     "copied-link-to-clipboard": "Copied link to clipboard", | ||||||
|  |     "copied-token-to-clipboard": "Copied token to clipboard", | ||||||
|     "count_organizations": "# Organizations", |     "count_organizations": "# Organizations", | ||||||
|     "count_teams": "# Teams", |     "count_teams": "# Teams", | ||||||
|     "create": "Create", |     "create": "Create", | ||||||
|     "create-a-new": "Create a new", |     "create-a-new": "Create a new", | ||||||
|  |     "create-a-new-card": "Create a new card", | ||||||
|     "create-a-new-contact": "Create a new contact", |     "create-a-new-contact": "Create a new contact", | ||||||
|     "create-a-new-distance-donation": "Create a new distance donation", |     "create-a-new-distance-donation": "Create a new distance donation", | ||||||
|     "create-a-new-donor": "Create a new donor", |     "create-a-new-donor": "Create a new donor", | ||||||
| @@ -74,14 +98,21 @@ | |||||||
|     "create-a-new-runner": "Create a new Runner", |     "create-a-new-runner": "Create a new Runner", | ||||||
|     "create-a-new-scan-fixed-only": "Create a new scan (fixed only)", |     "create-a-new-scan-fixed-only": "Create a new scan (fixed only)", | ||||||
|     "create-a-new-scanstation": "Create a new station", |     "create-a-new-scanstation": "Create a new station", | ||||||
|  |     "create-a-new-statsclient": "Create a new statsclient", | ||||||
|     "create-a-new-team": "Create a new team", |     "create-a-new-team": "Create a new team", | ||||||
|     "create-a-new-track": "Create a new Track", |     "create-a-new-track": "Create a new Track", | ||||||
|     "create-a-new-user": "Create a new User", |     "create-a-new-user": "Create a new User", | ||||||
|     "create-a-new-user-group": "Create a new user group", |     "create-a-new-user-group": "Create a new user group", | ||||||
|  |     "create-and-generate-pdf": "Create and generate PDF", | ||||||
|  |     "create-bulk-blanco-cards": "Create bulk blanco cards", | ||||||
|  |     "create-bulk-cards": "Add blanco cards", | ||||||
|     "create-organization": "Create Organization", |     "create-organization": "Create Organization", | ||||||
|     "create-team": "Create Team", |     "create-team": "Create Team", | ||||||
|     "create-track": "Create Track", |     "create-track": "Create Track", | ||||||
|     "create-user": "Create User", |     "create-user": "Create User", | ||||||
|  |     "create-without-pdf": "Create without PDF", | ||||||
|  |     "created-blanco-cards": "Created blanco cards", | ||||||
|  |     "creating-blanco-cards": "Creating blanco cards", | ||||||
|     "credits": "Credits", |     "credits": "Credits", | ||||||
|     "csv_import__class": "Class", |     "csv_import__class": "Class", | ||||||
|     "csv_import__firstname": "Firstname", |     "csv_import__firstname": "Firstname", | ||||||
| @@ -116,6 +147,7 @@ | |||||||
|     "delete-runner": "Delete Runner", |     "delete-runner": "Delete Runner", | ||||||
|     "delete-scan": "Delete scan", |     "delete-scan": "Delete scan", | ||||||
|     "delete-station": "Delete station", |     "delete-station": "Delete station", | ||||||
|  |     "delete-statsclient": "Delete statsclient", | ||||||
|     "delete-team": "Delete Team", |     "delete-team": "Delete Team", | ||||||
|     "delete-user": "Delete User", |     "delete-user": "Delete User", | ||||||
|     "deleted-scan": "Deleted scan", |     "deleted-scan": "Deleted scan", | ||||||
| @@ -124,6 +156,7 @@ | |||||||
|     "description-optional": "Description (optional)", |     "description-optional": "Description (optional)", | ||||||
|     "deselect-all": "deselect all", |     "deselect-all": "deselect all", | ||||||
|     "details": "Details", |     "details": "Details", | ||||||
|  |     "disabled": "disabled", | ||||||
|     "distance": "Distance", |     "distance": "Distance", | ||||||
|     "distance-donation": "distance donation", |     "distance-donation": "distance donation", | ||||||
|     "distance-in-km": "Distance in km", |     "distance-in-km": "Distance in km", | ||||||
| @@ -135,6 +168,9 @@ | |||||||
|     "documentation": "Documentation", |     "documentation": "Documentation", | ||||||
|     "donation-amount": "Donation amount", |     "donation-amount": "Donation amount", | ||||||
|     "donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€", |     "donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€", | ||||||
|  |     "donation-deleted": "Donation deleted", | ||||||
|  |     "donation-updated": "Donation updated", | ||||||
|  |     "donation_added": "Donation_added", | ||||||
|     "donations": "Donations", |     "donations": "Donations", | ||||||
|     "donor": "Donor", |     "donor": "Donor", | ||||||
|     "donor-added": "Donor added", |     "donor-added": "Donor added", | ||||||
| @@ -148,15 +184,21 @@ | |||||||
|     "dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌", |     "dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌", | ||||||
|     "e-mail-adress": "E-Mail Adress", |     "e-mail-adress": "E-Mail Adress", | ||||||
|     "edit": "Edit", |     "edit": "Edit", | ||||||
|  |     "edit-a-card": "Edit a card", | ||||||
|     "edit-permissions": "edit permissions", |     "edit-permissions": "edit permissions", | ||||||
|     "email_address_or_username": "Email / username", |     "email_address_or_username": "Email / username", | ||||||
|     "enabled": "enabled", |     "enabled": "enabled", | ||||||
|  |     "enabled_large": "Enabled", | ||||||
|     "english": "English", |     "english": "English", | ||||||
|  |     "enter-payment": "Enter payment", | ||||||
|  |     "error-during-import": "Error during import", | ||||||
|  |     "error-whyile-copying-to-clipboard": "Error while copying to clipboard", | ||||||
|     "error_on_login": "Error on login", |     "error_on_login": "Error on login", | ||||||
|     "erteilte": "Directly granted", |     "erteilte": "Directly granted", | ||||||
|     "everything-concerning-your-profile": "Everything concerning your profile", |     "everything-concerning-your-profile": "Everything concerning your profile", | ||||||
|     "everything-is-more-fun-together": "everything is more fun together 🏃♂️🏃♀️🏃♂️", |     "everything-is-more-fun-together": "everything is more fun together 🏃♂️🏃♀️🏃♂️", | ||||||
|     "faq": "FAQ", |     "faq": "FAQ", | ||||||
|  |     "filename_sponsoringquittungsliste": "DonorReceiptList", | ||||||
|     "filter-by-organization-team": "Filter by Organization/ Team", |     "filter-by-organization-team": "Filter by Organization/ Team", | ||||||
|     "first-name": "First name", |     "first-name": "First name", | ||||||
|     "first-name-is-required": "First Name is required", |     "first-name-is-required": "First Name is required", | ||||||
| @@ -166,6 +208,9 @@ | |||||||
|     "geerbte": "inherited", |     "geerbte": "inherited", | ||||||
|     "general-stats": "General Stats", |     "general-stats": "General Stats", | ||||||
|     "general_promise_error": "😢 Error", |     "general_promise_error": "😢 Error", | ||||||
|  |     "generate-runner-certificate": "Generate runner certificate", | ||||||
|  |     "generate-runner-certificates": "Generate runner certificates", | ||||||
|  |     "generate-runnercards": "Generate Runnercards", | ||||||
|     "generate-sponsoring-contract": "generate sponsoring contract", |     "generate-sponsoring-contract": "generate sponsoring contract", | ||||||
|     "generate-sponsoring-contracts": "generate sponsoring contracts", |     "generate-sponsoring-contracts": "generate sponsoring contracts", | ||||||
|     "generating-pdf": "generating PDF...", |     "generating-pdf": "generating PDF...", | ||||||
| @@ -184,6 +229,7 @@ | |||||||
|     "groups-are-being-loaded": "Groups are being loaded", |     "groups-are-being-loaded": "Groups are being loaded", | ||||||
|     "home": "Home", |     "home": "Home", | ||||||
|     "icon-image-credits": "We also want to thank these projects for illustrations and icons:", |     "icon-image-credits": "We also want to thank these projects for illustrations and icons:", | ||||||
|  |     "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button": "If you want to create multiple blanco cards: Try the 'Add blanco cards' button.", | ||||||
|     "import-finished": "Import finished", |     "import-finished": "Import finished", | ||||||
|     "import-runners": "Import runners", |     "import-runners": "Import runners", | ||||||
|     "import__target-organization": "Target Organization", |     "import__target-organization": "Target Organization", | ||||||
| @@ -194,6 +240,8 @@ | |||||||
|     "internal-error": "Internal Error", |     "internal-error": "Internal Error", | ||||||
|     "invalid": "Invalid", |     "invalid": "Invalid", | ||||||
|     "invalid-mail-reset": "the provided email is invalid", |     "invalid-mail-reset": "the provided email is invalid", | ||||||
|  |     "just-enter-how-many-you-want-and-the-system-will-create-them": "Just enter how many you want and the system will create them", | ||||||
|  |     "key": "Key", | ||||||
|     "laeufer-hinzufuegen": "Add runner", |     "laeufer-hinzufuegen": "Add runner", | ||||||
|     "laeufer-importieren": "Läufer importieren", |     "laeufer-importieren": "Läufer importieren", | ||||||
|     "laptime": "Laptime", |     "laptime": "Laptime", | ||||||
| @@ -202,6 +250,7 @@ | |||||||
|     "lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.", |     "lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.", | ||||||
|     "license": "License", |     "license": "License", | ||||||
|     "licenses-are-being-loaded": "Licenses are being loaded...", |     "licenses-are-being-loaded": "Licenses are being loaded...", | ||||||
|  |     "loading-cards": "Loading cards", | ||||||
|     "loading-contact-details": "Loading contact details...", |     "loading-contact-details": "Loading contact details...", | ||||||
|     "loading-donation-details": "Loading donation details", |     "loading-donation-details": "Loading donation details", | ||||||
|     "loading-donor-details": "Loading donor details", |     "loading-donor-details": "Loading donor details", | ||||||
| @@ -218,6 +267,10 @@ | |||||||
|     "middle-name": "Middle name", |     "middle-name": "Middle name", | ||||||
|     "minimum-lap-time-in-s": "minimum lap time in s", |     "minimum-lap-time-in-s": "minimum lap time in s", | ||||||
|     "minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0", |     "minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0", | ||||||
|  |     "must-be-at-least-10-characters-long": "Must be at least 10 characters long!", | ||||||
|  |     "must-contain-a-lowercase-letter": "Must contain a lowercase letter!", | ||||||
|  |     "must-contain-a-number": "Must contain a number!", | ||||||
|  |     "must-contain-a-uppercase-letter": "Must contain a uppercase letter!", | ||||||
|     "name": "Name", |     "name": "Name", | ||||||
|     "name-is-required": "Name is required", |     "name-is-required": "Name is required", | ||||||
|     "new-password": "New password", |     "new-password": "New password", | ||||||
| @@ -231,6 +284,8 @@ | |||||||
|     "no-organizations-found": "No organizations found", |     "no-organizations-found": "No organizations found", | ||||||
|     "no-runners-found": "No runners found", |     "no-runners-found": "No runners found", | ||||||
|     "no-tracks-added-yet": "there are no tracks added yet.", |     "no-tracks-added-yet": "there are no tracks added yet.", | ||||||
|  |     "non-blanko": "Non/Blanko", | ||||||
|  |     "open": "OPEN", | ||||||
|     "organization": "Organization", |     "organization": "Organization", | ||||||
|     "organization-added": "Organization added", |     "organization-added": "Organization added", | ||||||
|     "organization-deleted": "Organization deleted", |     "organization-deleted": "Organization deleted", | ||||||
| @@ -241,6 +296,8 @@ | |||||||
|     "organizations-are-being-loaded": "organizations are being loaded...", |     "organizations-are-being-loaded": "organizations are being loaded...", | ||||||
|     "orgs": "Organizations", |     "orgs": "Organizations", | ||||||
|     "oss_credit_description": "We use a lot of open source software on these projects, and would like to thank the following projects and contributors who help make open source great!", |     "oss_credit_description": "We use a lot of open source software on these projects, and would like to thank the following projects and contributors who help make open source great!", | ||||||
|  |     "paid": "PAID", | ||||||
|  |     "paid-amount": "Paid amount", | ||||||
|     "password": "Password", |     "password": "Password", | ||||||
|     "password-changed": "Password changed!", |     "password-changed": "Password changed!", | ||||||
|     "password-is-required": "Password is required", |     "password-is-required": "Password is required", | ||||||
| @@ -248,7 +305,8 @@ | |||||||
|     "password-reset-in-progress": "Password Reset in Progress...", |     "password-reset-in-progress": "Password Reset in Progress...", | ||||||
|     "password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".", |     "password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".", | ||||||
|     "password-reset-successful": "Password Reset successful!", |     "password-reset-successful": "Password Reset successful!", | ||||||
|     "passwords-dont-match": "Passwords don't match", |     "passwords-dont-match": "Passwords don't match!", | ||||||
|  |     "payment-amount-must-be-greater-than-0-00eur": "Payment amount must be greater than 0.00€!", | ||||||
|     "pdf-generation-failed": "PDF generation failed!", |     "pdf-generation-failed": "PDF generation failed!", | ||||||
|     "pdf-successfully-generated": "PDF successfully generated!", |     "pdf-successfully-generated": "PDF successfully generated!", | ||||||
|     "pdfs-successfully-generated": "PDFs successfully generated!", |     "pdfs-successfully-generated": "PDFs successfully generated!", | ||||||
| @@ -256,6 +314,7 @@ | |||||||
|     "permissions": "Permissions", |     "permissions": "Permissions", | ||||||
|     "permissions-updated": "Permissions updated!", |     "permissions-updated": "Permissions updated!", | ||||||
|     "phone": "Phone", |     "phone": "Phone", | ||||||
|  |     "please-copy-the-token-and-store-it-somewhere-save": "Please copy the token and store it somewhere safe.", | ||||||
|     "please-provide-a-password": "Please provide a password...", |     "please-provide-a-password": "Please provide a password...", | ||||||
|     "please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor", |     "please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor", | ||||||
|     "please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation", |     "please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation", | ||||||
| @@ -268,10 +327,15 @@ | |||||||
|     "please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.", |     "please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.", | ||||||
|     "please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.", |     "please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.", | ||||||
|     "please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.", |     "please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.", | ||||||
|  |     "please-provide-the-required-information-to-create-a-new-scanstation": "Please provide the required information to create a new scanstation", | ||||||
|  |     "please-provide-the-required-information-to-create-a-new-statsclient": "Please provide the required information to create a new statsclient", | ||||||
|     "please-request-a-new-reset-mail": "Please request a new reset mail...", |     "please-request-a-new-reset-mail": "Please request a new reset mail...", | ||||||
|  |     "please-wait-a-moment-your-login-is-still-being-processed": "Please wait a moment, your login is still being processed", | ||||||
|  |     "prefix": "Prefix", | ||||||
|     "privacy": "Privacy", |     "privacy": "Privacy", | ||||||
|     "privacy-loading": "Privacy loading...", |     "privacy-loading": "Privacy loading...", | ||||||
|     "profile": "Profile", |     "profile": "Profile", | ||||||
|  |     "profile-deleted": "Profile deleted!", | ||||||
|     "profile-picture": "Profile Picture", |     "profile-picture": "Profile Picture", | ||||||
|     "profile-updated": "Profile updated!", |     "profile-updated": "Profile updated!", | ||||||
|     "read-license": "Read License", |     "read-license": "Read License", | ||||||
| @@ -285,6 +349,7 @@ | |||||||
|     "runner-import": "Runner Import", |     "runner-import": "Runner Import", | ||||||
|     "runner-is-being-added": "Runner is being added...", |     "runner-is-being-added": "Runner is being added...", | ||||||
|     "runner-updated": "Runner updated!", |     "runner-updated": "Runner updated!", | ||||||
|  |     "runnercards": "Runnercards", | ||||||
|     "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"", |     "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"", | ||||||
|     "runners": "Runners", |     "runners": "Runners", | ||||||
|     "runners-are-being-imported": "Runners are being imported...", |     "runners-are-being-imported": "Runners are being imported...", | ||||||
| @@ -297,6 +362,8 @@ | |||||||
|     "scans": "Scans", |     "scans": "Scans", | ||||||
|     "scans-are-being-loaded": "Scans are being loaded", |     "scans-are-being-loaded": "Scans are being loaded", | ||||||
|     "scanstation": "Scanstation", |     "scanstation": "Scanstation", | ||||||
|  |     "scanstation-added": "Scanstation added", | ||||||
|  |     "scanstation-is-being-added": "Adding scanstation...", | ||||||
|     "scanstations": "Scanstations", |     "scanstations": "Scanstations", | ||||||
|     "scanstations-are-being-loaded": "Loading scanstations...", |     "scanstations-are-being-loaded": "Loading scanstations...", | ||||||
|     "search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)", |     "search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)", | ||||||
| @@ -306,23 +373,36 @@ | |||||||
|     "search-for-runner-by-name-or-id": "Search for runner (by name or id)", |     "search-for-runner-by-name-or-id": "Search for runner (by name or id)", | ||||||
|     "select-all": "select all", |     "select-all": "select all", | ||||||
|     "select-language": "Select language", |     "select-language": "Select language", | ||||||
|  |     "selfservice-registration": "Selfservice registration", | ||||||
|     "send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services", |     "send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services", | ||||||
|     "set-the-user-active-inactive": "set the user active/ inactive", |     "set-the-user-active-inactive": "set the user active/ inactive", | ||||||
|     "settings": "Settings", |     "settings": "Settings", | ||||||
|     "settings-for-your-profile": "Settings for your profile", |     "settings-for-your-profile": "Settings for your profile", | ||||||
|     "something-about-the-group": "Something about the group...", |     "something-about-the-group": "Something about the group...", | ||||||
|  |     "sponsoring-quittungs-liste_herunterladen": "Download donor receipt list", | ||||||
|  |     "sponsorings": "Sponsorings", | ||||||
|     "stats-are-being-loaded": "stats are being loaded...", |     "stats-are-being-loaded": "stats are being loaded...", | ||||||
|  |     "statsclient-deleted": "Deleted statsclient", | ||||||
|  |     "statsclient-is-being-added": "Statsclient is being added...", | ||||||
|  |     "statsclients": "Statsclients (aka Beamershow)", | ||||||
|  |     "statsclients-are-being-loaded": "Loading statsclients", | ||||||
|     "status": "Status", |     "status": "Status", | ||||||
|     "stuff-that-could-harm-your-profile": "Stuff that could harm your profile", |     "stuff-that-could-harm-your-profile": "Stuff that could harm your profile", | ||||||
|     "successful-password-reset": "Successful password reset!", |     "successful-password-reset": "Successful password reset!", | ||||||
|     "team": "Team", |     "team": "Team", | ||||||
|  |     "team-added": "Team added", | ||||||
|  |     "team-deleted": "Team deleted", | ||||||
|     "team-detail-is-being-loaded": "team detail is being loaded...", |     "team-detail-is-being-loaded": "team detail is being loaded...", | ||||||
|  |     "team-is-being-added": "Team is being added...", | ||||||
|     "team-name": "Team name", |     "team-name": "Team name", | ||||||
|     "team-name-is-required": "team name is required", |     "team-name-is-required": "team name is required", | ||||||
|     "teams": "Teams", |     "teams": "Teams", | ||||||
|     "teams-are-being-loaded": "teams are being loaded...", |     "teams-are-being-loaded": "teams are being loaded...", | ||||||
|     "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number": "the provided phone number is invalid.<br />please enter a valid international number...", |     "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number": "the provided phone number is invalid.<br />please enter a valid international number...", | ||||||
|     "the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m", |     "the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m", | ||||||
|  |     "the-scanstations-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again": "The scanstation api token will only get displayed once - you won't be able to change or view it again!", | ||||||
|  |     "the-statsclient-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again": "The statsclient api token will only get displayed once - you won't be able to change or view it again!", | ||||||
|  |     "there-are-no-cards-yet": "There are no cards yet.", | ||||||
|     "there-are-no-contacts-added-yet": "There are no contacts added yet.", |     "there-are-no-contacts-added-yet": "There are no contacts added yet.", | ||||||
|     "there-are-no-donations-yet": "There are no donations yet", |     "there-are-no-donations-yet": "There are no donations yet", | ||||||
|     "there-are-no-donors-yet": "There are no donors yet", |     "there-are-no-donors-yet": "There are no donors yet", | ||||||
| @@ -332,30 +412,42 @@ | |||||||
|     "there-are-no-scans-yet": "There are no scans yet", |     "there-are-no-scans-yet": "There are no scans yet", | ||||||
|     "there-are-no-teams-added-yet": "There are no teams added yet.", |     "there-are-no-teams-added-yet": "There are no teams added yet.", | ||||||
|     "there-are-no-users-added-yet": "There are no users added yet.", |     "there-are-no-users-added-yet": "There are no users added yet.", | ||||||
|  |     "this-card-is": "This card is", | ||||||
|     "this-might-take-a-moment": "This might take a moment 👀", |     "this-might-take-a-moment": "This might take a moment 👀", | ||||||
|     "this-scanstation-is": "This scanstation is", |     "this-scanstation-is": "This scanstation is", | ||||||
|  |     "token": "Token", | ||||||
|     "total-distance": "total distance", |     "total-distance": "total distance", | ||||||
|     "total-donation-amount": "total donation amount", |     "total-donation-amount": "total donation amount", | ||||||
|     "total-donations": "total donations", |     "total-donations": "total donations", | ||||||
|  |     "total-paid-amount": "Total paid amount", | ||||||
|     "total-scans": "total scans", |     "total-scans": "total scans", | ||||||
|  |     "total_donation_amount_in_eur": "Total donation amount in €", | ||||||
|     "track": "Track", |     "track": "Track", | ||||||
|     "track-added": "Track added", |     "track-added": "Track added", | ||||||
|     "track-data-is-being-loaded": "Track data is being loaded", |     "track-data-is-being-loaded": "Track data is being loaded", | ||||||
|     "track-is-being-added": "Track is being added...", |     "track-is-being-added": "Track is being added...", | ||||||
|  |     "track-is-being-updated": "Track is being updated...", | ||||||
|     "track-length-in-m": "Track Length in m", |     "track-length-in-m": "Track Length in m", | ||||||
|     "track-length-must-be-greater-than-0": "Track length must be greater than 0", |     "track-length-must-be-greater-than-0": "Track length must be greater than 0", | ||||||
|     "track-name": "Track name", |     "track-name": "Track name", | ||||||
|     "track-name-must-not-be-empty": "Track name must not be empty", |     "track-name-must-not-be-empty": "Track name must not be empty", | ||||||
|  |     "track-was-updated": "Track was updated!", | ||||||
|     "tracks": "Tracks", |     "tracks": "Tracks", | ||||||
|  |     "unpaid": "Unpaid", | ||||||
|  |     "update-card": "Update Card", | ||||||
|     "update-password": "Update password", |     "update-password": "Update password", | ||||||
|     "updated-contact": "Updated contact!", |     "updated-contact": "Updated contact!", | ||||||
|     "updated-donor": "updated donor", |     "updated-donor": "updated donor", | ||||||
|     "updated-organization": "updated organization", |     "updated-organization": "updated organization", | ||||||
|     "updated-scan": "updated scan", |     "updated-scan": "updated scan", | ||||||
|  |     "updated-team": "Updated team", | ||||||
|     "updateing-group": "updateing group...", |     "updateing-group": "updateing group...", | ||||||
|  |     "updating-card": "Updating card", | ||||||
|  |     "updating-donation": "Updating donation", | ||||||
|     "updating-organization": "updating organization", |     "updating-organization": "updating organization", | ||||||
|     "updating-permissions": "updating permissions...", |     "updating-permissions": "updating permissions...", | ||||||
|     "updating-runner": "Updating runner...", |     "updating-runner": "Updating runner...", | ||||||
|  |     "updating-team": "Updating team", | ||||||
|     "updating-user": "updating user...", |     "updating-user": "updating user...", | ||||||
|     "updating-your-profile": "Updating your profile...", |     "updating-your-profile": "Updating your profile...", | ||||||
|     "user-added": "User added", |     "user-added": "User added", | ||||||
| @@ -373,8 +465,13 @@ | |||||||
|     "welcome_wavinghand": "Welcome 👋", |     "welcome_wavinghand": "Welcome 👋", | ||||||
|     "yes-i-copied-the-token": "Yes, I copied the token", |     "yes-i-copied-the-token": "Yes, I copied the token", | ||||||
|     "you-are-going-to-loose-all-permissions-and-access-to-the-runner-system": "You are going to loose all permissions and access to the runner system!", |     "you-are-going-to-loose-all-permissions-and-access-to-the-runner-system": "You are going to loose all permissions and access to the runner system!", | ||||||
|  |     "you-can-enter-the-donations-paid-amount-manually-or-use-the-max-button-to-use-the-donations-exact-amount": "You can enter the donation's paid amount manually or use the MAX button to use the donation's exact amount.", | ||||||
|     "you-can-now-use-your-new-password-to-log-in-to-your-account": "You can now use your new password to log in to your account! 🎉", |     "you-can-now-use-your-new-password-to-log-in-to-your-account": "You can now use your new password to log in to your account! 🎉", | ||||||
|  |     "you-can-provide-a-runner-but-you-dont-have-to": "You can provide a runner, but you don't have to.", | ||||||
|  |     "you-dont-have-any-scanclients-yet": "You don't have any statsclients yet", | ||||||
|     "you-dont-have-any-scanstations-yet": "You don't have any scanstations yet", |     "you-dont-have-any-scanstations-yet": "You don't have any scanstations yet", | ||||||
|     "you-have-to-provide-an-organization": "You have to provide an organization", |     "you-have-to-provide-an-organization": "You have to provide an organization", | ||||||
|  |     "you-have-to-save-your-changes-to-generate-a-link": "You have to save your changes to generate a link.", | ||||||
|  |     "you-must-create-at-least-one-card-or-cancel": "You must create at least one card (or cancel).", | ||||||
|     "zip-postal-code": "ZIP/ postal code" |     "zip-postal-code": "ZIP/ postal code" | ||||||
| } | } | ||||||
							
								
								
									
										9
									
								
								src/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/main.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | import './style.css'; | ||||||
|  | import "toastify-js/src/toastify.css"; | ||||||
|  | import "gridjs/dist/theme/mermaid.css"; | ||||||
|  | import App from './App.svelte'; | ||||||
|  |  | ||||||
|  | const app = new App({ | ||||||
|  | 	target: document.body | ||||||
|  | }); | ||||||
|  | export default app; | ||||||
							
								
								
									
										3
									
								
								src/style.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/style.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | @tailwind base; | ||||||
|  | @tailwind components; | ||||||
|  | @tailwind utilities; | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| export const register = () => { |  | ||||||
| 	if ('serviceWorker' in navigator) { |  | ||||||
| 		window.addEventListener('load', () => { |  | ||||||
| 			navigator.serviceWorker.register('/sw.js').then( |  | ||||||
| 				(registration) => { |  | ||||||
| 					// console.log(`sw successful with scope: ${registration.scope}`); |  | ||||||
| 				}, |  | ||||||
| 				(err) => { |  | ||||||
| 					// console.log(`sw failed: ${err}`); |  | ||||||
| 				} |  | ||||||
| 			); |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
| @@ -1,14 +1,15 @@ | |||||||
| module.exports = { | module.exports = { | ||||||
| 	purge: { | 	mode: 'jit', | ||||||
| 		content: [ './src/**/*.svelte' ] | 	purge: [ './src/**/*.svelte' ], | ||||||
| 	}, |  | ||||||
| 	// darkMode: 'media', |  | ||||||
| 	variants: {}, |  | ||||||
| 	plugins: [], |  | ||||||
| 	theme: { | 	theme: { | ||||||
| 		container: { | 		extend: { | ||||||
| 			center: true, | 			colors: { | ||||||
| 			padding: '1.5rem' | 				reepolee: { | ||||||
|  | 					500: '#b40000', | ||||||
|  | 					600: '#9c0000', | ||||||
|  | 					700: '#750000' | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -1,19 +0,0 @@ | |||||||
| const fs = require('fs'); |  | ||||||
| let content_svelteconfig = fs.readFileSync('./s-config.template.js', { encoding: 'utf8' }); |  | ||||||
| let content_html = fs.readFileSync('./index.template.html', { encoding: 'utf8' }); |  | ||||||
| if (process.env.NODE_ENV_ODIT == 'development_fast') { |  | ||||||
| 	content_html = content_html.replace( |  | ||||||
| 		'__TAILWIND_INSERT__', |  | ||||||
| 		'<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/tailwindcss@2.0.2/dist/tailwind.min.css">' |  | ||||||
| 	); |  | ||||||
| 	content_svelteconfig = content_svelteconfig.replace('__insert__', '{postcss:{}}'); |  | ||||||
| } else { |  | ||||||
| 	content_html = content_html.replace('__TAILWIND_INSERT__', ''); |  | ||||||
| 	content_svelteconfig = content_svelteconfig.replace( |  | ||||||
| 		'__insert__', |  | ||||||
| 		"{postcss:{plugins:[require('tailwindcss'),require('autoprefixer')]}}" |  | ||||||
| 	); |  | ||||||
| } |  | ||||||
| fs.writeFileSync('./public/index.html', content_html); |  | ||||||
| fs.writeFileSync('./svelte.config.js', content_svelteconfig); |  | ||||||
| console.info('dev setup script done'); |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| const fs = require('fs'); | const fs = require('fs'); | ||||||
| const package = JSON.parse(fs.readFileSync(`./package.json`, { encoding: 'utf-8' })); | const package = JSON.parse(fs.readFileSync(`./package.json`, { encoding: 'utf-8' })); | ||||||
| const original = fs.readFileSync(`./index.template.html`, { encoding: 'utf-8' }); | const original = fs.readFileSync(`./index.html`, { encoding: 'utf-8' }); | ||||||
| let out = original.replace(/RELEASE_INFO-(\S)+-RELEASE_INFO/gi, 'RELEASE_INFO-' + package.version + '-RELEASE_INFO'); | let out = original.replace(/RELEASE_INFO-(\S)+-RELEASE_INFO/gi, 'RELEASE_INFO-' + package.version + '-RELEASE_INFO'); | ||||||
| fs.writeFileSync(`./index.template.html`, out); | fs.writeFileSync(`./index.html`, out); | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								vite.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								vite.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | import svelte from '@sveltejs/vite-plugin-svelte'; | ||||||
|  | import { minify } from 'html-minifier'; | ||||||
|  | import { defineConfig } from 'vite'; | ||||||
|  | // | ||||||
|  | const indexReplace = () => { | ||||||
|  | 	return { | ||||||
|  | 		name: 'html-transform', | ||||||
|  | 		transformIndexHtml(html) { | ||||||
|  | 			return minify(html, { | ||||||
|  | 				collapseWhitespace: true | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  | 	}; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default defineConfig(({ command, mode }) => { | ||||||
|  | 	const isProduction = mode === 'production'; | ||||||
|  | 	return { | ||||||
|  | 		// base: './', | ||||||
|  | 		build: { | ||||||
|  | 			polyfillDynamicImport: false, | ||||||
|  | 			cssCodeSplit: false, | ||||||
|  | 			minify: isProduction, | ||||||
|  | 			target: ["es2020", "esnext", "edge88", "chrome87", "safari14"] | ||||||
|  | 		}, | ||||||
|  | 		plugins: [ | ||||||
|  | 			svelte({ | ||||||
|  | 				//@ts-ignore | ||||||
|  | 				hot: !isProduction, | ||||||
|  | 				emitCss: true, | ||||||
|  | 				extensions: ['.md', '.svx', '.svelte'], | ||||||
|  | 				preprocess: [ | ||||||
|  | 					// | ||||||
|  | 				] | ||||||
|  | 			}), | ||||||
|  | 			indexReplace() | ||||||
|  | 		] | ||||||
|  | 	}; | ||||||
|  | }); | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| module.exports = { |  | ||||||
| 	globDirectory: 'public', |  | ||||||
| 	globPatterns: [ '**/*.{js,ico,png,svg,html,webmanifest,txt,json}' ], |  | ||||||
| 	globIgnores: [ 'env.js', 'env.sample.js' ], |  | ||||||
| 	swDest: 'public/sw.js', |  | ||||||
| 	cleanupOutdatedCaches: true, |  | ||||||
| 	mode: 'production', |  | ||||||
| 	sourcemap: false |  | ||||||
| }; |  | ||||||
		Reference in New Issue
	
	Block a user