Compare commits
	
		
			353 Commits
		
	
	
		
			fa55fce76e
			...
			0.18.4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 269d7a7def | |||
| e95f2333b0 | |||
| 950217e0a3 | |||
| 5e65fb3301 | |||
| 2a294cde04 | |||
| e95420d79c | |||
| cffbd17dc7 | |||
| 00de8c3d75 | |||
| 1f4711d07a | |||
| 30e3396897 | |||
| 5291e049a1 | |||
| 08fbb504c9 | |||
| e9ca1d3e5d | |||
| eb80406fdb | |||
| 9fe53b0b9c | |||
| 9ae5e62e5d | |||
| 1613ae7de6 | |||
| ed1caa7be7 | |||
| d88f3a5a27 | |||
| c98eb49ae3 | |||
| 6d9d8a4724 | |||
| 7d8c68a455 | |||
| 8f33640bec | |||
| f89023e24a | |||
| f5d14f2e18 | |||
| 9811ede3b2 | |||
| b2e51fea48 | |||
| 2a915620c9 | |||
| 526688935f | |||
| c7dcf7c66d | |||
| 06411dc147 | |||
| 195d182cc9 | |||
| 17217dae76 | |||
| f105cc0a41 | |||
| 2f62c7ae89 | |||
| f6c1fea17c | |||
| 178dc93319 | |||
| 8ffe8eff06 | |||
| bd4952ee57 | |||
| 4b171fd04f | |||
| 2c198cfde8 | |||
| 7c6d39b5fa | |||
| 9c13b2f9e9 | |||
| 1ec9556aa6 | |||
| adec38b50b | |||
| a35af6f020 | |||
| e74ff5e885 | |||
| c87561f63b | |||
| c681570134 | |||
| 53b945c72f | |||
| f6985daec7 | |||
| 5662c3b6da | |||
| 9def0b27c9 | |||
| 52a02c82d2 | |||
| c241961d0a | |||
| 8f50555a06 | |||
| b35375c929 | |||
| bf1e715261 | |||
| 0240e1dca2 | |||
| 7cec2a00c5 | |||
| 239f79fecb | |||
| 0265a59b82 | |||
| 57dce34fc5 | |||
| d8110580e9 | |||
| 03b7ada5ef | |||
| 1df505ea00 | |||
| 3e8dac3203 | |||
| 95707a71a9 | |||
| fc2c2907c4 | |||
| ebdd1c2c0c | |||
| 13254b24dd | |||
| e17eb64031 | |||
| a0727a0291 | |||
| f7f7926829 | |||
| 9b7dca341b | |||
| da3300562a | |||
| e2ddb5a14c | |||
| fb9645aed6 | |||
| a4ebc7e126 | |||
| fd5db7d68a | |||
| e7eddb4f08 | |||
| 94155845f0 | |||
| 3abf608b15 | |||
| d31fe2363b | |||
| 11a56f87e8 | |||
| 19793cdcd4 | |||
| 9363773fa1 | |||
| c7990882cf | |||
| d4ab76ea1b | |||
| 2c992a0e63 | |||
| 88f96acc3c | |||
| 245db06173 | |||
| 49c2cd5c4b | |||
| 64db553185 | |||
| a06a19ce9c | |||
| 592ddc1541 | |||
| cb5f2b73d0 | |||
| 2304b12c1c | |||
| 38d3e1912c | |||
| fbc14fd7b4 | |||
| 0283df22c8 | |||
| 845737ee8e | |||
| 6993511c67 | |||
| 9111ad147c | |||
| 333214aa8f | |||
| c0cde02fec | |||
| 070a20a2e5 | |||
| c5e8409079 | |||
| 67eff0eda9 | |||
| 3de7b632e0 | |||
| 842248e4c4 | |||
| c5d7ec25b5 | |||
| a9a965d698 | |||
| dc866dd540 | |||
| 89252619b1 | |||
| 2699b06d7c | |||
| fd0d45f721 | |||
| 5ecf838dd2 | |||
| 45a7a90cb8 | |||
| cac851f2b1 | |||
| 238082b657 | |||
| aecbabe522 | |||
| a9cdac4f74 | |||
| a59dbbe50e | |||
| 9bec95ede8 | |||
| 70307a9e82 | |||
| ef077b4e6a | |||
| dcabed4e93 | |||
| 1af047f66e | |||
| 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 | 
| @@ -1 +1,2 @@ | |||||||
| public/env.sample.js | public/env.sample.js | ||||||
|  | .pnpm-store | ||||||
							
								
								
									
										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 | ||||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,10 +1,6 @@ | |||||||
| node_modules | node_modules | ||||||
| package-lock.json |  | ||||||
| yarn.lock |  | ||||||
| *.map | *.map | ||||||
| public/env.js | public/env.js | ||||||
| public/index.html | public/index.html | ||||||
| /dist | /dist | ||||||
| .yarn | .pnpm-store | ||||||
| .pnp.js |  | ||||||
| .yarnrc.yml |  | ||||||
							
								
								
									
										519
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										519
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,15 +2,530 @@ | |||||||
|  |  | ||||||
| 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.18.4](https://git.odit.services/lfk/frontend/compare/0.18.3...0.18.4) | ||||||
|  |  | ||||||
|  | - Hide address2 in orgs by default [`e95f233`](https://git.odit.services/lfk/frontend/commit/e95f2333b0b958ed00c0e097b43aac2e70ad0d38) | ||||||
|  |  | ||||||
|  | #### [0.18.3](https://git.odit.services/lfk/frontend/compare/0.18.2...0.18.3) | ||||||
|  |  | ||||||
|  | > 15 April 2023 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.18.3 [`950217e`](https://git.odit.services/lfk/frontend/commit/950217e0a350f9999b879475edf41f2f11c48179) | ||||||
|  | - Dont show adress 2 in runner overview [`5e65fb3`](https://git.odit.services/lfk/frontend/commit/5e65fb33013c3dad38e7ad6740b017ae206f278f) | ||||||
|  |  | ||||||
|  | #### [0.18.2](https://git.odit.services/lfk/frontend/compare/0.18.1...0.18.2) | ||||||
|  |  | ||||||
|  | > 15 April 2023 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.18.2 [`2a294cd`](https://git.odit.services/lfk/frontend/commit/2a294cde040044bbebfb9c8b34b6c91b27772741) | ||||||
|  | - Added timestamps to scanoverview [`cffbd17`](https://git.odit.services/lfk/frontend/commit/cffbd17dc77054048cc9b14891f960f9a3fd18cb) | ||||||
|  | - Push in releaseit [`e95420d`](https://git.odit.services/lfk/frontend/commit/e95420d79c3227c0ca0cf0c0b599970c2b7d690e) | ||||||
|  |  | ||||||
|  | #### [0.18.1](https://git.odit.services/lfk/frontend/compare/0.18.0...0.18.1) | ||||||
|  |  | ||||||
|  | > 15 April 2023 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE v0.18.1 [`00de8c3`](https://git.odit.services/lfk/frontend/commit/00de8c3d75e90cd4614f42111f5f45bedde64130) | ||||||
|  | - Missing scanstation translations [`30e3396`](https://git.odit.services/lfk/frontend/commit/30e33968978bf33cedb31bcbf63fac273e1664f5) | ||||||
|  | - fix: button onclick a11y [`9fe53b0`](https://git.odit.services/lfk/frontend/commit/9fe53b0b9c71e8a6b4aa3f317327ffe729df0834) | ||||||
|  | - fix(ConfirmStatsClientDeletion): ScanStationService -> StatsClientService [`5291e04`](https://git.odit.services/lfk/frontend/commit/5291e049a1d2e880fbe277095da91b70d4812c3f) | ||||||
|  | - fix(ConfirmScanStationDeletion): donorControllerRemove -> scanStationControllerRemove [`08fbb50`](https://git.odit.services/lfk/frontend/commit/08fbb504c958415ce75e1e426296f870f0f1358d) | ||||||
|  |  | ||||||
|  | #### [0.18.0](https://git.odit.services/lfk/frontend/compare/0.17.3...0.18.0) | ||||||
|  |  | ||||||
|  | > 12 April 2023 | ||||||
|  |  | ||||||
|  | - Moved filter function to typed version [`#171`](https://git.odit.services/lfk/frontend/issues/171) | ||||||
|  | - ScansOverview: migrate to datatable [`#168`](https://git.odit.services/lfk/frontend/issues/168) | ||||||
|  | - 🚀RELEASE v0.18.0 [`eb80406`](https://git.odit.services/lfk/frontend/commit/eb80406fdb8abf3f76bca742095e8f1f03480a56) | ||||||
|  | - wip: ScansOverview -> new datatable [`c87561f`](https://git.odit.services/lfk/frontend/commit/c87561f63b90ab951daf91d9b8b54ba96a94cc7f) | ||||||
|  | - Basic card table replace [`5662c3b`](https://git.odit.services/lfk/frontend/commit/5662c3b6da67c00c94254bf39f8820e531fc93ef) | ||||||
|  | - RunnersOverview: table responsiveness [`bf1e715`](https://git.odit.services/lfk/frontend/commit/bf1e715261c0076fd8543dd1187c516209a73b16) | ||||||
|  | - Basic card delete modal [`8ffe8ef`](https://git.odit.services/lfk/frontend/commit/8ffe8eff0623767809cdc49ea15cf2d30b609285) | ||||||
|  | - wip: delete scans [`f105cc0`](https://git.odit.services/lfk/frontend/commit/f105cc0a41972261610d03015ecd5ee492dab8e2) | ||||||
|  | - Added delete runner modal [`fd5db7d`](https://git.odit.services/lfk/frontend/commit/fd5db7d68ab4b32f7bb14bd0275a447b749e1fd8) | ||||||
|  | - Extracted table bottom [`8f50555`](https://git.odit.services/lfk/frontend/commit/8f50555a06a35d0b11ac1d9201851d00d915c4b8) | ||||||
|  | - fix: formatting [`9def0b2`](https://git.odit.services/lfk/frontend/commit/9def0b27c9958c5c75466f56c17b6dc4d44d2b50) | ||||||
|  | - fix(CardsOverview): table scroll + checkbox posititioning [`9c13b2f`](https://git.odit.services/lfk/frontend/commit/9c13b2f9e938ad36cb74c66a8f91a5b872c8c2d1) | ||||||
|  | - Extracted table header [`c241961`](https://git.odit.services/lfk/frontend/commit/c241961d0ab0d6b8c3bf0453c1a21d84027ed62f) | ||||||
|  | - drop legacy dependencies [`c98eb49`](https://git.odit.services/lfk/frontend/commit/c98eb49ae31d1d28de79c7a9c945ade50828f07f) | ||||||
|  | - scan delete working [`c7dcf7c`](https://git.odit.services/lfk/frontend/commit/c7dcf7c66d7141117e08462ad05f6f441565d012) | ||||||
|  | - Add Card appends current cards [`195d182`](https://git.odit.services/lfk/frontend/commit/195d182cc977b4ac9f342f09a9ea69d461892a95) | ||||||
|  | - Moved data loading to onmount [`a35af6f`](https://git.odit.services/lfk/frontend/commit/a35af6f02055115d60d040baaf22dabbeda38498) | ||||||
|  | - Add runners reactivity [`06411dc`](https://git.odit.services/lfk/frontend/commit/06411dc14747fc9803da009a29688789c31dfc42) | ||||||
|  | - quick cleanup [`ebdd1c2`](https://git.odit.services/lfk/frontend/commit/ebdd1c2c0c2191c60dfacd0c4be989748628ca63) | ||||||
|  | - Extracted deletion into function of overview [`a0727a0`](https://git.odit.services/lfk/frontend/commit/a0727a02913708de06be8bf1b9f8cbae5489f080) | ||||||
|  | - Moved update card logic to overview [`53b945c`](https://git.odit.services/lfk/frontend/commit/53b945c72fa5b30f03eff8bb83f8f25d85d79b07) | ||||||
|  | - wip:ScanValid badge [`bd4952e`](https://git.odit.services/lfk/frontend/commit/bd4952ee575b4cd8caea3608b14d8c51b37a55eb) | ||||||
|  | - yeeted extra table styling [`52a02c8`](https://git.odit.services/lfk/frontend/commit/52a02c82d2bdabdf3e632910aa415c1ffb97fd7f) | ||||||
|  | - Added custom runner filter [`7c6d39b`](https://git.odit.services/lfk/frontend/commit/7c6d39b5fa60d1b834c9fb74dd26dfc2225161d6) | ||||||
|  | - Scan reactive add [`f5d14f2`](https://git.odit.services/lfk/frontend/commit/f5d14f2e1810462cba99b2482320746f0750d4a1) | ||||||
|  | - Moved code around [`239f79f`](https://git.odit.services/lfk/frontend/commit/239f79fecba732c807f4296cf524be15c1ccff1e) | ||||||
|  | - Scan deletion [`7d8c68a`](https://git.odit.services/lfk/frontend/commit/7d8c68a4550407e8c871545e7a0c5f622a1e3b9c) | ||||||
|  | - Added status filter function [`f6c1fea`](https://git.odit.services/lfk/frontend/commit/f6c1fea17ce64d890ac810d9dff058dd589c2989) | ||||||
|  | - Added delete toast [`e2ddb5a`](https://git.odit.services/lfk/frontend/commit/e2ddb5a14cc8b450c00129709bf5b933a47866dd) | ||||||
|  | - Fixed killing of the dom [`2f62c7a`](https://git.odit.services/lfk/frontend/commit/2f62c7ae89b7dda3bc0efd4d56e677898a3f197f) | ||||||
|  | - Cards details modal [`f6985da`](https://git.odit.services/lfk/frontend/commit/f6985daec71e4d922acd8a9b2673fd41317b075f) | ||||||
|  | - Reload table data on delete [`3e8dac3`](https://git.odit.services/lfk/frontend/commit/3e8dac3203f56723a3dad4a35887d60fc03d4ae3) | ||||||
|  | - Added middle-name [`d811058`](https://git.odit.services/lfk/frontend/commit/d8110580e9a33cfb0c6e9cbdce630262d5b5d4c1) | ||||||
|  | - Fixed id sorting [`7cec2a0`](https://git.odit.services/lfk/frontend/commit/7cec2a00c513678cae8643ef3905d566a538bde0) | ||||||
|  | - Import add to datatable [`2a91562`](https://git.odit.services/lfk/frontend/commit/2a915620c9adcb481be53e3c919bd5f14a5108ee) | ||||||
|  | - Scans deletion [`6d9d8a4`](https://git.odit.services/lfk/frontend/commit/6d9d8a4724135df98ed1bec74b0cc20a8bdbda53) | ||||||
|  | - fix(DeleteRunnerModal): ESC key [`1df505e`](https://git.odit.services/lfk/frontend/commit/1df505ea00968f65e7aff85baa80f8935fd2eb7b) | ||||||
|  | - Versionbuilder [`1613ae7`](https://git.odit.services/lfk/frontend/commit/1613ae7de6e84aa5f9df6774a8e820fcf515b14a) | ||||||
|  | - Fixed edit dispatch [`c681570`](https://git.odit.services/lfk/frontend/commit/c68157013431eeb1150e37fea87da02d5f7b032e) | ||||||
|  | - ScansOverview: use CardRunner link [`4b171fd`](https://git.odit.services/lfk/frontend/commit/4b171fd04f4fc895f9919294858e546686504c65) | ||||||
|  | - Added custom filter to scan overview [`2c198cf`](https://git.odit.services/lfk/frontend/commit/2c198cfde89363142c1c958e6961b7823196e04e) | ||||||
|  | - Disable sort for actions [`fc2c290`](https://git.odit.services/lfk/frontend/commit/fc2c2907c43e5b1fa192b571cf93b2f2e492158f) | ||||||
|  | - Delete modal logic [`da33005`](https://git.odit.services/lfk/frontend/commit/da3300562a25cf39945a94f873ec27f0d5fb9dc0) | ||||||
|  | - Module [`9ae5e62`](https://git.odit.services/lfk/frontend/commit/9ae5e62e5d84097082bc29fe20b329f6e614f6ed) | ||||||
|  | - new license file version [CI SKIP] [`ed1caa7`](https://git.odit.services/lfk/frontend/commit/ed1caa7be7673ef08f9c0074101f3cfb52963ab7) | ||||||
|  | - Merge pull request 'experiment/tanstack' (#172) from experiment/tanstack into dev [`d88f3a5`](https://git.odit.services/lfk/frontend/commit/d88f3a5a2737921536c5eb109734abafd7609e51) | ||||||
|  | - Removed pnpm store [`eb13f03`](https://git.odit.services/lfk/frontend/commit/eb13f038a1990ed2966552ab1d36df4a76e5a99a) | ||||||
|  | - dependency update, drop focusTrap, first tanstack experiment in RunnersOverview [`9bec95e`](https://git.odit.services/lfk/frontend/commit/9bec95ede80908e45ad0f18a66dbb45784f7e33e) | ||||||
|  | - Added translations [`dcabed4`](https://git.odit.services/lfk/frontend/commit/dcabed4e93134a1f04305c47536f1cc93b0cfb31) | ||||||
|  | - new license file version [CI SKIP] [`d0fe6a2`](https://git.odit.services/lfk/frontend/commit/d0fe6a2e8513638c4d8e2b3dafd48b91490eab3b) | ||||||
|  | - Bumped pnpm to 8 [`ee91748`](https://git.odit.services/lfk/frontend/commit/ee91748b3c7c2fb4f196d76121713ac42465b9dd) | ||||||
|  | - drop old datatables [`cb5f2b7`](https://git.odit.services/lfk/frontend/commit/cb5f2b73d0a5f71875f48d381382919f9bee364e) | ||||||
|  | - wip: pagination [`a59dbbe`](https://git.odit.services/lfk/frontend/commit/a59dbbe50ede47e0dada906d10887cc6b1ae3264) | ||||||
|  | - Added sort [`842248e`](https://git.odit.services/lfk/frontend/commit/842248e4c43bb94a0e73981d2d848e0723687932) | ||||||
|  | - cleanup headers [`5ecf838`](https://git.odit.services/lfk/frontend/commit/5ecf838dd231269c3c4f0d1e3376ff157457b04a) | ||||||
|  | - Moved filter function to shared [`2304b12`](https://git.odit.services/lfk/frontend/commit/2304b12c1cca6a64573223906ab0561ba9050ec5) | ||||||
|  | - Group filters [`0283df2`](https://git.odit.services/lfk/frontend/commit/0283df22c84740dd978e09120985215b65d7a12a) | ||||||
|  | - feat(RunnersOverview): row selection [`6993511`](https://git.odit.services/lfk/frontend/commit/6993511c67cc81303a1eece7fc2a4218dc818afc) | ||||||
|  | - Basic details and delete buttons [`9363773`](https://git.odit.services/lfk/frontend/commit/9363773fa1910975d48746572fe166363986c6ab) | ||||||
|  | - Added delete cards button [`70307a9`](https://git.odit.services/lfk/frontend/commit/70307a9e8272f22801c6765c947f8576cc1d1102) | ||||||
|  | - ScansOverview: add ThFilterTrack [`008835c`](https://git.odit.services/lfk/frontend/commit/008835c24f833aebbecc53921b0901a76f25bf06) | ||||||
|  | - Yeeted old datatable references [`8925261`](https://git.odit.services/lfk/frontend/commit/89252619b1f6d478c287178bfffc573cf574a8fa) | ||||||
|  | - Added filter [`a9cdac4`](https://git.odit.services/lfk/frontend/commit/a9cdac4f74a80afe3d0ce677ed5256201b75b29b) | ||||||
|  | - Moved sort onclick to header only [`3de7b63`](https://git.odit.services/lfk/frontend/commit/3de7b632e058e9ca9ab951de9e86cc2a40eceb68) | ||||||
|  | - Switched drone to pnpm cache [`d79608e`](https://git.odit.services/lfk/frontend/commit/d79608edbb7ca68b0a26f6c021a141373308081f) | ||||||
|  | - pagination size [`45a7a90`](https://git.odit.services/lfk/frontend/commit/45a7a90cb8ee933d064c5891b3532cb1ece38d38) | ||||||
|  | - RunnersOverview: add filter keyboard support [`9415584`](https://git.odit.services/lfk/frontend/commit/94155845f020679e8765d198b5ab4735e1d1142c) | ||||||
|  | - RunnersOverview: pass selectedRunners to actions [`d4ab76e`](https://git.odit.services/lfk/frontend/commit/d4ab76ea1be1789aaa53ca57d33b2356c6804ffd) | ||||||
|  | - fix pagination next prev [`cac851f`](https://git.odit.services/lfk/frontend/commit/cac851f2b19dafa01b08d1da612e34cbe3eb4133) | ||||||
|  | - Moved docker to pnpm with cache [`4cbd265`](https://git.odit.services/lfk/frontend/commit/4cbd26580e51b854fdfd369056f24a30b04100a1) | ||||||
|  | - fix: TracksOverview [`c799088`](https://git.odit.services/lfk/frontend/commit/c7990882cfa4ac6c3a2ffa696dcbf3b62790c1fe) | ||||||
|  | - Table header i18n [`070a20a`](https://git.odit.services/lfk/frontend/commit/070a20a2e599e9f00b1bd0e944f9d68d08dc1cba) | ||||||
|  | - default to 50 rows pagination [`dc866dd`](https://git.odit.services/lfk/frontend/commit/dc866dd5403ae681193c5e325b1f6d4068f80b61) | ||||||
|  | - improved tablefilters/groupFilter [`3abf608`](https://git.odit.services/lfk/frontend/commit/3abf608b15fcfc2cf65b705d28d394917e7ac333) | ||||||
|  | - Updated group name conversion [`9111ad1`](https://git.odit.services/lfk/frontend/commit/9111ad147c7537036de834f80db6e557d8d6f0d9) | ||||||
|  | - show certificate, runnercard, sponsoring contract section [`49c2cd5`](https://git.odit.services/lfk/frontend/commit/49c2cd5c4b32e1b75fe0dae48c1b470608936704) | ||||||
|  | - add basic table styling [`c5e8409`](https://git.odit.services/lfk/frontend/commit/c5e84090795636c7a80d6cabe46168e436d40f50) | ||||||
|  | - Cards deleted migrations [`ef077b4`](https://git.odit.services/lfk/frontend/commit/ef077b4e6ad722e6d1c9efd5060165f649fd94c9) | ||||||
|  | - ignore pnpm store from now on [`fe62ad5`](https://git.odit.services/lfk/frontend/commit/fe62ad5539bc94b242c30c2952bf4c115cc9abfd) | ||||||
|  | - ammendme [`d31fe23`](https://git.odit.services/lfk/frontend/commit/d31fe2363b0f167b937f7d67908e75c709b324d8) | ||||||
|  | - updated default table row count [`245db06`](https://git.odit.services/lfk/frontend/commit/245db0617306bab23993caf00dc5ba0405e95625) | ||||||
|  | - fix: checkbox styling [`64db553`](https://git.odit.services/lfk/frontend/commit/64db5531858b2275d27c53a347da6c0416a278e8) | ||||||
|  | - Added distance conversion [`333214a`](https://git.odit.services/lfk/frontend/commit/333214aa8f945975ee2dcbd1cfa285e5e75c9d02) | ||||||
|  | - Added filterfunctions [`a9a965d`](https://git.odit.services/lfk/frontend/commit/a9a965d6981b07b02e3f752e65e1e5c4e9df7147) | ||||||
|  | - card/ThFilterRunner: text align left [`7083b3d`](https://git.odit.services/lfk/frontend/commit/7083b3d8d2c81fa4ab14f21d379befbb2dab5b6c) | ||||||
|  | - import [`0240e1d`](https://git.odit.services/lfk/frontend/commit/0240e1dca2e162b8c79f294f5c651b4a38455235) | ||||||
|  | - fix: table sort button + search style [`57dce34`](https://git.odit.services/lfk/frontend/commit/57dce34fc5b04c297c5341db24ec47c2f1d39595) | ||||||
|  | - No longer switching pnpm store path [`e5241d6`](https://git.odit.services/lfk/frontend/commit/e5241d619beda90d271485b6c09142b1e7b9b5f1) | ||||||
|  | - Reactivated generate cards modal [`8f33640`](https://git.odit.services/lfk/frontend/commit/8f33640bec8f79987b1319fb0765535fe493d496) | ||||||
|  | - Always load bulk created cards [`f89023e`](https://git.odit.services/lfk/frontend/commit/f89023e24ab30c90a8cacf2e9590d1dfa1a743d5) | ||||||
|  | - ScansOverview: fallback for manual scans [`1ec9556`](https://git.odit.services/lfk/frontend/commit/1ec9556aa64b996d0bdd9cd3ee50a767a154f5f6) | ||||||
|  | - fix min-w th [`b35375c`](https://git.odit.services/lfk/frontend/commit/b35375c929ad8d525b2761c321ecef0399917828) | ||||||
|  | - Close modal on delete and import toastify [`03b7ada`](https://git.odit.services/lfk/frontend/commit/03b7ada5ef3cbf64f178d3d0d88c448d76a42885) | ||||||
|  | - Fixed styling [`13254b2`](https://git.odit.services/lfk/frontend/commit/13254b24dd75ca3eb47fd53d5368f047545e8420) | ||||||
|  | - Implemented table buttons [`11a56f8`](https://git.odit.services/lfk/frontend/commit/11a56f87e89866bd9b2f7029ef683fd139d60aa4) | ||||||
|  | - RunnersOverview: disable debug log [`2c992a0`](https://git.odit.services/lfk/frontend/commit/2c992a0e63fa72055d28785fe7f5c21fd26f1fea) | ||||||
|  | - RunnersOverview: TopActionSection: add margin top [`88f96ac`](https://git.odit.services/lfk/frontend/commit/88f96acc3c31c748d3da29b4564dd681a89208b9) | ||||||
|  | - Unused filter function value [`38d3e19`](https://git.odit.services/lfk/frontend/commit/38d3e1912cd6d5362db236c34e0c602b52060b73) | ||||||
|  | - Make the linter happy [`fbc14fd`](https://git.odit.services/lfk/frontend/commit/fbc14fd7b47009bf64a8824e7b60373eea031e3f) | ||||||
|  | - Dsable column filter for distance [`67eff0e`](https://git.odit.services/lfk/frontend/commit/67eff0eda9b7d49d1abe784247afdbfbd7a76b00) | ||||||
|  | - getPaginationRowModel [`238082b`](https://git.odit.services/lfk/frontend/commit/238082b657998f513111e832556926decbe84e21) | ||||||
|  | - -> onkeyup [`aecbabe`](https://git.odit.services/lfk/frontend/commit/aecbabe52221c0ab8d94a42dc8ac42f4f397fde8) | ||||||
|  | - removed unused import [`9811ede`](https://git.odit.services/lfk/frontend/commit/9811ede3b23a488a49962a43dae5abad60f93341) | ||||||
|  | - drop redundant button role [`e7eddb4`](https://git.odit.services/lfk/frontend/commit/e7eddb4f08632cc803b8752e3561319fab538843) | ||||||
|  | - Filter id as equals [`2699b06`](https://git.odit.services/lfk/frontend/commit/2699b06d7c42608b1b3ce2037908a78143038441) | ||||||
|  | - debug table [`fd0d45f`](https://git.odit.services/lfk/frontend/commit/fd0d45f721396787ad35fca2a7fd0f26d98b23e9) | ||||||
|  | - add TODO for ScansOverview status filter [`9505c2b`](https://git.odit.services/lfk/frontend/commit/9505c2b03005f4e667082cb4527fea863dbbf7fc) | ||||||
|  |  | ||||||
|  | #### [0.17.3](https://git.odit.services/lfk/frontend/compare/0.17.2...0.17.3) | ||||||
|  |  | ||||||
|  | > 15 March 2023 | ||||||
|  |  | ||||||
|  | - dependency fixes [`3ea7a01`](https://git.odit.services/lfk/frontend/commit/3ea7a015a9beba3c2e4d3eb966f24ff6d4ac786e) | ||||||
|  | - 🚀RELEASE v0.17.3 [`85705b6`](https://git.odit.services/lfk/frontend/commit/85705b6e684d0c41d3dc770cef7bffb199101576) | ||||||
|  | - 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) | #### [0.11.0](https://git.odit.services/lfk/frontend/compare/0.10.0...0.11.0) | ||||||
|  |  | ||||||
| - 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) | > 30 March 2021 | ||||||
|  |  | ||||||
| - Added button (including translations [`0614c76`](https://git.odit.services/lfk/frontend/commit/0614c76e924b18b512bab59933a26fec07cf483d) | - 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) | - Added button (including translations [`97e338f`](https://git.odit.services/lfk/frontend/commit/97e338f9d4f388596d550990457254c7fa1a3492) | ||||||
| - Sorted translations [`89bb9c2`](https://git.odit.services/lfk/frontend/commit/89bb9c215e356e0940678f5cabd9e38bc203040e) | - 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) | - 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) | - 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) | - 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) | - 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 version [`2d18686`](https://git.odit.services/lfk/frontend/commit/2d18686ce782a434ca7bd34c07c36a35b9497273) | ||||||
| - Bumped lfk-client-js [`7dfaa75`](https://git.odit.services/lfk/frontend/commit/7dfaa7579a22b13194fcdd1c02b4437958261472) | - Bumped lfk-client-js [`7dfaa75`](https://git.odit.services/lfk/frontend/commit/7dfaa7579a22b13194fcdd1c02b4437958261472) | ||||||
| @@ -958,7 +1473,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) | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,14 +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 | ||||||
| COPY package.json ./ |  | ||||||
| RUN yarn | COPY package.json pnpm-lock.yaml *.config.js *.config.cjs index.html ./ | ||||||
| COPY package.json *.config.js index.html ./ | RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 && pnpm i | ||||||
|  |  | ||||||
| COPY src ./src | COPY src ./src | ||||||
| COPY public ./public | COPY public ./public | ||||||
| RUN yarn 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/dist /app | COPY --from=build /app/dist /usr/share/nginx/html | ||||||
| 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 | ||||||
							
								
								
									
										42
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								index.html
									
									
									
									
									
								
							| @@ -1,22 +1,22 @@ | |||||||
| <!DOCTYPE html> | <!DOCTYPE html> | ||||||
| <html lang="en"> | <html lang="en"> | ||||||
|  |  | ||||||
| <head> | <head> | ||||||
|   <meta charset="utf-8" /> |   <meta charset="utf-8" /> | ||||||
|   <link rel="icon" href="/favicon.png" /> |   <link rel="icon" href="/favicon.png" /> | ||||||
|   <link rel="manifest" href="/manifest.webmanifest"> |   <link rel="manifest" href="/manifest.webmanifest"> | ||||||
|   <link rel="apple-touch-icon" href="/lfk-logo.png"> |   <link rel="apple-touch-icon" href="/lfk-logo.png"> | ||||||
|   <meta name="theme-color" content="#FFFFFF"> |   <meta name="theme-color" content="#FFFFFF"> | ||||||
|   <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> | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body> | <body> | ||||||
|   <span style="display: none;visibility: hidden;" id="buildinfo">RELEASE_INFO-0.8.4-RELEASE_INFO</span> |   <span style="display: none;visibility: hidden;" id="buildinfo">RELEASE_INFO-0.18.4-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 type="module" src="/src/main.js"></script> |   <script type="module" src="/src/main.js"></script> | ||||||
| </body> | </body> | ||||||
|  |  | ||||||
| </html> | </html> | ||||||
| @@ -6,6 +6,11 @@ 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 { |         location = /index.html { | ||||||
|             add_header Cache-Control 'no-store'; |             add_header Cache-Control 'no-store'; | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								order.js
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								order.js
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| const fs = require('fs'); | import fs from 'fs' | ||||||
| // get all language files | // get all language files | ||||||
| const files = fs.readdirSync('./src/locales/'); | const files = fs.readdirSync('./src/locales/'); | ||||||
| files.forEach((f) => { | files.forEach((f) => { | ||||||
|   | |||||||
							
								
								
									
										116
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										116
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,56 +1,60 @@ | |||||||
| { | { | ||||||
| 	"name": "@odit/lfk-frontend", | 	"name": "@odit/lfk-frontend", | ||||||
| 	"version": "0.8.4", | 	"version": "0.18.4", | ||||||
| 	"scripts": { | 	"type": "module", | ||||||
| 		"i18n-order": "node order.js", | 	"scripts": { | ||||||
| 		"dev": "vite", | 		"i18n-order": "node order.js", | ||||||
| 		"build": "vite build", | 		"dev": "vite", | ||||||
| 		"release": "release-it", | 		"build": "vite build", | ||||||
| 		"licenses:export": "license-exporter --json -o public" | 		"release": "release-it", | ||||||
| 	}, | 		"licenses:export": "license-exporter --json -o public" | ||||||
| 	"license": "CC-BY-NC-SA-4.0", | 	}, | ||||||
| 	"devDependencies": { | 	"license": "CC-BY-NC-SA-4.0", | ||||||
| 		"check-password-strength": "2.0.2", | 	"devDependencies": { | ||||||
| 		"@odit/lfk-client-js": "0.10.1", | 		"@odit/license-exporter": "0.0.12", | ||||||
| 		"@odit/license-exporter": "0.0.11", | 		"@sveltejs/vite-plugin-svelte": "2.0.4", | ||||||
| 		"@sveltejs/vite-plugin-svelte": "1.0.0-next.5", | 		"auto-changelog": "2.4.0", | ||||||
| 		"@types/html-minifier": "4.0.0", | 		"autoprefixer": "10.4.14", | ||||||
| 		"auto-changelog": "2.2.1", | 		"postcss": "8.4.21", | ||||||
| 		"autoprefixer": "10.2.5", | 		"release-it": "15.10.1", | ||||||
| 		"csvtojson": "2.0.10", | 		"svelte-select": "3.17.0", | ||||||
| 		"gridjs": "3.4.0", | 		"tailwindcss": "3.3.1", | ||||||
| 		"html-minifier": "4.0.0", | 		"vite": "4.2.1" | ||||||
| 		"localforage": "1.9.0", | 	}, | ||||||
| 		"marked": "2.0.1", | 	"release-it": { | ||||||
| 		"release-it": "14.5.1", | 		"git": { | ||||||
| 		"svelte": "3.37.0", | 			"commit": true, | ||||||
| 		"svelte-focus-trap": "1.2.0", | 			"requireCleanWorkingDir": false, | ||||||
| 		"svelte-i18n": "3.3.9", | 			"commitMessage": "🚀RELEASE v${version}", | ||||||
| 		"svelte-preprocess": "4.7.0", | 			"push": true, | ||||||
| 		"svelte-select": "3.17.0", | 			"tag": true, | ||||||
| 		"tailwindcss": "2.0.4", | 			"tagName": null, | ||||||
| 		"tinro": "0.6.1", | 			"tagAnnotation": "v${version}" | ||||||
| 		"toastify-js": "1.10.0", | 		}, | ||||||
| 		"validator": "13.5.2", | 		"npm": { | ||||||
| 		"vite": "2.1.5", | 			"publish": false | ||||||
| 		"vite-plugin-windicss": "0.12.2", | 		}, | ||||||
| 		"xlsx": "0.16.9" | 		"hooks": { | ||||||
| 	}, | 			"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" | ||||||
| 	"release-it": { | 		} | ||||||
| 		"git": { | 	}, | ||||||
| 			"commit": true, | 	"dependencies": { | ||||||
| 			"requireCleanWorkingDir": false, | 		"@odit/lfk-client-js": "0.14.3", | ||||||
| 			"commitMessage": "🚀RELEASE v${version}", | 		"@paralleldrive/cuid2": "^2.2.0", | ||||||
| 			"push": false, | 		"@tanstack/svelte-table": "^8.8.5", | ||||||
| 			"tag": true, | 		"check-password-strength": "2.0.7", | ||||||
| 			"tagName": null, | 		"csvtojson": "2.0.10", | ||||||
| 			"tagAnnotation": "v${version}" | 		"gridjs": "3.4.0", | ||||||
| 		}, | 		"localforage": "1.10.0", | ||||||
| 		"npm": { | 		"marked": "2.0.3", | ||||||
| 			"publish": false | 		"svelte": "3.58.0", | ||||||
| 		}, | 		"svelte-i18n": "3.6.0", | ||||||
| 		"hooks": { | 		"tinro": "0.6.12", | ||||||
| 			"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" | 		"toastify-js": "1.12.0", | ||||||
| 		} | 		"validator": "13.9.0", | ||||||
| 	} | 		"xlsx": "0.18.5" | ||||||
| } | 	}, | ||||||
|  | 	"volta": { | ||||||
|  | 		"node": "19.7.0" | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										3942
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										3942
									
								
								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,5 +1,6 @@ | |||||||
| const config = { | const config = { | ||||||
| 	baseurl: 'http://localhost:4010', | 	baseurl: 'http://localhost:4010', | ||||||
|  | 	baseurl_documentserver: 'http://localhost:4010/documents', | ||||||
| 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe', | 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe', | ||||||
| 	// optional | 	// optional | ||||||
| 	default_username: 'demo', | 	default_username: 'demo', | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -72,6 +72,8 @@ | |||||||
|   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 Cards from "./components/cards/Cards.svelte"; | ||||||
|  | 	import StatsClients from "./components/statsclients/StatsClients.svelte"; | ||||||
|  | 	import StatsClientDetail from "./components/statsclients/StatsClientDetail.svelte"; | ||||||
|   store.init(); |   store.init(); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| @@ -206,6 +208,14 @@ | |||||||
|             <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> | ||||||
|   | |||||||
| @@ -76,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%)", | ||||||
|   | |||||||
							
								
								
									
										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> | ||||||
| @@ -1,14 +1,16 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { RunnerCardService } from "@odit/lfk-client-js"; |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|   export let bulk_modal_open; |   export let bulk_modal_open; | ||||||
|   export let current_cards; |  | ||||||
|   function focus(el) { |   function focus(el) { | ||||||
|     el.focus(); |     el.focus(); | ||||||
|   } |   } | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |  | ||||||
|   $: card_count = 0; |   $: card_count = 0; | ||||||
|   $: is_card_count_valid = card_count > 0; |   $: is_card_count_valid = card_count > 0; | ||||||
|   $: processed_last_submit = true; |   $: processed_last_submit = true; | ||||||
| @@ -34,7 +36,7 @@ | |||||||
|         text: $_("creating-blanco-cards"), |         text: $_("creating-blanco-cards"), | ||||||
|         duration: -1, |         duration: -1, | ||||||
|       }).showToast(); |       }).showToast(); | ||||||
|       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, false) |       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true) | ||||||
|         .then((result) => { |         .then((result) => { | ||||||
|           bulk_modal_open = false; |           bulk_modal_open = false; | ||||||
|           // |           // | ||||||
| @@ -43,6 +45,7 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|  |           dispatch("created", {cards: result}) | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
| @@ -71,13 +74,13 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           current_cards = current_cards.concat(result); |  | ||||||
|           const toast = Toastify({ |           const toast = Toastify({ | ||||||
|             text: $_("generating-pdf"), |             text: $_("generating-pdf"), | ||||||
|             duration: -1, |             duration: -1, | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|  |           dispatch("created", {cards: result}) | ||||||
|           fetch( |           fetch( | ||||||
|             `${config.baseurl}/documents/cards?&download=true&key=${config.documentserver_key}`, |             `${config.baseurl_documentserver}/cards?&download=true&key=${config.documentserver_key}`, | ||||||
|             { |             { | ||||||
|               method: "POST", |               method: "POST", | ||||||
|               headers: { |               headers: { | ||||||
| @@ -133,7 +136,7 @@ | |||||||
| {#if bulk_modal_open} | {#if bulk_modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       bulk_modal_open = false; |       bulk_modal_open = false; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { |   import { | ||||||
|     RunnerCardService, |     RunnerCardService, | ||||||
|     RunnerService, |     RunnerService, | ||||||
| @@ -9,24 +9,40 @@ | |||||||
|   } 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 { createEventDispatcher } from "svelte"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_cards; |  | ||||||
|   const getRunnerLabel = (option) => |   const dispatch = createEventDispatcher(); | ||||||
|     option.firstname + " " + (option.middlename || "") + " " + option.lastname; |   const getRunnerLabel = (option) => { | ||||||
|   const filterRunners = (label, filterText, option) => |     if (option.middlename) { | ||||||
|     label.toLowerCase().includes(filterText.toLowerCase()) || |       return option.firstname + " " + option.middlename + " " + option.lastname; | ||||||
|     option.value.toString().startsWith(filterText.toLowerCase()); |     } | ||||||
|  |     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) { |   function focus(el) { | ||||||
|     el.focus(); |     el.focus(); | ||||||
|   } |   } | ||||||
|   $: runner = 0; |   $: runner = 0; | ||||||
|   $: runners = []; |  | ||||||
|   $: enabled = true; |   $: enabled = true; | ||||||
|   $: processed_last_submit = true; |   $: processed_last_submit = true; | ||||||
|  |  | ||||||
|  |   let loading = true; | ||||||
|  |   let runners = []; | ||||||
|   RunnerService.runnerControllerGetAll().then((val) => { |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|     runners = val.map((r) => { |     runners = val.map((r) => { | ||||||
|       return { label: getRunnerLabel(r), value: r }; |       return { label: getRunnerLabel(r), value: r }; | ||||||
|     }); |     }); | ||||||
|  |     loading = false; | ||||||
|   }); |   }); | ||||||
|   $: createbtnenabled = true; |   $: createbtnenabled = true; | ||||||
|   (() => { |   (() => { | ||||||
| @@ -64,8 +80,7 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           current_cards.push(result); |           dispatch("created", { cards: [result] }); | ||||||
|           current_cards = current_cards; |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
| @@ -82,65 +97,81 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|     }}> |     }} | ||||||
|  |   > | ||||||
|     <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" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <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" |         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" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|  |       > | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |         <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="sm:flex sm:items-start"> | ||||||
|             <div |             <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"> |               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 |               <svg | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |                 viewBox="0 0 24 24" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   fill="currentColor" |                   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> |                   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> | ||||||
|             <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"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('create-a-new-card')} |                 {$_("create-a-new-card")} | ||||||
|               </h3> |               </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"> | ||||||
|                   {$_('you-can-provide-a-runner-but-you-dont-have-to')} |                   {$_("you-can-provide-a-runner-but-you-dont-have-to")} | ||||||
|                   {$_('if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button')} |                   {$_( | ||||||
|  |                     "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button" | ||||||
|  |                   )} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <div class="grid grid-cols-6 gap-6"> |               <div class="grid grid-cols-6 gap-6"> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="donor" |                     for="donor" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('runner')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("runner")}</label | ||||||
|  |                   > | ||||||
|                   <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) => filterRunners(label, filterText, option)} |                     itemFilter={(label, filterText, option) => | ||||||
|  |                       filterRunners(label, filterText, option)} | ||||||
|                     items={runners} |                     items={runners} | ||||||
|                     showChevron={true} |                     bind:loading | ||||||
|                     placeholder={$_('search-for-runner-by-name-or-id')} |                     showChevron={!loading} | ||||||
|                     noOptionsMessage={$_('no-runners-found')} |                     placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|                     on:select={(selectedValue) => (runner = selectedValue.detail.value.id)} |                     noOptionsMessage={$_("no-runners-found")} | ||||||
|                     on:clear={() => (runner = null)} /> |                     on:select={(selectedValue) => | ||||||
|  |                       (runner = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (runner = null)} | ||||||
|  |                   /> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
| @@ -152,16 +183,18 @@ | |||||||
|             class:opacity-50={!createbtnenabled} |             class:opacity-50={!createbtnenabled} | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             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" | ||||||
|             {$_('create')} |           > | ||||||
|  |             {$_("create")} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               modal_open = false; |               modal_open = false; | ||||||
|             }} |             }} | ||||||
|             type="button" |             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"> |             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')} |           > | ||||||
|  |             {$_("cancel")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,26 +1,34 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { RunnerCardService, RunnerService } from "@odit/lfk-client-js"; |   import { RunnerCardService, RunnerService } 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 { createEventDispatcher } from "svelte"; | ||||||
|   export let edit_modal_open; |   export let edit_modal_open; | ||||||
|   export let current_cards; |  | ||||||
|   export let runner = {}; |   export let runner = {}; | ||||||
|   export let editable = {}; |   export let editable = {}; | ||||||
|   export let original_data = {}; |   export let original_data = {}; | ||||||
|   const getRunnerLabel = (option) => |   const getRunnerLabel = (option) => | ||||||
|     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()) || |     if (filterText.startsWith("#")) { | ||||||
|     option.value.toString().startsWith(filterText.toLowerCase()); |       return option.value.id == parseInt(filterText.replace("#","")) | ||||||
|  |     } | ||||||
|  |     return ( | ||||||
|  |       label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|  |       option.value.toString().startsWith(filterText.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   function focus(el) { |   function focus(el) { | ||||||
|     el.focus(); |     el.focus(); | ||||||
|   } |   } | ||||||
|   $: runners = []; |   $: runners = []; | ||||||
|   $: enabled = true; |   $: enabled = true; | ||||||
|   $: processed_last_submit = true; |   $: processed_last_submit = true; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|   RunnerService.runnerControllerGetAll().then((val) => { |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|     runners = val.map((r) => { |     runners = val.map((r) => { | ||||||
|       return { label: getRunnerLabel(r), value: r }; |       return { label: getRunnerLabel(r), value: r }; | ||||||
| @@ -63,8 +71,7 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           current_cards[current_cards.findIndex((c) => c.id === id)] = result; |           dispatch('dataUpdated', {card: result}); | ||||||
|           current_cards = current_cards; |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
| @@ -81,7 +88,7 @@ | |||||||
| {#if edit_modal_open} | {#if edit_modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       edit_modal_open = false; |       edit_modal_open = false; | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								src/components/cards/CardRunner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/cards/CardRunner.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let runner; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !runner} | ||||||
|  |   {$_("non-blanko")} | ||||||
|  | {:else} | ||||||
|  |   <a href={`/runners/${runner.id}`}> | ||||||
|  |     {#if runner.middlename} | ||||||
|  |       {runner.firstname} {runner.middlename} {runner.lastname} | ||||||
|  |     {:else} | ||||||
|  |       {runner.firstname} {runner.lastname} | ||||||
|  |     {/if} | ||||||
|  |   </a> | ||||||
|  | {/if} | ||||||
							
								
								
									
										16
									
								
								src/components/cards/CardStatus.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/cards/CardStatus.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let enabled = false; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if 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} | ||||||
| @@ -7,34 +7,48 @@ | |||||||
|   $: current_cards = []; |   $: current_cards = []; | ||||||
|   export let modal_open = false; |   export let modal_open = false; | ||||||
|   export let bulk_modal_open = false; |   export let bulk_modal_open = false; | ||||||
|  |   let addCards; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <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"> | ||||||
|     {$_('cards')} |     {$_("cards")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:CREATE')} |     {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:CREATE")} | ||||||
|       <button |       <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
|           modal_open = true; |           modal_open = true; | ||||||
|         }} |         }} | ||||||
|         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" | ||||||
|         {$_('add-card')} |       > | ||||||
|  |         {$_("add-card")} | ||||||
|       </button> |       </button> | ||||||
|       <button |       <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
|           bulk_modal_open = true; |           bulk_modal_open = true; | ||||||
|         }} |         }} | ||||||
|         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" | ||||||
|         {$_('create-bulk-cards')} |       > | ||||||
|  |         {$_("create-bulk-cards")} | ||||||
|       </button> |       </button> | ||||||
|     {/if} |     {/if} | ||||||
|   </span> |   </span> | ||||||
|   <CardsOverview bind:current_cards /> |   <CardsOverview bind:current_cards bind:addCards /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:CREATE")} | ||||||
|   <AddCardModal bind:current_cards bind:modal_open /> |   <AddCardModal | ||||||
|   <AddCardBulkModal bind:current_cards bind:bulk_modal_open /> |     bind:modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       console.log(event) | ||||||
|  |       addCards(event.detail.cards); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  |   <AddCardBulkModal | ||||||
|  |     bind:bulk_modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addCards(event.detail.cards); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -1,237 +1,300 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, json, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { RunnerCardService } from "@odit/lfk-client-js"; |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import CardsEmptyState from "./CardsEmptyState.svelte"; |   import CardsEmptyState from "./CardsEmptyState.svelte"; | ||||||
|   import CardDetailModal from "./CardDetailModal.svelte"; |   import CardDetailModal from "./CardDetailModal.svelte"; | ||||||
|   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|  |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|  |   import CardStatus from "./CardStatus.svelte"; | ||||||
|  |   import CardRunner from "./CardRunner.svelte"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import { runnerFilter, statusFilter } from "../shared/tablefilters"; | ||||||
|  |   import DeleteCardModal from "./DeleteCardModal.svelte"; | ||||||
|  |  | ||||||
|   export let edit_modal_open = false; |   export let edit_modal_open = false; | ||||||
|   export let runner = {}; |   export let runner = {}; | ||||||
|   export let editable = {}; |   export let editable = {}; | ||||||
|   export let original_data = {}; |   export let original_data = {}; | ||||||
|   export let current_cards = []; |   export let current_cards = []; | ||||||
|   $: searchvalue = ""; |   export const addCards = (cards) => { | ||||||
|   $: active_deletes = []; |     console.log(cards) | ||||||
|   $: cards_show = current_cards.some( |     current_cards = current_cards.concat(...cards); | ||||||
|     (r) => r.is_selected === true |     options.update((options) => ({ | ||||||
|   ); |       ...options, | ||||||
|   $: generate_cards = current_cards.filter((r) => r.is_selected === true); |       data: current_cards, | ||||||
|   const cards_promise = RunnerCardService.runnerCardControllerGetAll().then( |     })); | ||||||
|     (val) => { |   }; | ||||||
|       current_cards = val; |  | ||||||
|     } |   $: dataLoaded = false; | ||||||
|   ); |   $: selected = | ||||||
|   function should_display_based_on_id(id) { |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |   $: selectedCards = | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |     $table?.getSelectedRowModel().rows.map((row) => row.original) || []; | ||||||
|     } |   $: active_delete = undefined; | ||||||
|     return id.toString() === searchvalue; |   $: cards_show = generate_cards.length > 0; | ||||||
|   } |   $: generate_cards = []; | ||||||
|   const getRunnerLabel = (option) => |  | ||||||
|     option?.firstname + " " + (option?.middlename || "") + " " + (option?.lastname || "{$_('non-blanko')}"); |   const columns = [ | ||||||
|  |     { | ||||||
|  |       accessorKey: "code", | ||||||
|  |       header: () => $_("code"), | ||||||
|  |       filterFn: `includesString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "runner", | ||||||
|  |       header: () => $_("runner"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(CardRunner, { runner: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `runner`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "enabled", | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(CardStatus, { enabled: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       header: () => $_("status"), | ||||||
|  |       filterFn: `status`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(TableActions, { | ||||||
|  |           detailsAction: () => { | ||||||
|  |             open_edit_modal( | ||||||
|  |               current_cards[ | ||||||
|  |                 current_cards.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ] | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_delete = | ||||||
|  |               current_cards[ | ||||||
|  |                 current_cards.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ]; | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes("CARD:DELETE"), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |  | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     filterFns: { | ||||||
|  |       runner: runnerFilter, | ||||||
|  |       status: statusFilter, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const table = createSvelteTable(options); | ||||||
|  |  | ||||||
|   function open_edit_modal(card) { |   function open_edit_modal(card) { | ||||||
|     if(card.runner?.id){ |     const getRunnerLabel = (option) => | ||||||
|  |       option.firstname + | ||||||
|  |       " " + | ||||||
|  |       (option.middlename || "") + | ||||||
|  |       " " + | ||||||
|  |       option.lastname; | ||||||
|  |     if (card.runner?.id) { | ||||||
|       runner = Object.assign( |       runner = Object.assign( | ||||||
|         { runner }, |         { runner }, | ||||||
|         { label: getRunnerLabel(card.runner), value: card.runner } |         { label: getRunnerLabel(card.runner), value: card.runner } | ||||||
|       ); |       ); | ||||||
|       card.runner = card.runner.id; |       card.runner = card.runner.id; | ||||||
|     } |     } else { | ||||||
|     else{ |       card.runner = null; | ||||||
|       card.runner=null; |       runner = null; | ||||||
|       runner = null |  | ||||||
|     } |     } | ||||||
|     editable = Object.assign(editable, card); |     editable = Object.assign(editable, card); | ||||||
|     original_data = Object.assign(original_data, card); |     original_data = Object.assign(original_data, card); | ||||||
|     edit_modal_open = true; |     edit_modal_open = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   async function deleteCard(delete_card_id) { | ||||||
|  |     await RunnerCardService.runnerCardControllerRemove(delete_card_id, true); | ||||||
|  |     current_cards = current_cards.filter((r) => r.id !== delete_card_id); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_cards, | ||||||
|  |     })); | ||||||
|  |     Toastify({ | ||||||
|  |       text: $_("card-deleted"), | ||||||
|  |       duration: 3500, | ||||||
|  |       backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |     }).showToast(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   onMount(() => { | ||||||
|  |     RunnerCardService.runnerCardControllerGetAll().then((val) => { | ||||||
|  |       current_cards = val; | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_cards, | ||||||
|  |       })); | ||||||
|  |       dataLoaded = true; | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:UPDATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:UPDATE")} | ||||||
|   <CardDetailModal |   <CardDetailModal | ||||||
|     bind:current_cards |  | ||||||
|     bind:edit_modal_open |     bind:edit_modal_open | ||||||
|     bind:runner |     bind:runner | ||||||
|     bind:editable |     bind:editable | ||||||
|     bind:original_data /> |     bind:original_data | ||||||
|  |     on:dataUpdated={(event) => { | ||||||
|  |       current_cards[ | ||||||
|  |         current_cards.findIndex((c) => c.id === event.detail.card.id) | ||||||
|  |       ] = event.detail.card; | ||||||
|  |       current_cards = current_cards; | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_cards, | ||||||
|  |       })); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")} | ||||||
|   {#await cards_promise} |   <DeleteCardModal | ||||||
|  |     delete_card={active_delete} | ||||||
|  |     modal_open={active_delete != undefined} | ||||||
|  |     on:delete={(event) => { | ||||||
|  |       deleteCard(event.detail.id); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  |   {#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">{$_('loading-cards')}</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("loading-cards")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:else if current_cards.length === 0} | ||||||
|     {#if current_cards.length === 0} |     <CardsEmptyState /> | ||||||
|       <CardsEmptyState /> |   {:else} | ||||||
|     {:else} |     <div class="h-12 mt-1"> | ||||||
|       <input |       {#if selected.length > 0} | ||||||
|         type="search" |         <button | ||||||
|         bind:value={searchvalue} |           type="button" | ||||||
|         placeholder={$_('datatable.search')} |           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 inline-flex" | ||||||
|         aria-label={$_('datatable.search')} |           id="options-menu" | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |           on:click={async () => { | ||||||
|         <div class="h-12"> |             const prom = []; | ||||||
|           <GenerateRunnerCards |             for (const card of selectedCards) { | ||||||
|             bind:cards_show |               prom.push( | ||||||
|             bind:generate_cards /> |                 await RunnerCardService.runnerCardControllerRemove( | ||||||
|         </div> |                   card.id, | ||||||
|       <div |                   true | ||||||
|         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> |             await Promise.all(prom); | ||||||
|               <th |             for (const card of selectedCards) { | ||||||
|                 scope="col" |               current_cards = current_cards.filter((r) => r.id !== card.id); | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             } | ||||||
|                 <span |             options.update((options) => ({ | ||||||
|                   on:click={() => { |               ...options, | ||||||
|                     const newstate = !current_cards.some((r) => r.is_selected === true); |               data: current_cards, | ||||||
|                     current_cards = current_cards.map((r) => { |             })); | ||||||
|                       r.is_selected = newstate; |             $table.resetRowSelection() | ||||||
|                       return r; |           }} | ||||||
|                     }); |         > | ||||||
|                   }} |           {$_("delete-cards")} | ||||||
|                   class="underline cursor-pointer select-none">{#if current_cards.some((r) => r.is_selected === true)} |           <svg | ||||||
|                     {$_('deselect-all')} |             xmlns="http://www.w3.org/2000/svg" | ||||||
|                   {:else}{$_('select-all')}{/if} |             fill="none" | ||||||
|                 </span> |             viewBox="0 0 24 24" | ||||||
|               </th> |             stroke-width="1.5" | ||||||
|               <th |             stroke="currentColor" | ||||||
|                 scope="col" |             class="w-5 h-5" | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |           > | ||||||
|                 {$_('code')} |             <path | ||||||
|               </th> |               stroke-linecap="round" | ||||||
|               <th |               stroke-linejoin="round" | ||||||
|                 scope="col" |               d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0" | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             /> | ||||||
|                 {$_('runner')} |           </svg> | ||||||
|               </th> |         </button> | ||||||
|               <th |       {/if} | ||||||
|                 scope="col" |       <GenerateRunnerCards cards_show={selected.length>0} bind:generate_cards={selectedCards} /> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |     </div> | ||||||
|                 {$_('status')} |     <div class="overflow-x-auto"> | ||||||
|               </th> |       <table class="w-full"> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |         <thead> | ||||||
|                 <span class="sr-only">{$_('action')}</span> |           {#each $table.getHeaderGroups() as headerGroup} | ||||||
|  |             <tr class="select-none"> | ||||||
|  |               <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|  |                 <InputElement | ||||||
|  |                   type="checkbox" | ||||||
|  |                   checked={$table.getIsAllRowsSelected()} | ||||||
|  |                   indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|  |                   on:change={() => $table.toggleAllRowsSelected()} | ||||||
|  |                 /> | ||||||
|               </th> |               </th> | ||||||
|  |               {#each headerGroup.headers as header} | ||||||
|  |                 <TableHeader {header} /> | ||||||
|  |               {/each} | ||||||
|             </tr> |             </tr> | ||||||
|           </thead> |           {/each} | ||||||
|           <tbody class="divide-y divide-gray-200"> |         </thead> | ||||||
|             {#each current_cards as card} |         <tbody> | ||||||
|               {#if card.code |           {#each $table.getRowModel().rows as row} | ||||||
|                 .toLowerCase() |             <tr> | ||||||
|                 .includes( |               <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|                   searchvalue.toLowerCase() |                 <InputElement | ||||||
|                 ) || card.runner?.firstname |                   type="checkbox" | ||||||
|                   .toLowerCase() |                   checked={row.getIsSelected()} | ||||||
|                   .includes( |                   on:change={() => row.toggleSelected()} | ||||||
|                     searchvalue.toLowerCase() |                 /> | ||||||
|                   ) || card.runner?.middlename |               </td> | ||||||
|                   .toLowerCase() |               {#each row.getVisibleCells() as cell} | ||||||
|                   .includes( |                 <td> | ||||||
|                     searchvalue.toLowerCase() |                   <svelte:component | ||||||
|                   ) || card.runner?.lastname |                     this={flexRender( | ||||||
|                   .toLowerCase() |                       cell.column.columnDef.cell, | ||||||
|                   .includes( |                       cell.getContext() | ||||||
|                     searchvalue.toLowerCase() |                     )} | ||||||
|                   ) || should_display_based_on_id(card.id)} |                   /> | ||||||
|                 <tr data-rowid="card_{card.id}"> |                 </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |               {/each} | ||||||
|                     <input |             </tr> | ||||||
|                       bind:checked={card.is_selected} |           {/each} | ||||||
|                       type="checkbox" |         </tbody> | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |       </table> | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center">{card.code}</div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       {#if card.runner} |  | ||||||
|                         <a |  | ||||||
|                           href="../runners/{card.runner.id}" |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{card.runner.firstname} |  | ||||||
|                           {card.runner.middlename || ''} |  | ||||||
|                           {card.runner.lastname}</a> |  | ||||||
|                       {:else}{$_('non-blanko')}{/if} |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       {#if card.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} |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|  |  | ||||||
|                   {#if active_deletes[card.id] === true} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           active_deletes[card.id] = false; |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           RunnerCardService.runnerCardControllerRemove(card.id, false).then( |  | ||||||
|                             (resp) => { |  | ||||||
|                               current_cards = current_cards.filter( |  | ||||||
|                                 (obj) => obj.id !== card.id |  | ||||||
|                               ); |  | ||||||
|                               Toastify({ |  | ||||||
|                                 text: $_('card-deleted'), |  | ||||||
|                                 duration: 500, |  | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             } |  | ||||||
|                           ); |  | ||||||
|                         }} |  | ||||||
|                         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"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           open_edit_modal(card); |  | ||||||
|                         }} |  | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</button> |  | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:DELETE')} |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[card.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> |     </div> | ||||||
|   {/await} |     <TableBottom {table} {selected} /> | ||||||
|  |   {/if} | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
							
								
								
									
										128
									
								
								src/components/cards/DeleteCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								src/components/cards/DeleteCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_card = { | ||||||
|  |     id: 0, | ||||||
|  |     code: "", | ||||||
|  |     runner: { | ||||||
|  |       firstname: "", | ||||||
|  |       lastname: "", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   onMount(() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   async function submit() { | ||||||
|  |     dispatch("delete", { id: delete_card.id }); | ||||||
|  |     modal_open = false; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     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"> | ||||||
|  |                 {$_("confirm-delete")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_("please-confirm-the-deletion-of-card")} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="w-full"> | ||||||
|  |                 {$_("card")} #{delete_card.code}<br /> | ||||||
|  |                 <span class="inline-block"> | ||||||
|  |                   {$_("runner")}: | ||||||
|  |                   {#if delete_card.runner} | ||||||
|  |                     <span class="inline-block" | ||||||
|  |                       >{delete_card.runner.firstname} | ||||||
|  |                       {delete_card.runner.lastname}</span | ||||||
|  |                     > | ||||||
|  |                   {:else} | ||||||
|  |                     {$_("non-blanko")} | ||||||
|  |                   {/if}</span | ||||||
|  |                 > | ||||||
|  |               </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={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" | ||||||
|  |           > | ||||||
|  |             {$_("delete")} | ||||||
|  |           </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} | ||||||
							
								
								
									
										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,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     GroupContactService, |     GroupContactService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
| @@ -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(); | ||||||
| @@ -145,7 +145,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="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"> |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|   | |||||||
| @@ -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> | ||||||
| @@ -246,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" | ||||||
| @@ -297,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" | ||||||
| @@ -318,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} |  | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     DonationService, |     DonationService, | ||||||
|     DonorService, |     DonorService, | ||||||
| @@ -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> | ||||||
| @@ -150,7 +156,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
| @@ -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 { 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: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 | ||||||
| @@ -37,7 +49,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="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"> |         <table class="divide-y divide-gray-200 w-full"> | ||||||
| @@ -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> | ||||||
| @@ -132,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"> | ||||||
| @@ -149,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)', | ||||||
| @@ -163,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> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     DonorService |     DonorService | ||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
| @@ -134,7 +134,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { DonorService } from "@odit/lfk-client-js"; |   import { DonorService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { createEventDispatcher } from "svelte"; |   import { createEventDispatcher } from "svelte"; | ||||||
| @@ -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(); | ||||||
| @@ -32,7 +32,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> |     on:click_outside={cancelDelete}> | ||||||
|     <div |     <div | ||||||
|   | |||||||
| @@ -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"; | ||||||
| @@ -51,7 +51,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="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"> |         <table class="divide-y divide-gray-200 w-full"> | ||||||
| @@ -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> | ||||||
| @@ -114,7 +119,7 @@ | |||||||
|                     {/if} |                     {/if} | ||||||
|                     {#if donor.address.address1 !== null} |                     {#if donor.address.address1 !== null} | ||||||
|                       {donor.address.address1}<br /> |                       {donor.address.address1}<br /> | ||||||
|                       {donor.address.address2 || ''}<br /> |                       <!-- {donor.address.address2 || ''}<br /> --> | ||||||
|                       {donor.address.postalcode} |                       {donor.address.postalcode} | ||||||
|                       {donor.address.city} |                       {donor.address.city} | ||||||
|                       {donor.address.country} |                       {donor.address.country} | ||||||
| @@ -127,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 | ||||||
| @@ -145,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"> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   (function () { |   (function () { | ||||||
|     document.onkeydown = function (e) { |     document.onkeydown = function (e) { | ||||||
| @@ -25,7 +25,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -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> | ||||||
|     - |     - | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|   import { _, getLocaleFromNavigator } from "svelte-i18n"; |   import { _, getLocaleFromNavigator } from "svelte-i18n"; | ||||||
|   import marked from "marked"; |   import marked from "marked"; | ||||||
|   import Footer from "./Footer.svelte"; |   import Footer from "./Footer.svelte"; | ||||||
|   import * as css from "../base/simple.css"; |   // import * as css from "../base/simple.css?inline"; | ||||||
|   let html = ""; |   let html = ""; | ||||||
|   async function load() { |   async function load() { | ||||||
|     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); |     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { UserGroupService } from "@odit/lfk-client-js"; |   import { UserGroupService } from "@odit/lfk-client-js"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
| @@ -69,7 +69,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -30,7 +30,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="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"> |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |   import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
| @@ -89,7 +89,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |   import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { createEventDispatcher } from "svelte"; |   import { createEventDispatcher } from "svelte"; | ||||||
| @@ -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(); | ||||||
| @@ -32,7 +32,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> |     on:click_outside={cancelDelete}> | ||||||
|     <div |     <div | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ | |||||||
|   import Select from "svelte-select"; |   import Select from "svelte-select"; | ||||||
|   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   import { tick } from "svelte"; |   import { tick } from "svelte"; | ||||||
|   $: delete_triggered = false; |   $: delete_triggered = false; | ||||||
|   $: address_valid_or_none = |   $: address_valid_or_none = | ||||||
| @@ -34,6 +35,7 @@ | |||||||
|   $: iscityvalid = editable.address?.city?.trim().length !== 0; |   $: iscityvalid = editable.address?.city?.trim().length !== 0; | ||||||
|   $: sponsoring_contracts_show = true; |   $: sponsoring_contracts_show = true; | ||||||
|   $: cards_show = true; |   $: cards_show = true; | ||||||
|  |   $: certificates_show = true; | ||||||
|   $: generate_orgs = [original_object]; |   $: generate_orgs = [original_object]; | ||||||
|   $: registrationLink = `${config.baseurl}/selfservice/register/${editable.registrationKey}`; |   $: registrationLink = `${config.baseurl}/selfservice/register/${editable.registrationKey}`; | ||||||
|   const getContactLabel = (option) => |   const getContactLabel = (option) => | ||||||
| @@ -117,9 +119,9 @@ | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   async function copy() { |   async function copy() { | ||||||
|     if(!editable.registrationKey){ |     if (!editable.registrationKey) { | ||||||
|       Toastify({ |       Toastify({ | ||||||
|         text: $_('you-have-to-save-your-changes-to-generate-a-link'), |         text: $_("you-have-to-save-your-changes-to-generate-a-link"), | ||||||
|         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%)", | ||||||
| @@ -165,7 +167,8 @@ | |||||||
|   passed_orgs={[]} |   passed_orgs={[]} | ||||||
|   passed_org={editable} |   passed_org={editable} | ||||||
|   opened_from="OrgDetail" |   opened_from="OrgDetail" | ||||||
|   bind:import_modal_open /> |   bind:import_modal_open | ||||||
|  | /> | ||||||
| <ConfirmOrgDeletion bind:modal_open bind:delete_org /> | <ConfirmOrgDeletion bind:modal_open bind:delete_org /> | ||||||
| {#if data_loaded} | {#if data_loaded} | ||||||
|   <section class="container p-5"> |   <section class="container p-5"> | ||||||
| @@ -174,28 +177,35 @@ | |||||||
|       <span data-id="org_actions_${editable.id}"> |       <span data-id="org_actions_${editable.id}"> | ||||||
|         <GenerateSponsoringContracts |         <GenerateSponsoringContracts | ||||||
|           bind:sponsoring_contracts_show |           bind:sponsoring_contracts_show | ||||||
|           bind:generate_orgs /> |           bind:generate_orgs | ||||||
|  |         /> | ||||||
|         <GenerateRunnerCards bind:cards_show bind:generate_orgs /> |         <GenerateRunnerCards bind:cards_show bind:generate_orgs /> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} |         <GenerateRunnerCertificates bind:certificates_show bind:generate_orgs /> | ||||||
|  |         {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:IMPORT")} | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               import_modal_open = true; |               import_modal_open = true; | ||||||
|             }} |             }} | ||||||
|             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" | ||||||
|             {$_('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} | ||||||
|               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-delete')}</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-delete")}</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} | ||||||
|           {#if !delete_triggered} |           {#if !delete_triggered} | ||||||
|             <button |             <button | ||||||
| @@ -203,7 +213,9 @@ | |||||||
|                 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-organization')}</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-organization")}</button | ||||||
|  |             > | ||||||
|           {/if} |           {/if} | ||||||
|         {/if} |         {/if} | ||||||
|         {#if !delete_triggered} |         {#if !delete_triggered} | ||||||
| @@ -212,7 +224,9 @@ | |||||||
|             disabled={!save_enabled} |             disabled={!save_enabled} | ||||||
|             class:opacity-50={!save_enabled} |             class:opacity-50={!save_enabled} | ||||||
|             type="button" |             type="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> |             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> | ||||||
| @@ -231,12 +245,13 @@ | |||||||
|                 class="h-3 w-3 stroke-current" |                 class="h-3 w-3 stroke-current" | ||||||
|                 height="1em" |                 height="1em" | ||||||
|                 width="1em" |                 width="1em" | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><path |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                   d="M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z" /> |                 ><path d="M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z" /> | ||||||
|                 <polyline points="9 22 9 12 15 12 15 22" /></svg> |                 <polyline points="9 22 9 12 15 12 15 22" /></svg | ||||||
|  |               > | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <a class="mr-2" href="/">{$_('home')}</a><svg |               <a class="mr-2" href="/">{$_("home")}</a><svg | ||||||
|                 stroke="currentColor" |                 stroke="currentColor" | ||||||
|                 fill="none" |                 fill="none" | ||||||
|                 stroke-width="2" |                 stroke-width="2" | ||||||
| @@ -246,24 +261,25 @@ | |||||||
|                 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="mr-2 flex items-center"> |             <li class="mr-2 flex items-center"> | ||||||
|               <svg |               <svg | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |                 viewBox="0 0 24 24" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   d="M21 20h2v2H1v-2h2V3a1 1 0 0 1 1-1h16a1 1 0 0 1 1 1v17zm-2 0V4H5v16h14zM8 11h3v2H8v-2zm0-4h3v2H8V7zm0 8h3v2H8v-2zm5 0h3v2h-3v-2zm0-4h3v2h-3v-2zm0-4h3v2h-3V7z" /></svg> |                   d="M21 20h2v2H1v-2h2V3a1 1 0 0 1 1-1h16a1 1 0 0 1 1 1v17zm-2 0V4H5v16h14zM8 11h3v2H8v-2zm0-4h3v2H8V7zm0 8h3v2H8v-2zm5 0h3v2h-3v-2zm0-4h3v2h-3v-2zm0-4h3v2h-3V7z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <a class="mr-2" href="./">{$_('organizations')}</a><svg |               <a class="mr-2" href="./">{$_("organizations")}</a><svg | ||||||
|                 stroke="currentColor" |                 stroke="currentColor" | ||||||
|                 fill="none" |                 fill="none" | ||||||
|                 stroke-width="2" |                 stroke-width="2" | ||||||
| @@ -273,12 +289,10 @@ | |||||||
|                 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">Org-Details #{params.orgid}</span> |               <span class="mr-2">Org-Details #{params.orgid}</span> | ||||||
| @@ -288,81 +302,88 @@ | |||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <div class="text-sm w-full"> |     <div class="text-sm w-full"> | ||||||
|       <label for="name" class="font-medium text-gray-700">{$_('name')}</label> |       <label for="name" class="font-medium text-gray-700">{$_("name")}</label> | ||||||
|       <input |       <input | ||||||
|         autocomplete="off" |         autocomplete="off" | ||||||
|         placeholder={$_('name')} |         placeholder={$_("name")} | ||||||
|         type="text" |         type="text" | ||||||
|         bind:value={editable.name} |         bind:value={editable.name} | ||||||
|         name="name" |         name="name" | ||||||
|         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="contact" class="font-medium text-gray-700" | ||||||
|         for="contact" |         >{$_("contact")}</label | ||||||
|         class="font-medium text-gray-700">{$_('contact')}</label> |       > | ||||||
|       <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.value.id.toString().startsWith(filterText.toLowerCase())} | ||||||
|               filterText.toLowerCase() |  | ||||||
|             ) || option.value.id |  | ||||||
|             .toString() |  | ||||||
|             .startsWith(filterText.toLowerCase())} |  | ||||||
|         items={contacts} |         items={contacts} | ||||||
|         showChevron={true} |         showChevron={true} | ||||||
|         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) => (editable.contact = selectedValue.detail.value)} |         on:select={(selectedValue) => | ||||||
|         on:clear={() => (editable.contact = null)} /> |           (editable.contact = selectedValue.detail.value)} | ||||||
|  |         on:clear={() => (editable.contact = null)} | ||||||
|  |       /> | ||||||
|     </div> |     </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.registrationEnabled} |             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> | ||||||
|         <div class="ml-3 text-sm"> |         <div class="ml-3 text-sm"> | ||||||
|           <label |           <label | ||||||
|             for="comments" |             for="toggle_selfservice_feature" | ||||||
|             class="font-medium text-gray-700">{$_('selfservice-registration')}</label> |             class="font-medium text-gray-700" | ||||||
|  |             >{$_("selfservice-registration")}</label | ||||||
|  |           > | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|       <div> |       <div> | ||||||
|         {#if editable.registrationEnabled} |         {#if editable.registrationEnabled} | ||||||
|           <div class="text-sm w-full"> |           <div class="text-sm w-full"> | ||||||
|             <div on:click={copy} class="inline-flex w-full"> |             <button on:click={copy} class="inline-flex w-full"> | ||||||
|               <p |               <p | ||||||
|                 name="token" |                 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"> |                 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} |                 {#if editable.registrationKey} | ||||||
|                 {registrationLink} |                   {registrationLink} | ||||||
|                 {:else} |                 {:else} | ||||||
|                 {$_('you-have-to-save-your-changes-to-generate-a-link')} |                   {$_("you-have-to-save-your-changes-to-generate-a-link")} | ||||||
|                 {/if} |                 {/if} | ||||||
|               </p> |               </p> | ||||||
|               <div |               <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"> |                 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 |                 <svg | ||||||
|                   xmlns="http://www.w3.org/2000/svg" |                   xmlns="http://www.w3.org/2000/svg" | ||||||
|                   viewBox="0 0 24 24" |                   viewBox="0 0 24 24" | ||||||
|                   width="24" |                   width="24" | ||||||
|                   height="24"><path fill="none" d="M0 0h24v24H0z" /> |                   height="24" | ||||||
|  |                   ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                   <path |                   <path | ||||||
|                     fill="currentColor" |                     fill="currentColor" | ||||||
|                     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> |             </button> | ||||||
|             {#if editable.registrationKey} |             {#if editable.registrationKey} | ||||||
|             <p class="text-gray-500 text-xs"> |               <p class="text-gray-500 text-xs"> | ||||||
|               {$_('click-to-copy-the-link-into-your-clipboard')} |                 {$_("click-to-copy-the-link-into-your-clipboard")} | ||||||
|             </p> |               </p> | ||||||
|             {/if} |             {/if} | ||||||
|           </div> |           </div> | ||||||
|         {/if} |         {/if} | ||||||
| @@ -372,15 +393,17 @@ | |||||||
|             <div class="flex items-center h-5"> |             <div class="flex items-center h-5"> | ||||||
|               <input |               <input | ||||||
|                 bind:checked={editable.address_checked} |                 bind:checked={editable.address_checked} | ||||||
|                 id="comments" |                 id="toggle_address_checkbox" | ||||||
|                 name="comments" |                 name="toggle_address_checkbox" | ||||||
|                 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> | ||||||
|             <div class="ml-3 text-sm"> |             <div class="ml-3 text-sm"> | ||||||
|               <label |               <label | ||||||
|                 for="comments" |                 for="toggle_address_checkbox" | ||||||
|                 class="font-medium text-gray-700">{$_('address')}</label> |                 class="font-medium text-gray-700">{$_("address")}</label | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
| @@ -388,7 +411,9 @@ | |||||||
|           <div class="col-span-6"> |           <div class="col-span-6"> | ||||||
|             <label |             <label | ||||||
|               for="address1" |               for="address1" | ||||||
|               class="block text-sm font-medium text-gray-700">{$_('address')}</label> |               class="block text-sm font-medium text-gray-700" | ||||||
|  |               >{$_("address")}</label | ||||||
|  |             > | ||||||
|             <input |             <input | ||||||
|               autocomplete="off" |               autocomplete="off" | ||||||
|               placeholder="Address" |               placeholder="Address" | ||||||
| @@ -398,65 +423,74 @@ | |||||||
|               bind:value={editable.address.address1} |               bind:value={editable.address.address1} | ||||||
|               type="text" |               type="text" | ||||||
|               name="address1" |               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" /> |               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} |             {#if !isAddress1Valid} | ||||||
|               <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" | ||||||
|                 {$_('address-is-required')} |               > | ||||||
|  |                 {$_("address-is-required")} | ||||||
|               </span> |               </span> | ||||||
|             {/if} |             {/if} | ||||||
|           </div> |           </div> | ||||||
|           <div class="col-span-6"> |           <div class="col-span-6"> | ||||||
|             <label |             <label | ||||||
|               for="address2" |               for="address2" | ||||||
|               class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> |               class="block text-sm font-medium text-gray-700" | ||||||
|  |               >{$_("apartment-suite-etc")}</label | ||||||
|  |             > | ||||||
|             <input |             <input | ||||||
|               autocomplete="off" |               autocomplete="off" | ||||||
|               placeholder={$_('apartment-suite-etc')} |               placeholder={$_("apartment-suite-etc")} | ||||||
|               bind:value={editable.address.address2} |               bind:value={editable.address.address2} | ||||||
|               type="text" |               type="text" | ||||||
|               name="address2" |               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" /> |               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="col-span-6"> |           <div class="col-span-6"> | ||||||
|             <label |             <label for="zipcode" class="block text-sm font-medium text-gray-700" | ||||||
|               for="zipcode" |               >{$_("zip-postal-code")}</label | ||||||
|               class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> |             > | ||||||
|             <input |             <input | ||||||
|               autocomplete="off" |               autocomplete="off" | ||||||
|               placeholder={$_('zip-postal-code')} |               placeholder={$_("zip-postal-code")} | ||||||
|               class:border-red-500={!iszipcodevalid} |               class:border-red-500={!iszipcodevalid} | ||||||
|               class:focus:border-red-500={!iszipcodevalid} |               class:focus:border-red-500={!iszipcodevalid} | ||||||
|               class:focus:ring-red-500={!iszipcodevalid} |               class:focus:ring-red-500={!iszipcodevalid} | ||||||
|               bind:value={editable.address.postalcode} |               bind:value={editable.address.postalcode} | ||||||
|               type="text" |               type="text" | ||||||
|               name="zipcode" |               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" /> |               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} |             {#if !iszipcodevalid} | ||||||
|               <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-zipcode-postal-code-is-required')} |               > | ||||||
|  |                 {$_("valid-zipcode-postal-code-is-required")} | ||||||
|               </span> |               </span> | ||||||
|             {/if} |             {/if} | ||||||
|           </div> |           </div> | ||||||
|           <div class="col-span-6"> |           <div class="col-span-6"> | ||||||
|             <label |             <label for="city" class="block text-sm font-medium text-gray-700" | ||||||
|               for="city" |               >{$_("city")}</label | ||||||
|               class="block text-sm font-medium text-gray-700">{$_('city')}</label> |             > | ||||||
|             <input |             <input | ||||||
|               autocomplete="off" |               autocomplete="off" | ||||||
|               placeholder={$_('city')} |               placeholder={$_("city")} | ||||||
|               class:border-red-500={!iscityvalid} |               class:border-red-500={!iscityvalid} | ||||||
|               class:focus:border-red-500={!iscityvalid} |               class:focus:border-red-500={!iscityvalid} | ||||||
|               class:focus:ring-red-500={!iscityvalid} |               class:focus:ring-red-500={!iscityvalid} | ||||||
|               bind:value={editable.address.city} |               bind:value={editable.address.city} | ||||||
|               type="text" |               type="text" | ||||||
|               name="city" |               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" /> |               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} |             {#if !iscityvalid} | ||||||
|               <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-city-is-required')} |               > | ||||||
|  |                 {$_("valid-city-is-required")} | ||||||
|               </span> |               </span> | ||||||
|             {/if} |             {/if} | ||||||
|           </div> |           </div> | ||||||
| @@ -466,7 +500,7 @@ | |||||||
|   </section> |   </section> | ||||||
| {:else} | {:else} | ||||||
|   {#await promise} |   {#await promise} | ||||||
|     {$_('organization-detail-is-being-loaded')} |     {$_("organization-detail-is-being-loaded")} | ||||||
|   {:catch error} |   {:catch error} | ||||||
|     <PromiseError /> |     <PromiseError /> | ||||||
|   {/await} |   {/await} | ||||||
|   | |||||||
| @@ -1,212 +1,219 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; |   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"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   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 GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|   $: searchvalue = ""; |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: active_deletes = []; |   $: searchvalue = ""; | ||||||
|   $: sponsoring_contracts_show = current_organizations.some((r) => r.is_selected === true); |   $: active_deletes = []; | ||||||
|   $: cards_show = current_organizations.some((r) => r.is_selected === true); |   $: sponsoring_contracts_show = current_organizations.some((r) => r.is_selected === true); | ||||||
|   $: generate_orgs = current_organizations.filter((r) => r.is_selected === true); |   $: cards_show = current_organizations.some((r) => r.is_selected === true); | ||||||
|   export let current_organizations = []; |   $: generate_orgs = current_organizations.filter((r) => r.is_selected === true); | ||||||
|  |   $: certificates_show = current_organizations.some( | ||||||
|   const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then( |     (r) => r.is_selected === true | ||||||
|     (val) => { |   ); | ||||||
|       current_organizations = val; |   export let current_organizations = []; | ||||||
|     } |  | ||||||
|   ); |   const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
| </script> |     (val) => { | ||||||
|  |       current_organizations = val; | ||||||
| <ConfirmOrgDeletion |     } | ||||||
|   on:cancelDelete={(event) => { |   ); | ||||||
|     modal_open = false; | </script> | ||||||
|     active_deletes[event.detail.id] = false; |  | ||||||
|   }} | <ConfirmOrgDeletion | ||||||
|   bind:modal_open |   on:cancelDelete={(event) => { | ||||||
|   bind:delete_org /> |     modal_open = false; | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:GET')} |     active_deletes[event.detail.id] = false; | ||||||
|   {#await promise} |   }} | ||||||
|     <div |   bind:modal_open | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |   bind:delete_org /> | ||||||
|       role="alert"> | {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:GET')} | ||||||
|       <p class="font-bold">{$_('organizations-are-being-loaded')}</p> |   {#await promise} | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |     <div | ||||||
|     </div> |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|   {:then} |       role="alert"> | ||||||
|     {#if current_organizations.length === 0} |       <p class="font-bold">{$_('organizations-are-being-loaded')}</p> | ||||||
|       <OrgsEmptyState /> |       <p class="text-sm">{$_('this-might-take-a-moment')}</p> | ||||||
|     {:else} |     </div> | ||||||
|       <input |   {:then} | ||||||
|         type="search" |     {#if current_organizations.length === 0} | ||||||
|         bind:value={searchvalue} |       <OrgsEmptyState /> | ||||||
|         placeholder={$_('datatable.search')} |     {:else} | ||||||
|         aria-label={$_('datatable.search')} |       <input | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         type="search" | ||||||
|       <div class="h-12"> |         bind:value={searchvalue} | ||||||
|         <GenerateSponsoringContracts |         placeholder={$_('datatable.search')} | ||||||
|             bind:sponsoring_contracts_show |         aria-label={$_('datatable.search')} | ||||||
|             bind:generate_orgs /> |         class="mb-4" /> | ||||||
|         <GenerateRunnerCards |       <div class="h-12"> | ||||||
|             bind:cards_show |         <GenerateSponsoringContracts | ||||||
|             bind:generate_orgs /> |             bind:sponsoring_contracts_show | ||||||
|       </div> |             bind:generate_orgs /> | ||||||
|       <div |         <GenerateRunnerCards | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |             bind:cards_show | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |             bind:generate_orgs /> | ||||||
|           <thead class="bg-gray-50"> |         <GenerateRunnerCertificates | ||||||
|             <tr> |             bind:certificates_show | ||||||
|               <th |             bind:generate_orgs /> | ||||||
|                 scope="col" |       </div> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |       <div | ||||||
|                 <span |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> | ||||||
|                   on:click={() => { |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|                     const newstate = !current_organizations.some((r) => r.is_selected === true); |           <thead class="bg-gray-50"> | ||||||
|                     current_organizations = current_organizations.map((r) => { |             <tr> | ||||||
|                       r.is_selected = newstate; |               <th | ||||||
|                       return r; |                 scope="col" | ||||||
|                     }); |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|                   }} |                 <span | ||||||
|                   class="underline cursor-pointer select-none">{#if current_organizations.some((r) => r.is_selected === true)} |                   on:click={() => { | ||||||
|                     {$_('deselect-all')} |                     const newstate = !current_organizations.some((r) => r.is_selected === true); | ||||||
|                   {:else}{$_('select-all')}{/if} |                     current_organizations = current_organizations.map((r) => { | ||||||
|                 </span> |                       r.is_selected = newstate; | ||||||
|               </th> |                       return r; | ||||||
|               <th |                     }); | ||||||
|                 scope="col" |                   }} | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   class="underline cursor-pointer select-none">{#if current_organizations.some((r) => r.is_selected === true)} | ||||||
|                 {$_('name')} |                     {$_('deselect-all')} | ||||||
|               </th> |                   {:else}{$_('select-all')}{/if} | ||||||
|               <th |                 </span> | ||||||
|                 scope="col" |               </th> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th | ||||||
|                 {$_('address')} |                 scope="col" | ||||||
|               </th> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|               <th |                 {$_('name')} | ||||||
|                 scope="col" |               </th> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th | ||||||
|                 {$_('contact')} |                 scope="col" | ||||||
|               </th> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |                 {$_('address')} | ||||||
|                 <span class="sr-only">{$_('action')}</span> |               </th> | ||||||
|               </th> |               <th | ||||||
|             </tr> |                 scope="col" | ||||||
|           </thead> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|           <tbody class="divide-y divide-gray-200"> |                 {$_('contact')} | ||||||
|             {#each current_organizations as o} |               </th> | ||||||
|               {#if Object.values(o) |               <th scope="col" class="relative px-6 py-3"> | ||||||
|                 .toString() |                 <span class="sr-only">{$_('action')}</span> | ||||||
|                 .toLowerCase() |               </th> | ||||||
|                 .includes(searchvalue)} |             </tr> | ||||||
|                 <tr data-rowid="org_{o.id}"> |           </thead> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |           <tbody class="divide-y divide-gray-200"> | ||||||
|                     <input |             {#each current_organizations as o} | ||||||
|                       bind:checked={o.is_selected} |               {#if Object.values(o) | ||||||
|                       type="checkbox" |                 .toString() | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |                 .toLowerCase() | ||||||
|                   </td> |                 .includes(searchvalue)} | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                 <tr data-rowid="org_{o.id}"> | ||||||
|                     <div class="flex items-center"> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                       <div class="ml-4"> |                     <input | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                       bind:checked={o.is_selected} | ||||||
|                           {o.name} |                       type="checkbox" | ||||||
|                         </div> |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | ||||||
|                       </div> |                   </td> | ||||||
|                     </div> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                   </td> |                     <div class="flex items-center"> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                       <div class="ml-4"> | ||||||
|                     <div class="flex items-center"> |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                       <div class="ml-4"> |                           {o.name} | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                         </div> | ||||||
|                           {#if o.address.address1 !== null} |                       </div> | ||||||
|                             {o.address.address1}<br /> |                     </div> | ||||||
|                             {o.address.address2 || ''}<br /> |                   </td> | ||||||
|                             {o.address.postalcode} |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                             {o.address.city} |                     <div class="flex items-center"> | ||||||
|                             {o.address.country} |                       <div class="ml-4"> | ||||||
|                           {/if} |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                         </div> |                           {#if o.address.address1 !== null} | ||||||
|                       </div> |                             {o.address.address1}<br /> | ||||||
|                     </div> |                             <!-- {o.address.address2 || ''}<br /> --> | ||||||
|                   </td> |                             {o.address.postalcode} | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                             {o.address.city} | ||||||
|                     <div class="flex items-center"> |                             {o.address.country} | ||||||
|                       <div class="ml-4"> |                           {/if} | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                         </div> | ||||||
|                           {#if o.contact} |                       </div> | ||||||
|                             <a |                     </div> | ||||||
|                               href="../contacts/{o.contact.id}" |                   </td> | ||||||
|                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{o.contact.firstname} |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                               {o.contact.middlename || ''} |                     <div class="flex items-center"> | ||||||
|                               {o.contact.lastname}</a> |                       <div class="ml-4"> | ||||||
|                           {:else}{$_('no-contact-specified')}{/if} |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                         </div> |                           {#if o.contact} | ||||||
|                       </div> |                             <a | ||||||
|                     </div> |                               href="../contacts/{o.contact.id}" | ||||||
|                   </td> |                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{o.contact.firstname} | ||||||
|                   {#if active_deletes[o.id] === true} |                               {o.contact.middlename || ''} | ||||||
|                     <td |                               {o.contact.lastname}</a> | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                           {:else}{$_('no-contact-specified')}{/if} | ||||||
|                       <button |                         </div> | ||||||
|                         on:click={() => { |                       </div> | ||||||
|                           active_deletes[o.id] = false; |                     </div> | ||||||
|                         }} |                   </td> | ||||||
|                         tabindex="0" |                   {#if active_deletes[o.id] === true} | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |                     <td | ||||||
|                       <button |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|                         on:click={() => { |                       <button | ||||||
|                           RunnerOrganizationService.runnerOrganizationControllerRemove(o.id, false) |                         on:click={() => { | ||||||
|                             .then((resp) => { |                           active_deletes[o.id] = false; | ||||||
|                               current_organizations = current_organizations.filter((obj) => obj.id !== o.id); |                         }} | ||||||
|                               Toastify({ |                         tabindex="0" | ||||||
|                                 text: 'Organization deleted', |                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> | ||||||
|                                 duration: 500, |                       <button | ||||||
|                                 backgroundColor: |                         on:click={() => { | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |                           RunnerOrganizationService.runnerOrganizationControllerRemove(o.id, false) | ||||||
|                               }).showToast(); |                             .then((resp) => { | ||||||
|                             }) |                               current_organizations = current_organizations.filter((obj) => obj.id !== o.id); | ||||||
|                             .catch((err) => { |                               Toastify({ | ||||||
|                               modal_open = true; |                                 text: 'Organization deleted', | ||||||
|                               delete_org = o; |                                 duration: 500, | ||||||
|                             }); |                                 backgroundColor: | ||||||
|                         }} |                                   'linear-gradient(to right, #00b09b, #96c93d)', | ||||||
|                         tabindex="0" |                               }).showToast(); | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |                             }) | ||||||
|                     </td> |                             .catch((err) => { | ||||||
|                   {:else} |                               modal_open = true; | ||||||
|                     <td |                               delete_org = o; | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                             }); | ||||||
|                       <a |                         }} | ||||||
|                         href="./{o.id}" |                         tabindex="0" | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:DELETE')} |                     </td> | ||||||
|                         <button |                   {:else} | ||||||
|                           on:click={() => { |                     <td | ||||||
|                             active_deletes[o.id] = true; |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|                           }} |                       <a | ||||||
|                           tabindex="0" |                         href="./{o.id}" | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> | ||||||
|                       {/if} |                       {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:DELETE')} | ||||||
|                     </td> |                         <button | ||||||
|                   {/if} |                           on:click={() => { | ||||||
|                 </tr> |                             active_deletes[o.id] = true; | ||||||
|               {/if} |                           }} | ||||||
|             {/each} |                           tabindex="0" | ||||||
|           </tbody> |                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> | ||||||
|         </table> |                       {/if} | ||||||
|       </div> |                     </td> | ||||||
|     {/if} |                   {/if} | ||||||
|   {:catch error} |                 </tr> | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |               {/if} | ||||||
|       <span class="inline-block align-middle mr-8"> |             {/each} | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |           </tbody> | ||||||
|         {error} |         </table> | ||||||
|       </span> |       </div> | ||||||
|     </div> |     {/if} | ||||||
|   {/await} |   {:catch error} | ||||||
| {/if} |     <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,339 +1,418 @@ | |||||||
| <script> | <script> | ||||||
|     import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|     import { |   import { | ||||||
|         RunnerCardService, |     RunnerCardService, | ||||||
|         RunnerOrganizationService, |     RunnerOrganizationService, | ||||||
|         RunnerTeamService, |     RunnerTeamService, | ||||||
|     } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|     import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|     export let cards_show = false; |   import { init } from "@paralleldrive/cuid2"; | ||||||
|     export let generate_cards = []; |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|     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) { |   export let cards_show = false; | ||||||
|         cards_dropdown_open = 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; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|         if (generate_orgs.length > 0) { |   function generateRunnerCards(locale) { | ||||||
|             generateOrgCards(locale); |     cards_dropdown_open = false; | ||||||
|         } else if (generate_teams.length > 0) { |  | ||||||
|             generateTeamCards(locale); |     if (generate_orgs.length > 0) { | ||||||
|         } else if (generate_runners.length > 0) { |       generateOrgCards(locale); | ||||||
|             generateRunnersCards(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 { |         } else { | ||||||
|             generateCards(locale); |           return response.blob(); | ||||||
|         } |         } | ||||||
|     } |       }) | ||||||
|  |       .then((blob) => { | ||||||
|     function generateCards(locale) { |         const url = window.URL.createObjectURL(blob); | ||||||
|         const toast = Toastify({ |         let a = document.createElement("a"); | ||||||
|             text: $_("generating-pdf"), |         a.href = url; | ||||||
|             duration: -1, |         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(); |         }).showToast(); | ||||||
|         fetch( |       }) | ||||||
|             `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, |       .catch((err) => { | ||||||
|             { |         console.error(err); | ||||||
|                 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.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) { |   async function generateRunnersCards(locale) { | ||||||
|         const toast = Toastify({ |     const toast = Toastify({ | ||||||
|             text: $_("generating-pdf"), |       text: $_("generating-pdf"), | ||||||
|             duration: -1, |       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(); |         }).showToast(); | ||||||
|         const current_cards = await RunnerCardService.runnerCardControllerGetAll(); |       }) | ||||||
|  |       .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 = []; |         let cards = []; | ||||||
|         for (let runner of generate_runners) { |         for (let runner of runners) { | ||||||
|             let card = current_cards.find((c) => c.runner?.id == runner.id); |           let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|             if (!card) { |           if (!card) { | ||||||
|                 card = await RunnerCardService.runnerCardControllerPost({ |             card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|                     runner: runner.id, |               runner: runner.id, | ||||||
|                 }); |             }); | ||||||
|             } |           } | ||||||
|             cards.push(card); |           cards.push(card); | ||||||
|         } |         } | ||||||
|         fetch( |         await fetch( | ||||||
|             `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, |           `${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|             { |           { | ||||||
|                 method: "POST", |             method: "POST", | ||||||
|                 headers: { |             headers: { | ||||||
|                     "Content-Type": "application/json", |               "Content-Type": "application/json", | ||||||
|                 }, |             }, | ||||||
|                 body: JSON.stringify(cards), |             body: JSON.stringify(cards), | ||||||
|             } |           } | ||||||
|         ) |         ) | ||||||
|             .then((response) => { |           .then((response) => { | ||||||
|                 if (response.status != "200") { |             if (response.status != "200") { | ||||||
|                     toast.hideToast(); |               toast.hideToast(); | ||||||
|                     Toastify({ |               Toastify({ | ||||||
|                         text: $_("pdf-generation-failed"), |                 text: $_("pdf-generation-failed"), | ||||||
|                         duration: 3500, |                 duration: 3500, | ||||||
|                         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%)", | ||||||
|                     }).showToast(); |               }).showToast(); | ||||||
|                 } else { |             } else { | ||||||
|                     return response.blob(); |               return response.blob(); | ||||||
|                 } |  | ||||||
|             }) |  | ||||||
|             .then((blob) => { |  | ||||||
|                 const url = window.URL.createObjectURL(blob); |  | ||||||
|                 let a = document.createElement("a"); |  | ||||||
|                 a.href = url; |  | ||||||
|                 a.download = "Runnercards.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}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, |           .then((blob) => { | ||||||
|                 { |             const url = window.URL.createObjectURL(blob); | ||||||
|                     method: "POST", |             let a = document.createElement("a"); | ||||||
|                     headers: { |             a.href = url; | ||||||
|                         "Content-Type": "application/json", |             a.download = `${$_("runnercards")}_${o.name}_${ | ||||||
|                     }, |               t.name | ||||||
|                     body: JSON.stringify(cards), |             }-${locale}-${createId()}.pdf`; | ||||||
|                 } |             document.body.appendChild(a); | ||||||
|             ) |             a.click(); | ||||||
|                 .then((response) => { |             a.remove(); | ||||||
|                     if (response.status != "200") { |             if ( | ||||||
|                         toast.hideToast(); |               count === o.teams.length && | ||||||
|                         Toastify({ |               count_orgs === generate_orgs.length | ||||||
|                             text: $_("pdf-generation-failed"), |             ) { | ||||||
|                             duration: 3500, |               toast.hideToast(); | ||||||
|                             backgroundColor: |               Toastify({ | ||||||
|                                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |                 text: $_("pdfs-successfully-generated"), | ||||||
|                         }).showToast(); |                 duration: 3500, | ||||||
|                     } else { |                 backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|                         return response.blob(); |               }).showToast(); | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|                 .then((blob) => { |  | ||||||
|                     count++; |  | ||||||
|                     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 === 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-pdf"), |  | ||||||
|             duration: -1, |  | ||||||
|         }).showToast(); |  | ||||||
|         let count = 0; |  | ||||||
|         const current_cards = await RunnerCardService.runnerCardControllerGetAll(); |  | ||||||
|         for (const o of generate_orgs) { |  | ||||||
|             const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |  | ||||||
|                 o.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}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`, |           .catch((err) => {}); | ||||||
|                 { |       } | ||||||
|                     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 = "Sponsorings_" + o.name + ".pdf"; |  | ||||||
|                     document.body.appendChild(a); |  | ||||||
|                     a.click(); |  | ||||||
|                     a.remove(); |  | ||||||
|                     if (count === generate_orgs.length) { |  | ||||||
|                         toast.hideToast(); |  | ||||||
|                         Toastify({ |  | ||||||
|                             text: $_("pdfs-successfully-generated"), |  | ||||||
|                             duration: 3500, |  | ||||||
|                             backgroundColor: |  | ||||||
|                                 "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|                         }).showToast(); |  | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|                 .catch((err) => {}); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if cards_show} | {#if cards_show} | ||||||
|     <div id="cards:dropdown" class="relative inline-block"> |   <div id="cards:dropdown" class="relative inline-block"> | ||||||
|         <div> |     <div> | ||||||
|             <button |       <button | ||||||
|                 on:click={() => { |         on:click={() => { | ||||||
|                     cards_dropdown_open = !cards_dropdown_open; |           cards_dropdown_open = !cards_dropdown_open; | ||||||
|                 }} |         }} | ||||||
|                 type="button" |         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" |         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" |         id="options-menu" | ||||||
|                 aria-haspopup="true" |         aria-haspopup="true" | ||||||
|                 aria-expanded="true"> |         aria-expanded="true" | ||||||
|                 {$_('generate-runnercards')} |       > | ||||||
|                 <svg |         {$_("generate-runnercards")} | ||||||
|                     xmlns="http://www.w3.org/2000/svg" |         <svg | ||||||
|                     width="24" |           xmlns="http://www.w3.org/2000/svg" | ||||||
|                     height="24" |           width="24" | ||||||
|                     viewBox="0 0 24 24" |           height="24" | ||||||
|                     class="-mr-1 ml-2 h-5 w-5"><path |           viewBox="0 0 24 24" | ||||||
|                         fill="none" |           class="-mr-1 ml-2 h-5 w-5" | ||||||
|                         d="M0 0h24v24H0z" /> |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                     <path |           <path | ||||||
|                         fill="currentColor" |             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> |             d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" | ||||||
|             </button> |           /></svg | ||||||
|         </div> |         > | ||||||
|         {#if cards_dropdown_open} |       </button> | ||||||
|             <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="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> |     </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} | {/if} | ||||||
|   | |||||||
| @@ -1,273 +1,355 @@ | |||||||
| <script> | <script> | ||||||
|     import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|     import { |   import { | ||||||
|         DonationService, |     DonationService, | ||||||
|         RunnerTeamService, |     RunnerTeamService, | ||||||
|         RunnerOrganizationService |     RunnerOrganizationService, | ||||||
|     } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|     import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|     export let certificates_show = false; |   import { init } from "@paralleldrive/cuid2"; | ||||||
|     export let generate_runners = []; |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|     export let generate_orgs = []; |  | ||||||
|     export let generate_teams = []; |   export let certificates_show = false; | ||||||
|     $: certificates_dropdown_open = false; |   export let generate_runners = []; | ||||||
|     document.addEventListener("click", function (e) { |   export let generate_orgs = []; | ||||||
|         if ( |   export let generate_teams = []; | ||||||
|             e.target.parentNode?.parentNode?.id != "certificates:dropdown" && |   $: certificates_dropdown_open = false; | ||||||
|             e.target.parentNode?.parentNode?.id != "certificates:dropdown:menu" |   document.addEventListener("click", function (e) { | ||||||
|         ) { |     if ( | ||||||
|             certificates_dropdown_open = false; |       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) { |   function generateCertificates(locale) { | ||||||
|             generateOrgCertificates(locale); |     certificates_dropdown_open = false; | ||||||
|         } else if (generate_teams.length > 0) { |  | ||||||
|             generateTeamCertificates(locale); |     if (generate_orgs.length > 0) { | ||||||
|         } else { |       generateOrgCertificates(locale); | ||||||
|             generateRunnerCertificates(locale); |     } else if (generate_teams.length > 0) { | ||||||
|         } |       generateTeamCertificates(locale); | ||||||
|     } |     } else { | ||||||
|  |       generateRunnerCertificates(locale); | ||||||
|     async function generateRunnerCertificates(locale) { |     } | ||||||
|         const toast = Toastify({ |   } | ||||||
|             text: $_("generating-pdf"), |  | ||||||
|             duration: -1, |   async function generateRunnerCertificates(locale) { | ||||||
|         }).showToast(); |     const toast = Toastify({ | ||||||
|         const current_donations = await DonationService.donationControllerGetAll(); |       text: $_("generating-pdf"), | ||||||
|         let certificateRunners = []; |       duration: -1, | ||||||
|         for (let runner of generate_runners) { |     }).showToast(); | ||||||
|             runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; |     const current_donations = | ||||||
|             certificateRunners.push(runner); |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|         } |     let certificateRunners = []; | ||||||
|         fetch( |     for (let runner of generate_runners) { | ||||||
|             `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, |       runner.distanceDonations = | ||||||
|             { |         current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|                 method: "POST", |       console.log(runner.distanceDonations); | ||||||
|                 headers: { |       certificateRunners.push(runner); | ||||||
|                     "Content-Type": "application/json", |     } | ||||||
|                 }, |     fetch( | ||||||
|                 body: JSON.stringify(certificateRunners), |       `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|             } |       { | ||||||
|         ) |         method: "POST", | ||||||
|             .then((response) => { |         headers: { | ||||||
|                 if (response.status != "200") { |           "Content-Type": "application/json", | ||||||
|                     toast.hideToast(); |         }, | ||||||
|                     Toastify({ |         body: JSON.stringify(certificateRunners), | ||||||
|                         text: $_("pdf-generation-failed"), |       } | ||||||
|                         duration: 3500, |     ) | ||||||
|                         backgroundColor: |       .then((response) => { | ||||||
|                             "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |         if (response.status != "200") { | ||||||
|                     }).showToast(); |           toast.hideToast(); | ||||||
|                 } else { |           Toastify({ | ||||||
|                     return response.blob(); |             text: $_("pdf-generation-failed"), | ||||||
|                 } |             duration: 3500, | ||||||
|             }) |             backgroundColor: | ||||||
|             .then((blob) => { |               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|                 const url = window.URL.createObjectURL(blob); |           }).showToast(); | ||||||
|                 let a = document.createElement("a"); |         } else { | ||||||
|                 a.href = url; |           return response.blob(); | ||||||
|                 a.download = "Certificates.pdf"; |         } | ||||||
|                 document.body.appendChild(a); |       }) | ||||||
|                 a.click(); |       .then((blob) => { | ||||||
|                 a.remove(); |         const url = window.URL.createObjectURL(blob); | ||||||
|                 toast.hideToast(); |         let a = document.createElement("a"); | ||||||
|                 Toastify({ |         a.href = url; | ||||||
|                     text: $_("pdf-successfully-generated"), |         if (generate_runners.length == 1) { | ||||||
|                     duration: 3500, |           a.download = `${$_("certificates")}_${ | ||||||
|                     backgroundColor: |             generate_runners[0].firstname | ||||||
|                         "linear-gradient(to right, #00b09b, #96c93d)", |           }_${generate_runners[0].lastname}-${locale}-${createId()}.pdf`; | ||||||
|                 }).showToast(); |         } else { | ||||||
|             }) |           a.download = `${$_("certificates")}-${locale}.pdf`; | ||||||
|             .catch((err) => {}); |         } | ||||||
|     } |         document.body.appendChild(a); | ||||||
|  |         a.click(); | ||||||
|     async function generateTeamCertificates(locale) { |         a.remove(); | ||||||
|         const toast = Toastify({ |         toast.hideToast(); | ||||||
|             text: $_("generating-pdfs"), |         Toastify({ | ||||||
|             duration: -1, |           text: $_("pdf-successfully-generated"), | ||||||
|         }).showToast(); |           duration: 3500, | ||||||
|         let count = 0; |           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         const current_donations = await DonationService.donationControllerGetAll(); |         }).showToast(); | ||||||
|         for (const t of generate_teams) { |       }) | ||||||
|             const runners = await RunnerTeamService.runnerTeamControllerGetRunners( |       .catch((err) => {}); | ||||||
|                 t.id |   } | ||||||
|                 ); |  | ||||||
|             let certificateRunners = []; |   async function generateTeamCertificates(locale) { | ||||||
|             for (let runner of runners) { |     const toast = Toastify({ | ||||||
|                 runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; |       text: $_("generating-pdfs"), | ||||||
|                 certificateRunners.push(runner); |       duration: -1, | ||||||
|             } |     }).showToast(); | ||||||
|             console.log(certificateRunners) |     let count = 0; | ||||||
|             fetch( |     const current_donations = | ||||||
|                 `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|                 { |     for (const t of generate_teams) { | ||||||
|                     method: "POST", |       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|                     headers: { |         t.id | ||||||
|                         "Content-Type": "application/json", |       ); | ||||||
|                     }, |       let certificateRunners = []; | ||||||
|                     body: JSON.stringify(certificateRunners), |       for (let runner of runners) { | ||||||
|                 } |         runner.distanceDonations = | ||||||
|             ) |           current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|                 .then((response) => { |         certificateRunners.push(runner); | ||||||
|                     if (response.status != "200") { |       } | ||||||
|                         toast.hideToast(); |       fetch( | ||||||
|                         Toastify({ |         `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|                             text: $_("pdf-generation-failed"), |         { | ||||||
|                             duration: 3500, |           method: "POST", | ||||||
|                             backgroundColor: |           headers: { | ||||||
|                                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |             "Content-Type": "application/json", | ||||||
|                         }).showToast(); |           }, | ||||||
|                     } else { |           body: JSON.stringify(certificateRunners), | ||||||
|                         return response.blob(); |         } | ||||||
|                     } |       ) | ||||||
|                 }) |         .then((response) => { | ||||||
|                 .then((blob) => { |           if (response.status != "200") { | ||||||
|                     count++; |             toast.hideToast(); | ||||||
|                     const url = window.URL.createObjectURL(blob); |             Toastify({ | ||||||
|                     let a = document.createElement("a"); |               text: $_("pdf-generation-failed"), | ||||||
|                     a.href = url; |               duration: 3500, | ||||||
|                     a.download = "Certificates_" + t.name + ".pdf"; |               backgroundColor: | ||||||
|                     document.body.appendChild(a); |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|                     a.click(); |             }).showToast(); | ||||||
|                     a.remove(); |           } else { | ||||||
|                     if (count === generate_teams.length) { |             return response.blob(); | ||||||
|                         toast.hideToast(); |           } | ||||||
|                         Toastify({ |         }) | ||||||
|                             text: $_("pdfs-successfully-generated"), |         .then((blob) => { | ||||||
|                             duration: 3500, |           count++; | ||||||
|                             backgroundColor: |           const url = window.URL.createObjectURL(blob); | ||||||
|                                 "linear-gradient(to right, #00b09b, #96c93d)", |           let a = document.createElement("a"); | ||||||
|                         }).showToast(); |           a.href = url; | ||||||
|                     } |           a.download = `${$_("certificates")}_${t.name}-${locale}-${createId()}.pdf`; | ||||||
|                 }) |           document.body.appendChild(a); | ||||||
|                 .catch((err) => {}); |           a.click(); | ||||||
|         } |           a.remove(); | ||||||
|     } |           if (count === generate_teams.length) { | ||||||
|  |             toast.hideToast(); | ||||||
|     async function generateOrgCertificates(locale) { |             Toastify({ | ||||||
|         const toast = Toastify({ |               text: $_("pdfs-successfully-generated"), | ||||||
|             text: $_("generating-pdf"), |               duration: 3500, | ||||||
|             duration: -1, |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|         }).showToast(); |             }).showToast(); | ||||||
|         let count = 0; |           } | ||||||
|         const current_donations = await DonationService.donationControllerGetAll(); |         }) | ||||||
|         for (const o of generate_orgs) { |         .catch((err) => {}); | ||||||
|             const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |     } | ||||||
|                 o.id |   } | ||||||
|             ); |  | ||||||
|             let certificateRunners = []; |   async function generateOrgCertificates(locale) { | ||||||
|             for (let runner of runners) { |     const toast = Toastify({ | ||||||
|                 runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; |       text: $_("generating-pdfs"), | ||||||
|                 certificateRunners.push(runner); |       duration: -1, | ||||||
|             } |     }).showToast(); | ||||||
|             fetch( |     const current_donations = | ||||||
|                 `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, |       (await DonationService.donationControllerGetAll()) || []; | ||||||
|                 { |     let count = 0; | ||||||
|                     method: "POST", |     let count_orgs = 0; | ||||||
|                     headers: { |     for (const o of generate_orgs) { | ||||||
|                         "Content-Type": "application/json", |       count_orgs++; | ||||||
|                     }, |       let count = 0; | ||||||
|                     body: JSON.stringify(certificateRunners), |       let runners = | ||||||
|                 } |         await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|             ) |           o.id, | ||||||
|                 .then((response) => { |           true | ||||||
|                     if (response.status != "200") { |         ); | ||||||
|                         toast.hideToast(); |       let certificateRunners = []; | ||||||
|                         Toastify({ |       for (let runner of runners) { | ||||||
|                             text: $_("pdf-generation-failed"), |         runner.distanceDonations = | ||||||
|                             duration: 3500, |           current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|                             backgroundColor: |         certificateRunners.push(runner); | ||||||
|                                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |       } | ||||||
|                         }).showToast(); |       await fetch( | ||||||
|                     } else { |         `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|                         return response.blob(); |         { | ||||||
|                     } |           method: "POST", | ||||||
|                 }) |           headers: { | ||||||
|                 .then((blob) => { |             "Content-Type": "application/json", | ||||||
|                     count++; |           }, | ||||||
|                     const url = window.URL.createObjectURL(blob); |           body: JSON.stringify(certificateRunners), | ||||||
|                     let a = document.createElement("a"); |         } | ||||||
|                     a.href = url; |       ) | ||||||
|                     a.download = "Certificates_" + o.name + ".pdf"; |         .then((response) => { | ||||||
|                     document.body.appendChild(a); |           if (response.status != "200") { | ||||||
|                     a.click(); |             toast.hideToast(); | ||||||
|                     a.remove(); |             Toastify({ | ||||||
|                     if (count === generate_orgs.length) { |               text: $_("pdf-generation-failed"), | ||||||
|                         toast.hideToast(); |               duration: 3500, | ||||||
|                         Toastify({ |               backgroundColor: | ||||||
|                             text: $_("pdfs-successfully-generated"), |                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|                             duration: 3500, |             }).showToast(); | ||||||
|                             backgroundColor: |           } else { | ||||||
|                                 "linear-gradient(to right, #00b09b, #96c93d)", |             return response.blob(); | ||||||
|                         }).showToast(); |           } | ||||||
|                     } |         }) | ||||||
|                 }) |         .then((blob) => { | ||||||
|                 .catch((err) => {}); |           const url = window.URL.createObjectURL(blob); | ||||||
|         } |           let a = document.createElement("a"); | ||||||
|     } |           a.href = url; | ||||||
| </script> |           a.download = `${$_("certificates")}_${o.name}-${locale}-${createId()}.pdf`; | ||||||
|  |           document.body.appendChild(a); | ||||||
| {#if certificates_show} |           a.click(); | ||||||
|     <div id="certificates:dropdown" class="relative inline-block"> |           a.remove(); | ||||||
|         <div> |           if (count === o.teams.length && count_orgs === generate_orgs.length) { | ||||||
|             <button |             toast.hideToast(); | ||||||
|                 on:click={() => { |             console.log("here"); | ||||||
|                     certificates_dropdown_open = !certificates_dropdown_open; |             Toastify({ | ||||||
|                 }} |               text: $_("pdfs-successfully-generated"), | ||||||
|                 type="button" |               duration: 3500, | ||||||
|                 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" |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|                 id="options-menu" |             }).showToast(); | ||||||
|                 aria-haspopup="true" |           } | ||||||
|                 aria-expanded="true"> |         }) | ||||||
|                 {$_('generate-runner-certificates')} |         .catch((err) => {}); | ||||||
|                 <svg |       for (const t of o.teams) { | ||||||
|                     xmlns="http://www.w3.org/2000/svg" |         count++; | ||||||
|                     width="24" |         let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|                     height="24" |           t.id | ||||||
|                     viewBox="0 0 24 24" |         ); | ||||||
|                     class="-mr-1 ml-2 h-5 w-5"><path |         let certificateRunners = []; | ||||||
|                         fill="none" |         for (let runner of runners) { | ||||||
|                         d="M0 0h24v24H0z" /> |           runner.distanceDonations = | ||||||
|                     <path |             current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|                         fill="currentColor" |           certificateRunners.push(runner); | ||||||
|                         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> |         await fetch( | ||||||
|         </div> |           `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|         {#if certificates_dropdown_open} |           { | ||||||
|             <div |             method: "POST", | ||||||
|                 class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |             headers: { | ||||||
|                 id="certificates:dropdown:menu"> |               "Content-Type": "application/json", | ||||||
|                 <div |             }, | ||||||
|                     class="py-1" |             body: JSON.stringify(certificateRunners), | ||||||
|                     role="menu" |           } | ||||||
|                     aria-orientation="vertical" |         ) | ||||||
|                     aria-labelledby="options-menu"> |           .then((response) => { | ||||||
|                     <span |             if (response.status != "200") { | ||||||
|                         class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |               toast.hideToast(); | ||||||
|                     <button |               Toastify({ | ||||||
|                         on:click={() => { |                 text: $_("pdf-generation-failed"), | ||||||
|                             generateCertificates('de'); |                 duration: 3500, | ||||||
|                         }} |                 backgroundColor: | ||||||
|                         type="submit" |                   "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", | ||||||
|                         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" |               }).showToast(); | ||||||
|                         role="menuitem"> |             } else { | ||||||
|                         {$_('german')} |               return response.blob(); | ||||||
|                     </button> |             } | ||||||
|                     <button |           }) | ||||||
|                         on:click={() => { |           .then((blob) => { | ||||||
|                             generateCertificates('en'); |             const url = window.URL.createObjectURL(blob); | ||||||
|                         }} |             let a = document.createElement("a"); | ||||||
|                         type="submit" |             a.href = url; | ||||||
|                         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" |             a.download = `${$_("certificates")}_${o.name}_${ | ||||||
|                         role="menuitem"> |               t.name | ||||||
|                         {$_('english')} |             }-${locale}-${createId()}.pdf`; | ||||||
|                     </button> |             document.body.appendChild(a); | ||||||
|                 </div> |             a.click(); | ||||||
|             </div> |             a.remove(); | ||||||
|         {/if} |             if ( | ||||||
|     </div> |               count === o.teams.length && | ||||||
| {/if} |               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} | ||||||
|   | |||||||
| @@ -1,254 +1,324 @@ | |||||||
| <script> | <script> | ||||||
|     import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|     import { |   import { | ||||||
|         RunnerOrganizationService, |     RunnerOrganizationService, | ||||||
|         RunnerTeamService, |     RunnerTeamService, | ||||||
|     } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|     import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|     export let sponsoring_contracts_show = false; |   import { init } from "@paralleldrive/cuid2"; | ||||||
|     export let generate_runners = []; |   const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|     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) { |   export let sponsoring_contracts_show = false; | ||||||
|         sponsoring_contracts_download_open = false; |   export let generate_runners = []; | ||||||
|  |   export let generate_orgs = []; | ||||||
|         if (generate_orgs.length > 0) { |   export let generate_teams = []; | ||||||
|             generateOrgContracts(locale); |   $: sponsoring_contracts_download_open = false; | ||||||
|         } else if (generate_teams.length > 0) { |   document.addEventListener("click", function (e) { | ||||||
|             generateTeamContracts(locale); |     if ( | ||||||
|         } else { |       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && | ||||||
|             generateRunnerContracts(locale); |       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" | ||||||
|         } |     ) { | ||||||
|  |       sponsoring_contracts_download_open = false; | ||||||
|     } |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|     async function generateTeamContracts(locale) { |   function generateSponsoringContract(locale) { | ||||||
|         const toast = Toastify({ |     sponsoring_contracts_download_open = false; | ||||||
|             text: $_("generating-pdfs"), |  | ||||||
|             duration: -1, |     if (generate_orgs.length > 0) { | ||||||
|         }).showToast(); |       generateOrgContracts(locale); | ||||||
|         let count = 0; |     } else if (generate_teams.length > 0) { | ||||||
|         for (const t of generate_teams) { |       generateTeamContracts(locale); | ||||||
|             count++; |     } else { | ||||||
|             const runners = await RunnerTeamService.runnerTeamControllerGetRunners( |       generateRunnerContracts(locale); | ||||||
|                 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 === 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) { |   async function generateTeamContracts(locale) { | ||||||
|         const toast = Toastify({ |     const toast = Toastify({ | ||||||
|             text: $_("generating-pdf"), |       text: $_("generating-pdfs"), | ||||||
|             duration: -1, |       duration: -1, | ||||||
|         }).showToast(); |     }).showToast(); | ||||||
|         for (const o of generate_orgs) { |     let count = 0; | ||||||
|             const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |     for (const t of generate_teams) { | ||||||
|                 o.id |       count++; | ||||||
|             ); |       const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|             fetch( |         t.id | ||||||
|                 `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |       ); | ||||||
|                 { |       fetch( | ||||||
|                     method: "POST", |         `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, | ||||||
|                     headers: { |         { | ||||||
|                         "Content-Type": "application/json", |           method: "POST", | ||||||
|                     }, |           headers: { | ||||||
|                     body: JSON.stringify(runners), |             "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 === generate_orgs.length) { |  | ||||||
|                         toast.hideToast(); |  | ||||||
|                         Toastify({ |  | ||||||
|                             text: $_("pdfs-successfully-generated"), |  | ||||||
|                             duration: 3500, |  | ||||||
|                             backgroundColor: |  | ||||||
|                                 "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|                         }).showToast(); |  | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|                 .catch((err) => {}); |  | ||||||
|         } |         } | ||||||
|  |       ) | ||||||
|  |         .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) => {}); | ||||||
|     } |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|     function generateRunnerContracts(locale) { |   async function generateOrgContracts(locale) { | ||||||
|         const toast = Toastify({ |     const toast = Toastify({ | ||||||
|             text: $_("generating-pdf"), |       text: $_("generating-pdf"), | ||||||
|             duration: -1, |       duration: -1, | ||||||
|         }).showToast(); |     }).showToast(); | ||||||
|         fetch( |     let count_orgs = 0; | ||||||
|             `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |     for (const o of generate_orgs) { | ||||||
|             { |       count_orgs++; | ||||||
|                 method: "POST", |       let count = 0; | ||||||
|                 headers: { |       let runners = | ||||||
|                     "Content-Type": "application/json", |         await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|                 }, |           o.id, | ||||||
|                 body: JSON.stringify(generate_runners), |           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) => { |           .then((response) => { | ||||||
|                 if (response.status != "200") { |             if (response.status != "200") { | ||||||
|                     toast.hideToast(); |               toast.hideToast(); | ||||||
|                     Toastify({ |               Toastify({ | ||||||
|                         text: $_("pdf-generation-failed"), |                 text: $_("pdf-generation-failed"), | ||||||
|                         duration: 3500, |                 duration: 3500, | ||||||
|                         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%)", | ||||||
|                     }).showToast(); |               }).showToast(); | ||||||
|                 } else { |             } else { | ||||||
|                     return response.blob(); |               return response.blob(); | ||||||
|                 } |             } | ||||||
|             }) |           }) | ||||||
|             .then((blob) => { |           .then((blob) => { | ||||||
|                 const url = window.URL.createObjectURL(blob); |             const url = window.URL.createObjectURL(blob); | ||||||
|                 let a = document.createElement("a"); |             let a = document.createElement("a"); | ||||||
|                 a.href = url; |             a.href = url; | ||||||
|                 a.download = "Sponsoring.pdf"; |             a.download = `${$_("sponsorings")}_${o.name}_${ | ||||||
|                 document.body.appendChild(a); |               t.name | ||||||
|                 a.click(); |             }-${locale}-${createId()}.pdf`; | ||||||
|                 a.remove(); |             document.body.appendChild(a); | ||||||
|                 toast.hideToast(); |             a.click(); | ||||||
|                 Toastify({ |             a.remove(); | ||||||
|                     text: $_("pdf-successfully-generated"), |             if ( | ||||||
|                     duration: 3500, |               count === o.teams.length && | ||||||
|                     backgroundColor: |               count_orgs === generate_orgs.length | ||||||
|                         "linear-gradient(to right, #00b09b, #96c93d)", |             ) { | ||||||
|                 }).showToast(); |               toast.hideToast(); | ||||||
|             }) |               Toastify({ | ||||||
|             .catch((err) => { |                 text: $_("pdfs-successfully-generated"), | ||||||
|                 console.error(err); |                 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> | </script> | ||||||
|  |  | ||||||
| {#if sponsoring_contracts_show} | {#if sponsoring_contracts_show} | ||||||
|     <div id="sponsoring:dropdown" class="relative inline-block"> |   <div id="sponsoring:dropdown" class="relative inline-block"> | ||||||
|         <div> |     <div> | ||||||
|             <button |       <button | ||||||
|                 on:click={() => { |         on:click={() => { | ||||||
|                     sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |           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" |         type="button" | ||||||
|                 id="options-menu" |         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" | ||||||
|                 aria-haspopup="true" |         id="options-menu" | ||||||
|                 aria-expanded="true"> |         aria-haspopup="true" | ||||||
|                 {$_('generate-sponsoring-contracts')} |         aria-expanded="true" | ||||||
|                 <svg |       > | ||||||
|                     xmlns="http://www.w3.org/2000/svg" |         {$_("generate-sponsoring-contracts")} | ||||||
|                     width="24" |         <svg | ||||||
|                     height="24" |           xmlns="http://www.w3.org/2000/svg" | ||||||
|                     viewBox="0 0 24 24" |           width="24" | ||||||
|                     class="-mr-1 ml-2 h-5 w-5"><path |           height="24" | ||||||
|                         fill="none" |           viewBox="0 0 24 24" | ||||||
|                         d="M0 0h24v24H0z" /> |           class="-mr-1 ml-2 h-5 w-5" | ||||||
|                     <path |           ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                         fill="currentColor" |           <path | ||||||
|                         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> |             fill="currentColor" | ||||||
|             </button> |             d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" | ||||||
|         </div> |           /></svg | ||||||
|         {#if sponsoring_contracts_download_open} |         > | ||||||
|             <div |       </button> | ||||||
|                 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" |  | ||||||
|                         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> |     </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} | {/if} | ||||||
|   | |||||||
| @@ -1,30 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import { _, json } from "svelte-i18n"; |  | ||||||
|   import { getlang } from "./datatable_i18n"; |  | ||||||
|   import { Grid } from "gridjs"; |  | ||||||
|   // |  | ||||||
|   let table; |  | ||||||
|   const datatable = new Grid({ |  | ||||||
|     columns: ["Name", "Email", "Phone Number"], |  | ||||||
|     language: getlang($json("datatable")), |  | ||||||
|     sort: true, |  | ||||||
|     search: { enabled: true }, |  | ||||||
|     data: [ |  | ||||||
|       ["John", "john@example.com", "(353) 01 222 3333"], |  | ||||||
|       ["Mark", "mark@gmail.com", "(01) 22 888 4444"], |  | ||||||
|       ["Eoin", "eoin@gmail.com", "0097 22 654 00033"], |  | ||||||
|       ["Sarah", "sarahcdd@gmail.com", "+322 876 1233"], |  | ||||||
|       ["Afshin", "afshin@mail.com", "(353) 22 87 8356"], |  | ||||||
|     ], |  | ||||||
|     pagination: { |  | ||||||
|       enabled: true, |  | ||||||
|       limit: 2, |  | ||||||
|       summary: false, |  | ||||||
|     }, |  | ||||||
|   }); |  | ||||||
|   setTimeout(() => { |  | ||||||
|     datatable.render(table); |  | ||||||
|   }, 0); |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <div bind:this={table} /> |  | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     RunnerService, |     RunnerService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
| @@ -11,8 +11,10 @@ | |||||||
|   import isMobilePhone from "validator/es/lib/isMobilePhone"; |   import isMobilePhone from "validator/es/lib/isMobilePhone"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import Select from "svelte-select"; |   import Select from "svelte-select"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |  | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_runners; |  | ||||||
|   $: selected_team = undefined; |   $: selected_team = undefined; | ||||||
|   let firstname_input; |   let firstname_input; | ||||||
|   let lastname_input; |   let lastname_input; | ||||||
| @@ -107,8 +109,7 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           current_runners.push(result); |           dispatch("created", { runners: [result] }); | ||||||
|           current_runners = current_runners; |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
| @@ -125,7 +126,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
							
								
								
									
										115
									
								
								src/components/runners/DeleteRunnerModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								src/components/runners/DeleteRunnerModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_runner = { | ||||||
|  |     id: 0, | ||||||
|  |     firstname: "", | ||||||
|  |     lastname: "", | ||||||
|  |   }; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   onMount(() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   async function submit() { | ||||||
|  |     dispatch("delete", { id: delete_runner.id }); | ||||||
|  |     modal_open=false; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     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 | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <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 | ||||||
|  |               > | ||||||
|  |             </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"> | ||||||
|  |                 {$_("confirm-delete")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_("please-confirm-the-deletion-of-runner")} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="w-full"> | ||||||
|  |                 <span class="inline-block" | ||||||
|  |                   >{delete_runner.firstname} {delete_runner.lastname}</span | ||||||
|  |                 > | ||||||
|  |               </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={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" | ||||||
|  |           > | ||||||
|  |             {$_("delete")} | ||||||
|  |           </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} | ||||||
| @@ -3,7 +3,7 @@ | |||||||
|   import { read as readXlsx, utils as xlsx_utils } from "xlsx"; |   import { read as readXlsx, utils as xlsx_utils } from "xlsx"; | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { |   import { | ||||||
|     ImportService, |     ImportService, | ||||||
| @@ -16,7 +16,6 @@ | |||||||
|   export let passed_org; |   export let passed_org; | ||||||
|   export let passed_orgs; |   export let passed_orgs; | ||||||
|   export let passed_team; |   export let passed_team; | ||||||
|   export let current_runners; |  | ||||||
|   export let import_modal_open; |   export let import_modal_open; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: importButtonEnabled = |   $: importButtonEnabled = | ||||||
| @@ -34,7 +33,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) { | ||||||
|         // |         // | ||||||
| @@ -169,7 +168,7 @@ | |||||||
|             mapped |             mapped | ||||||
|           ) |           ) | ||||||
|             .then((resp) => { |             .then((resp) => { | ||||||
|               current_runners = current_runners.concat(resp); |               dispatch("created", { runners: resp }); | ||||||
|               toast.hideToast(); |               toast.hideToast(); | ||||||
|               recent_processed = true; |               recent_processed = true; | ||||||
|               Toastify({ |               Toastify({ | ||||||
| @@ -198,11 +197,11 @@ | |||||||
|             mapped |             mapped | ||||||
|           ) |           ) | ||||||
|             .then((resp) => { |             .then((resp) => { | ||||||
|               current_runners = current_runners.concat(resp); |               dispatch("created", { runners: resp }); | ||||||
|               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(); | ||||||
| @@ -228,7 +227,7 @@ | |||||||
| {#if import_modal_open} | {#if import_modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       cancelModal(); |       cancelModal(); | ||||||
| @@ -281,6 +280,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'} | ||||||
|   | |||||||
| @@ -71,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); | ||||||
| @@ -95,7 +98,7 @@ | |||||||
| </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"> | ||||||
| @@ -109,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" | ||||||
| @@ -124,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> | ||||||
| @@ -142,36 +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} | ||||||
|           <GenerateSponsoringContracts |           <GenerateSponsoringContracts | ||||||
|             bind:sponsoring_contracts_show |             bind:sponsoring_contracts_show | ||||||
|             bind:generate_runners /> |             bind:generate_runners | ||||||
|           <GenerateRunnerCards |           /> | ||||||
|             bind:cards_show |           <GenerateRunnerCards bind:cards_show bind:generate_runners /> | ||||||
|             bind:generate_runners /> |  | ||||||
|           <GenerateRunnerCertificates |           <GenerateRunnerCertificates | ||||||
|             bind:certificates_show |             bind:certificates_show | ||||||
|             bind:generate_runners /> |             bind:generate_runners | ||||||
|  |           /> | ||||||
|           {#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} | ||||||
| @@ -180,121 +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) => { |         on:select={(selectedValue) => { | ||||||
|           editable.group = selectedValue.detail.value.id; |           editable.group = selectedValue.detail.value.id; | ||||||
|         }} |         }} | ||||||
|         on:clear={() => (editable.group = null)} /> |         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} | ||||||
|   | |||||||
| @@ -7,35 +7,43 @@ | |||||||
|   $: current_runners = []; |   $: current_runners = []; | ||||||
|   export let modal_open = false; |   export let modal_open = false; | ||||||
|   export let import_modal_open = false; |   export let import_modal_open = false; | ||||||
|  |   let addRunners; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <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"> | ||||||
|     {$_('runners')} |     {$_("runners")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:CREATE')} |     {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:CREATE")} | ||||||
|       <button |       <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
|           modal_open = true; |           modal_open = true; | ||||||
|         }} |         }} | ||||||
|         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" | ||||||
|         {$_('laeufer-hinzufuegen')} |       > | ||||||
|  |         {$_("laeufer-hinzufuegen")} | ||||||
|       </button> |       </button> | ||||||
|       <button |       <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
|           import_modal_open = true; |           import_modal_open = true; | ||||||
|         }} |         }} | ||||||
|         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" | ||||||
|         {$_('import-runners')} |       > | ||||||
|  |         {$_("import-runners")} | ||||||
|       </button> |       </button> | ||||||
|     {/if} |     {/if} | ||||||
|   </span> |   </span> | ||||||
|   <RunnersOverview bind:current_runners /> |   <RunnersOverview bind:current_runners bind:addRunners /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:CREATE")} | ||||||
|   <AddRunnerModal bind:current_runners bind:modal_open /> |   <AddRunnerModal | ||||||
|  |     bind:modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addRunners(event.detail.runners); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|   <ImportRunnerModal |   <ImportRunnerModal | ||||||
|     on:cancelDelete={(event) => { |     on:cancelDelete={(event) => { | ||||||
|       import_modal_open = false; |       import_modal_open = false; | ||||||
| @@ -43,7 +51,10 @@ | |||||||
|     passed_team={{}} |     passed_team={{}} | ||||||
|     passed_orgs={[]} |     passed_orgs={[]} | ||||||
|     passed_org={{}} |     passed_org={{}} | ||||||
|     bind:current_runners |  | ||||||
|     opened_from="RunnerOverview" |     opened_from="RunnerOverview" | ||||||
|     bind:import_modal_open /> |     bind:import_modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addRunners(event.detail.runners); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -1,263 +1,259 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |  | ||||||
|   import { |   import { | ||||||
|  |     RunnerOrganizationService, | ||||||
|     RunnerService, |     RunnerService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
|     RunnerOrganizationService, |  | ||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import RunnersEmptyState from "./RunnersEmptyState.svelte"; |  | ||||||
|   import Select from "svelte-select"; |  | ||||||
|   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; |  | ||||||
|   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: searchvalue = ""; |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   $: active_deletes = []; |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|  |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |   import { groupFilter } from "../shared/tablefilters"; | ||||||
|  |   import DeleteRunnerModal from "./DeleteRunnerModal.svelte"; | ||||||
|  |   import Toastify from "toastify-js"; | ||||||
|  |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|  |  | ||||||
|  |   $: selectedRunners = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.original) || []; | ||||||
|  |   $: selected = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|  |  | ||||||
|  |   $: active_delete = undefined; | ||||||
|  |   let dataLoaded = false; | ||||||
|   export let current_runners = []; |   export let current_runners = []; | ||||||
|   const runners_promise = RunnerService.runnerControllerGetAll().then((val) => { |   $: sponsoring_contracts_show = selected.length > 0; | ||||||
|     current_runners = val; |   $: cards_show = selected.length > 0; | ||||||
|   }); |   $: certificates_show = selected.length > 0; | ||||||
|   $: selectedFilter_teams = null; |  | ||||||
|   $: selectedFilter = null; |  | ||||||
|   $: filter__teams = selectedFilter_teams || []; |  | ||||||
|   $: filter__orgs = selectedFilter || []; |  | ||||||
|   $: filterGroupIDs = filter__teams.concat(filter__orgs).map((i) => i.value); |  | ||||||
|   $: sponsoring_contracts_show = current_runners.some( |  | ||||||
|     (r) => r.is_selected === true |  | ||||||
|   ); |  | ||||||
|   $: cards_show = current_runners.some( |  | ||||||
|     (r) => r.is_selected === true |  | ||||||
|   ); |  | ||||||
|   $: certificates_show = current_runners.some( |  | ||||||
|     (r) => r.is_selected === true |  | ||||||
|   ); |  | ||||||
|   $: generate_runners = current_runners.filter((r) => r.is_selected === true); |  | ||||||
|   $: teams = []; |   $: teams = []; | ||||||
|   $: orgs = []; |   $: orgs = []; | ||||||
|   $: mappedteams = teams.map(function (g) { |  | ||||||
|     return { value: g.id, label: g.parentGroup.name + " > " + g.name }; |  | ||||||
|   }); |  | ||||||
|   $: selectgroups = orgs |  | ||||||
|     .map(function (g) { |  | ||||||
|       return { value: g.id, label: g.name }; |  | ||||||
|     }) |  | ||||||
|     .concat(mappedteams); |  | ||||||
|  |  | ||||||
|   RunnerTeamService.runnerTeamControllerGetAll().then((val) => { |   export const addRunners = (runners) => { | ||||||
|     teams = val; |     current_runners = current_runners.concat(...runners); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_runners, | ||||||
|  |     })); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   //Section table | ||||||
|  |   const columns = [ | ||||||
|  |     { | ||||||
|  |       accessorKey: "id", | ||||||
|  |       header: () => "id", | ||||||
|  |       filterFn: `equalsString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "firstname", | ||||||
|  |       header: () => $_("first-name"), | ||||||
|  |       filterFn: `includesString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "middlename", | ||||||
|  |       header: () => $_("middle-name"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         if (!info || !info.getValue()) { | ||||||
|  |           return ""; | ||||||
|  |         } | ||||||
|  |         return info.getValue(); | ||||||
|  |       }, | ||||||
|  |       filterFn: `includesString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "lastname", | ||||||
|  |       header: () => $_("last-name"), | ||||||
|  |       filterFn: `includesString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "group", | ||||||
|  |       header: () => $_("group"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         const group = info.getValue(); | ||||||
|  |         if (group.responseType === "RUNNERORGANIZATION") { | ||||||
|  |           return group.name; | ||||||
|  |         } | ||||||
|  |         return `${group.parentGroup.name} > ${group.name}`; | ||||||
|  |       }, | ||||||
|  |       filterFn: `group`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "distance", | ||||||
|  |       header: () => $_("distance"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         if (info.getValue() < 1000) { | ||||||
|  |           return `${info.getValue()} m`; | ||||||
|  |         } | ||||||
|  |         return `${(info.getValue() / 1000).toFixed(1)} km`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(TableActions, { | ||||||
|  |           detailsLink: `./${info.row.original.id}`, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_delete = | ||||||
|  |               current_runners[ | ||||||
|  |                 current_runners.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ]; | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes( | ||||||
|  |               "RUNNER:DELETE" | ||||||
|  |             ), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     filterFns: { | ||||||
|  |       group: groupFilter, | ||||||
|  |     }, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|   }); |   }); | ||||||
|   RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { |   const table = createSvelteTable(options); | ||||||
|     orgs = val; |  | ||||||
|   }); |   async function deleteRunner(delete_runner_id) { | ||||||
|   function should_display_based_on_id(id) { |     await RunnerService.runnerControllerRemove(delete_runner_id, true); | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |     current_runners = current_runners.filter((r) => r.id !== delete_runner_id); | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |     options.update((options) => ({ | ||||||
|     } |       ...options, | ||||||
|     return id.toString() === searchvalue; |       data: current_runners, | ||||||
|  |     })); | ||||||
|  |     Toastify({ | ||||||
|  |       text: $_("runner-deleted"), | ||||||
|  |       duration: 3500, | ||||||
|  |       backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |     }).showToast(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   onMount(() => { | ||||||
|  |     RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|  |       current_runners = val; | ||||||
|  |       dataLoaded = true; | ||||||
|  |  | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_runners, | ||||||
|  |       })); | ||||||
|  |     }); | ||||||
|  |     RunnerTeamService.runnerTeamControllerGetAll().then((val) => { | ||||||
|  |       teams = val; | ||||||
|  |     }); | ||||||
|  |     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|  |       (val) => { | ||||||
|  |         orgs = val; | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   }); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:GET')} | <DeleteRunnerModal | ||||||
|   {#await runners_promise} |   delete_runner={active_delete} | ||||||
|  |   modal_open={active_delete != undefined} | ||||||
|  |   on:delete={(event) => { | ||||||
|  |     deleteRunner(event.detail.id); | ||||||
|  |   }} | ||||||
|  | /> | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:GET")} | ||||||
|  |   {#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 mt-2"> | ||||||
|       <RunnersEmptyState /> |       <GenerateSponsoringContracts | ||||||
|     {:else} |         bind:sponsoring_contracts_show | ||||||
|       <input |         bind:generate_runners={selectedRunners} | ||||||
|         type="search" |       /> | ||||||
|         bind:value={searchvalue} |       <GenerateRunnerCards | ||||||
|         placeholder={$_('datatable.search')} |         bind:cards_show | ||||||
|         aria-label={$_('datatable.search')} |         bind:generate_runners={selectedRunners} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |       /> | ||||||
|       <div class="block mb-6"> |       <GenerateRunnerCertificates | ||||||
|         <label |         bind:certificates_show | ||||||
|           for="country" |         bind:generate_runners={selectedRunners} | ||||||
|           class="text-sm font-medium text-gray-700">{$_('filter-by-organization-team')}</label> |       /> | ||||||
|         <Select |     </div> | ||||||
|           on:select={(event) => { |     <div class="overflow-x-auto"> | ||||||
|             selectedFilter = event.detail; |       <table class="w-full"> | ||||||
|           }} |         <thead> | ||||||
|           selectedValue={selectedFilter} |           {#each $table.getHeaderGroups() as headerGroup} | ||||||
|           placeholder={$_('filter-by-organization-team')} |             <tr class="select-none"> | ||||||
|           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" |               <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|           items={selectgroups} |                 <InputElement | ||||||
|           isMulti={true} /> |                   type="checkbox" | ||||||
|       </div> |                   checked={$table.getIsAllRowsSelected()} | ||||||
|       <div class="h-12"> |                   indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|         <GenerateSponsoringContracts |                   on:change={() => $table.toggleAllRowsSelected()} | ||||||
|           bind:sponsoring_contracts_show |                 /> | ||||||
|           bind:generate_runners /> |  | ||||||
|         <GenerateRunnerCards |  | ||||||
|           bind:cards_show |  | ||||||
|           bind:generate_runners /> |  | ||||||
|         <GenerateRunnerCertificates |  | ||||||
|           bind:certificates_show |  | ||||||
|           bind:generate_runners /> |  | ||||||
|       </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> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 <span |  | ||||||
|                   on:click={() => { |  | ||||||
|                     const newstate = !current_runners.some((r) => r.is_selected === true); |  | ||||||
|                     current_runners = current_runners.map((r) => { |  | ||||||
|                       r.is_selected = newstate; |  | ||||||
|                       return r; |  | ||||||
|                     }); |  | ||||||
|                   }} |  | ||||||
|                   class="underline cursor-pointer select-none">{#if current_runners.some((r) => r.is_selected === true)} |  | ||||||
|                     {$_('deselect-all')} |  | ||||||
|                   {:else}{$_('select-all')}{/if} |  | ||||||
|                 </span> |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('name')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 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> |               </th> | ||||||
|  |               {#each headerGroup.headers as header} | ||||||
|  |                 <TableHeader {header} /> | ||||||
|  |               {/each} | ||||||
|             </tr> |             </tr> | ||||||
|           </thead> |           {/each} | ||||||
|           <tbody class="divide-y divide-gray-200"> |         </thead> | ||||||
|             {#each current_runners as runner} |         <tbody> | ||||||
|               {#if runner.firstname |           {#each $table.getRowModel().rows as row} | ||||||
|                 .toLowerCase() |             <tr> | ||||||
|                 .includes( |               <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|                   searchvalue.toLowerCase() |                 <InputElement | ||||||
|                 ) || runner.lastname |                   type="checkbox" | ||||||
|                   .toLowerCase() |                   checked={row.getIsSelected()} | ||||||
|                   .includes( |                   on:change={() => row.toggleSelected()} | ||||||
|                     searchvalue.toLowerCase() |                 /> | ||||||
|                   ) || should_display_based_on_id(runner.id)} |               </td> | ||||||
|                 {#if filterGroupIDs.includes(runner.group.id) || filterGroupIDs.includes(runner.group.parentGroup?.id) || filterGroupIDs.length === 0} |               {#each row.getVisibleCells() as cell} | ||||||
|                   <tr |                 <td> | ||||||
|                     data-rowid="user_{runner.id}" |                   <svelte:component | ||||||
|                     data-groupid={runner.group.id}> |                     this={flexRender( | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |                       cell.column.columnDef.cell, | ||||||
|                       <input |                       cell.getContext() | ||||||
|                         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> |               {/each} | ||||||
|                     <td class="px-6 py-4 whitespace-nowrap"> |             </tr> | ||||||
|                       <div class="flex items-center"> |           {/each} | ||||||
|                         <div class="ml-4"> |         </tbody> | ||||||
|                           <div class="text-sm font-medium text-gray-900"> |       </table> | ||||||
|                             {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} |  | ||||||
|             {/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> |     </div> | ||||||
|   {/await} |     <div class="h-2" /> | ||||||
|  |   {/if} | ||||||
| {/if} | {/if} | ||||||
|  | <TableBottom {table} {selected} /> | ||||||
|   | |||||||
							
								
								
									
										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> | ||||||
| @@ -1,23 +1,24 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     RunnerService, |     RunnerService, | ||||||
|     ScanService, |     ScanService, | ||||||
|   } 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 { createEventDispatcher } from "svelte"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_scans; |  | ||||||
|   const getRunnerLabel = (option) => |   const getRunnerLabel = (option) => | ||||||
|     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(); | ||||||
|   } |   } | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|   $: runner = 0; |   $: runner = 0; | ||||||
|   $: runners = []; |   $: runners = []; | ||||||
|   RunnerService.runnerControllerGetAll().then((val) => { |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
| @@ -63,8 +64,7 @@ | |||||||
|             duration: 500, |             duration: 500, | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|           }).showToast(); |           }).showToast(); | ||||||
|           current_scans.push(result); |           dispatch("created", { scans: [result] }); | ||||||
|           current_scans = current_scans; |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
| @@ -81,7 +81,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
							
								
								
									
										110
									
								
								src/components/scans/DeleteScanModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								src/components/scans/DeleteScanModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_scan = { | ||||||
|  |     id: 0, | ||||||
|  |     runner: { | ||||||
|  |       firstname: "", | ||||||
|  |       lastname: "", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   onMount(() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   async function submit() { | ||||||
|  |     dispatch("delete", { id: delete_scan.id }); | ||||||
|  |     modal_open = false; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|  |     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 | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <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 | ||||||
|  |               > | ||||||
|  |             </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"> | ||||||
|  |                 {$_("confirm-delete")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 {$_("please-confirm-the-deletion-of-scan")} #{delete_scan.id} | ||||||
|  |               </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={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" | ||||||
|  |           > | ||||||
|  |             {$_("delete")} | ||||||
|  |           </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} | ||||||
							
								
								
									
										16
									
								
								src/components/scans/ScanValid.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/scans/ScanValid.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let valid = false; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if valid} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800" | ||||||
|  |     >{$_("valid")}</span | ||||||
|  |   > | ||||||
|  | {:else} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800" | ||||||
|  |     >{$_("invalid")}</span | ||||||
|  |   > | ||||||
|  | {/if} | ||||||
| @@ -5,6 +5,7 @@ | |||||||
|   import ScansOverview from "./ScansOverview.svelte"; |   import ScansOverview from "./ScansOverview.svelte"; | ||||||
|   $: current_scans = []; |   $: current_scans = []; | ||||||
|   export let modal_open = false; |   export let modal_open = false; | ||||||
|  |   let addScans; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
| @@ -21,9 +22,11 @@ | |||||||
|       </button> |       </button> | ||||||
|     {/if} |     {/if} | ||||||
|   </span> |   </span> | ||||||
|   <ScansOverview bind:current_scans /> |   <ScansOverview bind:current_scans bind:addScans /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:CREATE')} | ||||||
|   <AddScanModal bind:current_scans bind:modal_open /> |   <AddScanModal bind:modal_open on:created={(event)=>{ | ||||||
|  |     addScans(event.detail.scans) | ||||||
|  |   }} /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -1,195 +1,303 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { |   import { ScanService, TrackService } from "@odit/lfk-client-js"; | ||||||
|     ScanService, |  | ||||||
|   } from "@odit/lfk-client-js"; |  | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|  |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|   import ScansEmptyState from "./ScansEmptyState.svelte"; |   import ScansEmptyState from "./ScansEmptyState.svelte"; | ||||||
|   $: searchvalue = ""; |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|   $: active_deletes = []; |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |   import { runnerFilter, statusFilter } from "../shared/tablefilters"; | ||||||
|  |   import CardRunner from "../cards/CardRunner.svelte"; | ||||||
|  |   import ScanValid from "./ScanValid.svelte"; | ||||||
|  |   import DeleteScanModal from "./DeleteScanModal.svelte"; | ||||||
|  |  | ||||||
|  |   $: selectedScans = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.original) || []; | ||||||
|  |   $: selected = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|  |  | ||||||
|  |   $: active_delete = undefined; | ||||||
|   export let current_scans = []; |   export let current_scans = []; | ||||||
|  |   export const addScans = (scans) => { | ||||||
|  |     current_scans = current_scans.concat(...scans); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_scans, | ||||||
|  |     })); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   const scans_promise = ScanService.scanControllerGetAll().then((val) => { |   const scans_promise = ScanService.scanControllerGetAll().then((val) => { | ||||||
|     current_scans = val; |     current_scans = val; | ||||||
|  |     // handler.setRows(val); | ||||||
|  |     current_scans = val; | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_scans, | ||||||
|  |     })); | ||||||
|   }); |   }); | ||||||
|   function should_display_based_on_id(id) { |   let 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`; | ||||||
|  |     } | ||||||
|  |     if (laptime < 3600) { | ||||||
|  |       return `${Math.floor(laptime / 60)}min ${ | ||||||
|  |         laptime - 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 | ||||||
|  |     }`; | ||||||
|   } |   } | ||||||
|   function format_laptime(laptime){ |  | ||||||
|     if(laptime == 0 || laptime == null){return $_('first-scan-of-the-day')} |   const columns = [ | ||||||
|     if(laptime < 60){return `${laptime}s`} |     { | ||||||
|     if(laptime < 3600){return `${Math.floor(laptime / 60)}min ${laptime - (Math.floor(laptime / 60)*60)}s`} |       accessorKey: "id", | ||||||
|     return `${Math.floor(laptime / 3600)}h ${laptime - (Math.floor(laptime / 3600)*3600)}min ${laptime - (Math.floor(laptime / 3600)*3600) - (Math.floor(laptime / 60)*60)}` |       header: () => "id", | ||||||
|  |       filterFn: `equalsString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "runner", | ||||||
|  |       header: () => $_("runner"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(CardRunner, { runner: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `runner`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "lapTime", | ||||||
|  |       header: () => $_("laptime"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return format_laptime(info.getValue()); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "timestamp", | ||||||
|  |       header: () => $_("timestamp"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return (new Date(parseInt(info.getValue())*1000)).toLocaleString() | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "distance", | ||||||
|  |       header: () => $_("distance"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         if (info.getValue() < 1000) { | ||||||
|  |           return `${info.getValue()}m`; | ||||||
|  |         } | ||||||
|  |         return `${(info.getValue() / 1000).toFixed(1)}km`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "track", | ||||||
|  |       header: () => $_("track"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         const track = info.getValue(); | ||||||
|  |         return track?.name || "?"; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "valid", | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(ScanValid, { valid: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       header: () => $_("status"), | ||||||
|  |       filterFn: `status`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(TableActions, { | ||||||
|  |           detailsLink: `./${info.row.original.id}`, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_delete = | ||||||
|  |               current_scans[ | ||||||
|  |                 current_scans.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ]; | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes("SCAN:DELETE"), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     filterFns: { | ||||||
|  |       runner: runnerFilter, | ||||||
|  |       status: statusFilter, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|  |   }); | ||||||
|  |   const table = createSvelteTable(options); | ||||||
|  |   async function deleteScan(scan_id) { | ||||||
|  |     await ScanService.scanControllerRemove(scan_id, true); | ||||||
|  |     current_scans = current_scans.filter((r) => r.id !== scan_id); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_scans, | ||||||
|  |     })); | ||||||
|  |     Toastify({ | ||||||
|  |       text: $_("scan-deleted"), | ||||||
|  |       duration: 3500, | ||||||
|  |       backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|  |     }).showToast(); | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')} | <DeleteScanModal | ||||||
|  |   delete_scan={active_delete} | ||||||
|  |   modal_open={active_delete != undefined} | ||||||
|  |   on:delete={(event) => { | ||||||
|  |     deleteScan(event.detail.id); | ||||||
|  |   }} | ||||||
|  | /> | ||||||
|  | {#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 |       {#if selected.length > 0} | ||||||
|         type="search" |         <button | ||||||
|         bind:value={searchvalue} |           type="button" | ||||||
|         placeholder={$_('datatable.search')} |           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 inline-flex" | ||||||
|         aria-label={$_('datatable.search')} |           id="options-menu" | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |           on:click={async () => { | ||||||
|       <div |             const prom = []; | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |             for (const scan of selectedScans) { | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |               prom.push(ScanService.scanControllerRemove(scan.id, true)); | ||||||
|           <thead class="bg-gray-50"> |             } | ||||||
|             <tr> |             await Promise.all(prom); | ||||||
|               <th |             for (const scan of selectedScans) { | ||||||
|                 scope="col" |               current_scans = current_scans.filter((r) => r.id !== scan.id); | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             } | ||||||
|                 {$_('runner')} |             options.update((options) => ({ | ||||||
|               </th> |               ...options, | ||||||
|               <th |               data: current_scans, | ||||||
|                 scope="col" |             })); | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             $table.resetRowSelection(); | ||||||
|                 {$_('distance-track')} |             Toastify({ | ||||||
|               </th> |               text: $_("scan-deleted"), | ||||||
|               <th |               duration: 3500, | ||||||
|                 scope="col" |               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             }).showToast(); | ||||||
|                 {$_('laptime')} |           }} | ||||||
|               </th> |         > | ||||||
|               <th |           {$_("delete-scans")} | ||||||
|                 scope="col" |           <svg | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |             xmlns="http://www.w3.org/2000/svg" | ||||||
|                 {$_('status')} |             fill="none" | ||||||
|               </th> |             viewBox="0 0 24 24" | ||||||
|               <th scope="col" class="relative px-6 py-3"> |             stroke-width="1.5" | ||||||
|                 <span class="sr-only">{$_('action')}</span> |             stroke="currentColor" | ||||||
|               </th> |             class="w-5 h-5" | ||||||
|             </tr> |           > | ||||||
|  |             <path | ||||||
|  |               stroke-linecap="round" | ||||||
|  |               stroke-linejoin="round" | ||||||
|  |               d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0" | ||||||
|  |             /> | ||||||
|  |           </svg> | ||||||
|  |         </button> | ||||||
|  |       {/if} | ||||||
|  |       <div class="overflow-x-auto"> | ||||||
|  |         <table class="w-full"> | ||||||
|  |           <thead> | ||||||
|  |             {#each $table.getHeaderGroups() as headerGroup} | ||||||
|  |               <tr class="select-none"> | ||||||
|  |                 <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|  |                   <InputElement | ||||||
|  |                     type="checkbox" | ||||||
|  |                     checked={$table.getIsAllRowsSelected()} | ||||||
|  |                     indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|  |                     on:change={() => $table.toggleAllRowsSelected()} | ||||||
|  |                   /> | ||||||
|  |                 </th> | ||||||
|  |                 {#each headerGroup.headers as header} | ||||||
|  |                   <TableHeader {header} /> | ||||||
|  |                 {/each} | ||||||
|  |               </tr> | ||||||
|  |             {/each} | ||||||
|           </thead> |           </thead> | ||||||
|           <tbody class="divide-y divide-gray-200"> |           <tbody> | ||||||
|             {#each current_scans as scan} |             {#each $table.getRowModel().rows as row} | ||||||
|               {#if scan.track?.name |               <tr> | ||||||
|                 .toLowerCase() |                 <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|                 .includes( |                   <InputElement | ||||||
|                   searchvalue.toLowerCase() |                     type="checkbox" | ||||||
|                 ) || scan.runner?.firstname |                     checked={row.getIsSelected()} | ||||||
|                   .toLowerCase() |                     on:change={() => row.toggleSelected()} | ||||||
|                   .includes( |                   /> | ||||||
|                     searchvalue.toLowerCase() |                 </td> | ||||||
|                   ) || scan.runner?.lastname |                 {#each row.getVisibleCells() as cell} | ||||||
|                   .toLowerCase() |                   <td> | ||||||
|                   .includes( |                     <svelte:component | ||||||
|                     searchvalue.toLowerCase() |                       this={flexRender( | ||||||
|                   ) || should_display_based_on_id(scan.id)} |                         cell.column.columnDef.cell, | ||||||
|                 <tr data-rowid="scan_{scan.id}"> |                         cell.getContext() | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                       )} | ||||||
|                     <div class="flex items-center"> |                     /> | ||||||
|                       <a |  | ||||||
|                         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} |  | ||||||
|                         {scan.runner.middlename || ''} |  | ||||||
|                         {scan.runner.lastname}</a> |  | ||||||
|                     </div> |  | ||||||
|                   </td> |                   </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                 {/each} | ||||||
|                     <div class="text-sm font-medium text-gray-900"> |               </tr> | ||||||
|                       {#if scan.distance < 1000} |  | ||||||
|                         {scan.distance}m |  | ||||||
|                       {:else}{scan.distance / 1000}km{/if} |  | ||||||
|                       {#if scan.track} |  | ||||||
|                         <a |  | ||||||
|                           href="../tracks" |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{scan.track.name} |  | ||||||
|                         </a> |  | ||||||
|                       {/if} |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     {#if scan.responseType === "TRACKSCAN"} |  | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                         {format_laptime(scan.lapTime)} |  | ||||||
|                       </div> |  | ||||||
|                     {:else} |  | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                         {$_('scan-with-fixed-distance')} |  | ||||||
|                       </div> |  | ||||||
|                     {/if} |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       {#if scan.valid} |  | ||||||
|                         <span |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('valid')}</span> |  | ||||||
|                       {:else} |  | ||||||
|                         <span |  | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('invalid')}</span> |  | ||||||
|                       {/if} |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|  |  | ||||||
|                   {#if active_deletes[scan.id] === true} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           active_deletes[scan.id] = false; |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           ScanService.scanControllerRemove(scan.id, false).then( |  | ||||||
|                             (resp) => { |  | ||||||
|                               current_scans = current_scans.filter( |  | ||||||
|                                 (obj) => obj.id !== scan.id |  | ||||||
|                               ); |  | ||||||
|                               Toastify({ |  | ||||||
|                                 text: 'Scan deleted', |  | ||||||
|                                 duration: 500, |  | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             } |  | ||||||
|                           ); |  | ||||||
|                         }} |  | ||||||
|                         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="./{scan.id}" |  | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |  | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:DELETE')} |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[scan.id] = true; |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                   {/if} |  | ||||||
|                 </tr> |  | ||||||
|               {/if} |  | ||||||
|             {/each} |             {/each} | ||||||
|           </tbody> |           </tbody> | ||||||
|         </table> |         </table> | ||||||
|       </div> |       </div> | ||||||
|  |       <TableBottom {table} {selected} /> | ||||||
|     {/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> | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { ScanStationService, TrackService } from "@odit/lfk-client-js"; |   import { ScanStationService, TrackService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import Select from "svelte-select"; |   import Select from "svelte-select"; | ||||||
| @@ -81,7 +81,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { ScanStationService } from "@odit/lfk-client-js"; |   import { ScanStationService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { createEventDispatcher } from "svelte"; |   import { createEventDispatcher } from "svelte"; | ||||||
| @@ -13,7 +13,7 @@ | |||||||
|     dispatch("cancelDelete", { id: delete_station.id }); |     dispatch("cancelDelete", { id: delete_station.id }); | ||||||
|   } |   } | ||||||
|   function deleteStation() { |   function deleteStation() { | ||||||
|     ScanStationService.donorControllerRemove( |     ScanStationService.scanStationControllerRemove( | ||||||
|       delete_station.id, |       delete_station.id, | ||||||
|       true |       true | ||||||
|     ) |     ) | ||||||
| @@ -32,7 +32,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> |     on:click_outside={cancelDelete}> | ||||||
|     <div |     <div | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { tick, createEventDispatcher } from "svelte"; |   import { tick, createEventDispatcher } from "svelte"; | ||||||
|   export let copy_modal_open; |   export let copy_modal_open; | ||||||
| @@ -46,70 +46,87 @@ | |||||||
|   {#if valueCopy != null} |   {#if valueCopy != null} | ||||||
|     <textarea bind:this={areaDom}>{valueCopy}</textarea> |     <textarea bind:this={areaDom}>{valueCopy}</textarea> | ||||||
|   {/if} |   {/if} | ||||||
|   <div class="fixed z-10 inset-0 overflow-y-auto" use:focusTrap> |   <div class="fixed z-10 inset-0 overflow-y-auto"> | ||||||
|     <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" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <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" |         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" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|  |       > | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |         <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="sm:flex sm:items-start"> | ||||||
|             <div |             <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"> |               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 |               <svg | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> |                 viewBox="0 0 24 24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   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"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('token')} |                 {$_("token")} | ||||||
|               </h3> |               </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" | ||||||
|  |                   )} | ||||||
|                   <br /> |                   <br /> | ||||||
|                   {$_('please-copy-the-token-and-store-it-somewhere-save')} |                   {$_("please-copy-the-token-and-store-it-somewhere-save")} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <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" | ||||||
|                 <div on:click={copy} class="inline-flex"> |                   >{$_("token")}</label | ||||||
|  |                 > | ||||||
|  |                 <button on:click={copy} class="inline-flex"> | ||||||
|                   <p |                   <p | ||||||
|                     name="token" |                     name="token" | ||||||
|                     class:bg-green-200={copied} |                     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"> |                     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_station.key} |                     {new_station.key} | ||||||
|                   </p> |                   </p> | ||||||
|                   <div |                   <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"> |                     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 |                     <svg | ||||||
|                       xmlns="http://www.w3.org/2000/svg" |                       xmlns="http://www.w3.org/2000/svg" | ||||||
|                       viewBox="0 0 24 24" |                       viewBox="0 0 24 24" | ||||||
|                       width="24" |                       width="24" | ||||||
|                       height="24"><path fill="none" d="M0 0h24v24H0z" /> |                       height="24" | ||||||
|  |                       ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                       <path |                       <path | ||||||
|                         fill="currentColor" |                         fill="currentColor" | ||||||
|                         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> |                 </button> | ||||||
|                 <p class="text-gray-500 text-xs"> |                 <p class="text-gray-500 text-xs"> | ||||||
|                   {$_('click-to-copy-token-to-clipboard')} |                   {$_("click-to-copy-token-to-clipboard")} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
| @@ -119,8 +136,9 @@ | |||||||
|           <button |           <button | ||||||
|             on:click={close} |             on:click={close} | ||||||
|             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-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"> |             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')} |           > | ||||||
|  |             {$_("yes-i-copied-the-token")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="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"> |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   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 { createEventDispatcher } from "svelte"; |   import { createEventDispatcher } from "svelte"; | ||||||
| @@ -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(); | ||||||
| @@ -30,7 +30,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> |     on:click_outside={cancelDelete}> | ||||||
|     <div |     <div | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								src/components/shared/InputElement.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/components/shared/InputElement.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | <script> | ||||||
|  |   let className = ""; | ||||||
|  |   export { className as class }; | ||||||
|  |  | ||||||
|  |   export let type; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <input | ||||||
|  |   class={`border-1 border-stone-300 border rounded-md shadow ${className} ${ | ||||||
|  |     type === "checkbox" && "w-5 h-5 text-orange-400" | ||||||
|  |   }`} | ||||||
|  |   {type} | ||||||
|  |   {...$$restProps} | ||||||
|  |   on:click | ||||||
|  |   on:change | ||||||
|  |   on:keydown | ||||||
|  |   on:keyup | ||||||
|  |   on:mouseenter | ||||||
|  |   on:mouseleave | ||||||
|  | /> | ||||||
							
								
								
									
										26
									
								
								src/components/shared/TableActions.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/components/shared/TableActions.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |  | ||||||
|  |   export let detailsLink; | ||||||
|  |   export let detailsAction; | ||||||
|  |   export let deleteEnabled; | ||||||
|  |   export let deleteAction; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if detailsLink} | ||||||
|  |   <a href={detailsLink} class="text-indigo-600 hover:text-indigo-900" | ||||||
|  |     >{$_("details")}</a | ||||||
|  |   > | ||||||
|  | {:else if detailsAction} | ||||||
|  |   <button on:click={detailsAction} class="text-indigo-600 hover:text-indigo-900" | ||||||
|  |     >{$_("details")}</button | ||||||
|  |   > | ||||||
|  | {/if} | ||||||
|  | {#if deleteEnabled} | ||||||
|  |   <button | ||||||
|  |     tabindex="0" | ||||||
|  |     on:click={deleteAction} | ||||||
|  |     class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  |     >{$_("delete")}</button | ||||||
|  |   > | ||||||
|  | {/if} | ||||||
							
								
								
									
										71
									
								
								src/components/shared/TableBottom.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/components/shared/TableBottom.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | |||||||
|  | <script> | ||||||
|  |   export let table; | ||||||
|  |   export let selected; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="flex items-center gap-2"> | ||||||
|  |   <button | ||||||
|  |     class="border rounded p-1" | ||||||
|  |     on:click={() => $table.setPageIndex(0)} | ||||||
|  |     disabled={!$table.getCanPreviousPage()} | ||||||
|  |   > | ||||||
|  |     {"<<"} | ||||||
|  |   </button> | ||||||
|  |   <button | ||||||
|  |     class="border rounded p-1" | ||||||
|  |     on:click={() => $table.previousPage()} | ||||||
|  |     disabled={!$table.getCanPreviousPage()} | ||||||
|  |   > | ||||||
|  |     {"<"} | ||||||
|  |   </button> | ||||||
|  |   <button | ||||||
|  |     class="border rounded p-1" | ||||||
|  |     on:click={() => $table.nextPage()} | ||||||
|  |     disabled={!$table.getCanNextPage()} | ||||||
|  |   > | ||||||
|  |     {">"} | ||||||
|  |   </button> | ||||||
|  |   <button | ||||||
|  |     class="border rounded p-1" | ||||||
|  |     on:click={() => $table.setPageIndex($table.getPageCount() - 1)} | ||||||
|  |     disabled={!$table.getCanNextPage()} | ||||||
|  |   > | ||||||
|  |     {">>"} | ||||||
|  |   </button> | ||||||
|  |   <span class="flex items-center gap-1"> | ||||||
|  |     <div>Page</div> | ||||||
|  |     <strong> | ||||||
|  |       {$table.getState().pagination.pageIndex + 1} of{" "} | ||||||
|  |       {$table.getPageCount()} | ||||||
|  |     </strong> | ||||||
|  |   </span> | ||||||
|  |   <span class="flex items-center gap-1"> | ||||||
|  |     | Go to page: | ||||||
|  |     <input | ||||||
|  |       type="number" | ||||||
|  |       defaultValue={$table.getState().pagination.pageIndex + 1} | ||||||
|  |       on:change={(e) => { | ||||||
|  |         const page = e.target.value ? Number(e.target.value) - 1 : 0; | ||||||
|  |         $table.setPageIndex(page); | ||||||
|  |       }} | ||||||
|  |       class="border p-1 rounded w-16" | ||||||
|  |     /> | ||||||
|  |   </span> | ||||||
|  |   <select | ||||||
|  |     value={$table.getState().pagination.pageSize} | ||||||
|  |     on:input={(e) => { | ||||||
|  |       const ps = Number(e.target.value); | ||||||
|  |       console.log({ ps }); | ||||||
|  |       $table.setPageSize(Number(e.target.value)); | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     {#each [25, 50, 100, 250, 500] as pageSize} | ||||||
|  |       <option value={pageSize}>{pageSize}</option> | ||||||
|  |     {/each} | ||||||
|  |   </select> | ||||||
|  | </div> | ||||||
|  | <!-- <pre>{JSON.stringify($table.getState(), null, 2)}</pre> --> | ||||||
|  | <div> | ||||||
|  |   {Object.keys(selected).length} of{" "} | ||||||
|  |   {$table.getPreFilteredRowModel().rows.length} Total Rows Selected | ||||||
|  | </div> | ||||||
							
								
								
									
										57
									
								
								src/components/shared/TableHeader.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/components/shared/TableHeader.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | <script> | ||||||
|  |   import { flexRender } from "@tanstack/svelte-table"; | ||||||
|  |   export let header; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <th class="cursor-pointer min-w-[5rem]"> | ||||||
|  |   {#if !header.isPlaceholder} | ||||||
|  |     <button | ||||||
|  |       class="w-full" | ||||||
|  |       tabindex="0" | ||||||
|  |       on:click={header.column.getToggleSortingHandler()} | ||||||
|  |     > | ||||||
|  |       <svelte:component | ||||||
|  |         this={flexRender(header.column.columnDef.header, header.getContext())} | ||||||
|  |       /> | ||||||
|  |       {#if header.column | ||||||
|  |         .getIsSorted() | ||||||
|  |         .toString() == "asc" && header.column.getCanSort()} | ||||||
|  |         🔼 | ||||||
|  |       {:else if header.column | ||||||
|  |         .getIsSorted() | ||||||
|  |         .toString() == "desc" && header.column.getCanSort()} | ||||||
|  |         🔽 | ||||||
|  |       {/if} | ||||||
|  |     </button> | ||||||
|  |   {/if} | ||||||
|  |   {#if header.column.getCanFilter()} | ||||||
|  |     <div class="relative max-w-xs"> | ||||||
|  |       <input | ||||||
|  |         title="name-search" | ||||||
|  |         value={header.column.getFilterValue() || ""} | ||||||
|  |         on:keyup={(e) => { | ||||||
|  |           header.column.setFilterValue(e.target.value); | ||||||
|  |         }} | ||||||
|  |         type="text" | ||||||
|  |         class="block w-full rounded-md border-gray-200 py-2 pl-8 text-xs focus:border-blue-500 focus:ring-blue-500" | ||||||
|  |         placeholder="" | ||||||
|  |       /> | ||||||
|  |       <div | ||||||
|  |         class="pointer-events-none absolute inset-y-0 left-0 flex items-center pl-2" | ||||||
|  |       > | ||||||
|  |         <svg | ||||||
|  |           class="h-3.5 w-3.5 text-gray-400" | ||||||
|  |           xmlns="http://www.w3.org/2000/svg" | ||||||
|  |           width={16} | ||||||
|  |           height={16} | ||||||
|  |           fill="currentColor" | ||||||
|  |           viewBox="0 0 16 16" | ||||||
|  |         > | ||||||
|  |           <path | ||||||
|  |             d="M11.742 10.344a6.5 6.5 0 1 0-1.397 1.398h-.001c.03.04.062.078.098.115l3.85 3.85a1 1 0 0 0 1.415-1.414l-3.85-3.85a1.007 1.007 0 0 0-.115-.1zM12 6.5a5.5 5.5 0 1 1-11 0 5.5 5.5 0 0 1 11 0z" | ||||||
|  |           /> | ||||||
|  |         </svg> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   {/if} | ||||||
|  | </th> | ||||||
							
								
								
									
										34
									
								
								src/components/shared/tablefilters.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								src/components/shared/tablefilters.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | export const groupFilter = (row, columnId, value) => { | ||||||
|  |     const group = row.getValue(columnId); | ||||||
|  |     if (group.responseType === "RUNNERORGANIZATION") { | ||||||
|  |         return group.name.toLowerCase().includes(value.toLowerCase()); | ||||||
|  |     } else if (value.includes(">")) { | ||||||
|  |         return ( | ||||||
|  |             `${group.parentGroup.name} > ${group.name}`.toLowerCase().includes(value.toLowerCase()) | ||||||
|  |         ); | ||||||
|  |     } else { | ||||||
|  |         return ( | ||||||
|  |             group.name.toLowerCase().includes(value.toLowerCase()) || | ||||||
|  |             group.parentGroup.name.toLowerCase().includes(value.toLowerCase()) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | export const runnerFilter = (row, columnId, value) => { | ||||||
|  |     const runner = row.getValue(columnId); | ||||||
|  |     if(!runner && value == "blanko"){return true} | ||||||
|  |     if(!runner){return false} | ||||||
|  |  | ||||||
|  |     if(value.startsWith("#")){ | ||||||
|  |         return runner.id == value.replace("#","") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if(runner.middlename){ | ||||||
|  |         return `${runner.firstname} ${runner.middlename} ${runner.lastname}`.toLowerCase().includes(value.toLowerCase()) | ||||||
|  |     } | ||||||
|  |     return `${runner.firstname} ${runner.lastname}`.toLowerCase().includes(value.toLowerCase()) | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const statusFilter = (row, columnId, value) => { | ||||||
|  |     const status = row.getValue(columnId); | ||||||
|  |     return status.toString().includes(value); | ||||||
|  | }; | ||||||
							
								
								
									
										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 { 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: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,88 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { StatsClientService } 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() { | ||||||
|  |     StatsClientService.statsClientControllerRemove(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: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} | ||||||
							
								
								
									
										147
									
								
								src/components/statsclients/CopyStatsClientTokenModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								src/components/statsclients/CopyStatsClientTokenModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,147 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |  | ||||||
|  |   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"> | ||||||
|  |     <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 | ||||||
|  |                 > | ||||||
|  |                 <button 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> | ||||||
|  |                 </button> | ||||||
|  |                 <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="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 | 
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { |   import { | ||||||
|     RunnerOrganizationService, |     RunnerOrganizationService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
| @@ -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(); | ||||||
| @@ -77,7 +77,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { RunnerTeamService } from "@odit/lfk-client-js"; |   import { RunnerTeamService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { createEventDispatcher } from "svelte"; |   import { createEventDispatcher } from "svelte"; | ||||||
| @@ -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(); | ||||||
| @@ -29,7 +29,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> |     on:click_outside={cancelDelete}> | ||||||
|     <div |     <div | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ | |||||||
|   import Teams from "./Teams.svelte"; |   import Teams from "./Teams.svelte"; | ||||||
|   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; |   import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.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] = [ | ||||||
|     {}, |     {}, | ||||||
|     {}, |     {}, | ||||||
| @@ -25,11 +26,12 @@ | |||||||
|   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_show = true; |   $: sponsoring_contracts_show = true; | ||||||
|   $: cards_show = true; |   $: cards_show = true; | ||||||
|  |   $: certificates_show = true; | ||||||
|   $: generate_teams = [original]; |   $: generate_teams = [original]; | ||||||
|   $: group = {}; |   $: group = {}; | ||||||
|   $: contact = {}; |   $: contact = {}; | ||||||
| @@ -45,6 +47,8 @@ | |||||||
|     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( |     RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|       (val) => { |       (val) => { | ||||||
|         orgs = val.map((r) => { |         orgs = val.map((r) => { | ||||||
|  |           delete r.contact; | ||||||
|  |           r.teams = []; | ||||||
|           return { label: r.name, value: r }; |           return { label: r.name, value: r }; | ||||||
|         }); |         }); | ||||||
|         group = orgs.find((g) => g.value.id == teamdata.parentGroup.id); |         group = orgs.find((g) => g.value.id == teamdata.parentGroup.id); | ||||||
| @@ -65,7 +69,7 @@ | |||||||
|     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(); | ||||||
| @@ -79,7 +83,7 @@ | |||||||
|   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; | ||||||
| @@ -90,7 +94,7 @@ | |||||||
|           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(); | ||||||
| @@ -122,6 +126,9 @@ | |||||||
|         <GenerateRunnerCards |         <GenerateRunnerCards | ||||||
|           bind:cards_show |           bind:cards_show | ||||||
|           bind:generate_teams /> |           bind:generate_teams /> | ||||||
|  |         <GenerateRunnerCertificates | ||||||
|  |           bind:certificates_show | ||||||
|  |           bind:generate_teams /> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} |         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|   | |||||||
| @@ -1,217 +1,224 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, t, _ } from "svelte-i18n"; |   import { getLocaleFromNavigator, t, _ } from "svelte-i18n"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
|   import { RunnerTeamService } from "@odit/lfk-client-js"; |   import { RunnerTeamService } from "@odit/lfk-client-js"; | ||||||
|   const teams_promise = RunnerTeamService.runnerTeamControllerGetAll(); |   const teams_promise = RunnerTeamService.runnerTeamControllerGetAll(); | ||||||
|   import store, { users as usersstore } from "../../store.js"; |   import store, { users as usersstore } from "../../store.js"; | ||||||
|   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 GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|   $: searchvalue = ""; |   import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|   $: active_deletes = []; |   $: searchvalue = ""; | ||||||
|   $: sponsoring_contracts_show = current_teams.some( |   $: active_deletes = []; | ||||||
|     (r) => r.is_selected === true |   $: sponsoring_contracts_show = current_teams.some( | ||||||
|   ); |     (r) => r.is_selected === true | ||||||
|   $: cards_show = current_teams.some( |   ); | ||||||
|     (r) => r.is_selected === true |   $: cards_show = current_teams.some( | ||||||
|   ); |     (r) => r.is_selected === true | ||||||
|   $: generate_teams = current_teams.filter((r) => r.is_selected === true); |   ); | ||||||
|   export let current_teams = []; |   $: certificates_show = current_teams.some( | ||||||
|   let modal_open = false; |     (r) => r.is_selected === true | ||||||
|   let delete_team = {}; |   ); | ||||||
|   usersstore.subscribe((val) => { |   $: generate_teams = current_teams.filter((r) => r.is_selected === true); | ||||||
|     current_teams = val; |   export let current_teams = []; | ||||||
|   }); |   let modal_open = false; | ||||||
|   teams_promise.then((data) => { |   let delete_team = {}; | ||||||
|     usersstore.set(data); |   usersstore.subscribe((val) => { | ||||||
|   }); |     current_teams = val; | ||||||
| </script> |   }); | ||||||
|  |   teams_promise.then((data) => { | ||||||
| <ConfirmTeamDeletion |     usersstore.set(data); | ||||||
|   on:cancelDelete={(event) => { |   }); | ||||||
|     modal_open = false; | </script> | ||||||
|     active_deletes[event.detail.id] = false; |  | ||||||
|   }} | <ConfirmTeamDeletion | ||||||
|   bind:modal_open |   on:cancelDelete={(event) => { | ||||||
|   bind:delete_team /> |     modal_open = false; | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')} |     active_deletes[event.detail.id] = false; | ||||||
|   {#await teams_promise} |   }} | ||||||
|     <div |   bind:modal_open | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |   bind:delete_team /> | ||||||
|       role="alert"> | {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')} | ||||||
|       <p class="font-bold">{$_('teams-are-being-loaded')}</p> |   {#await teams_promise} | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |     <div | ||||||
|     </div> |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|   {:then} |       role="alert"> | ||||||
|     {#if current_teams.length === 0} |       <p class="font-bold">{$_('teams-are-being-loaded')}</p> | ||||||
|       <TeamsEmptyState /> |       <p class="text-sm">{$_('this-might-take-a-moment')}</p> | ||||||
|     {:else} |     </div> | ||||||
|       <input |   {:then} | ||||||
|         type="search" |     {#if current_teams.length === 0} | ||||||
|         bind:value={searchvalue} |       <TeamsEmptyState /> | ||||||
|         placeholder={$_('datatable.search')} |     {:else} | ||||||
|         aria-label={$_('datatable.search')} |       <input | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         type="search" | ||||||
|       <div class="h-12"> |         bind:value={searchvalue} | ||||||
|         <GenerateSponsoringContracts |         placeholder={$_('datatable.search')} | ||||||
|           bind:sponsoring_contracts_show |         aria-label={$_('datatable.search')} | ||||||
|           bind:generate_teams /> |         class="mb-4" /> | ||||||
|         <GenerateRunnerCards |       <div class="h-12"> | ||||||
|           bind:cards_show |         <GenerateSponsoringContracts | ||||||
|           bind:generate_teams /> |           bind:sponsoring_contracts_show | ||||||
|       </div> |           bind:generate_teams /> | ||||||
|       <div |         <GenerateRunnerCards | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |           bind:cards_show | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |           bind:generate_teams /> | ||||||
|           <thead class="bg-gray-50"> |         <GenerateRunnerCertificates | ||||||
|             <tr> |           bind:certificates_show | ||||||
|               <th |           bind:generate_teams /> | ||||||
|                 scope="col" |       </div> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |       <div | ||||||
|                 <span |         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> | ||||||
|                   on:click={() => { |         <table class="divide-y divide-gray-200 w-full"> | ||||||
|                     const newstate = !current_teams.some((r) => r.is_selected === true); |           <thead class="bg-gray-50"> | ||||||
|                     current_teams = current_teams.map((r) => { |             <tr> | ||||||
|                       r.is_selected = newstate; |               <th | ||||||
|                       return r; |                 scope="col" | ||||||
|                     }); |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|                   }} |                 <span | ||||||
|                   class="underline cursor-pointer select-none">{#if current_teams.some((r) => r.is_selected === true)} |                   on:click={() => { | ||||||
|                     {$_('deselect-all')} |                     const newstate = !current_teams.some((r) => r.is_selected === true); | ||||||
|                   {:else}{$_('select-all')}{/if} |                     current_teams = current_teams.map((r) => { | ||||||
|                 </span> |                       r.is_selected = newstate; | ||||||
|               </th> |                       return r; | ||||||
|               <th |                     }); | ||||||
|                 scope="col" |                   }} | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   class="underline cursor-pointer select-none">{#if current_teams.some((r) => r.is_selected === true)} | ||||||
|                 {$_('name')} |                     {$_('deselect-all')} | ||||||
|               </th> |                   {:else}{$_('select-all')}{/if} | ||||||
|               <th |                 </span> | ||||||
|                 scope="col" |               </th> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th | ||||||
|                 {$_('organization')} |                 scope="col" | ||||||
|               </th> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|               <th |                 {$_('name')} | ||||||
|                 scope="col" |               </th> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th | ||||||
|                 {$_('contact')} |                 scope="col" | ||||||
|               </th> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |                 {$_('organization')} | ||||||
|                 <span class="sr-only">{$_('action')}</span> |               </th> | ||||||
|               </th> |               <th | ||||||
|             </tr> |                 scope="col" | ||||||
|           </thead> |                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||||
|           <tbody class="divide-y divide-gray-200"> |                 {$_('contact')} | ||||||
|             {#each current_teams as t} |               </th> | ||||||
|               {#if Object.values(t) |               <th scope="col" class="relative px-6 py-3"> | ||||||
|                 .toString() |                 <span class="sr-only">{$_('action')}</span> | ||||||
|                 .toLowerCase() |               </th> | ||||||
|                 .includes(searchvalue)} |             </tr> | ||||||
|                 <tr data-rowid="team_{t.id}"> |           </thead> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |           <tbody class="divide-y divide-gray-200"> | ||||||
|                     <input |             {#each current_teams as t} | ||||||
|                       bind:checked={t.is_selected} |               {#if Object.values(t) | ||||||
|                       type="checkbox" |                 .toString() | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |                 .toLowerCase() | ||||||
|                   </td> |                 .includes(searchvalue)} | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                 <tr data-rowid="team_{t.id}"> | ||||||
|                     <div class="flex items-center"> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                       <div class="ml-4"> |                     <input | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                       bind:checked={t.is_selected} | ||||||
|                           {t.name} |                       type="checkbox" | ||||||
|                         </div> |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | ||||||
|                       </div> |                   </td> | ||||||
|                     </div> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                   </td> |                     <div class="flex items-center"> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                       <div class="ml-4"> | ||||||
|                     <div class="flex items-center"> |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                       <div class="ml-4"> |                           {t.name} | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                         </div> | ||||||
|                           {#if t.parentGroup} |                       </div> | ||||||
|                             <a |                     </div> | ||||||
|                               href="../orgs/{t.parentGroup.id}" |                   </td> | ||||||
|                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                           {:else}{$_('no-organization-specified')}{/if} |                     <div class="flex items-center"> | ||||||
|                         </div> |                       <div class="ml-4"> | ||||||
|                       </div> |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                     </div> |                           {#if t.parentGroup} | ||||||
|                   </td> |                             <a | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                               href="../orgs/{t.parentGroup.id}" | ||||||
|                     <div class="flex items-center"> |                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a> | ||||||
|                       <div class="ml-4"> |                           {:else}{$_('no-organization-specified')}{/if} | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |                         </div> | ||||||
|                           {#if t.contact} |                       </div> | ||||||
|                             <a |                     </div> | ||||||
|                               href="../contacts/{t.contact.id}" |                   </td> | ||||||
|                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname} |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                               {t.contact.middlename || ''} |                     <div class="flex items-center"> | ||||||
|                               {t.contact.lastname}</a> |                       <div class="ml-4"> | ||||||
|                           {:else}{$_('no-contact-specified')}{/if} |                         <div class="text-sm font-medium text-gray-900"> | ||||||
|                         </div> |                           {#if t.contact} | ||||||
|                       </div> |                             <a | ||||||
|                     </div> |                               href="../contacts/{t.contact.id}" | ||||||
|                   </td> |                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname} | ||||||
|                   {#if active_deletes[t.id] === true} |                               {t.contact.middlename || ''} | ||||||
|                     <td |                               {t.contact.lastname}</a> | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                           {:else}{$_('no-contact-specified')}{/if} | ||||||
|                       <button |                         </div> | ||||||
|                         on:click={() => { |                       </div> | ||||||
|                           active_deletes[t.id] = false; |                     </div> | ||||||
|                         }} |                   </td> | ||||||
|                         tabindex="0" |                   {#if active_deletes[t.id] === true} | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">Cancel |                     <td | ||||||
|                         Delete</button> |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|                       <button |                       <button | ||||||
|                         on:click={() => { |                         on:click={() => { | ||||||
|                           RunnerTeamService.runnerTeamControllerRemove(t.id, false) |                           active_deletes[t.id] = false; | ||||||
|                             .then((resp) => { |                         }} | ||||||
|                               current_teams = current_teams.filter((obj) => obj.id !== t.id); |                         tabindex="0" | ||||||
|                               Toastify({ |                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">Cancel | ||||||
|                                 text: $_('organization-deleted'), |                         Delete</button> | ||||||
|                                 duration: 500, |                       <button | ||||||
|                                 backgroundColor: |                         on:click={() => { | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |                           RunnerTeamService.runnerTeamControllerRemove(t.id, false) | ||||||
|                               }).showToast(); |                             .then((resp) => { | ||||||
|                             }) |                               current_teams = current_teams.filter((obj) => obj.id !== t.id); | ||||||
|                             .catch((err) => { |                               Toastify({ | ||||||
|                               modal_open = true; |                                 text: $_('organization-deleted'), | ||||||
|                               delete_team = t; |                                 duration: 500, | ||||||
|                             }); |                                 backgroundColor: | ||||||
|                         }} |                                   'linear-gradient(to right, #00b09b, #96c93d)', | ||||||
|                         tabindex="0" |                               }).showToast(); | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |                             }) | ||||||
|                     </td> |                             .catch((err) => { | ||||||
|                   {:else} |                               modal_open = true; | ||||||
|                     <td |                               delete_team = t; | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |                             }); | ||||||
|                       <a |                         }} | ||||||
|                         href="./{t.id}" |                         tabindex="0" | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')} |                     </td> | ||||||
|                         <button |                   {:else} | ||||||
|                           on:click={() => { |                     <td | ||||||
|                             active_deletes[t.id] = true; |                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||||
|                           }} |                       <a | ||||||
|                           tabindex="0" |                         href="./{t.id}" | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> | ||||||
|                       {/if} |                       {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')} | ||||||
|                     </td> |                         <button | ||||||
|                   {/if} |                           on:click={() => { | ||||||
|                 </tr> |                             active_deletes[t.id] = true; | ||||||
|               {/if} |                           }} | ||||||
|             {/each} |                           tabindex="0" | ||||||
|           </tbody> |                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> | ||||||
|         </table> |                       {/if} | ||||||
|       </div> |                     </td> | ||||||
|     {/if} |                   {/if} | ||||||
|   {:catch error} |                 </tr> | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |               {/if} | ||||||
|       <span class="inline-block align-middle mr-8"> |             {/each} | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |           </tbody> | ||||||
|         {error} |         </table> | ||||||
|       </span> |       </div> | ||||||
|     </div> |     {/if} | ||||||
|   {/await} |   {:catch error} | ||||||
| {/if} |     <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,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   import { tracks as tracksstore } from "../../store.js"; |   import { tracks as tracksstore } from "../../store.js"; | ||||||
|   import { TrackService } from "@odit/lfk-client-js"; |   import { TrackService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import Toastify from "toastify-js"; | ||||||
| @@ -75,7 +75,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -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(); | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |    | ||||||
|   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"; | ||||||
| @@ -92,7 +92,7 @@ | |||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-auto" | ||||||
|     use:focusTrap |      | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|   | |||||||
| @@ -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) => { | ||||||
|   | |||||||
| @@ -36,7 +36,7 @@ | |||||||
|         bind:value={searchvalue} |         bind:value={searchvalue} | ||||||
|         placeholder={$_('datatable.search')} |         placeholder={$_('datatable.search')} | ||||||
|         aria-label={$_('datatable.search')} |         aria-label={$_('datatable.search')} | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |         class="mb-4" /> | ||||||
|       <!-- {/if} --> |       <!-- {/if} --> | ||||||
|       <!-- <button |       <!-- <button | ||||||
|         on:click={() => { |         on:click={() => { | ||||||
| @@ -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} | ||||||
|   | |||||||
| @@ -1,428 +1,491 @@ | |||||||
| { | { | ||||||
|   "404message": "Die gesuchte Seite wurde leider nicht gefunden.", |     "404message": "Die gesuchte Seite wurde leider nicht gefunden.", | ||||||
|   "404title": "Fehler 404", |     "404title": "Fehler 404", | ||||||
|   "about": "Über", |     "about": "Über", | ||||||
|   "action": "Aktionen", |     "action": "Aktionen", | ||||||
|   "active": "Aktiv", |     "active": "Aktiv", | ||||||
|   "add-card": "Karte erstellen", |     "add-card": "Karte erstellen", | ||||||
|   "add-donation": "Sponsoring erstellen", |     "add-donation": "Sponsoring erstellen", | ||||||
|   "add-donor": "Sponsor:in erstellen", |     "add-donor": "Sponsor:in erstellen", | ||||||
|   "add-scan": "Scan erstellen", |     "add-or-update-a-payment": "Zahlung hinzufügen oder bearbeiten", | ||||||
|   "add-the-first-scanstation": "Erstelle deine erste Scannerstation.", |     "add-scan": "Scan erstellen", | ||||||
|   "add-user-group": "Neue Gruppe erstellen", |     "add-the-first-scanstation": "Erstelle deine erste Scannerstation.", | ||||||
|   "add-your-first-card": "Erstelle deine erste Läuferkarte", |     "add-the-first-statsclient": "Erstelle deinen ersten Statsclient.", | ||||||
|   "add-your-first-contact": "Erstelle den ersten Kontakt", |     "add-user-group": "Neue Gruppe erstellen", | ||||||
|   "add-your-first-donor": "Erstelle die erste Sponsor:in", |     "add-your-first-card": "Erstelle deine erste Läuferkarte", | ||||||
|   "add-your-first-group": "Erstelle die erste Gruppe", |     "add-your-first-contact": "Erstelle den ersten Kontakt", | ||||||
|   "add-your-first-organization": "Erstelle die erste Organisation", |     "add-your-first-donor": "Erstelle die erste Sponsor:in", | ||||||
|   "add-your-first-runner": "Erstelle die erste Läufer:in", |     "add-your-first-group": "Erstelle die erste Gruppe", | ||||||
|   "add-your-first-team": "Erstelle das erste Team", |     "add-your-first-organization": "Erstelle die erste Organisation", | ||||||
|   "add-your-first-track": "Erstelle den ersten Track (Laufstrecke).", |     "add-your-first-runner": "Erstelle die erste Läufer:in", | ||||||
|   "add-your-first-user": "Erstelle die erste Benutzer:in", |     "add-your-first-team": "Erstelle das erste Team", | ||||||
|   "add-your-fist-donation": "Erstelle dein erstes Sponsoring", |     "add-your-first-track": "Erstelle den ersten Track (Laufstrecke).", | ||||||
|   "add-your-fist-scan": "Füge deinen ersten Scan hinzu", |     "add-your-first-user": "Erstelle die erste Benutzer:in", | ||||||
|   "adding-card": "Karte wird erstellt", |     "add-your-fist-donation": "Erstelle dein erstes Sponsoring", | ||||||
|   "adding-scan": "Scan wird hinzugefügt", |     "add-your-fist-scan": "Füge deinen ersten Scan hinzu", | ||||||
|   "address": "Adresse", |     "adding-card": "Karte wird erstellt", | ||||||
|   "address-is-required": "Du musst eine Adresse angeben", |     "adding-donation": "Sponsoring wird erstellt...", | ||||||
|   "after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!", |     "adding-scan": "Scan wird hinzugefügt", | ||||||
|   "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.", |     "address": "Adresse", | ||||||
|   "all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls gelöscht", |     "address-is-required": "Du musst eine Adresse angeben", | ||||||
|   "all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!", |     "after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!", | ||||||
|   "all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!", |     "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.", | ||||||
|   "amount": "Anzahl", |     "all": "Alle", | ||||||
|   "amount-per-kilometer": "Betrag pro Kilometer", |     "all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls gelöscht", | ||||||
|   "apartment-suite-etc": "Apartment, Wohnung, etc.", |     "all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!", | ||||||
|   "application_name": "Lauf für Kaya! - Admin", |     "all-associated-scans-will-get-deleted-as-well": "Alle Scans dieser Station werden ebenfalls gelöscht", | ||||||
|   "applying-changes": "Änderungen anwenden", |     "all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!", | ||||||
|   "attention": "Achtung!", |     "already-paid": "Bereits bezahlt", | ||||||
|   "author": "Autor:in", |     "amount": "Anzahl", | ||||||
|   "bitte-bestaetige-diese-laeufer-fuer-den-import": "Bitte die Läufer:innen für den Import bestätigen.", |     "amount-per-kilometer": "Betrag pro Kilometer", | ||||||
|   "by": "von", |     "apartment-suite-etc": "Apartment, Wohnung, etc.", | ||||||
|   "cancel": "Abbrechen", |     "application_name": "Lauf für Kaya! - Admin", | ||||||
|   "cancel-delete": "Löschen abbrechen", |     "applying-changes": "Änderungen anwenden", | ||||||
|   "cancel-keep-donor": "Abbrechen, Sponsor:in behalten", |     "attention": "Achtung!", | ||||||
|   "cancel-keep-my-profile": "Abbrechen, mein Profil behalten", |     "author": "Autor:in", | ||||||
|   "cancel-keep-organization": "Abbrechen und Organisation bearbeiten", |     "bitte-bestaetige-diese-laeufer-fuer-den-import": "Bitte die Läufer:innen für den Import bestätigen.", | ||||||
|   "cancel-keep-team": "Abbrechen, Team behalten", |     "by": "von", | ||||||
|   "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.", |     "cancel": "Abbrechen", | ||||||
|   "card-added": "Karte wurde hinzugefügt", |     "cancel-delete": "Löschen abbrechen", | ||||||
|   "card-deleted": "Karte gelöscht", |     "cancel-keep-donor": "Abbrechen, Sponsor:in behalten", | ||||||
|   "card-updated": "Karte aktualisiert", |     "cancel-keep-my-profile": "Abbrechen, mein Profil behalten", | ||||||
|   "cards": "Läuferkarten", |     "cancel-keep-organization": "Abbrechen und Organisation bearbeiten", | ||||||
|   "change-your-password-here": "Hier kannst du dein Passwort ändern", |     "cancel-keep-station": "Abbrechen und Station behalten", | ||||||
|   "changing-your-password": "Passwort wird geändert", |     "cancel-keep-statsclient": "Abbrechen und Statsclient behalten", | ||||||
|   "city": "Stadt", |     "cancel-keep-team": "Abbrechen, Team behalten", | ||||||
|   "click-to-copy-the-link-into-your-clipboard": "Klicke auf den Link, um ihn in deine Zwischenablage zu kopieren", |     "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.", | ||||||
|   "click-to-copy-token-to-clipboard": "Klicke auf den Token, um ihn in deine Zwischenablage zu kopieren", |     "card": "Läuferkarte", | ||||||
|   "close": "Schließen", |     "card-added": "Karte wurde hinzugefügt", | ||||||
|   "code": "Code", |     "card-deleted": "Karte gelöscht", | ||||||
|   "configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit", |     "card-updated": "Karte aktualisiert", | ||||||
|   "confirm": "Bestätigen", |     "cards": "Läuferkarten", | ||||||
|   "confirm-delete": "Löschung Bestätigen", |     "cards-deleted": "Karten gelöscht", | ||||||
|   "confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen", |     "certificates": "Urkunden", | ||||||
|   "confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen", |     "change-your-password-here": "Hier kannst du dein Passwort ändern", | ||||||
|   "confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.", |     "changing-your-password": "Passwort wird geändert", | ||||||
|   "confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.", |     "city": "Stadt", | ||||||
|   "confirm-deletion": "Löschung Bestätigen", |     "click-to-copy-the-link-into-your-clipboard": "Klicke auf den Link, um ihn in deine Zwischenablage zu kopieren", | ||||||
|   "confirm-the-new-password": "Neues Passwort bestätigen", |     "click-to-copy-token-to-clipboard": "Klicke auf den Token, um ihn in deine Zwischenablage zu kopieren", | ||||||
|   "contact": "Kontakt", |     "close": "Schließen", | ||||||
|   "contact-deleted": "Kontakt gelöscht", |     "code": "Code", | ||||||
|   "contact-information": "Kontaktinformation", |     "configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit", | ||||||
|   "contact-is-being-updated": "Kontakt wird aktualisiert ...", |     "confirm": "Bestätigen", | ||||||
|   "contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe", |     "confirm-delete": "Löschung Bestätigen", | ||||||
|   "contacts": "Kontakte", |     "confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen", | ||||||
|   "contacts-are-being-loaded": "Kontakte werden geladen ...", |     "confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen", | ||||||
|   "copied-link-to-clipboard": "Link wurde in die Zwischenablage kopiert", |     "confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.", | ||||||
|   "copied-token-to-clipboard": "Token wurde in die Zwischenablage kopiert", |     "confirm-delete-station-with-all-scans": "Löschen der Scannerstation mit allen Scans bestätigen", | ||||||
|   "count_organizations": "Organisationen (Anzahl)", |     "confirm-delete-statsclient": "Bestätigung, Statsclient löschen", | ||||||
|   "count_teams": "Teams (Anzahl)", |     "confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.", | ||||||
|   "create": "Erstellen", |     "confirm-deletion": "Löschung Bestätigen", | ||||||
|   "create-a-new": "Erstelle eine neue", |     "confirm-the-new-password": "Neues Passwort bestätigen", | ||||||
|   "create-a-new-card": "Neue Läuferkarte erstellen", |     "contact": "Kontakt", | ||||||
|   "create-a-new-contact": "Kontakt erstellen", |     "contact-added": "Kontakt wurde hinzugefügt", | ||||||
|   "create-a-new-distance-donation": "Erstelle ein neues Sponsoring", |     "contact-deleted": "Kontakt gelöscht", | ||||||
|   "create-a-new-donor": "Neue Sponsor:in erstellen", |     "contact-information": "Kontaktinformation", | ||||||
|   "create-a-new-fixed-donation": "Erstelle eine neue Festbetragsspende", |     "contact-is-being-added": "Kontakt wird erstellt...", | ||||||
|   "create-a-new-organization": "Neue Organisation anlegen", |     "contact-is-being-updated": "Kontakt wird aktualisiert ...", | ||||||
|   "create-a-new-runner": "Neue Läufer:in erstellen", |     "contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe", | ||||||
|   "create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)", |     "contacts": "Kontakte", | ||||||
|   "create-a-new-scanstation": "Neue Station erstellen", |     "contacts-are-being-loaded": "Kontakte werden geladen ...", | ||||||
|   "create-a-new-team": "Erstelle ein neues Team", |     "copied-link-to-clipboard": "Link wurde in die Zwischenablage kopiert", | ||||||
|   "create-a-new-track": "Neuen Track erstellen", |     "copied-token-to-clipboard": "Token wurde in die Zwischenablage kopiert", | ||||||
|   "create-a-new-user": "Neue Benutzer:in anlegen", |     "count_organizations": "Organisationen (Anzahl)", | ||||||
|   "create-a-new-user-group": "Erstelle eine neue Gruppe", |     "count_teams": "Teams (Anzahl)", | ||||||
|   "create-and-generate-pdf": "Erstellen und PDF herunterladen", |     "create": "Erstellen", | ||||||
|   "create-bulk-blanco-cards": "Blankokarten erstellen", |     "create-a-new": "Erstelle eine neue", | ||||||
|   "create-bulk-cards": "Blankokarten erstellen", |     "create-a-new-card": "Neue Läuferkarte erstellen", | ||||||
|   "create-organization": "Organisation erstellen", |     "create-a-new-contact": "Kontakt erstellen", | ||||||
|   "create-team": "Team erstellen", |     "create-a-new-distance-donation": "Erstelle ein neues Sponsoring", | ||||||
|   "create-track": "Track erstellen", |     "create-a-new-donor": "Neue Sponsor:in erstellen", | ||||||
|   "create-user": "Benutzer anlegen", |     "create-a-new-fixed-donation": "Erstelle eine neue Festbetragsspende", | ||||||
|   "create-without-pdf": "Ohne PDF erstellen", |     "create-a-new-organization": "Neue Organisation anlegen", | ||||||
|   "created-blanco-cards": "Blankokarten wurden erstellt", |     "create-a-new-runner": "Neue Läufer:in erstellen", | ||||||
|   "creating-blanco-cards": "Erstelle Blankokarten", |     "create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)", | ||||||
|   "credits": "Credits", |     "create-a-new-scanstation": "Neue Station erstellen", | ||||||
|   "csv_import__class": "Klasse", |     "create-a-new-statsclient": "Neuen Statsclient erstellen", | ||||||
|   "csv_import__firstname": "Vorname", |     "create-a-new-team": "Erstelle ein neues Team", | ||||||
|   "csv_import__lastname": "Nachname", |     "create-a-new-track": "Neuen Track erstellen", | ||||||
|   "csv_import__middlename": "Mittelname", |     "create-a-new-user": "Neue Benutzer:in anlegen", | ||||||
|   "csv_import__team": "Team", |     "create-a-new-user-group": "Erstelle eine neue Gruppe", | ||||||
|   "danger-zone": "Gefahrenzone", |     "create-and-generate-pdf": "Erstellen und PDF herunterladen", | ||||||
|   "dashboard-greeting": "Hallo", |     "create-bulk-blanco-cards": "Blankokarten erstellen", | ||||||
|   "dashboard-title": "Dashboard", |     "create-bulk-cards": "Blankokarten erstellen", | ||||||
|   "datatable": { |     "create-organization": "Organisation erstellen", | ||||||
|     "search": "🔍 Suche ...", |     "create-team": "Team erstellen", | ||||||
|     "an_error_happened_while_fetching_the_data": "Beim Abrufen der Daten ist ein Fehler aufgetreten", |     "create-track": "Track erstellen", | ||||||
|     "loading": "Wird geladen...", |     "create-user": "Benutzer anlegen", | ||||||
|     "next": "Nächste", |     "create-without-pdf": "Ohne PDF erstellen", | ||||||
|     "of": "von", |     "created-blanco-cards": "Blankokarten wurden erstellt", | ||||||
|     "previous": "Vorherige", |     "creating-blanco-cards": "Erstelle Blankokarten", | ||||||
|     "to": "bis", |     "credits": "Credits", | ||||||
|     "showing": "Zeige", |     "csv_import__class": "Klasse", | ||||||
|     "no_matching_records_found": "Keine passenden Einträge gefunden", |     "csv_import__firstname": "Vorname", | ||||||
|     "page": "Seite", |     "csv_import__lastname": "Nachname", | ||||||
|     "records": "Einträge", |     "csv_import__middlename": "Mittelname", | ||||||
|     "sort_column_ascending": "Spalte aufsteigend sortieren", |     "csv_import__team": "Team", | ||||||
|     "sort_column_descending": "Spalte absteigend sortieren" |     "danger-zone": "Gefahrenzone", | ||||||
|   }, |     "dashboard-greeting": "Hallo", | ||||||
|   "delete": "Löschen", |     "dashboard-title": "Dashboard", | ||||||
|   "delete-contact": "Kontakt löschen", |     "datatable": { | ||||||
|   "delete-donation": "Sponsporing löschen", |         "search": "🔍 Suche ...", | ||||||
|   "delete-donor": "Sponsor:in löschen", |         "an_error_happened_while_fetching_the_data": "Beim Abrufen der Daten ist ein Fehler aufgetreten", | ||||||
|   "delete-group": "Gruppe löschen", |         "loading": "Wird geladen...", | ||||||
|   "delete-organization": "Organisation löschen", |         "next": "Nächste", | ||||||
|   "delete-profile": "Profil löschen", |         "of": "von", | ||||||
|   "delete-runner": "Läufer:in löschen", |         "previous": "Vorherige", | ||||||
|   "delete-scan": "Scan löschen", |         "to": "bis", | ||||||
|   "delete-station": "Station löschen", |         "showing": "Zeige", | ||||||
|   "delete-team": "Team Löschen", |         "no_matching_records_found": "Keine passenden Einträge gefunden", | ||||||
|   "delete-user": "Benutzer:in löschen", |         "page": "Seite", | ||||||
|   "deleted-scan": "Scan wurde gelöscht", |         "records": "Einträge", | ||||||
|   "dependency_name": "Name", |         "sort_column_ascending": "Spalte aufsteigend sortieren", | ||||||
|   "description": "Beschreibung", |         "sort_column_descending": "Spalte absteigend sortieren" | ||||||
|   "description-optional": "Beschreibung (optional)", |     }, | ||||||
|   "deselect-all": "Alle abwählen", |     "delete": "Löschen", | ||||||
|   "details": "Details", |     "delete-cards": "Karten löschen", | ||||||
|   "disabled": "deaktiviert", |     "delete-contact": "Kontakt löschen", | ||||||
|   "distance": "Distanz", |     "delete-donation": "Sponsoring löschen", | ||||||
|   "distance-donation": "Sponsoring", |     "delete-donor": "Sponsor:in löschen", | ||||||
|   "distance-in-km": "Distanz (in KM)", |     "delete-group": "Gruppe löschen", | ||||||
|   "distance-track": "Distanz (+Track)", |     "delete-organization": "Organisation löschen", | ||||||
|   "do-you-really-want-to-delete-your-profile": "Möchtest du dein Profil wirklich löschen?", |     "delete-profile": "Profil löschen", | ||||||
|   "do-you-want-to-delete-the-organization-delete_org-name": "Möchtest du die Organisation {orgname} löschen?", |     "delete-runner": "Läufer:in löschen", | ||||||
|   "do-you-want-to-delete-the-team-delete_team-name": "Möchtest du das Team {teamname} löschen?", |     "delete-scan": "Scan löschen", | ||||||
|   "do-you-want-to-delete-this-donor-with-all-related-donations": "Möchtest du diese Sponsor:in mit all ihren Sponsorings löschen?", |     "delete-scans": "Scans löschen", | ||||||
|   "documentation": "Dokumentation", |     "delete-station": "Station löschen", | ||||||
|   "donation-amount": "Sponsoringbetrag", |     "delete-statsclient": "Statsclient löschen", | ||||||
|   "donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.", |     "delete-team": "Team Löschen", | ||||||
|   "donations": "Sponsorings", |     "delete-user": "Benutzer:in löschen", | ||||||
|   "donor": "Sponsor:in", |     "deleted-scan": "Scan wurde gelöscht", | ||||||
|   "donor-added": "Sponsor:in hinzugefügt", |     "dependency_name": "Name", | ||||||
|   "donor-deleted": "Sponsor:in gelöscht", |     "description": "Beschreibung", | ||||||
|   "donor-has-no-associated-donations": "Zur Sponsor:in gibt es noch keine Sponsorings", |     "description-optional": "Beschreibung (optional)", | ||||||
|   "donor-is-being-added": "Sponsor:in wird hinzugefügt...", |     "deselect-all": "Alle abwählen", | ||||||
|   "donor-is-being-updated": "Sponsor:in wird aktualisiert", |     "details": "Details", | ||||||
|   "donors": "Sponsor:innen", |     "disabled": "deaktiviert", | ||||||
|   "donors-are-being-loaded": "Sponsor:innen werden geladen", |     "distance": "Distanz", | ||||||
|   "dont-have-your-email-connected": "Deine E-Mail ist nicht verknüpft?", |     "distance-donation": "Sponsoring", | ||||||
|   "dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌", |     "distance-in-km": "Distanz (in KM)", | ||||||
|   "e-mail-adress": "E-Mail-Adresse", |     "distance-track": "Distanz (+Track)", | ||||||
|   "edit": "Bearbeiten", |     "do-you-really-want-to-delete-your-profile": "Möchtest du dein Profil wirklich löschen?", | ||||||
|   "edit-a-card": "Läuferkarte bearbeiten", |     "do-you-want-to-delete-the-organization-delete_org-name": "Möchtest du die Organisation {orgname} löschen?", | ||||||
|   "edit-permissions": "Berechtigungen bearbeiten", |     "do-you-want-to-delete-the-team-delete_team-name": "Möchtest du das Team {teamname} löschen?", | ||||||
|   "email_address_or_username": "E-Mail-Adresse/ Benutzername", |     "do-you-want-to-delete-this-donor-with-all-related-donations": "Möchtest du diese Sponsor:in mit all ihren Sponsorings löschen?", | ||||||
|   "enabled": "aktiviert", |     "documentation": "Dokumentation", | ||||||
|   "enabled_large": "Aktiviert", |     "donation-amount": "Sponsoringbetrag", | ||||||
|   "english": "Englisch", |     "donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.", | ||||||
|   "error-during-import": "Fehler beim Importieren", |     "donation-deleted": "Sponsoring gelöscht", | ||||||
|   "error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage", |     "donation-updated": "Sponsoring wurde aktualisiert", | ||||||
|   "error_on_login": "😢Fehler beim Login", |     "donation_added": "Sponsoring hinzugefügt", | ||||||
|   "erteilte": "Direkt erteilte", |     "donations": "Sponsorings", | ||||||
|   "everything-concerning-your-profile": "Alles zu deinem Profil", |     "donor": "Sponsor:in", | ||||||
|   "everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃♂️🏃♀️🏃♂️", |     "donor-added": "Sponsor:in hinzugefügt", | ||||||
|   "faq": "FAQ", |     "donor-deleted": "Sponsor:in gelöscht", | ||||||
|   "filter-by-organization-team": "Filtern nach Organisation / Team", |     "donor-has-no-associated-donations": "Zur Sponsor:in gibt es noch keine Sponsorings", | ||||||
|   "first-name": "Vorname", |     "donor-is-being-added": "Sponsor:in wird hinzugefügt...", | ||||||
|   "first-name-is-required": "Vorname muss angegeben werden", |     "donor-is-being-updated": "Sponsor:in wird aktualisiert", | ||||||
|   "first-scan-of-the-day": "Erster Scan des Tages", |     "donors": "Sponsor:innen", | ||||||
|   "fixed-donation": "Festbetragsspende", |     "donors-are-being-loaded": "Sponsor:innen werden geladen", | ||||||
|   "forgot_password": "Passwort vergessen?", |     "dont-have-your-email-connected": "Deine E-Mail ist nicht verknüpft?", | ||||||
|   "geerbte": "geerbte", |     "dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌", | ||||||
|   "general-stats": "Allgemeine Statistiken", |     "e-mail-adress": "E-Mail-Adresse", | ||||||
|   "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten", |     "edit": "Bearbeiten", | ||||||
|   "generate-runnercards": "Läuferkarten generieren", |     "edit-a-card": "Läuferkarte bearbeiten", | ||||||
|   "generate-sponsoring-contract": "Sponsoringvertrag generieren", |     "edit-permissions": "Berechtigungen bearbeiten", | ||||||
|   "generate-sponsoring-contracts": "Sponsoringverträge generieren", |     "email_address_or_username": "E-Mail-Adresse/ Benutzername", | ||||||
|   "generating-pdf": "Pdf wird generiert...", |     "enabled": "aktiviert", | ||||||
|   "generating-pdfs": "PDFs werden generiert...", |     "enabled_large": "Aktiviert", | ||||||
|   "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.", |     "english": "Englisch", | ||||||
|   "german": "Deutsch", |     "enter-payment": "Zahlung eingeben", | ||||||
|   "go-to-login": "Zum Login", |     "error-during-import": "Fehler beim Importieren", | ||||||
|   "goback": "Zur Startseite", |     "error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage", | ||||||
|   "granted": "Gewährt", |     "error_on_login": "😢Fehler beim Login", | ||||||
|   "group": "Gruppe", |     "erteilte": "Direkt erteilte", | ||||||
|   "group-added": "Gruppe hinzugefügt", |     "everything-concerning-your-profile": "Alles zu deinem Profil", | ||||||
|   "group-is-being-added": "Gruppe wird erstellt", |     "everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃♂️🏃♀️🏃♂️", | ||||||
|   "group-name-is-required": "Der Gruppenname muss angegeben werden.", |     "faq": "FAQ", | ||||||
|   "group-updated": "Gruppe aktualisiert", |     "filename_sponsoringquittungsliste": "SponsoringQuittungsListe", | ||||||
|   "groups": "Gruppen", |     "filter-by-organization-team": "Filtern nach Organisation / Team", | ||||||
|   "groups-are-being-loaded": "Gruppen werden geladen", |     "first-name": "Vorname", | ||||||
|   "home": "Start", |     "first-name-is-required": "Vorname muss angegeben werden", | ||||||
|   "icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:", |     "first-scan-of-the-day": "Erster Scan des Tages", | ||||||
|   "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.", |     "fixed-donation": "Festbetragsspende", | ||||||
|   "import-finished": "Import abgeschlossen", |     "forgot_password": "Passwort vergessen?", | ||||||
|   "import-runners": "Läufer:innen importieren", |     "geerbte": "geerbte", | ||||||
|   "import__target-organization": "Ziel Organisation", |     "general-stats": "Allgemeine Statistiken", | ||||||
|   "imprint": "Impressum ", |     "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten", | ||||||
|   "imprint-loading": "Impressum lädt...", |     "generate-runner-certificate": "Urkunde generieren", | ||||||
|   "inactive": "Inaktiv", |     "generate-runner-certificates": "Urkunden generieren", | ||||||
|   "installed-version": "Installierte Version", |     "generate-runnercards": "Läuferkarten generieren", | ||||||
|   "internal-error": "Interner Fehler", |     "generate-sponsoring-contract": "Sponsoringvertrag generieren", | ||||||
|   "invalid": "Ungültig", |     "generate-sponsoring-contracts": "Sponsoringverträge generieren", | ||||||
|   "invalid-mail-reset": "Das ist keine gültige E-Mail", |     "generating-pdf": "PDF wird generiert...", | ||||||
|   "just-enter-how-many-you-want-and-the-system-will-create-them": "Gebe einfach ein, wie viele Blankokarten das System erstellen soll.", |     "generating-pdfs": "PDFs werden generiert...", | ||||||
|   "laeufer-hinzufuegen": "Läufer:in hinzufügen", |     "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.", | ||||||
|   "laeufer-importieren": "Läufer:innen importieren", |     "german": "Deutsch", | ||||||
|   "laptime": "Rundenzeit", |     "go-to-login": "Zum Login", | ||||||
|   "last-name": "Nachname", |     "goback": "Zur Startseite", | ||||||
|   "last-name-is-required": "Nachname muss angegeben werden", |     "granted": "Gewährt", | ||||||
|   "lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.", |     "group": "Gruppe", | ||||||
|   "license": "Lizenz", |     "group-added": "Gruppe hinzugefügt", | ||||||
|   "licenses-are-being-loaded": "Lizenzen werden geladen...", |     "group-is-being-added": "Gruppe wird erstellt", | ||||||
|   "loading-cards": "Läuferkarten werden geladen", |     "group-name-is-required": "Der Gruppenname muss angegeben werden.", | ||||||
|   "loading-contact-details": "Kontaktdaten werden geladen ...", |     "group-updated": "Gruppe aktualisiert", | ||||||
|   "loading-donation-details": "Lade Sponsoringdetails", |     "groups": "Gruppen", | ||||||
|   "loading-donor-details": "Lade Details", |     "groups-are-being-loaded": "Gruppen werden geladen", | ||||||
|   "loading-group-detail": "Lade Gruppendetails...", |     "home": "Start", | ||||||
|   "loading-profile-data": "Lade Profildaten", |     "icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:", | ||||||
|   "loading-runners": "Läufer:innen werden geladen...", |     "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.", | ||||||
|   "loading-station-details": "Lade Scanstation-Details ...", |     "import-finished": "Import abgeschlossen", | ||||||
|   "log_in": "Anmelden", |     "import-runners": "Läufer:innen importieren", | ||||||
|   "log_in_to_your_account": "Bitte melde dich an", |     "import__target-organization": "Ziel Organisation", | ||||||
|   "login_is_checked": "Login wird überprüft", |     "imprint": "Impressum ", | ||||||
|   "logout": "Abmelden", |     "imprint-loading": "Impressum lädt...", | ||||||
|   "mail-validation-in-progress": "E-Mail Verifizierung läuft... ", |     "inactive": "Inaktiv", | ||||||
|   "manage-admin-users": "Nutzer verwalten", |     "installed-version": "Installierte Version", | ||||||
|   "middle-name": "Mittelname", |     "internal-error": "Interner Fehler", | ||||||
|   "minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)", |     "invalid": "Ungültig", | ||||||
|   "minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein", |     "invalid-mail-reset": "Das ist keine gültige E-Mail", | ||||||
|   "must-be-at-least-10-characters-long": "Passwort muss mindestens 10 Zeichen lang sein!", |     "just-enter-how-many-you-want-and-the-system-will-create-them": "Gebe einfach ein, wie viele Blankokarten das System erstellen soll.", | ||||||
|   "must-contain-a-lowercase-letter": "Passwort muss einen Großbuchstaben enthalten!", |     "key": "Schlüssel", | ||||||
|   "must-contain-a-number": "Passwort muss eine Zahl enthalten!", |     "laeufer-hinzufuegen": "Läufer:in hinzufügen", | ||||||
|   "must-contain-a-uppercase-letter": "Passwort muss einen Kleinbuchstaben enthalten!", |     "laeufer-importieren": "Läufer:innen importieren", | ||||||
|   "name": "Name", |     "laptime": "Rundenzeit", | ||||||
|   "name-is-required": "Der Gruppenname muss angegeben werden", |     "last-name": "Nachname", | ||||||
|   "new-password": "Neues Passwort", |     "last-name-is-required": "Nachname muss angegeben werden", | ||||||
|   "no-contact-found": "Keine Kontakte gefunden", |     "lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.", | ||||||
|   "no-contact-selected": "Kein Kontakt ausgewählt", |     "license": "Lizenz", | ||||||
|   "no-contact-specified": "Kein Kontakt angegeben", |     "licenses-are-being-loaded": "Lizenzen werden geladen...", | ||||||
|   "no-donors-found": "Keine Spender:innen gefunden", |     "loading-cards": "Läuferkarten werden geladen", | ||||||
|   "no-license-text-could-be-found": "Kein Lizenz-Text gefunden 😢", |     "loading-contact-details": "Kontaktdaten werden geladen ...", | ||||||
|   "no-organization-or-team-found": "Keine Organisationen oder Teams gefunden", |     "loading-donation-details": "Lade Sponsoringdetails", | ||||||
|   "no-organization-specified": "Keine Organisation angegeben", |     "loading-donor-details": "Lade Details", | ||||||
|   "no-organizations-found": "Keine Organisationen gefunden", |     "loading-group-detail": "Lade Gruppendetails...", | ||||||
|   "no-runners-found": "Keine Läufer:innen gefunden", |     "loading-profile-data": "Lade Profildaten", | ||||||
|   "no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.", |     "loading-runners": "Läufer:innen werden geladen...", | ||||||
|   "non-blanko": "Keine/Blankokarte", |     "loading-station-details": "Lade Scanstation-Details ...", | ||||||
|   "organization": "Organisation", |     "log_in": "Anmelden", | ||||||
|   "organization-added": "Organisation hinzugefügt", |     "log_in_to_your_account": "Bitte melde dich an", | ||||||
|   "organization-deleted": "Organisation gelöscht", |     "login_is_checked": "Login wird überprüft", | ||||||
|   "organization-detail-is-being-loaded": "Organisationsdetails werden geladen ...", |     "logout": "Abmelden", | ||||||
|   "organization-is-being-added": "Organisation wird hinzugefügt ...", |     "mail-validation-in-progress": "E-Mail Verifizierung läuft... ", | ||||||
|   "organization-name-is-required": "Der Name muss angegeben werden", |     "manage-admin-users": "Nutzer verwalten", | ||||||
|   "organizations": "Organisationen", |     "middle-name": "Mittelname", | ||||||
|   "organizations-are-being-loaded": "Organisationen werden geladen ...", |     "minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)", | ||||||
|   "orgs": "Organisationen", |     "minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein", | ||||||
|   "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!", |     "must-be-at-least-10-characters-long": "Passwort muss mindestens 10 Zeichen lang sein!", | ||||||
|   "password": "Passwort", |     "must-contain-a-lowercase-letter": "Passwort muss einen Großbuchstaben enthalten!", | ||||||
|   "password-changed": "Passwort wurde aktualisiert!", |     "must-contain-a-number": "Passwort muss eine Zahl enthalten!", | ||||||
|   "password-is-required": "Passwort muss angegeben werden", |     "must-contain-a-uppercase-letter": "Passwort muss einen Kleinbuchstaben enthalten!", | ||||||
|   "password-reset-failed": "Passwort zurücksetzen ist fehlgeschlagen!", |     "name": "Name", | ||||||
|   "password-reset-in-progress": "Passwort wird zurückgesetzt...", |     "name-is-required": "Der Gruppenname muss angegeben werden", | ||||||
|   "password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.", |     "new-password": "Neues Passwort", | ||||||
|   "password-reset-successful": "Passwort erfolgreich zurückgesetzt!", |     "no-contact-found": "Keine Kontakte gefunden", | ||||||
|   "passwords-dont-match": "Die Passwörter stimmen nicht überein!", |     "no-contact-selected": "Kein Kontakt ausgewählt", | ||||||
|   "pdf-generation-failed": "PDF Generierung fehlgeschlagen!", |     "no-contact-specified": "Kein Kontakt angegeben", | ||||||
|   "pdf-successfully-generated": "PDF wurde erfolgreich generiert!", |     "no-donors-found": "Keine Spender:innen gefunden", | ||||||
|   "pdfs-successfully-generated": "Alle PDFs wurden generiert!", |     "no-license-text-could-be-found": "Kein Lizenz-Text gefunden 😢", | ||||||
|   "per-kilometer": "pro Kilometer", |     "no-organization-or-team-found": "Keine Organisationen oder Teams gefunden", | ||||||
|   "permissions": "Berechtigungen", |     "no-organization-specified": "Keine Organisation angegeben", | ||||||
|   "permissions-updated": "Berechtigungen aktualisiert!", |     "no-organizations-found": "Keine Organisationen gefunden", | ||||||
|   "phone": "Telefon", |     "no-runners-found": "Keine Läufer:innen gefunden", | ||||||
|   "please-copy-the-token-and-store-it-somewhere-save": "Bitte kopiere dir den Token und bewahre ihn gut auf.", |     "no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.", | ||||||
|   "please-provide-a-password": "Bitte gebe ein Passwort an...", |     "non-blanko": "Keine/Blankokarte", | ||||||
|   "please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen", |     "open": "OFFEN", | ||||||
|   "please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.", |     "organization": "Organisation", | ||||||
|   "please-provide-the-nessecary-information-to-create-a-new-scan": "Bitte gebe alle notwendigen Informationen an, um einen neuen Scan zu erstellen.", |     "organization-added": "Organisation hinzugefügt", | ||||||
|   "please-provide-the-required-csv-xlsx-file": "Bitte eine CSV oder XLSX Datei hochladen.", |     "organization-deleted": "Organisation gelöscht", | ||||||
|   "please-provide-the-required-information-for-creating-a-new-user-group": "Bitte gebe alle für eine neue Gruppe notwendigen Informationen an.", |     "organization-detail-is-being-loaded": "Organisationsdetails werden geladen ...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-contact": "Bitte gebe alle nötigen Informationen an, im den neuen Kontakt zu erstellen.", |     "organization-is-being-added": "Organisation wird hinzugefügt ...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-organization": "Bitte gebe alle nötigen Informationen an, im die neue Organisation zu erstellen.", |     "organization-name-is-required": "Der Name muss angegeben werden", | ||||||
|   "please-provide-the-required-information-to-add-a-new-runner": "Bitte die benötigten Informationen angeben.", |     "organizations": "Organisationen", | ||||||
|   "please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.", |     "organizations-are-being-loaded": "Organisationen werden geladen ...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.", |     "orgs": "Organisationen", | ||||||
|   "please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.", |     "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!", | ||||||
|   "please-provide-the-required-information-to-create-a-new-scanstation": "Bitte gebe alle für eine Scannerstation notwendigen Informationen an", |     "paid": "BEZAHLT", | ||||||
|   "please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...", |     "paid-amount": "Gezahlter Betrag", | ||||||
|   "privacy": "Datenschutz", |     "password": "Passwort", | ||||||
|   "privacy-loading": "Datenschutzerklärung lädt...", |     "password-changed": "Passwort wurde aktualisiert!", | ||||||
|   "profile": "Profil", |     "password-is-required": "Passwort muss angegeben werden", | ||||||
|   "profile-picture": "Profilbild", |     "password-reset-failed": "Passwort zurücksetzen ist fehlgeschlagen!", | ||||||
|   "profile-updated": "Profil wurde aktualisiert!", |     "password-reset-in-progress": "Passwort wird zurückgesetzt...", | ||||||
|   "read-license": "Lizenz-Text lesen", |     "password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.", | ||||||
|   "receipt-needed": "Spendenquittung benötigt", |     "password-reset-successful": "Passwort erfolgreich zurückgesetzt!", | ||||||
|   "repo_link": "Link", |     "passwords-dont-match": "Die Passwörter stimmen nicht überein!", | ||||||
|   "request-a-new-reset-mail": "Neue Reset-Mail anfordern", |     "payment-amount-must-be-greater-than-0-00eur": "Der Zahlungsbetrag muss größer als 0.00€ sein!", | ||||||
|   "reset-my-password": "Passwort zurücksetzen", |     "pdf-generation-failed": "PDF Generierung fehlgeschlagen!", | ||||||
|   "reset-password": "Passwort zurücksetzen", |     "pdf-successfully-generated": "PDF wurde erfolgreich generiert!", | ||||||
|   "runner": "Läufer:in", |     "pdfs-successfully-generated": "Alle PDFs wurden generiert!", | ||||||
|   "runner-added": "Läufer:in hinzugefügt", |     "per-kilometer": "pro Kilometer", | ||||||
|   "runner-import": "Läufer:innen Import", |     "permissions": "Berechtigungen", | ||||||
|   "runner-is-being-added": "Läufer:in wird hinzugefügt...", |     "permissions-updated": "Berechtigungen aktualisiert!", | ||||||
|   "runner-updated": "Läufer:in aktualisiert!", |     "phone": "Telefon", | ||||||
|   "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen", |     "please-confirm-the-deletion-of-card": "Bitte bestätige die Löschung der Karte", | ||||||
|   "runners": "Läufer", |     "please-confirm-the-deletion-of-runner": "Bitte bestätige die Löschung der Läufer:in", | ||||||
|   "runners-are-being-imported": "Läufer:innen werden importiert ...", |     "please-confirm-the-deletion-of-scan": "Bitte bestätige die Löschung des Scans", | ||||||
|   "runners-are-being-loaded": "Läufer:innen werden geladen ...", |     "please-copy-the-token-and-store-it-somewhere-save": "Bitte kopiere dir den Token und bewahre ihn gut auf.", | ||||||
|   "save": "Speichern", |     "please-provide-a-password": "Bitte gebe ein Passwort an...", | ||||||
|   "save-changes": "Änderungen speichern", |     "please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen", | ||||||
|   "scan-added": "Scan hinzugefügt", |     "please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.", | ||||||
|   "scan-is-being-updated": "Scan wird aktualisiert", |     "please-provide-the-nessecary-information-to-create-a-new-scan": "Bitte gebe alle notwendigen Informationen an, um einen neuen Scan zu erstellen.", | ||||||
|   "scan-with-fixed-distance": "Scan mit Festdistanz", |     "please-provide-the-required-csv-xlsx-file": "Bitte eine CSV oder XLSX Datei hochladen.", | ||||||
|   "scans": "Scans", |     "please-provide-the-required-information-for-creating-a-new-user-group": "Bitte gebe alle für eine neue Gruppe notwendigen Informationen an.", | ||||||
|   "scans-are-being-loaded": "Scans werden geladen", |     "please-provide-the-required-information-to-add-a-new-contact": "Bitte gebe alle nötigen Informationen an, im den neuen Kontakt zu erstellen.", | ||||||
|   "scanstation": "Scanner Station", |     "please-provide-the-required-information-to-add-a-new-organization": "Bitte gebe alle nötigen Informationen an, im die neue Organisation zu erstellen.", | ||||||
|   "scanstation-added": "Station wurde erstellt", |     "please-provide-the-required-information-to-add-a-new-runner": "Bitte die benötigten Informationen angeben.", | ||||||
|   "scanstation-is-being-added": "Scannerstation wird angelegt...", |     "please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.", | ||||||
|   "scanstations": "Scanner Stationen", |     "please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.", | ||||||
|   "scanstations-are-being-loaded": "Scannerstationen werden geladen...", |     "please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.", | ||||||
|   "search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)", |     "please-provide-the-required-information-to-create-a-new-scanstation": "Bitte gebe alle für eine Scannerstation notwendigen Informationen an", | ||||||
|   "search-for-an-organization-or-team-by-name-or-id": "Suche eine Organisation oder ein Team (via Name oder Id)", |     "please-provide-the-required-information-to-create-a-new-statsclient": "Bitte gebe alle für einen Statsclient notwendigen Informationen an", | ||||||
|   "search-for-donor-name-or-id": "Suche eine Spender:in (via Name oder Id)", |     "please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...", | ||||||
|   "search-for-permission": "Berechtigungen durchsuchen", |     "please-wait-a-moment-your-login-is-still-being-processed": "Bitte warte einen Moment, deine Anmeldung wird verarbeitet", | ||||||
|   "search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)", |     "prefix": "Prefix", | ||||||
|   "select-all": "Alle auswählen", |     "privacy": "Datenschutz", | ||||||
|   "select-language": "Sprache auswählen", |     "privacy-loading": "Datenschutzerklärung lädt...", | ||||||
|   "selfservice-registration": "Selfservice Registrierung", |     "profile": "Profil", | ||||||
|   "send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services", |     "profile-deleted": "Profil gelöscht!", | ||||||
|   "set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen", |     "profile-picture": "Profilbild", | ||||||
|   "settings": "Einstellungen", |     "profile-updated": "Profil wurde aktualisiert!", | ||||||
|   "settings-for-your-profile": "Die Einstellungen deines Accounts", |     "read-license": "Lizenz-Text lesen", | ||||||
|   "something-about-the-group": "Infos zur Gruppe", |     "receipt-needed": "Spendenquittung benötigt", | ||||||
|   "stats-are-being-loaded": "Die Statistiken werden geladen...", |     "repo_link": "Link", | ||||||
|   "status": "Status", |     "request-a-new-reset-mail": "Neue Reset-Mail anfordern", | ||||||
|   "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können", |     "reset-my-password": "Passwort zurücksetzen", | ||||||
|   "successful-password-reset": "Passwort erfolgreich zurückgesetzt!", |     "reset-password": "Passwort zurücksetzen", | ||||||
|   "team": "Team", |     "runner": "Läufer:in", | ||||||
|   "team-detail-is-being-loaded": "Team wird geladen...", |     "runner-added": "Läufer:in hinzugefügt", | ||||||
|   "team-name": "Teamname", |     "runner-deleted": "Läufer:in gelöscht", | ||||||
|   "team-name-is-required": "Teamname ist erforderlich", |     "runner-import": "Läufer:innen Import", | ||||||
|   "teams": "Teams", |     "runner-is-being-added": "Läufer:in wird hinzugefügt...", | ||||||
|   "teams-are-being-loaded": "Teams werden geladen ...", |     "runner-updated": "Läufer:in aktualisiert!", | ||||||
|   "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...", |     "runnercards": "Laeuferkarten", | ||||||
|   "the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.", |     "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen", | ||||||
|   "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!", |     "runners": "Läufer", | ||||||
|   "there-are-no-cards-yet": "Es gibt noch keine Läuferkarten.", |     "runners-are-being-imported": "Läufer:innen werden importiert ...", | ||||||
|   "there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.", |     "runners-are-being-loaded": "Läufer:innen werden geladen ...", | ||||||
|   "there-are-no-donations-yet": "Es gibt noch keine Sponsorings", |     "save": "Speichern", | ||||||
|   "there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen", |     "save-changes": "Änderungen speichern", | ||||||
|   "there-are-no-groups-yet": "Es gibt noch keine Gruppen", |     "scan-added": "Scan hinzugefügt", | ||||||
|   "there-are-no-organizations-added-yet": "Es wurden noch keine Organisationen hinzugefügt.", |     "scan-deleted": "Scan gelöscht", | ||||||
|   "there-are-no-runners-added-yet": "Es wurden noch keine Läufer:innen hinzugefügt.", |     "scan-is-being-updated": "Scan wird aktualisiert", | ||||||
|   "there-are-no-scans-yet": "Es gibt noch keine Scans", |     "scan-with-fixed-distance": "Scan mit Festdistanz", | ||||||
|   "there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.", |     "scans": "Scans", | ||||||
|   "there-are-no-users-added-yet": "Es wurden noch keine Benutzer hinzugefügt.", |     "scans-are-being-loaded": "Scans werden geladen", | ||||||
|   "this-card-is": "Diese Karte ist", |     "scanstation": "Scanner Station", | ||||||
|   "this-might-take-a-moment": "Das könnte einen kleinen Moment dauern", |     "scanstation-added": "Station wurde erstellt", | ||||||
|   "this-scanstation-is": "Diese Station ist", |     "scanstation-is-being-added": "Scannerstation wird angelegt...", | ||||||
|   "token": "Token", |     "scanstations": "Scanner Stationen", | ||||||
|   "total-distance": "gelaufene Strecke", |     "scanstations-are-being-loaded": "Scannerstationen werden geladen...", | ||||||
|   "total-donation-amount": "Gesamtbetrag", |     "search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)", | ||||||
|   "total-donations": "Spendensumme", |     "search-for-an-organization-or-team-by-name-or-id": "Suche eine Organisation oder ein Team (via Name oder Id)", | ||||||
|   "total-scans": "gesamte Scans", |     "search-for-donor-name-or-id": "Suche eine Spender:in (via Name oder Id)", | ||||||
|   "track": "Track", |     "search-for-permission": "Berechtigungen durchsuchen", | ||||||
|   "track-added": "Track hinzugefügt", |     "search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)", | ||||||
|   "track-data-is-being-loaded": "Trackdaten werden geladen", |     "select-all": "Alle auswählen", | ||||||
|   "track-is-being-added": "Track wird hinzugefügt...", |     "select-language": "Sprache auswählen", | ||||||
|   "track-length-in-m": "Tracklänge (in Metern)", |     "selfservice-registration": "Selfservice Registrierung", | ||||||
|   "track-length-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein", |     "send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services", | ||||||
|   "track-name": "Trackname", |     "set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen", | ||||||
|   "track-name-must-not-be-empty": "Der Name muss angegeben werden", |     "settings": "Einstellungen", | ||||||
|   "tracks": "Tracks", |     "settings-for-your-profile": "Die Einstellungen deines Accounts", | ||||||
|   "update-password": "Passwort ändern", |     "something-about-the-group": "Infos zur Gruppe", | ||||||
|   "updated-contact": "Kontakt aktualisiert!", |     "sponsoring-quittungs-liste_herunterladen": "Sponsoring-Quittungs-Liste herunterladen", | ||||||
|   "updated-donor": "Sponsor:in wurde aktualisiert", |     "sponsorings": "Sponsoringerklaerungen", | ||||||
|   "updated-organization": "Organisation wurde aktualisiert", |     "station-deleted": "Scannerstation gelöscht", | ||||||
|   "updated-scan": "Scan wurde aktualisiert", |     "stats-are-being-loaded": "Die Statistiken werden geladen...", | ||||||
|   "updateing-group": "Gruppe wird aktualisiert...", |     "statsclient-deleted": "Statsclient wurde gelöscht", | ||||||
|   "updating-card": "Karte wird aktualisiert", |     "statsclient-is-being-added": "Statsclient wird angelegt...", | ||||||
|   "updating-organization": "Organisation wird aktualisiert", |     "statsclients": "Statsclient (aka Beamershow)", | ||||||
|   "updating-permissions": "Berechtigungen werden aktualisiert...", |     "statsclients-are-being-loaded": "Statsclients werden geladen", | ||||||
|   "updating-runner": "Läufer:in wird aktualisiert.", |     "status": "Status", | ||||||
|   "updating-user": "Benutzer:in wird aktualisiert...", |     "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können", | ||||||
|   "updating-your-profile": "Profil wird aktualisiert...", |     "successful-password-reset": "Passwort erfolgreich zurückgesetzt!", | ||||||
|   "user-added": "Benutzer hinzugefügt", |     "team": "Team", | ||||||
|   "user-groups": "Benutzergruppen", |     "team-added": "Team wurde hinzugefügt", | ||||||
|   "user-is-being-added": "Benutzer wird hinzugefügt ...", |     "team-deleted": "Team gelöscht", | ||||||
|   "user-updated": "Benutzer:in wurde aktualisiert", |     "team-detail-is-being-loaded": "Team wird geladen...", | ||||||
|   "username": "Benutzername", |     "team-is-being-added": "Team wird erstellt...", | ||||||
|   "users": "Benutzer", |     "team-name": "Teamname", | ||||||
|   "valid": "Gültig", |     "team-name-is-required": "Teamname ist erforderlich", | ||||||
|   "valid-city-is-required": "Du musst eine Stadt angeben", |     "teams": "Teams", | ||||||
|   "valid-email-is-required": "Es wird eine valide E-Mail Adresse benötigt", |     "teams-are-being-loaded": "Teams werden geladen ...", | ||||||
|   "valid-international-phone-number-is-required": "Du musst eine Telefonnummer im internationalen Format angeben...", |     "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...", | ||||||
|   "valid-zipcode-postal-code-is-required": "Du musst eine valide Postleitzahl angeben", |     "the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.", | ||||||
|   "verfuegbare": "Verfügbar", |     "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!", | ||||||
|   "welcome_wavinghand": "Willkommen 👋", |     "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!", | ||||||
|   "yes-i-copied-the-token": "Ja, ich habe den Token kopiert", |     "there-are-no-cards-yet": "Es gibt noch keine Läuferkarten.", | ||||||
|   "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!", |     "there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.", | ||||||
|   "you-can-now-use-your-new-password-to-log-in-to-your-account": "Du kannst dich jetzt mit deinem neuen Passwort anmelden! 🎉", |     "there-are-no-donations-yet": "Es gibt noch keine Sponsorings", | ||||||
|   "you-can-provide-a-runner-but-you-dont-have-to": "Du kannst eine Läufer:in angeben, musst aber nicht.", |     "there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen", | ||||||
|   "you-dont-have-any-scanstations-yet": "Es gibt noch keine Scannerstationen", |     "there-are-no-groups-yet": "Es gibt noch keine Gruppen", | ||||||
|   "you-have-to-provide-an-organization": "Du musst eine Organisation angeben", |     "there-are-no-organizations-added-yet": "Es wurden noch keine Organisationen hinzugefügt.", | ||||||
|   "you-have-to-save-your-changes-to-generate-a-link": "Du musst deine Änderungen speichern, um einen Link zu generieren.", |     "there-are-no-runners-added-yet": "Es wurden noch keine Läufer:innen hinzugefügt.", | ||||||
|   "you-must-create-at-least-one-card-or-cancel": "Du musst mindestens eine Blankokarte erstellen (oder abbrechen).", |     "there-are-no-scans-yet": "Es gibt noch keine Scans", | ||||||
|   "zip-postal-code": "Postleitzahl", |     "there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.", | ||||||
|   "generate-runner-certificates": "Urkunden generieren" |     "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-scanstation-is": "Diese Station ist", | ||||||
|  |     "timestamp": "Timestamp", | ||||||
|  |     "token": "Token", | ||||||
|  |     "total-distance": "gelaufene Strecke", | ||||||
|  |     "total-donation-amount": "Gesamtbetrag", | ||||||
|  |     "total-donations": "Spendensumme", | ||||||
|  |     "total-paid-amount": "Gezahlter Gesamtbetrag", | ||||||
|  |     "total-scans": "gesamte Scans", | ||||||
|  |     "total_donation_amount_in_eur": "Gesamtbetrag in €", | ||||||
|  |     "track": "Track", | ||||||
|  |     "track-added": "Track hinzugefügt", | ||||||
|  |     "track-data-is-being-loaded": "Trackdaten werden geladen", | ||||||
|  |     "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-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein", | ||||||
|  |     "track-name": "Trackname", | ||||||
|  |     "track-name-must-not-be-empty": "Der Name muss angegeben werden", | ||||||
|  |     "track-was-updated": "Track wurde aktualisiert", | ||||||
|  |     "tracks": "Tracks", | ||||||
|  |     "unpaid": "Offen", | ||||||
|  |     "update-card": "Karte aktualisieren", | ||||||
|  |     "update-password": "Passwort ändern", | ||||||
|  |     "updated-contact": "Kontakt aktualisiert!", | ||||||
|  |     "updated-donor": "Sponsor:in wurde aktualisiert", | ||||||
|  |     "updated-organization": "Organisation wurde aktualisiert", | ||||||
|  |     "updated-scan": "Scan wurde aktualisiert", | ||||||
|  |     "updated-team": "Team wurde aktualisiert", | ||||||
|  |     "updateing-group": "Gruppe wird aktualisiert...", | ||||||
|  |     "updating-card": "Karte wird aktualisiert", | ||||||
|  |     "updating-donation": "Sponsoring wird aktualisiert", | ||||||
|  |     "updating-organization": "Organisation wird aktualisiert", | ||||||
|  |     "updating-permissions": "Berechtigungen werden aktualisiert...", | ||||||
|  |     "updating-runner": "Läufer:in wird aktualisiert.", | ||||||
|  |     "updating-team": "Team wird aktualisiert", | ||||||
|  |     "updating-user": "Benutzer:in wird aktualisiert...", | ||||||
|  |     "updating-your-profile": "Profil wird aktualisiert...", | ||||||
|  |     "user-added": "Benutzer hinzugefügt", | ||||||
|  |     "user-groups": "Benutzergruppen", | ||||||
|  |     "user-is-being-added": "Benutzer wird hinzugefügt ...", | ||||||
|  |     "user-updated": "Benutzer:in wurde aktualisiert", | ||||||
|  |     "username": "Benutzername", | ||||||
|  |     "users": "Benutzer", | ||||||
|  |     "valid": "Gültig", | ||||||
|  |     "valid-city-is-required": "Du musst eine Stadt angeben", | ||||||
|  |     "valid-email-is-required": "Es wird eine valide E-Mail Adresse benötigt", | ||||||
|  |     "valid-international-phone-number-is-required": "Du musst eine Telefonnummer im internationalen Format angeben...", | ||||||
|  |     "valid-zipcode-postal-code-is-required": "Du musst eine valide Postleitzahl angeben", | ||||||
|  |     "verfuegbare": "Verfügbar", | ||||||
|  |     "welcome_wavinghand": "Willkommen 👋", | ||||||
|  |     "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-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-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-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" | ||||||
|  | } | ||||||
| @@ -1,429 +1,491 @@ | |||||||
| { | { | ||||||
|   "404message": "Sorry, the page you are looking for could not be found.", |     "404message": "Sorry, the page you are looking for could not be found.", | ||||||
|   "404title": "Error 404", |     "404title": "Error 404", | ||||||
|   "about": "About", |     "about": "About", | ||||||
|   "action": "Action", |     "action": "Action", | ||||||
|   "active": "Active", |     "active": "Active", | ||||||
|   "add-card": "Add Card", |     "add-card": "Add Card", | ||||||
|   "add-donation": "Add donation", |     "add-donation": "Add donation", | ||||||
|   "add-donor": "add donor", |     "add-donor": "Add donor", | ||||||
|   "add-scan": "Add scan", |     "add-or-update-a-payment": "Add or update a payment", | ||||||
|   "add-the-first-scanstation": "Add your first scanstation.", |     "add-scan": "Add scan", | ||||||
|   "add-user-group": "Add User Group", |     "add-the-first-scanstation": "Add your first scanstation.", | ||||||
|   "add-your-first-card": "Add your first card", |     "add-the-first-statsclient": "Add your first statsclient.", | ||||||
|   "add-your-first-contact": "Add your first contact", |     "add-user-group": "Add User Group", | ||||||
|   "add-your-first-donor": "add your first donor", |     "add-your-first-card": "Add your first card", | ||||||
|   "add-your-first-group": "Add your first group", |     "add-your-first-contact": "Add your first contact", | ||||||
|   "add-your-first-organization": "Add your first organization", |     "add-your-first-donor": "add your first donor", | ||||||
|   "add-your-first-runner": "Add your first runner", |     "add-your-first-group": "Add your first group", | ||||||
|   "add-your-first-team": "Add your first team", |     "add-your-first-organization": "Add your first organization", | ||||||
|   "add-your-first-track": "Add your first track.", |     "add-your-first-runner": "Add your first runner", | ||||||
|   "add-your-first-user": "Add your first user", |     "add-your-first-team": "Add your first team", | ||||||
|   "add-your-fist-donation": "Add your fist donation", |     "add-your-first-track": "Add your first track.", | ||||||
|   "add-your-fist-scan": "Add your fist scan", |     "add-your-first-user": "Add your first user", | ||||||
|   "adding-card": "Adding Card", |     "add-your-fist-donation": "Add your fist donation", | ||||||
|   "adding-scan": "Adding Scan", |     "add-your-fist-scan": "Add your fist scan", | ||||||
|   "address": "Address", |     "adding-card": "Adding Card", | ||||||
|   "address-is-required": "Address is required", |     "adding-donation": "Adding donation...", | ||||||
|   "after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!", |     "adding-scan": "Adding Scan", | ||||||
|   "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.", |     "address": "Address", | ||||||
|   "all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well", |     "address-is-required": "Address is required", | ||||||
|   "all-associated-runners-will-be-deleted-too": "All associated runners will be deleted too!", |     "after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!", | ||||||
|   "all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!", |     "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.", | ||||||
|   "amount": "Amount", |     "all": "all", | ||||||
|   "amount-per-kilometer": "Amount per kilometer", |     "all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well", | ||||||
|   "apartment-suite-etc": "Apartment, suite, etc.", |     "all-associated-runners-will-be-deleted-too": "All associated runners will be deleted too!", | ||||||
|   "application_name": "Lauf für Kaya! - Admin", |     "all-associated-scans-will-get-deleted-as-well": "All associated scans will get deleted as well", | ||||||
|   "applying-changes": "Applying Changes", |     "all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!", | ||||||
|   "attention": "Attention!", |     "already-paid": "Already paid", | ||||||
|   "author": "Author", |     "amount": "Amount", | ||||||
|   "bitte-bestaetige-diese-laeufer-fuer-den-import": "Please confirm these runners for import.", |     "amount-per-kilometer": "Amount per kilometer", | ||||||
|   "by": "by", |     "apartment-suite-etc": "Apartment, suite, etc.", | ||||||
|   "cancel": "Cancel", |     "application_name": "Lauf für Kaya! - Admin", | ||||||
|   "cancel-delete": "Cancel Delete", |     "applying-changes": "Applying Changes", | ||||||
|   "cancel-keep-donor": "Cancel, keep donor", |     "attention": "Attention!", | ||||||
|   "cancel-keep-my-profile": "Cancel, keep my profile", |     "author": "Author", | ||||||
|   "cancel-keep-organization": "Cancel, keep organization", |     "bitte-bestaetige-diese-laeufer-fuer-den-import": "Please confirm these runners for import.", | ||||||
|   "cancel-keep-team": "Cancel, keep team", |     "by": "by", | ||||||
|   "cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity", |     "cancel": "Cancel", | ||||||
|   "card-added": "Card added", |     "cancel-delete": "Cancel Delete", | ||||||
|   "card-deleted": "Card deleted", |     "cancel-keep-donor": "Cancel, keep donor", | ||||||
|   "card-updated": "Card updated", |     "cancel-keep-my-profile": "Cancel, keep my profile", | ||||||
|   "cards": "Cards", |     "cancel-keep-organization": "Cancel, keep organization", | ||||||
|   "change-your-password-here": "Change your password here", |     "cancel-keep-station": "Cancel, keep station", | ||||||
|   "changing-your-password": "Changing your password", |     "cancel-keep-statsclient": "Cancel and keep statsclient", | ||||||
|   "city": "City", |     "cancel-keep-team": "Cancel, keep team", | ||||||
|   "click-to-copy-the-link-into-your-clipboard": "Click to copy the link into your clipboard", |     "cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity", | ||||||
|   "click-to-copy-token-to-clipboard": "Click to copy the token to your clipboard", |     "card": "card", | ||||||
|   "close": "Close", |     "card-added": "Card added", | ||||||
|   "code": "Code", |     "card-deleted": "Card deleted", | ||||||
|   "configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times", |     "card-updated": "Card updated", | ||||||
|   "confirm": "Confirm", |     "cards": "Cards", | ||||||
|   "confirm-delete": "Confirm Delete", |     "cards-deleted": "Cards deleted", | ||||||
|   "confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations", |     "certificates": "Certificates", | ||||||
|   "confirm-delete-my-user-profile": "Confirm, delete my user profile", |     "change-your-password-here": "Change your password here", | ||||||
|   "confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.", |     "changing-your-password": "Changing your password", | ||||||
|   "confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.", |     "city": "City", | ||||||
|   "confirm-deletion": "Confirm Deletion", |     "click-to-copy-the-link-into-your-clipboard": "Click to copy the link into your clipboard", | ||||||
|   "confirm-the-new-password": "Confirm the new password", |     "click-to-copy-token-to-clipboard": "Click to copy the token to your clipboard", | ||||||
|   "contact": "Contact", |     "close": "Close", | ||||||
|   "contact-deleted": "Contact deleted", |     "code": "Code", | ||||||
|   "contact-information": "Contact Information", |     "configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times", | ||||||
|   "contact-is-being-updated": "Contact is being updated...", |     "confirm": "Confirm", | ||||||
|   "contact-is-not-a-member-in-any-group": "Contact is not a member in any group", |     "confirm-delete": "Confirm Delete", | ||||||
|   "contacts": "Contacts", |     "confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations", | ||||||
|   "contacts-are-being-loaded": "contacts are being loaded...", |     "confirm-delete-my-user-profile": "Confirm, delete my user profile", | ||||||
|   "copied-link-to-clipboard": "Copied link to clipboard", |     "confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.", | ||||||
|   "copied-token-to-clipboard": "Copied token to clipboard", |     "confirm-delete-station-with-all-scans": "Confirm deletion of station with all scans", | ||||||
|   "count_organizations": "# Organizations", |     "confirm-delete-statsclient": "Confirm, delete statsclient", | ||||||
|   "count_teams": "# Teams", |     "confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.", | ||||||
|   "create": "Create", |     "confirm-deletion": "Confirm Deletion", | ||||||
|   "create-a-new": "Create a new", |     "confirm-the-new-password": "Confirm the new password", | ||||||
|   "create-a-new-card": "Create a new card", |     "contact": "Contact", | ||||||
|   "create-a-new-contact": "Create a new contact", |     "contact-added": "Contact added", | ||||||
|   "create-a-new-distance-donation": "Create a new distance donation", |     "contact-deleted": "Contact deleted", | ||||||
|   "create-a-new-donor": "Create a new donor", |     "contact-information": "Contact Information", | ||||||
|   "create-a-new-fixed-donation": "Create a new fixed donation", |     "contact-is-being-added": "Contact is being added...", | ||||||
|   "create-a-new-organization": "Create a new Organization", |     "contact-is-being-updated": "Contact is being updated...", | ||||||
|   "create-a-new-runner": "Create a new Runner", |     "contact-is-not-a-member-in-any-group": "Contact is not a member in any group", | ||||||
|   "create-a-new-scan-fixed-only": "Create a new scan (fixed only)", |     "contacts": "Contacts", | ||||||
|   "create-a-new-scanstation": "Create a new station", |     "contacts-are-being-loaded": "contacts are being loaded...", | ||||||
|   "create-a-new-team": "Create a new team", |     "copied-link-to-clipboard": "Copied link to clipboard", | ||||||
|   "create-a-new-track": "Create a new Track", |     "copied-token-to-clipboard": "Copied token to clipboard", | ||||||
|   "create-a-new-user": "Create a new User", |     "count_organizations": "# Organizations", | ||||||
|   "create-a-new-user-group": "Create a new user group", |     "count_teams": "# Teams", | ||||||
|   "create-and-generate-pdf": "Create and generate PDF", |     "create": "Create", | ||||||
|   "create-bulk-blanco-cards": "Create bulk blanco cards", |     "create-a-new": "Create a new", | ||||||
|   "create-bulk-cards": "Add blanco cards", |     "create-a-new-card": "Create a new card", | ||||||
|   "create-organization": "Create Organization", |     "create-a-new-contact": "Create a new contact", | ||||||
|   "create-team": "Create Team", |     "create-a-new-distance-donation": "Create a new distance donation", | ||||||
|   "create-track": "Create Track", |     "create-a-new-donor": "Create a new donor", | ||||||
|   "create-user": "Create User", |     "create-a-new-fixed-donation": "Create a new fixed donation", | ||||||
|   "create-without-pdf": "Create without PDF", |     "create-a-new-organization": "Create a new Organization", | ||||||
|   "created-blanco-cards": "Created blanco cards", |     "create-a-new-runner": "Create a new Runner", | ||||||
|   "creating-blanco-cards": "Creating blanco cards", |     "create-a-new-scan-fixed-only": "Create a new scan (fixed only)", | ||||||
|   "credits": "Credits", |     "create-a-new-scanstation": "Create a new station", | ||||||
|   "csv_import__class": "Class", |     "create-a-new-statsclient": "Create a new statsclient", | ||||||
|   "csv_import__firstname": "Firstname", |     "create-a-new-team": "Create a new team", | ||||||
|   "csv_import__lastname": "Lastname", |     "create-a-new-track": "Create a new Track", | ||||||
|   "csv_import__middlename": "Middlename", |     "create-a-new-user": "Create a new User", | ||||||
|   "csv_import__team": "Team", |     "create-a-new-user-group": "Create a new user group", | ||||||
|   "danger-zone": "Danger zone", |     "create-and-generate-pdf": "Create and generate PDF", | ||||||
|   "dashboard-greeting": "Hello", |     "create-bulk-blanco-cards": "Create bulk blanco cards", | ||||||
|   "dashboard-title": "Dashboard", |     "create-bulk-cards": "Add blanco cards", | ||||||
|   "datatable": { |     "create-organization": "Create Organization", | ||||||
|     "search": "🔍 Search...", |     "create-team": "Create Team", | ||||||
|     "sort_column_ascending": "Sort column ascending", |     "create-track": "Create Track", | ||||||
|     "sort_column_descending": "Sort column descending", |     "create-user": "Create User", | ||||||
|     "previous": "Previous", |     "create-without-pdf": "Create without PDF", | ||||||
|     "next": "Next", |     "created-blanco-cards": "Created blanco cards", | ||||||
|     "page": "Page", |     "creating-blanco-cards": "Creating blanco cards", | ||||||
|     "showing": "Showing", |     "credits": "Credits", | ||||||
|     "records": "Records", |     "csv_import__class": "Class", | ||||||
|     "of": "of", |     "csv_import__firstname": "Firstname", | ||||||
|     "to": "to", |     "csv_import__lastname": "Lastname", | ||||||
|     "loading": "Loading...", |     "csv_import__middlename": "Middlename", | ||||||
|     "no_matching_records_found": "No matching records found", |     "csv_import__team": "Team", | ||||||
|     "an_error_happened_while_fetching_the_data": "An error happened while fetching the data" |     "danger-zone": "Danger zone", | ||||||
|   }, |     "dashboard-greeting": "Hello", | ||||||
|   "delete": "Delete", |     "dashboard-title": "Dashboard", | ||||||
|   "delete-contact": "Delete Contact", |     "datatable": { | ||||||
|   "delete-donation": "Delete Donation", |         "search": "🔍 Search...", | ||||||
|   "delete-donor": "Delete donor", |         "sort_column_ascending": "Sort column ascending", | ||||||
|   "delete-group": "Delete Group", |         "sort_column_descending": "Sort column descending", | ||||||
|   "delete-organization": "Delete Organization", |         "previous": "Previous", | ||||||
|   "delete-profile": "Delete Profile", |         "next": "Next", | ||||||
|   "delete-runner": "Delete Runner", |         "page": "Page", | ||||||
|   "delete-scan": "Delete scan", |         "showing": "Showing", | ||||||
|   "delete-station": "Delete station", |         "records": "Records", | ||||||
|   "delete-team": "Delete Team", |         "of": "of", | ||||||
|   "delete-user": "Delete User", |         "to": "to", | ||||||
|   "deleted-scan": "Deleted scan", |         "loading": "Loading...", | ||||||
|   "dependency_name": "Name", |         "no_matching_records_found": "No matching records found", | ||||||
|   "description": "description", |         "an_error_happened_while_fetching_the_data": "An error happened while fetching the data" | ||||||
|   "description-optional": "Description (optional)", |     }, | ||||||
|   "deselect-all": "deselect all", |     "delete": "Delete", | ||||||
|   "details": "Details", |     "delete-cards": "Delete cards", | ||||||
|   "disabled": "disabled", |     "delete-contact": "Delete Contact", | ||||||
|   "distance": "Distance", |     "delete-donation": "Delete Donation", | ||||||
|   "distance-donation": "distance donation", |     "delete-donor": "Delete donor", | ||||||
|   "distance-in-km": "Distance in km", |     "delete-group": "Delete Group", | ||||||
|   "distance-track": "Distance (+Track)", |     "delete-organization": "Delete Organization", | ||||||
|   "do-you-really-want-to-delete-your-profile": "Do you really want to delete your profile?", |     "delete-profile": "Delete Profile", | ||||||
|   "do-you-want-to-delete-the-organization-delete_org-name": "Do you want to delete the organization {orgname}?", |     "delete-runner": "Delete Runner", | ||||||
|   "do-you-want-to-delete-the-team-delete_team-name": "Do you want to delete the team {teamname}?", |     "delete-scan": "Delete scan", | ||||||
|   "do-you-want-to-delete-this-donor-with-all-related-donations": "Do you want to delete this donor with all related donations", |     "delete-scans": "Delete scans", | ||||||
|   "documentation": "Documentation", |     "delete-station": "Delete station", | ||||||
|   "donation-amount": "Donation amount", |     "delete-statsclient": "Delete statsclient", | ||||||
|   "donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€", |     "delete-team": "Delete Team", | ||||||
|   "donations": "Donations", |     "delete-user": "Delete User", | ||||||
|   "donor": "Donor", |     "deleted-scan": "Deleted scan", | ||||||
|   "donor-added": "Donor added", |     "dependency_name": "Name", | ||||||
|   "donor-deleted": "donor deleted", |     "description": "description", | ||||||
|   "donor-has-no-associated-donations": "Donor has no associated donations.", |     "description-optional": "Description (optional)", | ||||||
|   "donor-is-being-added": "Donor is being added...", |     "deselect-all": "deselect all", | ||||||
|   "donor-is-being-updated": "Donor is being updated", |     "details": "Details", | ||||||
|   "donors": "Donors", |     "disabled": "disabled", | ||||||
|   "donors-are-being-loaded": "donors are being loaded", |     "distance": "Distance", | ||||||
|   "dont-have-your-email-connected": "Don't have your email connected?", |     "distance-donation": "distance donation", | ||||||
|   "dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌", |     "distance-in-km": "Distance in km", | ||||||
|   "e-mail-adress": "E-Mail Adress", |     "distance-track": "Distance (+Track)", | ||||||
|   "edit": "Edit", |     "do-you-really-want-to-delete-your-profile": "Do you really want to delete your profile?", | ||||||
|   "edit-a-card": "Edit a card", |     "do-you-want-to-delete-the-organization-delete_org-name": "Do you want to delete the organization {orgname}?", | ||||||
|   "edit-permissions": "edit permissions", |     "do-you-want-to-delete-the-team-delete_team-name": "Do you want to delete the team {teamname}?", | ||||||
|   "email_address_or_username": "Email / username", |     "do-you-want-to-delete-this-donor-with-all-related-donations": "Do you want to delete this donor with all related donations", | ||||||
|   "enabled": "enabled", |     "documentation": "Documentation", | ||||||
|   "enabled_large": "Enabled", |     "donation-amount": "Donation amount", | ||||||
|   "english": "English", |     "donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€", | ||||||
|   "error-during-import": "Error during import", |     "donation-deleted": "Donation deleted", | ||||||
|   "error-whyile-copying-to-clipboard": "Error while copying to clipboard", |     "donation-updated": "Donation updated", | ||||||
|   "error_on_login": "Error on login", |     "donation_added": "Donation_added", | ||||||
|   "erteilte": "Directly granted", |     "donations": "Donations", | ||||||
|   "everything-concerning-your-profile": "Everything concerning your profile", |     "donor": "Donor", | ||||||
|   "everything-is-more-fun-together": "everything is more fun together 🏃♂️🏃♀️🏃♂️", |     "donor-added": "Donor added", | ||||||
|   "faq": "FAQ", |     "donor-deleted": "donor deleted", | ||||||
|   "filter-by-organization-team": "Filter by Organization/ Team", |     "donor-has-no-associated-donations": "Donor has no associated donations.", | ||||||
|   "first-name": "First name", |     "donor-is-being-added": "Donor is being added...", | ||||||
|   "first-name-is-required": "First Name is required", |     "donor-is-being-updated": "Donor is being updated", | ||||||
|   "first-scan-of-the-day": "First scan of the day.", |     "donors": "Donors", | ||||||
|   "fixed-donation": "fixed donation", |     "donors-are-being-loaded": "donors are being loaded", | ||||||
|   "forgot_password": "Forgot your password?", |     "dont-have-your-email-connected": "Don't have your email connected?", | ||||||
|   "geerbte": "inherited", |     "dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌", | ||||||
|   "general-stats": "General Stats", |     "e-mail-adress": "E-Mail Adress", | ||||||
|   "general_promise_error": "😢 Error", |     "edit": "Edit", | ||||||
|   "generate-runnercards": "Generate Runnercards", |     "edit-a-card": "Edit a card", | ||||||
|   "generate-sponsoring-contract": "generate sponsoring contract", |     "edit-permissions": "edit permissions", | ||||||
|   "generate-sponsoring-contracts": "generate sponsoring contracts", |     "email_address_or_username": "Email / username", | ||||||
|   "generating-pdf": "generating PDF...", |     "enabled": "enabled", | ||||||
|   "generating-pdfs": "generating PDFs...", |     "enabled_large": "Enabled", | ||||||
|   "generic-ui-logic-error": "Something went wrong in the UI logic", |     "english": "English", | ||||||
|   "german": "German", |     "enter-payment": "Enter payment", | ||||||
|   "go-to-login": "Go To Login", |     "error-during-import": "Error during import", | ||||||
|   "goback": "Go Home", |     "error-whyile-copying-to-clipboard": "Error while copying to clipboard", | ||||||
|   "granted": "granted", |     "error_on_login": "Error on login", | ||||||
|   "group": "Group", |     "erteilte": "Directly granted", | ||||||
|   "group-added": "Group added", |     "everything-concerning-your-profile": "Everything concerning your profile", | ||||||
|   "group-is-being-added": "Group is being added...", |     "everything-is-more-fun-together": "everything is more fun together 🏃♂️🏃♀️🏃♂️", | ||||||
|   "group-name-is-required": "Group name is required", |     "faq": "FAQ", | ||||||
|   "group-updated": "group updated", |     "filename_sponsoringquittungsliste": "DonorReceiptList", | ||||||
|   "groups": "Groups", |     "filter-by-organization-team": "Filter by Organization/ Team", | ||||||
|   "groups-are-being-loaded": "Groups are being loaded", |     "first-name": "First name", | ||||||
|   "home": "Home", |     "first-name-is-required": "First Name is required", | ||||||
|   "icon-image-credits": "We also want to thank these projects for illustrations and icons:", |     "first-scan-of-the-day": "First scan of the day.", | ||||||
|   "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.", |     "fixed-donation": "fixed donation", | ||||||
|   "import-finished": "Import finished", |     "forgot_password": "Forgot your password?", | ||||||
|   "import-runners": "Import runners", |     "geerbte": "inherited", | ||||||
|   "import__target-organization": "Target Organization", |     "general-stats": "General Stats", | ||||||
|   "imprint": "Imprint", |     "general_promise_error": "😢 Error", | ||||||
|   "imprint-loading": "Imprint loading...", |     "generate-runner-certificate": "Generate runner certificate", | ||||||
|   "inactive": "Inactive", |     "generate-runner-certificates": "Generate runner certificates", | ||||||
|   "installed-version": "Installed version", |     "generate-runnercards": "Generate Runnercards", | ||||||
|   "internal-error": "Internal Error", |     "generate-sponsoring-contract": "generate sponsoring contract", | ||||||
|   "invalid": "Invalid", |     "generate-sponsoring-contracts": "generate sponsoring contracts", | ||||||
|   "invalid-mail-reset": "the provided email is invalid", |     "generating-pdf": "generating PDF...", | ||||||
|   "just-enter-how-many-you-want-and-the-system-will-create-them": "Just enter how many you want and the system will create them", |     "generating-pdfs": "generating PDFs...", | ||||||
|   "laeufer-hinzufuegen": "Add runner", |     "generic-ui-logic-error": "Something went wrong in the UI logic", | ||||||
|   "laeufer-importieren": "Läufer importieren", |     "german": "German", | ||||||
|   "laptime": "Laptime", |     "go-to-login": "Go To Login", | ||||||
|   "last-name": "Last name", |     "goback": "Go Home", | ||||||
|   "last-name-is-required": "Last Name is required", |     "granted": "granted", | ||||||
|   "lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.", |     "group": "Group", | ||||||
|   "license": "License", |     "group-added": "Group added", | ||||||
|   "licenses-are-being-loaded": "Licenses are being loaded...", |     "group-is-being-added": "Group is being added...", | ||||||
|   "loading-cards": "Loading cards", |     "group-name-is-required": "Group name is required", | ||||||
|   "loading-contact-details": "Loading contact details...", |     "group-updated": "group updated", | ||||||
|   "loading-donation-details": "Loading donation details", |     "groups": "Groups", | ||||||
|   "loading-donor-details": "Loading donor details", |     "groups-are-being-loaded": "Groups are being loaded", | ||||||
|   "loading-group-detail": "Loading group detail...", |     "home": "Home", | ||||||
|   "loading-profile-data": "Loading profile data", |     "icon-image-credits": "We also want to thank these projects for illustrations and icons:", | ||||||
|   "loading-runners": "loading runners...", |     "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.", | ||||||
|   "loading-station-details": "Loading station details", |     "import-finished": "Import finished", | ||||||
|   "log_in": "Log in", |     "import-runners": "Import runners", | ||||||
|   "log_in_to_your_account": "Log in to your account", |     "import__target-organization": "Target Organization", | ||||||
|   "login_is_checked": "Login is being checked...", |     "imprint": "Imprint", | ||||||
|   "logout": "Logout", |     "imprint-loading": "Imprint loading...", | ||||||
|   "mail-validation-in-progress": "mail validation in progress...", |     "inactive": "Inactive", | ||||||
|   "manage-admin-users": "manage admin users", |     "installed-version": "Installed version", | ||||||
|   "middle-name": "Middle name", |     "internal-error": "Internal Error", | ||||||
|   "minimum-lap-time-in-s": "minimum lap time in s", |     "invalid": "Invalid", | ||||||
|   "minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0", |     "invalid-mail-reset": "the provided email is invalid", | ||||||
|   "must-be-at-least-10-characters-long": "Must be at least 10 characters long!", |     "just-enter-how-many-you-want-and-the-system-will-create-them": "Just enter how many you want and the system will create them", | ||||||
|   "must-contain-a-lowercase-letter": "Must contain a lowercase letter!", |     "key": "Key", | ||||||
|   "must-contain-a-number": "Must contain a number!", |     "laeufer-hinzufuegen": "Add runner", | ||||||
|   "must-contain-a-uppercase-letter": "Must contain a uppercase letter!", |     "laeufer-importieren": "Läufer importieren", | ||||||
|   "name": "Name", |     "laptime": "Laptime", | ||||||
|   "name-is-required": "Name is required", |     "last-name": "Last name", | ||||||
|   "new-password": "New password", |     "last-name-is-required": "Last Name is required", | ||||||
|   "no-contact-found": "No contacts found", |     "lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.", | ||||||
|   "no-contact-selected": "No contact selected", |     "license": "License", | ||||||
|   "no-contact-specified": "no contact specified", |     "licenses-are-being-loaded": "Licenses are being loaded...", | ||||||
|   "no-donors-found": "No donors found", |     "loading-cards": "Loading cards", | ||||||
|   "no-license-text-could-be-found": "No license text could be found 😢", |     "loading-contact-details": "Loading contact details...", | ||||||
|   "no-organization-or-team-found": "No organization or team found", |     "loading-donation-details": "Loading donation details", | ||||||
|   "no-organization-specified": "no organization specified", |     "loading-donor-details": "Loading donor details", | ||||||
|   "no-organizations-found": "No organizations found", |     "loading-group-detail": "Loading group detail...", | ||||||
|   "no-runners-found": "No runners found", |     "loading-profile-data": "Loading profile data", | ||||||
|   "no-tracks-added-yet": "there are no tracks added yet.", |     "loading-runners": "loading runners...", | ||||||
|   "non-blanko": "Non/Blanko", |     "loading-station-details": "Loading station details", | ||||||
|   "organization": "Organization", |     "log_in": "Log in", | ||||||
|   "organization-added": "Organization added", |     "log_in_to_your_account": "Log in to your account", | ||||||
|   "organization-deleted": "Organization deleted", |     "login_is_checked": "Login is being checked...", | ||||||
|   "organization-detail-is-being-loaded": "organization detail is being loaded...", |     "logout": "Logout", | ||||||
|   "organization-is-being-added": "Organization is being added...", |     "mail-validation-in-progress": "mail validation in progress...", | ||||||
|   "organization-name-is-required": "Organization name is required", |     "manage-admin-users": "manage admin users", | ||||||
|   "organizations": "Organizations", |     "middle-name": "Middle name", | ||||||
|   "organizations-are-being-loaded": "organizations are being loaded...", |     "minimum-lap-time-in-s": "minimum lap time in s", | ||||||
|   "orgs": "Organizations", |     "minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0", | ||||||
|   "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!", |     "must-be-at-least-10-characters-long": "Must be at least 10 characters long!", | ||||||
|   "password": "Password", |     "must-contain-a-lowercase-letter": "Must contain a lowercase letter!", | ||||||
|   "password-changed": "Password changed!", |     "must-contain-a-number": "Must contain a number!", | ||||||
|   "password-is-required": "Password is required", |     "must-contain-a-uppercase-letter": "Must contain a uppercase letter!", | ||||||
|   "password-reset-failed": "Password reset failed!", |     "name": "Name", | ||||||
|   "password-reset-in-progress": "Password Reset in Progress...", |     "name-is-required": "Name is required", | ||||||
|   "password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".", |     "new-password": "New password", | ||||||
|   "password-reset-successful": "Password Reset successful!", |     "no-contact-found": "No contacts found", | ||||||
|   "passwords-dont-match": "Passwords don't match!", |     "no-contact-selected": "No contact selected", | ||||||
|   "pdf-generation-failed": "PDF generation failed!", |     "no-contact-specified": "no contact specified", | ||||||
|   "pdf-successfully-generated": "PDF successfully generated!", |     "no-donors-found": "No donors found", | ||||||
|   "pdfs-successfully-generated": "PDFs successfully generated!", |     "no-license-text-could-be-found": "No license text could be found 😢", | ||||||
|   "per-kilometer": "per Kilometer", |     "no-organization-or-team-found": "No organization or team found", | ||||||
|   "permissions": "Permissions", |     "no-organization-specified": "no organization specified", | ||||||
|   "permissions-updated": "Permissions updated!", |     "no-organizations-found": "No organizations found", | ||||||
|   "phone": "Phone", |     "no-runners-found": "No runners found", | ||||||
|   "please-copy-the-token-and-store-it-somewhere-save": "Please copy the token and store it somewhere safe.", |     "no-tracks-added-yet": "there are no tracks added yet.", | ||||||
|   "please-provide-a-password": "Please provide a password...", |     "non-blanko": "Non/Blanko", | ||||||
|   "please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor", |     "open": "OPEN", | ||||||
|   "please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation", |     "organization": "Organization", | ||||||
|   "please-provide-the-nessecary-information-to-create-a-new-scan": "Please provide the nessecary information to create a new scan.", |     "organization-added": "Organization added", | ||||||
|   "please-provide-the-required-csv-xlsx-file": "Please provide the required csv/ xlsx file", |     "organization-deleted": "Organization deleted", | ||||||
|   "please-provide-the-required-information-for-creating-a-new-user-group": "Please provide the required information for creating a new user group.", |     "organization-detail-is-being-loaded": "organization detail is being loaded...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-contact": "Please provide the required information to add a new contact.", |     "organization-is-being-added": "Organization is being added...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-organization": "Please provide the required information to add a new organization.", |     "organization-name-is-required": "Organization name is required", | ||||||
|   "please-provide-the-required-information-to-add-a-new-runner": "Please provide the required information to add a new runner.", |     "organizations": "Organizations", | ||||||
|   "please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.", |     "organizations-are-being-loaded": "organizations are being loaded...", | ||||||
|   "please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.", |     "orgs": "Organizations", | ||||||
|   "please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.", |     "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!", | ||||||
|   "please-provide-the-required-information-to-create-a-new-scanstation": "Please provide the required information to create a new scanstation", |     "paid": "PAID", | ||||||
|   "please-request-a-new-reset-mail": "Please request a new reset mail...", |     "paid-amount": "Paid amount", | ||||||
|   "privacy": "Privacy", |     "password": "Password", | ||||||
|   "privacy-loading": "Privacy loading...", |     "password-changed": "Password changed!", | ||||||
|   "profile": "Profile", |     "password-is-required": "Password is required", | ||||||
|   "profile-picture": "Profile Picture", |     "password-reset-failed": "Password reset failed!", | ||||||
|   "profile-updated": "Profile updated!", |     "password-reset-in-progress": "Password Reset in Progress...", | ||||||
|   "read-license": "Read License", |     "password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".", | ||||||
|   "receipt-needed": "Receipt needed", |     "password-reset-successful": "Password Reset successful!", | ||||||
|   "repo_link": "Link", |     "passwords-dont-match": "Passwords don't match!", | ||||||
|   "request-a-new-reset-mail": "Request a new reset mail", |     "payment-amount-must-be-greater-than-0-00eur": "Payment amount must be greater than 0.00€!", | ||||||
|   "reset-my-password": "Reset my password", |     "pdf-generation-failed": "PDF generation failed!", | ||||||
|   "reset-password": "Reset your password", |     "pdf-successfully-generated": "PDF successfully generated!", | ||||||
|   "runner": "Runner", |     "pdfs-successfully-generated": "PDFs successfully generated!", | ||||||
|   "runner-added": "Runner added", |     "per-kilometer": "per Kilometer", | ||||||
|   "runner-import": "Runner Import", |     "permissions": "Permissions", | ||||||
|   "runner-is-being-added": "Runner is being added...", |     "permissions-updated": "Permissions updated!", | ||||||
|   "runner-updated": "Runner updated!", |     "phone": "Phone", | ||||||
|   "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"", |     "please-confirm-the-deletion-of-card": "Please confirm the deletion of this card", | ||||||
|   "runners": "Runners", |     "please-confirm-the-deletion-of-runner": "Please confirm the deletion of this runner", | ||||||
|   "runners-are-being-imported": "Runners are being imported...", |     "please-confirm-the-deletion-of-scan": "Please confirm the deletion of scan", | ||||||
|   "runners-are-being-loaded": "runners are being loaded...", |     "please-copy-the-token-and-store-it-somewhere-save": "Please copy the token and store it somewhere safe.", | ||||||
|   "save": "Save", |     "please-provide-a-password": "Please provide a password...", | ||||||
|   "save-changes": "Save Changes", |     "please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor", | ||||||
|   "scan-added": "Scan added", |     "please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation", | ||||||
|   "scan-is-being-updated": "Scan is being updated", |     "please-provide-the-nessecary-information-to-create-a-new-scan": "Please provide the nessecary information to create a new scan.", | ||||||
|   "scan-with-fixed-distance": "Scan with fixed distance", |     "please-provide-the-required-csv-xlsx-file": "Please provide the required csv/ xlsx file", | ||||||
|   "scans": "Scans", |     "please-provide-the-required-information-for-creating-a-new-user-group": "Please provide the required information for creating a new user group.", | ||||||
|   "scans-are-being-loaded": "Scans are being loaded", |     "please-provide-the-required-information-to-add-a-new-contact": "Please provide the required information to add a new contact.", | ||||||
|   "scanstation": "Scanstation", |     "please-provide-the-required-information-to-add-a-new-organization": "Please provide the required information to add a new organization.", | ||||||
|   "scanstation-added": "Scanstation added", |     "please-provide-the-required-information-to-add-a-new-runner": "Please provide the required information to add a new runner.", | ||||||
|   "scanstation-is-being-added": "Adding scanstation...", |     "please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.", | ||||||
|   "scanstations": "Scanstations", |     "please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.", | ||||||
|   "scanstations-are-being-loaded": "Loading scanstations...", |     "please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.", | ||||||
|   "search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)", |     "please-provide-the-required-information-to-create-a-new-scanstation": "Please provide the required information to create a new scanstation", | ||||||
|   "search-for-an-organization-or-team-by-name-or-id": "Search for an organization or team (by name or id)", |     "please-provide-the-required-information-to-create-a-new-statsclient": "Please provide the required information to create a new statsclient", | ||||||
|   "search-for-donor-name-or-id": "Search for donor (by name or id)", |     "please-request-a-new-reset-mail": "Please request a new reset mail...", | ||||||
|   "search-for-permission": "Search for permission", |     "please-wait-a-moment-your-login-is-still-being-processed": "Please wait a moment, your login is still being processed", | ||||||
|   "search-for-runner-by-name-or-id": "Search for runner (by name or id)", |     "prefix": "Prefix", | ||||||
|   "select-all": "select all", |     "privacy": "Privacy", | ||||||
|   "select-language": "Select language", |     "privacy-loading": "Privacy loading...", | ||||||
|   "selfservice-registration": "Selfservice registration", |     "profile": "Profile", | ||||||
|   "send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services", |     "profile-deleted": "Profile deleted!", | ||||||
|   "set-the-user-active-inactive": "set the user active/ inactive", |     "profile-picture": "Profile Picture", | ||||||
|   "settings": "Settings", |     "profile-updated": "Profile updated!", | ||||||
|   "settings-for-your-profile": "Settings for your profile", |     "read-license": "Read License", | ||||||
|   "something-about-the-group": "Something about the group...", |     "receipt-needed": "Receipt needed", | ||||||
|   "stats-are-being-loaded": "stats are being loaded...", |     "repo_link": "Link", | ||||||
|   "status": "Status", |     "request-a-new-reset-mail": "Request a new reset mail", | ||||||
|   "stuff-that-could-harm-your-profile": "Stuff that could harm your profile", |     "reset-my-password": "Reset my password", | ||||||
|   "successful-password-reset": "Successful password reset!", |     "reset-password": "Reset your password", | ||||||
|   "team": "Team", |     "runner": "Runner", | ||||||
|   "team-detail-is-being-loaded": "team detail is being loaded...", |     "runner-added": "Runner added", | ||||||
|   "team-name": "Team name", |     "runner-deleted": "runner deleted", | ||||||
|   "team-name-is-required": "team name is required", |     "runner-import": "Runner Import", | ||||||
|   "teams": "Teams", |     "runner-is-being-added": "Runner is being added...", | ||||||
|   "teams-are-being-loaded": "teams are being loaded...", |     "runner-updated": "Runner updated!", | ||||||
|   "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...", |     "runnercards": "Runnercards", | ||||||
|   "the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m", |     "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"", | ||||||
|   "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!", |     "runners": "Runners", | ||||||
|   "there-are-no-cards-yet": "There are no cards yet.", |     "runners-are-being-imported": "Runners are being imported...", | ||||||
|   "there-are-no-contacts-added-yet": "There are no contacts added yet.", |     "runners-are-being-loaded": "runners are being loaded...", | ||||||
|   "there-are-no-donations-yet": "There are no donations yet", |     "save": "Save", | ||||||
|   "there-are-no-donors-yet": "There are no donors yet", |     "save-changes": "Save Changes", | ||||||
|   "there-are-no-groups-yet": "There are no groups yet", |     "scan-added": "Scan added", | ||||||
|   "there-are-no-organizations-added-yet": "There are no organizations added yet.", |     "scan-deleted": "scan deleted", | ||||||
|   "there-are-no-runners-added-yet": "There are no runners added yet.", |     "scan-is-being-updated": "Scan is being updated", | ||||||
|   "there-are-no-scans-yet": "There are no scans yet", |     "scan-with-fixed-distance": "Scan with fixed distance", | ||||||
|   "there-are-no-teams-added-yet": "There are no teams added yet.", |     "scans": "Scans", | ||||||
|   "there-are-no-users-added-yet": "There are no users added yet.", |     "scans-are-being-loaded": "Scans are being loaded", | ||||||
|   "this-card-is": "This card is", |     "scanstation": "Scanstation", | ||||||
|   "this-might-take-a-moment": "This might take a moment 👀", |     "scanstation-added": "Scanstation added", | ||||||
|   "this-scanstation-is": "This scanstation is", |     "scanstation-is-being-added": "Adding scanstation...", | ||||||
|   "token": "Token", |     "scanstations": "Scanstations", | ||||||
|   "total-distance": "total distance", |     "scanstations-are-being-loaded": "Loading scanstations...", | ||||||
|   "total-donation-amount": "total donation amount", |     "search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)", | ||||||
|   "total-donations": "total donations", |     "search-for-an-organization-or-team-by-name-or-id": "Search for an organization or team (by name or id)", | ||||||
|   "total-scans": "total scans", |     "search-for-donor-name-or-id": "Search for donor (by name or id)", | ||||||
|   "track": "Track", |     "search-for-permission": "Search for permission", | ||||||
|   "track-added": "Track added", |     "search-for-runner-by-name-or-id": "Search for runner (by name or id)", | ||||||
|   "track-data-is-being-loaded": "Track data is being loaded", |     "select-all": "select all", | ||||||
|   "track-is-being-added": "Track is being added...", |     "select-language": "Select language", | ||||||
|   "track-length-in-m": "Track Length in m", |     "selfservice-registration": "Selfservice registration", | ||||||
|   "track-length-must-be-greater-than-0": "Track length must be greater than 0", |     "send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services", | ||||||
|   "track-name": "Track name", |     "set-the-user-active-inactive": "set the user active/ inactive", | ||||||
|   "track-name-must-not-be-empty": "Track name must not be empty", |     "settings": "Settings", | ||||||
|   "tracks": "Tracks", |     "settings-for-your-profile": "Settings for your profile", | ||||||
|   "update-card": "Update Card", |     "something-about-the-group": "Something about the group...", | ||||||
|   "update-password": "Update password", |     "sponsoring-quittungs-liste_herunterladen": "Download donor receipt list", | ||||||
|   "updated-contact": "Updated contact!", |     "sponsorings": "Sponsorings", | ||||||
|   "updated-donor": "updated donor", |     "station-deleted": "station deleted", | ||||||
|   "updated-organization": "updated organization", |     "stats-are-being-loaded": "stats are being loaded...", | ||||||
|   "updated-scan": "updated scan", |     "statsclient-deleted": "Deleted statsclient", | ||||||
|   "updateing-group": "updateing group...", |     "statsclient-is-being-added": "Statsclient is being added...", | ||||||
|   "updating-card": "Updating card", |     "statsclients": "Statsclients (aka Beamershow)", | ||||||
|   "updating-organization": "updating organization", |     "statsclients-are-being-loaded": "Loading statsclients", | ||||||
|   "updating-permissions": "updating permissions...", |     "status": "Status", | ||||||
|   "updating-runner": "Updating runner...", |     "stuff-that-could-harm-your-profile": "Stuff that could harm your profile", | ||||||
|   "updating-user": "updating user...", |     "successful-password-reset": "Successful password reset!", | ||||||
|   "updating-your-profile": "Updating your profile...", |     "team": "Team", | ||||||
|   "user-added": "User added", |     "team-added": "Team added", | ||||||
|   "user-groups": "User Groups", |     "team-deleted": "Team deleted", | ||||||
|   "user-is-being-added": "User is being added...", |     "team-detail-is-being-loaded": "team detail is being loaded...", | ||||||
|   "user-updated": "User updated", |     "team-is-being-added": "Team is being added...", | ||||||
|   "username": "Username", |     "team-name": "Team name", | ||||||
|   "users": "Users", |     "team-name-is-required": "team name is required", | ||||||
|   "valid": "Valid", |     "teams": "Teams", | ||||||
|   "valid-city-is-required": "Valid city is required", |     "teams-are-being-loaded": "teams are being loaded...", | ||||||
|   "valid-email-is-required": "valid email is required", |     "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...", | ||||||
|   "valid-international-phone-number-is-required": "valid international phone number is required...", |     "the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m", | ||||||
|   "valid-zipcode-postal-code-is-required": "Valid zipcode/ postal code is required", |     "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!", | ||||||
|   "verfuegbare": "availdable", |     "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!", | ||||||
|   "welcome_wavinghand": "Welcome 👋", |     "there-are-no-cards-yet": "There are no cards yet.", | ||||||
|   "yes-i-copied-the-token": "Yes, I copied the token", |     "there-are-no-contacts-added-yet": "There are no contacts added yet.", | ||||||
|   "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!", |     "there-are-no-donations-yet": "There are no donations yet", | ||||||
|   "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! 🎉", |     "there-are-no-donors-yet": "There are no donors yet", | ||||||
|   "you-can-provide-a-runner-but-you-dont-have-to": "You can provide a runner, but you don't have to.", |     "there-are-no-groups-yet": "There are no groups yet", | ||||||
|   "you-dont-have-any-scanstations-yet": "You don't have any scanstations yet", |     "there-are-no-organizations-added-yet": "There are no organizations added yet.", | ||||||
|   "you-have-to-provide-an-organization": "You have to provide an organization", |     "there-are-no-runners-added-yet": "There are no runners added yet.", | ||||||
|   "you-have-to-save-your-changes-to-generate-a-link": "You have to save your changes to generate a link.", |     "there-are-no-scans-yet": "There are no scans yet", | ||||||
|   "you-must-create-at-least-one-card-or-cancel": "You must create at least one card (or cancel).", |     "there-are-no-teams-added-yet": "There are no teams added yet.", | ||||||
|   "zip-postal-code": "ZIP/ postal code", |     "there-are-no-users-added-yet": "There are no users added yet.", | ||||||
|   "generate-runner-certificates": "Generate runner certificates" |     "this-card-is": "This card is", | ||||||
| } |     "this-might-take-a-moment": "This might take a moment 👀", | ||||||
|  |     "this-scanstation-is": "This scanstation is", | ||||||
|  |     "timestamp": "timestamp", | ||||||
|  |     "token": "Token", | ||||||
|  |     "total-distance": "total distance", | ||||||
|  |     "total-donation-amount": "total donation amount", | ||||||
|  |     "total-donations": "total donations", | ||||||
|  |     "total-paid-amount": "Total paid amount", | ||||||
|  |     "total-scans": "total scans", | ||||||
|  |     "total_donation_amount_in_eur": "Total donation amount in €", | ||||||
|  |     "track": "Track", | ||||||
|  |     "track-added": "Track added", | ||||||
|  |     "track-data-is-being-loaded": "Track data is being loaded", | ||||||
|  |     "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-must-be-greater-than-0": "Track length must be greater than 0", | ||||||
|  |     "track-name": "Track name", | ||||||
|  |     "track-name-must-not-be-empty": "Track name must not be empty", | ||||||
|  |     "track-was-updated": "Track was updated!", | ||||||
|  |     "tracks": "Tracks", | ||||||
|  |     "unpaid": "Unpaid", | ||||||
|  |     "update-card": "Update Card", | ||||||
|  |     "update-password": "Update password", | ||||||
|  |     "updated-contact": "Updated contact!", | ||||||
|  |     "updated-donor": "updated donor", | ||||||
|  |     "updated-organization": "updated organization", | ||||||
|  |     "updated-scan": "updated scan", | ||||||
|  |     "updated-team": "Updated team", | ||||||
|  |     "updateing-group": "updateing group...", | ||||||
|  |     "updating-card": "Updating card", | ||||||
|  |     "updating-donation": "Updating donation", | ||||||
|  |     "updating-organization": "updating organization", | ||||||
|  |     "updating-permissions": "updating permissions...", | ||||||
|  |     "updating-runner": "Updating runner...", | ||||||
|  |     "updating-team": "Updating team", | ||||||
|  |     "updating-user": "updating user...", | ||||||
|  |     "updating-your-profile": "Updating your profile...", | ||||||
|  |     "user-added": "User added", | ||||||
|  |     "user-groups": "User Groups", | ||||||
|  |     "user-is-being-added": "User is being added...", | ||||||
|  |     "user-updated": "User updated", | ||||||
|  |     "username": "Username", | ||||||
|  |     "users": "Users", | ||||||
|  |     "valid": "Valid", | ||||||
|  |     "valid-city-is-required": "Valid city is required", | ||||||
|  |     "valid-email-is-required": "valid email is required", | ||||||
|  |     "valid-international-phone-number-is-required": "valid international phone number is required...", | ||||||
|  |     "valid-zipcode-postal-code-is-required": "Valid zipcode/ postal code is required", | ||||||
|  |     "verfuegbare": "availdable", | ||||||
|  |     "welcome_wavinghand": "Welcome 👋", | ||||||
|  |     "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-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-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-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" | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| import 'windi.css'; | import './style.css'; | ||||||
| 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 App from './App.svelte'; | import App from './App.svelte'; | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								src/style.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/style.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | @tailwind base; | ||||||
|  | @tailwind components; | ||||||
|  | @tailwind utilities; | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| module.exports = { | module.exports = { | ||||||
|  | 	mode: 'jit', | ||||||
|  | 	content: [ './src/**/*.svelte' ], | ||||||
| 	theme: { | 	theme: { | ||||||
| 		extend: { | 		extend: { | ||||||
| 			colors: { | 			colors: { | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user