Compare commits

...

100 Commits

Author SHA1 Message Date
76b69d851a 🚀RELEASE v0.15.3
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-16 18:22:27 +02:00
224f586368 Small bugfix (null got displayed) 🛠 2021-04-16 18:22:00 +02:00
9add6c8ff1 🚀RELEASE v0.15.2
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-16 18:10:50 +02:00
7a63d4eed1 Merge branch 'dev' of https://git.odit.services/lfk/frontend into dev
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-16 18:09:37 +02:00
e54a4807f7 NGINX cache assets 2021-04-16 18:09:30 +02:00
cee04c1d6f Footer - noopener link 2021-04-16 18:09:22 +02:00
cbec78589d Hotfix: Team change recognition 🐞
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-16 18:06:53 +02:00
a85db7cb3f 🚀RELEASE v0.15.1
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-16 17:53:38 +02:00
2bd3779839 Merge pull request '🐞🐳 fix Dockerfile' (#138) from bugfix/136-opacity_reactivity into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #138
2021-04-16 15:51:37 +00:00
303e33cafb 🐞🐳 fix Dockerfile
ref #136
2021-04-16 17:46:15 +02:00
b4e689dddf Dockerfile now uses selfhosted registry
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-15 19:50:37 +02:00
98a0b036c5 new license file version [CI SKIP] 2021-04-15 17:46:04 +00:00
fb3f30fb10 Merge pull request 'Opacity import fix bugfix/136-opacity_reactivity' (#137) from bugfix/136-opacity_reactivity into dev
Some checks failed
continuous-integration/drone/push Build is failing
Reviewed-on: #137
2021-04-15 17:44:49 +00:00
6213952007 Added bs import fix
ref #136
2021-04-15 19:43:43 +02:00
07ac041d69 🚚 move to tailwind
ref #136
2021-04-15 19:22:57 +02:00
5c02028841 🚀RELEASE v0.15.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-15 18:30:38 +02:00
c561b53670 Merge pull request 'Mark donations as payed feature/133-donation_payments' (#135) from feature/133-donation_payments into dev
Some checks failed
continuous-integration/drone/push Build is failing
Reviewed-on: #135
2021-04-15 16:29:39 +00:00
dcd0d5a362 Merge branch 'feature/133-donation_payments' of https://git.odit.services/lfk/frontend into feature/133-donation_payments 2021-04-15 18:27:47 +02:00
18acac83bc AddDonationModal - vertical alignment for paid status
ref #133
2021-04-15 18:27:35 +02:00
d7d44470bb DonationsOverview contrast on action
ref #133
2021-04-15 18:25:52 +02:00
0f0aae7ba4 Fixed chante recognition bug for fixed donation
ref #133
2021-04-15 18:21:23 +02:00
4c0886a5d9 Fixed typo
ref #133
2021-04-15 18:04:03 +02:00
04a3038369 Added missing updated comparison
ref #133
2021-04-15 16:56:03 +02:00
bdcf5d3fc0 Added payment updating via detail
ref #133
2021-04-15 15:54:14 +02:00
c7a858eed7 Sorted translations
ref #133
2021-04-15 15:42:47 +02:00
de5aa9237d Added **all** missing toast translations
ref #133
2021-04-15 15:42:29 +02:00
d015f97395 Added translations 🌎
ref #113
2021-04-15 15:34:36 +02:00
57618156b4 Added msiisng runner id conversion
ref #133
2021-04-15 15:30:23 +02:00
865254d646 Fixed styling
ref #133
2021-04-15 15:25:17 +02:00
1dbab03fe7 You can now add payments from the donation overview
ref #133
2021-04-15 15:24:31 +02:00
a943aaf5fc You can now open a modal to add a payment to a donation from the donation overview
ref #133
2021-04-15 15:05:05 +02:00
6e6e8b2617 Added Add Payment button to donor overview
ref #133
2021-04-15 14:40:46 +02:00
4c2c24af2c Changed top info style for donation overview
ref #133
2021-04-15 14:33:35 +02:00
3d3a10aafb You can now mark fixed donations as already paid on creation
ref #133
2021-04-15 14:31:24 +02:00
000fc97beb Changed top info style for donation detail
ref #133
2021-04-15 14:18:28 +02:00
5645eeaafa Added paid donation amount and status to donation detail
ref #133
2021-04-15 14:17:28 +02:00
961477d522 Added total donation amount to donation overview
ref #133
2021-04-15 14:12:11 +02:00
a5f71015a6 Added total donation amount to donor detail
ref #133
2021-04-15 14:10:35 +02:00
e42ea943b7 Added total donation amount to donor overview
ref #133
2021-04-15 14:09:23 +02:00
9c5fc6b61c 🚀RELEASE v0.14.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-14 19:28:00 +02:00
302caf015f new license file version [CI SKIP] 2021-04-14 17:27:30 +00:00
e11296071a Merge pull request 'added donor receipt list download to DonorsOverview' (#134) from feature/132-export-donors-receipt-list into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #134
close #132
2021-04-14 17:26:10 +00:00
112eb29f93 Switched to selfhosted images
ref #132
2021-04-14 19:24:06 +02:00
c6c97516b3 Sorted translations 🌎
ref #132
2021-04-14 19:23:26 +02:00
03676b2894 Fixed typos in translations
ref #132
2021-04-14 19:23:07 +02:00
9ca57fac2e bump @odit/lfk-client-js@0.11.0
ref #132
2021-04-14 18:58:57 +02:00
18f151c1fb general version bump
ref #132
2021-04-14 18:57:14 +02:00
e90e56d8b2 replace donationAmount with paidDonationAmount
ref #132
2021-04-14 18:54:05 +02:00
d241ca5698 added donor receipt list download to DonorsOverview
ref #132
2021-04-14 18:43:51 +02:00
b512cf8667 🚀RELEASE v0.13.1
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-11 21:14:03 +02:00
a24d2923c6 For await fix
Some checks failed
continuous-integration/drone/push Build is failing
2021-04-11 21:13:44 +02:00
467808abef 🚀RELEASE v0.13.0
Some checks failed
continuous-integration/drone/push Build is failing
2021-04-11 21:08:58 +02:00
861f1f2216 Merge pull request 'Better org pdf generation feature/130-org_doc_splitting' (#131) from feature/130-org_doc_splitting into dev
Some checks failed
continuous-integration/drone/push Build is failing
Reviewed-on: #131
2021-04-11 19:07:53 +00:00
509b22bea0 Org certificate generation now runs in sequence
ref #130
2021-04-11 21:00:33 +02:00
7447b2f4c1 Fixed const -> let
ref #130
2021-04-11 20:54:38 +02:00
fef14b6e4f Org card generation now runs in sequence
ref #130
2021-04-11 20:53:58 +02:00
01d2a7e6aa Org contract generation now runs in sequence
ref #130
2021-04-11 20:48:20 +02:00
ac586fec5a Hotfix: Org * generation🐞
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-11 20:17:34 +02:00
5476808683 Emergency document server url change
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-11 20:10:08 +02:00
331d737796 🚀RELEASE v0.12.5
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-08 19:30:20 +02:00
ef81b8adf9 Merge pull request 'Added runner team's parentorg name to runenr overciew' (#129) from feature/128-runner_orgs into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #129
2021-04-08 17:29:27 +00:00
8a7d635cef Switched to html entity
ref #128
2021-04-08 18:00:47 +02:00
4c259c1eef Added runner team's parentorg name to runenr overciew
ref #128
2021-04-08 17:58:49 +02:00
5b4ede5e2f 🚀RELEASE v0.12.4
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-08 17:34:32 +02:00
d0ab3dda78 🚑 [HOTFIX] - drop "svelte-infinite-loading"
Some checks failed
continuous-integration/drone/push Build is failing
2021-04-08 17:34:04 +02:00
d9cf51b4bb 🚀RELEASE v0.12.3
Some checks failed
continuous-integration/drone/push Build is failing
2021-04-08 17:30:23 +02:00
aa17f24220 new license file version [CI SKIP] 2021-04-08 15:28:50 +00:00
cf60edf7d4 Merge pull request 'fix' (#126) from bugfix/125-mobile into dev
Some checks failed
continuous-integration/drone/push Build is failing
Reviewed-on: #126
close #125
2021-04-08 15:27:22 +00:00
ffbc243194 custom css fix for collapsed_navigation
ref #125
2021-04-07 21:44:29 +02:00
b6b07cf30c 🐞 bugfix for svelte x tailwind class names
ref #125
2021-04-07 21:35:01 +02:00
495a6b22bd almost fixed... 2021-04-07 21:28:21 +02:00
0acaffbdfa fix
ref #125
2021-04-07 20:25:04 +02:00
6043bc4517 🚀RELEASE v0.12.2
Some checks failed
continuous-integration/drone/push Build is failing
2021-04-07 20:11:21 +02:00
e6ed066e3f Merge pull request 'feature/110-virtual_list' (#124) from feature/110-virtual_list into dev
Some checks failed
continuous-integration/drone/push Build is failing
Reviewed-on: #124
2021-04-07 18:10:02 +00:00
ee4e8655b8 Merge branch 'dev' into feature/110-virtual_list 2021-04-07 20:09:35 +02:00
37970d2be6 pre-merge fixes
ref #110
2021-04-07 18:59:46 +02:00
1376788016 updated virtual scroll list 2021-04-07 18:38:52 +02:00
4cad86cf85 fixed height table 2021-04-07 18:19:58 +02:00
6304116edb wip on virtuallist 2021-04-06 22:16:24 +02:00
834ff8fa63 🚀RELEASE v0.12.1
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-06 16:47:37 +02:00
1f428a535e Merge pull request 'ImportRunnerModal Cancel Button feature/122-import_cancel' (#123) from feature/112-import_cancel into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #123
2021-04-06 14:42:37 +00:00
0c40966970 Added cancel button for the first stage of runner import
ref #112
2021-04-05 16:23:24 +02:00
9da071fe9b Escape now triggers foll modal close (including reset) instead of just hiding th modal
ref #112
2021-04-05 16:14:43 +02:00
892a04f289 🚀RELEASE v0.12.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-05 16:09:17 +02:00
27cc9727f1 Fixed package version
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-05 16:08:41 +02:00
f0738d451b Merge pull request 'Implmented certificate generation feature/119-Certificate_generation' (#120) from feature/119-Certificate_generation into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #120
2021-04-05 14:05:38 +00:00
9e6a8daf2c Sorted translations 🌍
ref #119
2021-04-05 16:04:44 +02:00
bfacfec765 The PFS Prefixes now get translated via i18n
ref #119
2021-04-05 16:04:26 +02:00
0bae5bf32b sponsoring pdf names now include their locale
ref #119
2021-04-05 15:37:12 +02:00
22b09d16d0 Cleaned up generation strings and added the schem for single runner generations for sponsoring contracts
ref #119
2021-04-05 15:36:01 +02:00
9c867e106e Cleaned up generation strings and added the schem for single runner generations for cards
ref #119
2021-04-05 15:33:55 +02:00
304f28a3c1 certificate pdf names now include their locale
ref #119
2021-04-05 15:31:52 +02:00
d65d3793de Changed the basic nameing generation for runenr certificate files
ref #119
2021-04-05 15:31:01 +02:00
3638d87bd2 Runnercard pdfs now include their locale
ref #119
2021-04-05 15:28:13 +02:00
b97a92860d Fixed wrong permissiong getting checked
ref #119
2021-04-05 15:24:42 +02:00
7c86a5eeb3 added missing/ wrong translations + formatting!
ref #119
2021-04-05 12:02:18 +02:00
d23dbaaf69 Removed useless console.log
ref #119
2021-04-03 20:05:27 +02:00
e6ffc371e1 Certificate generation from org detail
ref #119
2021-04-03 20:03:57 +02:00
3177c6eaa3 Certificate generation from org overview
ref #119
2021-04-03 20:02:52 +02:00
acd2f0519d Certificate generation from team detail
ref #119
2021-04-03 20:00:51 +02:00
45 changed files with 2773 additions and 1866 deletions

View File

@@ -27,7 +27,7 @@ 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:alpine
commands: commands:
- yarn - yarn
- yarn licenses:export - yarn licenses:export

View File

@@ -2,15 +2,216 @@
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.15.3](https://git.odit.services/lfk/frontend/compare/0.15.2...0.15.3)
- 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)

View File

@@ -1,14 +1,12 @@
FROM node:15.5.1-alpine3.12 FROM registry.odit.services/hub/library/node:15.14.0-alpine3.13
WORKDIR /app WORKDIR /app
COPY package.json ./ COPY package.json ./
RUN yarn RUN yarn
COPY package.json *.config.js index.html ./ COPY package.json *.config.js postcss.config.cjs index.html ./
COPY src ./src COPY src ./src
COPY public ./public COPY public ./public
RUN yarn build RUN yarn build
# final image # final image
FROM alpine FROM registry.odit.services/hub/fholzer/nginx-brotli:v1.19.1
COPY --from=0 /app/dist /app COPY --from=0 /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

View File

@@ -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.15.3-RELEASE_INFO</span>
<noscript>You need to enable JavaScript to run this app.</noscript> <noscript>You need to enable JavaScript to run this app.</noscript>
<script src="/env.js"></script> <script src="/env.js"></script>
<script type="module" src="/src/main.js"></script> <script type="module" src="/src/main.js"></script>
</body> </body>
</html> </html>

View File

@@ -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';
} }

View File

@@ -1,6 +1,6 @@
{ {
"name": "@odit/lfk-frontend", "name": "@odit/lfk-frontend",
"version": "0.8.4", "version": "0.15.3",
"scripts": { "scripts": {
"i18n-order": "node order.js", "i18n-order": "node order.js",
"dev": "vite", "dev": "vite",
@@ -10,30 +10,30 @@
}, },
"license": "CC-BY-NC-SA-4.0", "license": "CC-BY-NC-SA-4.0",
"devDependencies": { "devDependencies": {
"check-password-strength": "2.0.2", "@odit/lfk-client-js": "0.11.0",
"@odit/lfk-client-js": "0.10.1",
"@odit/license-exporter": "0.0.11", "@odit/license-exporter": "0.0.11",
"@sveltejs/vite-plugin-svelte": "1.0.0-next.5", "@sveltejs/vite-plugin-svelte": "1.0.0-next.6",
"@types/html-minifier": "4.0.0", "@types/html-minifier": "4.0.0",
"auto-changelog": "2.2.1", "auto-changelog": "2.2.1",
"autoprefixer": "10.2.5", "autoprefixer": "^10.2.5",
"check-password-strength": "2.0.2",
"csvtojson": "2.0.10", "csvtojson": "2.0.10",
"gridjs": "3.4.0", "gridjs": "3.4.0",
"html-minifier": "4.0.0", "html-minifier": "4.0.0",
"localforage": "1.9.0", "localforage": "1.9.0",
"marked": "2.0.1", "marked": "2.0.3",
"release-it": "14.5.1", "postcss": "^8.2.10",
"release-it": "14.6.1",
"svelte": "3.37.0", "svelte": "3.37.0",
"svelte-focus-trap": "1.2.0", "svelte-focus-trap": "1.2.0",
"svelte-i18n": "3.3.9", "svelte-i18n": "3.3.9",
"svelte-preprocess": "4.7.0", "svelte-preprocess": "^4.7.0",
"svelte-select": "3.17.0", "svelte-select": "3.17.0",
"tailwindcss": "2.0.4", "tailwindcss": "^2.1.1",
"tinro": "0.6.1", "tinro": "0.6.1",
"toastify-js": "1.10.0", "toastify-js": "1.10.0",
"validator": "13.5.2", "validator": "13.5.2",
"vite": "2.1.5", "vite": "2.1.5",
"vite-plugin-windicss": "0.12.2",
"xlsx": "0.16.9" "xlsx": "0.16.9"
}, },
"release-it": { "release-it": {

6
postcss.config.cjs Normal file
View File

@@ -0,0 +1,6 @@
module.exports = {
plugins: {
tailwindcss: {},
autoprefixer: {}
}
};

View File

@@ -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

View File

@@ -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%)",

View 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>

View File

@@ -77,7 +77,7 @@
duration: -1, duration: -1,
}).showToast(); }).showToast();
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: {

View File

@@ -1,5 +1,5 @@
<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";
@@ -11,11 +11,21 @@
export let editable = {}; export let editable = {};
export let original_data = {}; export let original_data = {};
export let current_cards = []; export let current_cards = [];
$: filtered_cards = current_cards.filter(function (c) {
if (
c.code.toLowerCase().includes(searchvalue_lowercase) ||
c.runner?.firstname.toLowerCase().includes(searchvalue_lowercase) ||
c.runner?.middlename.toLowerCase().includes(searchvalue_lowercase) ||
c.runner?.lastname.toLowerCase().includes(searchvalue_lowercase) ||
should_display_based_on_id(c.id)
) {
return true;
}
});
$: searchvalue = ""; $: searchvalue = "";
$: searchvalue_lowercase = searchvalue.toLowerCase();
$: active_deletes = []; $: active_deletes = [];
$: cards_show = current_cards.some( $: cards_show = current_cards.some((r) => r.is_selected === true);
(r) => r.is_selected === true
);
$: generate_cards = current_cards.filter((r) => r.is_selected === true); $: generate_cards = current_cards.filter((r) => r.is_selected === true);
const cards_promise = RunnerCardService.runnerCardControllerGetAll().then( const cards_promise = RunnerCardService.runnerCardControllerGetAll().then(
(val) => { (val) => {
@@ -46,8 +56,38 @@
original_data = Object.assign(original_data, card); original_data = Object.assign(original_data, card);
edit_modal_open = true; edit_modal_open = true;
} }
// -----------------
let scrollTop = 0;
$: rendered = filtered_cards;
let innerHeight = 0;
let ele;
$: updateSlice(scrollTop);
$: innerHeight = `${filtered_cards.length * 25}px`;
$: if (ele) updateSlice();
function updateSlice() {
const height = ele ? parseInt(ele.clientHeight) : 100;
const init = scrollTop / 25;
const end = Math.ceil((scrollTop + height) / 25);
rendered = filtered_cards.slice(init, end + 15);
}
function updateScroll($event) {
scrollTop = $event.target.scrollTop;
}
</script> </script>
<style>
table tbody {
display: block;
overflow-y: scroll;
}
table thead, table tbody tr {
display: table;
width: 100%;
table-layout: fixed;
}
</style>
{#if store.state.jwtinfo.userdetails.permissions.includes('CARD:UPDATE')} {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:UPDATE')}
<CardDetailModal <CardDetailModal
bind:current_cards bind:current_cards
@@ -121,107 +161,111 @@
</th> </th>
</tr> </tr>
</thead> </thead>
<tbody class="divide-y divide-gray-200"> <tbody class="divide-y divide-gray-200 virtual-wrapper"
{#each current_cards as card} on:scroll={updateScroll}
{#if card.code style="height: 70vh; width:100%"
.toLowerCase() bind:this={ele}
.includes( >
searchvalue.toLowerCase() {#each filtered_cards as card, index}
) || card.runner?.firstname {#if card.code
.toLowerCase() .toLowerCase()
.includes( .includes(
searchvalue.toLowerCase() searchvalue.toLowerCase()
) || card.runner?.middlename ) || card.runner?.firstname
.toLowerCase() .toLowerCase()
.includes( .includes(
searchvalue.toLowerCase() searchvalue.toLowerCase()
) || card.runner?.lastname ) || card.runner?.middlename
.toLowerCase() .toLowerCase()
.includes( .includes(
searchvalue.toLowerCase() searchvalue.toLowerCase()
) || should_display_based_on_id(card.id)} ) || card.runner?.lastname
<tr data-rowid="card_{card.id}"> .toLowerCase()
<td class="px-6 py-4 whitespace-nowrap"> .includes(
<input searchvalue.toLowerCase()
bind:checked={card.is_selected} ) || should_display_based_on_id(card.id)}
type="checkbox" <tr data-rowid="card_{card.id}">
class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> <td class="px-6 py-4 whitespace-nowrap">
</td> <input
<td class="px-6 py-4 whitespace-nowrap"> bind:checked={card.is_selected}
<div class="flex items-center">{card.code}</div> type="checkbox"
</td> class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" />
<td class="px-6 py-4 whitespace-nowrap"> </td>
<div class="flex items-center"> <td class="px-6 py-4 whitespace-nowrap">
{#if card.runner} <div class="flex items-center">{card.code}</div>
<a </td>
href="../runners/{card.runner.id}" <td class="px-6 py-4 whitespace-nowrap">
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{card.runner.firstname} <div class="flex items-center">
{card.runner.middlename || ''} {#if card.runner}
{card.runner.lastname}</a> <a
{:else}{$_('non-blanko')}{/if} href="../runners/{card.runner.id}"
</div> class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{card.runner.firstname}
</td> {card.runner.middlename || ''}
<td class="px-6 py-4 whitespace-nowrap"> {card.runner.lastname}</a>
<div class="flex items-center"> {:else}{$_('non-blanko')}{/if}
{#if card.enabled} </div>
<span </td>
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('enabled')}</span> <td class="px-6 py-4 whitespace-nowrap">
{:else} <div class="flex items-center">
<span {#if card.enabled}
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('disabled')}</span> <span
{/if} class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('enabled')}</span>
</div> {:else}
</td> <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} {#if active_deletes[card.id] === true}
<td <td
class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
<button <button
on:click={() => { on:click={() => {
active_deletes[card.id] = false; active_deletes[card.id] = false;
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button>
<button <button
on:click={() => { on:click={() => {
RunnerCardService.runnerCardControllerRemove(card.id, false).then( RunnerCardService.runnerCardControllerRemove(card.id, false).then(
(resp) => { (resp) => {
current_cards = current_cards.filter( current_cards = current_cards.filter(
(obj) => obj.id !== card.id (obj) => obj.id !== card.id
); );
Toastify({ Toastify({
text: $_('card-deleted'), text: $_('card-deleted'),
duration: 500, duration: 500,
backgroundColor: backgroundColor:
'linear-gradient(to right, #00b09b, #96c93d)', 'linear-gradient(to right, #00b09b, #96c93d)',
}).showToast(); }).showToast();
} }
); );
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button>
</td> </td>
{:else} {:else}
<td <td
class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
<button <button
on:click={() => { on:click={() => {
open_edit_modal(card); open_edit_modal(card);
}} }}
class="text-indigo-600 hover:text-indigo-900">{$_('details')}</button> class="text-indigo-600 hover:text-indigo-900">{$_('details')}</button>
{#if store.state.jwtinfo.userdetails.permissions.includes('CARD:DELETE')} {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:DELETE')}
<button <button
on:click={() => { on:click={() => {
active_deletes[card.id] = true; active_deletes[card.id] = true;
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button>
{/if} {/if}
</td> </td>
{/if} {/if}
</tr> </tr>
{/if} {/if}
{/each} {/each}
</tbody> </tbody>
</table> </table>
</div> </div>

View File

@@ -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();

View File

@@ -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>
@@ -297,14 +307,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 +329,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>

View File

@@ -9,6 +9,7 @@
} from "@odit/lfk-client-js"; } from "@odit/lfk-client-js";
import Select from "svelte-select"; import Select from "svelte-select";
import Toastify from "toastify-js"; import Toastify from "toastify-js";
import { is_promise } from "svelte/internal";
export let modal_open; export let modal_open;
export let current_donations; export let current_donations;
const getDonorLabel = (option) => const getDonorLabel = (option) =>
@@ -24,6 +25,7 @@
$: donors = []; $: donors = [];
$: runners = []; $: runners = [];
$: is_fixed = false; $: is_fixed = false;
$: is_paid = false;
DonorService.donorControllerGetAll().then((val) => { DonorService.donorControllerGetAll().then((val) => {
donors = val.map((r) => { donors = val.map((r) => {
return { label: getDonorLabel(r), value: r }; return { label: getDonorLabel(r), value: r };
@@ -57,14 +59,18 @@
let amount_cent = Math.floor(amount_input * 100); let amount_cent = Math.floor(amount_input * 100);
processed_last_submit = false; processed_last_submit = false;
const toast = Toastify({ const toast = Toastify({
text: "adding donation", text: $_('adding-donation'),
duration: -1, duration: -1,
}).showToast(); }).showToast();
if (is_fixed) { if (is_fixed) {
let postdata = { let postdata = {
donor, donor,
amount: amount_cent, amount: amount_cent,
paidAmount: 0
}; };
if(is_paid){
postdata.paidAmount = amount_cent;
}
DonationService.donationControllerPostFixed(postdata) DonationService.donationControllerPostFixed(postdata)
.then((result) => { .then((result) => {
donor = donors[0].id || 0; donor = donors[0].id || 0;
@@ -73,7 +79,7 @@
modal_open = false; modal_open = false;
// //
Toastify({ Toastify({
text: "donation_added", text: $_('donation_added'),
duration: 500, duration: 500,
backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast(); }).showToast();
@@ -102,7 +108,7 @@
modal_open = false; modal_open = false;
// //
Toastify({ Toastify({
text: "donation_added", text: $_('donation_added'),
duration: 500, duration: 500,
backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast(); }).showToast();
@@ -123,7 +129,7 @@
</script> </script>
<style> <style>
input:before { .toggle:before {
content: ""; content: "";
position: absolute; position: absolute;
width: 1.25rem; width: 1.25rem;
@@ -137,12 +143,12 @@
transition: 0.2s ease-in-out; transition: 0.2s ease-in-out;
} }
input:checked { .toggle:checked {
/* @apply: bg-indigo-400; */ /* @apply: bg-indigo-400; */
background-color: #7f9cf5; background-color: #7f9cf5;
} }
input:checked:before { .toggle:checked:before {
left: 1.25rem; left: 1.25rem;
} }
</style> </style>
@@ -195,7 +201,7 @@
class="ml-2 text-base" class="ml-2 text-base"
class:text-gray-300={is_fixed}>{$_('distance-donation')}</span> class:text-gray-300={is_fixed}>{$_('distance-donation')}</span>
<input <input
class="relative w-10 h-5 transition-all duration-200 ease-in-out bg-gray-400 rounded-full shadow-inner outline-none appearance-none align-middle" class="toggle relative w-10 h-5 transition-all duration-200 ease-in-out bg-gray-400 rounded-full shadow-inner outline-none appearance-none align-middle"
type="checkbox" type="checkbox"
bind:checked={is_fixed} /> bind:checked={is_fixed} />
<span <span
@@ -267,6 +273,29 @@
</span> </span>
{/if} {/if}
</div> </div>
{#if is_fixed}
<div class="col-span-6">
<label
for="paid"
class="block text-sm font-medium text-gray-700">{$_('already-paid')}</label>
<p class="text-gray-500">
<input
id="paid"
bind:checked={is_paid}
name="paid"
type="checkbox"
class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" >
<span class="align-text-bottom">
{#if is_paid}
{$_('paid')}
{:else}
{$_('open')}
{/if}
</span>
</p>
</div>
{/if}
</div> </div>
</div> </div>
</div> </div>

View File

@@ -0,0 +1,202 @@
<script>
import { _ } from "svelte-i18n";
import { clickOutside } from "../base/outsideclick";
import { focusTrap } from "svelte-focus-trap";
import { DonationService } from "@odit/lfk-client-js";
import Toastify from "toastify-js";
export let payment_modal_open = false;
export let current_donations = [];
export let editable = {};
export let original_data = {};
export let paid_amount_input = 0;
$:processed_last_submit=true;
function focus(el) {
el.focus();
}
$: createbtnenabled = is_paid_amount_valid && !(paid_amount_input*100 == original_data.paidAmount)
$: is_paid_amount_valid = paid_amount_input > 0;
(() => {
document.onkeydown = (e) => {
e = e || window.event;
if (e.key === "Escape") {
payment_modal_open = false;
}
if (e.keyCode === 13) {
if (createbtnenabled === true) {
createbtnenabled = false;
submit();
}
}
};
})();
function submit() {
if (processed_last_submit === true) {
processed_last_submit = false;
const toast = Toastify({
text: $_('updating-donation'),
duration: -1,
}).showToast();
editable.donor = editable.donor.id;
editable.paidAmount = paid_amount_input*100;
if(editable.responseType == "DISTANCEDONATION" || editable.runner){
editable.runner = editable.runner.id;
DonationService.donationControllerPutDistance(original_data.id, editable)
.then((result) => {
let id = original_data.id;
editable = {};
original_data = {};
payment_modal_open = false;
//
Toastify({
text: $_('donation-updated'),
duration: 500,
backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast();
current_donations[current_donations.findIndex((c) => c.id === id)] = result;
current_donations = current_donations;
})
.catch((err) => {
//
})
.finally(() => {
processed_last_submit = true;
//
toast.hideToast();
});
}
else{
DonationService.donationControllerPutFixed(original_data.id, editable)
.then((result) => {
let id = original_data.id;
editable = {};
original_data = {};
payment_modal_open = false;
//
Toastify({
text: $_('donation-updated'),
duration: 500,
backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast();
current_donations[current_donations.findIndex((c) => c.id === id)] = result;
current_donations = current_donations;
})
.catch((err) => {
//
})
.finally(() => {
processed_last_submit = true;
//
toast.hideToast();
});
}
}
}
</script>
{#if payment_modal_open}
<div
class="fixed z-10 inset-0 overflow-y-auto"
use:focusTrap
use:clickOutside
on:click_outside={() => {
payment_modal_open = false;
}}>
<div
class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
<div class="fixed inset-0 transition-opacity" aria-hidden="true">
<div
class="absolute inset-0 bg-gray-500 opacity-75"
data-id="modal_backdrop" />
</div>
<span
class="hidden sm:inline-block sm:align-middle sm:h-screen"
aria-hidden="true">&#8203;</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}

View File

@@ -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} />

View File

@@ -4,9 +4,14 @@
import store from "../../store"; import store from "../../store";
import Toastify from "toastify-js"; import Toastify from "toastify-js";
import DonationsEmptyState from "./DonationsEmptyState.svelte"; import DonationsEmptyState from "./DonationsEmptyState.svelte";
import AddDonationPaymentModal from "./AddDonationPaymentModal.svelte";
$: searchvalue = ""; $: searchvalue = "";
$: active_deletes = []; $: active_deletes = [];
export let current_donations = []; export let current_donations = [];
export let payment_modal_open = false;
export let editable = {};
export let original_data = {};
export let paid_amount_input = 0;
const donations_promise = DonationService.donationControllerGetAll().then( const donations_promise = DonationService.donationControllerGetAll().then(
(val) => { (val) => {
current_donations = val; current_donations = val;
@@ -18,8 +23,15 @@
} }
return id.toString() === searchvalue; return id.toString() === searchvalue;
} }
function open_payment_modal(donation) {
editable = Object.assign({}, donation);
original_data = Object.assign({}, donation);
paid_amount_input = (donation.paidAmount/100).toFixed(2);
payment_modal_open = true;
}
</script> </script>
<AddDonationPaymentModal bind:current_donations bind:original_data bind:editable bind:paid_amount_input bind:payment_modal_open />
{#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')} {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')}
{#await donations_promise} {#await donations_promise}
<div <div
@@ -63,6 +75,16 @@
class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
{$_('donation-amount')} {$_('donation-amount')}
</th> </th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
{$_('paid-amount')}
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
{$_('status')}
</th>
<th scope="col" class="relative px-6 py-3"> <th scope="col" class="relative px-6 py-3">
<span class="sr-only">{$_('action')}</span> <span class="sr-only">{$_('action')}</span>
</th> </th>
@@ -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>

View File

@@ -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();

View File

@@ -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}

View File

@@ -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>

View File

@@ -1,5 +1,5 @@
<script> <script>
import { getLocaleFromNavigator, _ } from "svelte-i18n"; import { _ } from "svelte-i18n";
import { DonationService, DonorService } from "@odit/lfk-client-js"; import { DonationService, DonorService } from "@odit/lfk-client-js";
import store from "../../store"; import store from "../../store";
import DonorsEmptyState from "./DonorsEmptyState.svelte"; import DonorsEmptyState from "./DonorsEmptyState.svelte";
@@ -77,6 +77,11 @@
class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
{$_('total-donation-amount')} {$_('total-donation-amount')}
</th> </th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
{$_('total-paid-amount')}
</th>
<th scope="col" class="relative px-6 py-3"> <th scope="col" class="relative px-6 py-3">
<span class="sr-only">{$_('action')}</span> <span class="sr-only">{$_('action')}</span>
</th> </th>
@@ -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">

View File

@@ -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>
- -

View File

@@ -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();

View File

@@ -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) =>
@@ -176,6 +178,7 @@
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 />
<GenerateRunnerCertificates bind:certificates_show bind:generate_orgs />
{#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')}
<button <button
on:click={() => { on:click={() => {
@@ -186,7 +189,7 @@
{$_('import-runners')} {$_('import-runners')}
</button> </button>
{/if} {/if}
{#if store.state.jwtinfo.userdetails.permissions.includes('USER:DELETE')} {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:DELETE')}
{#if delete_triggered} {#if delete_triggered}
<button <button
on:click={deleteOrganization} on:click={deleteOrganization}

View File

@@ -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="gridjs-input gridjs-search-input 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}

View File

@@ -41,7 +41,7 @@
duration: -1, duration: -1,
}).showToast(); }).showToast();
fetch( 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: {
@@ -67,7 +67,7 @@
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 = "Runnercards.pdf"; a.download = `${$_('runnercards')}-${locale}.pdf`;
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
a.remove(); a.remove();
@@ -101,7 +101,7 @@
cards.push(card); cards.push(card);
} }
fetch( 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: {
@@ -127,7 +127,12 @@
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 = "Runnercards.pdf"; if(generate_runners.length == 1){
a.download = `${$_('runnercards')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
}
else{
a.download = `Runnercards-${locale}.pdf`;
}
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
a.remove(); a.remove();
@@ -164,7 +169,7 @@
cards.push(card); cards.push(card);
} }
fetch( 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: {
@@ -191,7 +196,7 @@
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 = "Sponsorings_" + t.name + ".pdf"; a.download = `${$_('runnercards')}_${t.name}-${locale}.pdf`;
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
a.remove(); a.remove();
@@ -211,15 +216,16 @@
async function generateOrgCards(locale) { async function generateOrgCards(locale) {
const toast = Toastify({ const toast = Toastify({
text: $_("generating-pdf"), text: $_("generating-pdfs"),
duration: -1, duration: -1,
}).showToast(); }).showToast();
let count = 0;
const current_cards = await RunnerCardService.runnerCardControllerGetAll(); const current_cards = await RunnerCardService.runnerCardControllerGetAll();
let count = 0;
let count_orgs =0;
for (const o of generate_orgs) { for (const o of generate_orgs) {
const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( count_orgs++;
o.id let count = 0;
); let runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners(o.id, true)
let cards = []; let cards = [];
for (let runner of 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);
@@ -230,8 +236,8 @@
} }
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: {
@@ -240,39 +246,97 @@
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) => { .then((blob) => {
count++; 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 = `${$_('runnercards')}_${o.name}_direct-${locale}.pdf`;
a.download = "Sponsorings_" + o.name + ".pdf"; document.body.appendChild(a);
document.body.appendChild(a); a.click();
a.click(); a.remove();
a.remove(); if (count === o.teams.length && count_orgs === generate_orgs.length) {
if (count === generate_orgs.length) { toast.hideToast();
toast.hideToast(); console.log("here")
Toastify({ Toastify({
text: $_("pdfs-successfully-generated"), text: $_("pdfs-successfully-generated"),
duration: 3500, duration: 3500,
backgroundColor: backgroundColor:
"linear-gradient(to right, #00b09b, #96c93d)", "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast(); }).showToast();
} }
}) })
.catch((err) => {}); .catch((err) => {});
for (const t of o.teams) {
count++;
let runners = await RunnerTeamService.runnerTeamControllerGetRunners(
t.id
);
let cards = [];
for (let runner of runners) {
let card = current_cards.find((c) => c.runner?.id == runner.id);
if (!card) {
card = await RunnerCardService.runnerCardControllerPost({
runner: runner.id,
});
}
cards.push(card);
}
await fetch(
`${config.baseurl_documentserver}/cards?locale=${locale}&download=true&key=${config.documentserver_key}`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(cards),
}
)
.then((response) => {
if (response.status != "200") {
toast.hideToast();
Toastify({
text: $_("pdf-generation-failed"),
duration: 3500,
backgroundColor:
"linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
}).showToast();
} else {
return response.blob();
}
})
.then((blob) => {
const url = window.URL.createObjectURL(blob);
let a = document.createElement("a");
a.href = url;
a.download = `${$_('runnercards')}_${o.name}_${t.name}-${locale}.pdf`;
document.body.appendChild(a);
a.click();
a.remove();
if (count === o.teams.length && count_orgs === generate_orgs.length) {
toast.hideToast();
Toastify({
text: $_("pdfs-successfully-generated"),
duration: 3500,
backgroundColor:
"linear-gradient(to right, #00b09b, #96c93d)",
}).showToast();
}
})
.catch((err) => {});
}
} }
} }
</script> </script>

View File

@@ -1,273 +1,331 @@
<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; export let certificates_show = false;
export let generate_runners = []; export let generate_runners = [];
export let generate_orgs = []; export let generate_orgs = [];
export let generate_teams = []; export let generate_teams = [];
$: certificates_dropdown_open = false; $: certificates_dropdown_open = false;
document.addEventListener("click", function (e) { document.addEventListener("click", function (e) {
if ( if (
e.target.parentNode?.parentNode?.id != "certificates:dropdown" && e.target.parentNode?.parentNode?.id != "certificates:dropdown" &&
e.target.parentNode?.parentNode?.id != "certificates:dropdown:menu" e.target.parentNode?.parentNode?.id != "certificates:dropdown:menu"
) { ) {
certificates_dropdown_open = false; certificates_dropdown_open = false;
} }
}); });
function generateCertificates(locale) { function generateCertificates(locale) {
certificates_dropdown_open = false; certificates_dropdown_open = false;
if (generate_orgs.length > 0) { if (generate_orgs.length > 0) {
generateOrgCertificates(locale); generateOrgCertificates(locale);
} else if (generate_teams.length > 0) { } else if (generate_teams.length > 0) {
generateTeamCertificates(locale); generateTeamCertificates(locale);
} else { } else {
generateRunnerCertificates(locale); generateRunnerCertificates(locale);
} }
} }
async function generateRunnerCertificates(locale) { async function generateRunnerCertificates(locale) {
const toast = Toastify({ const toast = Toastify({
text: $_("generating-pdf"), text: $_("generating-pdf"),
duration: -1, duration: -1,
}).showToast(); }).showToast();
const current_donations = await DonationService.donationControllerGetAll(); const current_donations = (await DonationService.donationControllerGetAll()) || [];
let certificateRunners = []; let certificateRunners = [];
for (let runner of generate_runners) { for (let runner of generate_runners) {
runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
certificateRunners.push(runner); certificateRunners.push(runner);
} }
fetch( fetch(
`${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, `${config.baseurl_documentserver}/certificates?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(certificateRunners), body: JSON.stringify(certificateRunners),
} }
) )
.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 = "Certificates.pdf"; if(generate_runners.length == 1){
document.body.appendChild(a); a.download = `${$_('certificates')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
a.click(); }
a.remove(); else{
toast.hideToast(); a.download = `${$_('certificates')}-${locale}.pdf`;
Toastify({ }
text: $_("pdf-successfully-generated"), document.body.appendChild(a);
duration: 3500, a.click();
backgroundColor: a.remove();
"linear-gradient(to right, #00b09b, #96c93d)", toast.hideToast();
}).showToast(); Toastify({
}) text: $_("pdf-successfully-generated"),
.catch((err) => {}); duration: 3500,
} backgroundColor:
"linear-gradient(to right, #00b09b, #96c93d)",
async function generateTeamCertificates(locale) { }).showToast();
const toast = Toastify({ })
text: $_("generating-pdfs"), .catch((err) => {});
duration: -1, }
}).showToast();
let count = 0; async function generateTeamCertificates(locale) {
const current_donations = await DonationService.donationControllerGetAll(); const toast = Toastify({
for (const t of generate_teams) { text: $_("generating-pdfs"),
const runners = await RunnerTeamService.runnerTeamControllerGetRunners( duration: -1,
t.id }).showToast();
); let count = 0;
let certificateRunners = []; const current_donations = (await DonationService.donationControllerGetAll()) || [];
for (let runner of runners) { for (const t of generate_teams) {
runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; const runners = await RunnerTeamService.runnerTeamControllerGetRunners(
certificateRunners.push(runner); t.id
} );
console.log(certificateRunners) let certificateRunners = [];
fetch( for (let runner of runners) {
`${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
{ certificateRunners.push(runner);
method: "POST", }
headers: { fetch(
"Content-Type": "application/json", `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
}, {
body: JSON.stringify(certificateRunners), method: "POST",
} headers: {
) "Content-Type": "application/json",
.then((response) => { },
if (response.status != "200") { body: JSON.stringify(certificateRunners),
toast.hideToast(); }
Toastify({ )
text: $_("pdf-generation-failed"), .then((response) => {
duration: 3500, if (response.status != "200") {
backgroundColor: toast.hideToast();
"linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", Toastify({
}).showToast(); text: $_("pdf-generation-failed"),
} else { duration: 3500,
return response.blob(); backgroundColor:
} "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
}) }).showToast();
.then((blob) => { } else {
count++; return response.blob();
const url = window.URL.createObjectURL(blob); }
let a = document.createElement("a"); })
a.href = url; .then((blob) => {
a.download = "Certificates_" + t.name + ".pdf"; count++;
document.body.appendChild(a); const url = window.URL.createObjectURL(blob);
a.click(); let a = document.createElement("a");
a.remove(); a.href = url;
if (count === generate_teams.length) { a.download = `${$_('certificates')}_${t.name}-${locale}.pdf`;
toast.hideToast(); document.body.appendChild(a);
Toastify({ a.click();
text: $_("pdfs-successfully-generated"), a.remove();
duration: 3500, if (count === generate_teams.length) {
backgroundColor: toast.hideToast();
"linear-gradient(to right, #00b09b, #96c93d)", Toastify({
}).showToast(); text: $_("pdfs-successfully-generated"),
} duration: 3500,
}) backgroundColor:
.catch((err) => {}); "linear-gradient(to right, #00b09b, #96c93d)",
} }).showToast();
} }
})
async function generateOrgCertificates(locale) { .catch((err) => {});
const toast = Toastify({ }
text: $_("generating-pdf"), }
duration: -1,
}).showToast(); async function generateOrgCertificates(locale) {
let count = 0; const toast = Toastify({
const current_donations = await DonationService.donationControllerGetAll(); text: $_("generating-pdfs"),
for (const o of generate_orgs) { duration: -1,
const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( }).showToast();
o.id const current_donations = (await DonationService.donationControllerGetAll()) || [];
); let count = 0;
let certificateRunners = []; let count_orgs =0;
for (let runner of runners) { for (const o of generate_orgs) {
runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || []; count_orgs++;
certificateRunners.push(runner); let count = 0;
} let runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners(o.id, true)
fetch( let certificateRunners = [];
`${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`, for (let runner of runners) {
{ runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
method: "POST", certificateRunners.push(runner);
headers: { }
"Content-Type": "application/json", await fetch(
}, `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
body: JSON.stringify(certificateRunners), {
} method: "POST",
) headers: {
.then((response) => { "Content-Type": "application/json",
if (response.status != "200") { },
toast.hideToast(); body: JSON.stringify(certificateRunners),
Toastify({ }
text: $_("pdf-generation-failed"), )
duration: 3500, .then((response) => {
backgroundColor: if (response.status != "200") {
"linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", toast.hideToast();
}).showToast(); Toastify({
} else { text: $_("pdf-generation-failed"),
return response.blob(); duration: 3500,
} backgroundColor:
}) "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
.then((blob) => { }).showToast();
count++; } else {
const url = window.URL.createObjectURL(blob); return response.blob();
let a = document.createElement("a"); }
a.href = url; })
a.download = "Certificates_" + o.name + ".pdf"; .then((blob) => {
document.body.appendChild(a); const url = window.URL.createObjectURL(blob);
a.click(); let a = document.createElement("a");
a.remove(); a.href = url;
if (count === generate_orgs.length) { a.download = `${$_('certificates')}_${o.name}_direct-${locale}.pdf`;
toast.hideToast(); document.body.appendChild(a);
Toastify({ a.click();
text: $_("pdfs-successfully-generated"), a.remove();
duration: 3500, if (count === o.teams.length && count_orgs === generate_orgs.length) {
backgroundColor: toast.hideToast();
"linear-gradient(to right, #00b09b, #96c93d)", console.log("here")
}).showToast(); Toastify({
} text: $_("pdfs-successfully-generated"),
}) duration: 3500,
.catch((err) => {}); backgroundColor:
} "linear-gradient(to right, #00b09b, #96c93d)",
} }).showToast();
</script> }
})
{#if certificates_show} .catch((err) => {});
<div id="certificates:dropdown" class="relative inline-block"> for (const t of o.teams) {
<div> count++;
<button let runners = await RunnerTeamService.runnerTeamControllerGetRunners(
on:click={() => { t.id
certificates_dropdown_open = !certificates_dropdown_open; );
}} let certificateRunners = [];
type="button" for (let runner of runners) {
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" runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
id="options-menu" certificateRunners.push(runner);
aria-haspopup="true" }
aria-expanded="true"> await fetch(
{$_('generate-runner-certificates')} `${config.baseurl_documentserver}/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
<svg {
xmlns="http://www.w3.org/2000/svg" method: "POST",
width="24" headers: {
height="24" "Content-Type": "application/json",
viewBox="0 0 24 24" },
class="-mr-1 ml-2 h-5 w-5"><path body: JSON.stringify(certificateRunners),
fill="none" }
d="M0 0h24v24H0z" /> )
<path .then((response) => {
fill="currentColor" if (response.status != "200") {
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> toast.hideToast();
</button> Toastify({
</div> text: $_("pdf-generation-failed"),
{#if certificates_dropdown_open} duration: 3500,
<div backgroundColor:
class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
id="certificates:dropdown:menu"> }).showToast();
<div } else {
class="py-1" return response.blob();
role="menu" }
aria-orientation="vertical" })
aria-labelledby="options-menu"> .then((blob) => {
<span const url = window.URL.createObjectURL(blob);
class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> let a = document.createElement("a");
<button a.href = url;
on:click={() => { a.download = `${$_('certificates')}_${o.name}_${t.name}-${locale}.pdf`;
generateCertificates('de'); document.body.appendChild(a);
}} a.click();
type="submit" a.remove();
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" if (count === o.teams.length && count_orgs === generate_orgs.length) {
role="menuitem"> toast.hideToast();
{$_('german')} Toastify({
</button> text: $_("pdfs-successfully-generated"),
<button duration: 3500,
on:click={() => { backgroundColor:
generateCertificates('en'); "linear-gradient(to right, #00b09b, #96c93d)",
}} }).showToast();
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"> .catch((err) => {});
{$_('english')} }
</button> }
</div> }
</div> </script>
{/if}
</div> {#if certificates_show}
{/if} <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"
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}

View File

@@ -43,7 +43,7 @@
t.id t.id
); );
fetch( fetch(
`${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`,
{ {
method: "POST", method: "POST",
headers: { headers: {
@@ -69,7 +69,7 @@
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 = "Sponsorings_" + t.name + ".pdf"; a.download = `${$_('sponsorings')}_${t.name}-${locale}.pdf`;
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
a.remove(); a.remove();
@@ -92,12 +92,13 @@
text: $_("generating-pdf"), text: $_("generating-pdf"),
duration: -1, duration: -1,
}).showToast(); }).showToast();
let count_orgs =0;
for (const o of generate_orgs) { for (const o of generate_orgs) {
const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( count_orgs++;
o.id let count = 0;
); let runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners(o.id, true)
fetch( await fetch(
`${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`,
{ {
method: "POST", method: "POST",
headers: { headers: {
@@ -106,39 +107,87 @@
body: JSON.stringify(runners), 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) => {
count++; 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 = `${$_('sponsorings')}_${o.name}_direct-${locale}.pdf`;
a.download = "Sponsorings_" + o.name + ".pdf"; document.body.appendChild(a);
document.body.appendChild(a); a.click();
a.click(); a.remove();
a.remove(); if (count === o.teams.length && count_orgs === generate_orgs.length) {
if (count === generate_orgs.length) { toast.hideToast();
toast.hideToast(); console.log("here")
Toastify({ Toastify({
text: $_("pdfs-successfully-generated"), text: $_("pdfs-successfully-generated"),
duration: 3500, duration: 3500,
backgroundColor: backgroundColor:
"linear-gradient(to right, #00b09b, #96c93d)", "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast(); }).showToast();
} }
}) })
.catch((err) => {}); .catch((err) => {});
for (const t of o.teams) {
count++;
let runners = await RunnerTeamService.runnerTeamControllerGetRunners(
t.id
);
await fetch(
`${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(runners),
}
)
.then((response) => {
if (response.status != "200") {
toast.hideToast();
Toastify({
text: $_("pdf-generation-failed"),
duration: 3500,
backgroundColor:
"linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
}).showToast();
} else {
return response.blob();
}
})
.then((blob) => {
const url = window.URL.createObjectURL(blob);
let a = document.createElement("a");
a.href = url;
a.download = `${$_('sponsorings')}_${o.name}_${t.name}-${locale}.pdf`;
document.body.appendChild(a);
a.click();
a.remove();
if (count === o.teams.length && count_orgs === generate_orgs.length) {
toast.hideToast();
Toastify({
text: $_("pdfs-successfully-generated"),
duration: 3500,
backgroundColor:
"linear-gradient(to right, #00b09b, #96c93d)",
}).showToast();
}
})
.catch((err) => {});
}
} }
} }
@@ -148,7 +197,7 @@
duration: -1, duration: -1,
}).showToast(); }).showToast();
fetch( fetch(
`${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, `${config.baseurl_documentserver}/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`,
{ {
method: "POST", method: "POST",
headers: { headers: {
@@ -174,7 +223,10 @@
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"; if(generate_runners.length == 1){
a.download = `${$_('sponsorings')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
}
a.download = `${$_('sponsorings')}-${locale}.pdf`;
document.body.appendChild(a); document.body.appendChild(a);
a.click(); a.click();
a.remove(); a.remove();

View File

@@ -34,7 +34,7 @@
document.onkeydown = (e) => { document.onkeydown = (e) => {
e = e || window.event; e = e || window.event;
if (e.key === "Escape") { if (e.key === "Escape") {
import_modal_open = false; cancelModal();
} }
if (e.keyCode === 13) { if (e.keyCode === 13) {
// //
@@ -202,7 +202,7 @@
toast.hideToast(); toast.hideToast();
recent_processed = true; recent_processed = true;
Toastify({ Toastify({
text: "Import finished", text: $_('import-finished'),
duration: 500, duration: 500,
backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
}).showToast(); }).showToast();
@@ -281,6 +281,16 @@
bind:files bind:files
type="file" /> type="file" />
</div> </div>
<div class="overflow-hidden relative mt-4 mb-4">
<button
on:click={() => {
cancelModal();
}}
type="button"
class="w-full rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 md:ml-40 mr-0 sm:ml-0 sm:w-auto sm:text-sm">
{$_('cancel')}
</button>
</div>
{/if} {/if}
{#if json_output.length > 0} {#if json_output.length > 0}
{#if opened_from === 'OrgOverview'} {#if opened_from === 'OrgOverview'}

View File

@@ -197,7 +197,7 @@
{#if runner.group.responseType === 'RUNNERTEAM'} {#if runner.group.responseType === 'RUNNERTEAM'}
<a <a
href="../teams/{runner.group.id}" 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> class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{runner.group.parentGroup.name} &gt; {runner.group.name}</a>
{/if} {/if}
{#if runner.group.responseType === 'RUNNERORGANIZATION'} {#if runner.group.responseType === 'RUNNERORGANIZATION'}
<a <a

View File

@@ -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();

View File

@@ -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();

View File

@@ -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();

View File

@@ -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={() => {

View File

@@ -1,224 +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";
import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
$: searchvalue = ""; $: searchvalue = "";
$: active_deletes = []; $: active_deletes = [];
$: sponsoring_contracts_show = current_teams.some( $: sponsoring_contracts_show = current_teams.some(
(r) => r.is_selected === true (r) => r.is_selected === true
); );
$: cards_show = current_teams.some( $: cards_show = current_teams.some(
(r) => r.is_selected === true (r) => r.is_selected === true
); );
$: certificates_show = current_teams.some( $: certificates_show = current_teams.some(
(r) => r.is_selected === true (r) => r.is_selected === true
); );
$: generate_teams = current_teams.filter((r) => r.is_selected === true); $: generate_teams = current_teams.filter((r) => r.is_selected === true);
export let current_teams = []; export let current_teams = [];
let modal_open = false; let modal_open = false;
let delete_team = {}; let delete_team = {};
usersstore.subscribe((val) => { usersstore.subscribe((val) => {
current_teams = val; current_teams = val;
}); });
teams_promise.then((data) => { teams_promise.then((data) => {
usersstore.set(data); usersstore.set(data);
}); });
</script> </script>
<ConfirmTeamDeletion <ConfirmTeamDeletion
on:cancelDelete={(event) => { on:cancelDelete={(event) => {
modal_open = false; modal_open = false;
active_deletes[event.detail.id] = false; active_deletes[event.detail.id] = false;
}} }}
bind:modal_open bind:modal_open
bind:delete_team /> bind:delete_team />
{#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')} {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')}
{#await teams_promise} {#await teams_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">{$_('teams-are-being-loaded')}</p> <p class="font-bold">{$_('teams-are-being-loaded')}</p>
<p class="text-sm">{$_('this-might-take-a-moment')}</p> <p class="text-sm">{$_('this-might-take-a-moment')}</p>
</div> </div>
{:then} {:then}
{#if current_teams.length === 0} {#if current_teams.length === 0}
<TeamsEmptyState /> <TeamsEmptyState />
{:else} {:else}
<input <input
type="search" type="search"
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="gridjs-input gridjs-search-input mb-4" />
<div class="h-12"> <div class="h-12">
<GenerateSponsoringContracts <GenerateSponsoringContracts
bind:sponsoring_contracts_show bind:sponsoring_contracts_show
bind:generate_teams /> bind:generate_teams />
<GenerateRunnerCards <GenerateRunnerCards
bind:cards_show bind:cards_show
bind:generate_teams /> bind:generate_teams />
<GenerateRunnerCertificates <GenerateRunnerCertificates
bind:certificates_show bind:certificates_show
bind:generate_teams /> bind:generate_teams />
</div> </div>
<div <div
class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll">
<table class="divide-y divide-gray-200 w-full"> <table class="divide-y divide-gray-200 w-full">
<thead class="bg-gray-50"> <thead class="bg-gray-50">
<tr> <tr>
<th <th
scope="col" scope="col"
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">
<span <span
on:click={() => { on:click={() => {
const newstate = !current_teams.some((r) => r.is_selected === true); const newstate = !current_teams.some((r) => r.is_selected === true);
current_teams = current_teams.map((r) => { current_teams = current_teams.map((r) => {
r.is_selected = newstate; r.is_selected = newstate;
return r; return r;
}); });
}} }}
class="underline cursor-pointer select-none">{#if current_teams.some((r) => r.is_selected === true)} class="underline cursor-pointer select-none">{#if current_teams.some((r) => r.is_selected === true)}
{$_('deselect-all')} {$_('deselect-all')}
{:else}{$_('select-all')}{/if} {:else}{$_('select-all')}{/if}
</span> </span>
</th> </th>
<th <th
scope="col" scope="col"
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">
{$_('name')} {$_('name')}
</th> </th>
<th <th
scope="col" scope="col"
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">
{$_('organization')} {$_('organization')}
</th> </th>
<th <th
scope="col" scope="col"
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">
{$_('contact')} {$_('contact')}
</th> </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>
</tr> </tr>
</thead> </thead>
<tbody class="divide-y divide-gray-200"> <tbody class="divide-y divide-gray-200">
{#each current_teams as t} {#each current_teams as t}
{#if Object.values(t) {#if Object.values(t)
.toString() .toString()
.toLowerCase() .toLowerCase()
.includes(searchvalue)} .includes(searchvalue)}
<tr data-rowid="team_{t.id}"> <tr data-rowid="team_{t.id}">
<td class="px-6 py-4 whitespace-nowrap"> <td class="px-6 py-4 whitespace-nowrap">
<input <input
bind:checked={t.is_selected} bind:checked={t.is_selected}
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" />
</td> </td>
<td class="px-6 py-4 whitespace-nowrap"> <td class="px-6 py-4 whitespace-nowrap">
<div class="flex items-center"> <div class="flex items-center">
<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">
{t.name} {t.name}
</div> </div>
</div> </div>
</div> </div>
</td> </td>
<td class="px-6 py-4 whitespace-nowrap"> <td class="px-6 py-4 whitespace-nowrap">
<div class="flex items-center"> <div class="flex items-center">
<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">
{#if t.parentGroup} {#if t.parentGroup}
<a <a
href="../orgs/{t.parentGroup.id}" href="../orgs/{t.parentGroup.id}"
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a> class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a>
{:else}{$_('no-organization-specified')}{/if} {:else}{$_('no-organization-specified')}{/if}
</div> </div>
</div> </div>
</div> </div>
</td> </td>
<td class="px-6 py-4 whitespace-nowrap"> <td class="px-6 py-4 whitespace-nowrap">
<div class="flex items-center"> <div class="flex items-center">
<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">
{#if t.contact} {#if t.contact}
<a <a
href="../contacts/{t.contact.id}" href="../contacts/{t.contact.id}"
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname} class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname}
{t.contact.middlename || ''} {t.contact.middlename || ''}
{t.contact.lastname}</a> {t.contact.lastname}</a>
{:else}{$_('no-contact-specified')}{/if} {:else}{$_('no-contact-specified')}{/if}
</div> </div>
</div> </div>
</div> </div>
</td> </td>
{#if active_deletes[t.id] === true} {#if active_deletes[t.id] === true}
<td <td
class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
<button <button
on:click={() => { on:click={() => {
active_deletes[t.id] = false; active_deletes[t.id] = false;
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">Cancel class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">Cancel
Delete</button> Delete</button>
<button <button
on:click={() => { on:click={() => {
RunnerTeamService.runnerTeamControllerRemove(t.id, false) RunnerTeamService.runnerTeamControllerRemove(t.id, false)
.then((resp) => { .then((resp) => {
current_teams = current_teams.filter((obj) => obj.id !== t.id); current_teams = current_teams.filter((obj) => obj.id !== t.id);
Toastify({ Toastify({
text: $_('organization-deleted'), text: $_('organization-deleted'),
duration: 500, duration: 500,
backgroundColor: backgroundColor:
'linear-gradient(to right, #00b09b, #96c93d)', 'linear-gradient(to right, #00b09b, #96c93d)',
}).showToast(); }).showToast();
}) })
.catch((err) => { .catch((err) => {
modal_open = true; modal_open = true;
delete_team = t; delete_team = t;
}); });
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button>
</td> </td>
{:else} {:else}
<td <td
class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
<a <a
href="./{t.id}" href="./{t.id}"
class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a>
{#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')} {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')}
<button <button
on:click={() => { on:click={() => {
active_deletes[t.id] = true; active_deletes[t.id] = true;
}} }}
tabindex="0" tabindex="0"
class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button>
{/if} {/if}
</td> </td>
{/if} {/if}
</tr> </tr>
{/if} {/if}
{/each} {/each}
</tbody> </tbody>
</table> </table>
</div> </div>
{/if} {/if}
{:catch error} {:catch error}
<div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500">
<span class="inline-block align-middle mr-8"> <span class="inline-block align-middle mr-8">
<b class="capitalize">{$_('general_promise_error')}</b> <b class="capitalize">{$_('general_promise_error')}</b>
{error} {error}
</span> </span>
</div> </div>
{/await} {/await}
{/if} {/if}

View File

@@ -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();

View File

@@ -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) => {

View File

@@ -1,428 +1,461 @@
{ {
"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-scan": "Scan erstellen",
"add-the-first-scanstation": "Erstelle deine erste Scannerstation.", "add-the-first-scanstation": "Erstelle deine erste Scannerstation.",
"add-user-group": "Neue Gruppe erstellen", "add-user-group": "Neue Gruppe erstellen",
"add-your-first-card": "Erstelle deine erste Läuferkarte", "add-your-first-card": "Erstelle deine erste Läuferkarte",
"add-your-first-contact": "Erstelle den ersten Kontakt", "add-your-first-contact": "Erstelle den ersten Kontakt",
"add-your-first-donor": "Erstelle die erste Sponsor:in", "add-your-first-donor": "Erstelle die erste Sponsor:in",
"add-your-first-group": "Erstelle die erste Gruppe", "add-your-first-group": "Erstelle die erste Gruppe",
"add-your-first-organization": "Erstelle die erste Organisation", "add-your-first-organization": "Erstelle die erste Organisation",
"add-your-first-runner": "Erstelle die erste Läufer:in", "add-your-first-runner": "Erstelle die erste Läufer:in",
"add-your-first-team": "Erstelle das erste Team", "add-your-first-team": "Erstelle das erste Team",
"add-your-first-track": "Erstelle den ersten Track (Laufstrecke).", "add-your-first-track": "Erstelle den ersten Track (Laufstrecke).",
"add-your-first-user": "Erstelle die erste Benutzer:in", "add-your-first-user": "Erstelle die erste Benutzer:in",
"add-your-fist-donation": "Erstelle dein erstes Sponsoring", "add-your-fist-donation": "Erstelle dein erstes Sponsoring",
"add-your-fist-scan": "Füge deinen ersten Scan hinzu", "add-your-fist-scan": "Füge deinen ersten Scan hinzu",
"adding-card": "Karte wird erstellt", "adding-card": "Karte wird erstellt",
"adding-scan": "Scan wird hinzugefügt", "adding-donation": "Sponsoring wird erstellt...",
"address": "Adresse", "adding-scan": "Scan wird hinzugefügt",
"address-is-required": "Du musst eine Adresse angeben", "address": "Adresse",
"after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!", "address-is-required": "Du musst eine Adresse angeben",
"after-the-update-youll-get-logged-out-please-login-with-your-new-password-after-that": "Nach der Änderung wirst du abgemeldet - bitte melde dich dann mit deinem neuen Passwort an.", "after-deletion-we-cant-restore-your-old-profile": "Nach der Löschung können auch die Admins dein Profil nicht wiederherstellen!",
"all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls 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.",
"all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!", "all-associated-donations-will-get-deleted-as-well": "Alle Sponsorings dieser Sponsor:in werden ebenfalls gelöscht",
"all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!", "all-associated-runners-will-be-deleted-too": "Alle zugehörigen Läufer:innen werden auch gelöscht!",
"amount": "Anzahl", "all-associated-teams-and-runners-will-be-deleted-too": "Alle assoziierten Teams und Läufer:innen werden auch gelöscht!",
"amount-per-kilometer": "Betrag pro Kilometer", "already-paid": "Bereits bezahlt",
"apartment-suite-etc": "Apartment, Wohnung, etc.", "amount": "Anzahl",
"application_name": "Lauf für Kaya! - Admin", "amount-per-kilometer": "Betrag pro Kilometer",
"applying-changes": "Änderungen anwenden", "apartment-suite-etc": "Apartment, Wohnung, etc.",
"attention": "Achtung!", "application_name": "Lauf für Kaya! - Admin",
"author": "Autor:in", "applying-changes": "Änderungen anwenden",
"bitte-bestaetige-diese-laeufer-fuer-den-import": "Bitte die Läufer:innen für den Import bestätigen.", "attention": "Achtung!",
"by": "von", "author": "Autor:in",
"cancel": "Abbrechen", "bitte-bestaetige-diese-laeufer-fuer-den-import": "Bitte die Läufer:innen für den Import bestätigen.",
"cancel-delete": "Löschen abbrechen", "by": "von",
"cancel-keep-donor": "Abbrechen, Sponsor:in behalten", "cancel": "Abbrechen",
"cancel-keep-my-profile": "Abbrechen, mein Profil behalten", "cancel-delete": "Löschen abbrechen",
"cancel-keep-organization": "Abbrechen und Organisation bearbeiten", "cancel-keep-donor": "Abbrechen, Sponsor:in behalten",
"cancel-keep-team": "Abbrechen, Team behalten", "cancel-keep-my-profile": "Abbrechen, mein Profil behalten",
"cannot-reset-your-password-directly": "Schade. \nWir können das Passwort leider nicht direkt zurücksetzen.\nBitte sende uns eine Mail in der du deine Identität bestätigst.", "cancel-keep-organization": "Abbrechen und Organisation bearbeiten",
"card-added": "Karte wurde hinzugefügt", "cancel-keep-team": "Abbrechen, Team behalten",
"card-deleted": "Karte gelöscht", "cannot-reset-your-password-directly": "Schade. \nWir können das Passwort leider nicht direkt zurücksetzen.\nBitte sende uns eine Mail in der du deine Identität bestätigst.",
"card-updated": "Karte aktualisiert", "card-added": "Karte wurde hinzugefügt",
"cards": "Läuferkarten", "card-deleted": "Karte gelöscht",
"change-your-password-here": "Hier kannst du dein Passwort ändern", "card-updated": "Karte aktualisiert",
"changing-your-password": "Passwort wird geändert", "cards": "Läuferkarten",
"city": "Stadt", "certificates": "Urkunden",
"click-to-copy-the-link-into-your-clipboard": "Klicke auf den Link, um ihn in deine Zwischenablage zu kopieren", "change-your-password-here": "Hier kannst du dein Passwort ändern",
"click-to-copy-token-to-clipboard": "Klicke auf den Token, um ihn in deine Zwischenablage zu kopieren", "changing-your-password": "Passwort wird geändert",
"close": "Schließen", "city": "Stadt",
"code": "Code", "click-to-copy-the-link-into-your-clipboard": "Klicke auf den Link, um ihn in deine Zwischenablage zu kopieren",
"configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit", "click-to-copy-token-to-clipboard": "Klicke auf den Token, um ihn in deine Zwischenablage zu kopieren",
"confirm": "Bestätigen", "close": "Schließen",
"confirm-delete": "Löschung Bestätigen", "code": "Code",
"confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen", "configure-the-tracks-and-minimum-lap-times": "Bearbeite die Tracks und ihre minimale Rundenzeit",
"confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen", "confirm": "Bestätigen",
"confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.", "confirm-delete": "Löschung Bestätigen",
"confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.", "confirm-delete-donor-with-all-donations": "Bestätigen, Sponsor:in mit allen Sponsorings löschen",
"confirm-deletion": "Löschung Bestätigen", "confirm-delete-my-user-profile": "Bestätigung, mein Benutzerprofil löschen",
"confirm-the-new-password": "Neues Passwort bestätigen", "confirm-delete-organization-and-associated-teams-runners": "Bestätugung, lösche die Organisation und alle zugehörigen Teams und Läufer:innen.",
"contact": "Kontakt", "confirm-delete-team-and-associated-runners": "Bestätigung, lösche das Team mitsamt seinen Läufer:innen.",
"contact-deleted": "Kontakt gelöscht", "confirm-deletion": "Löschung Bestätigen",
"contact-information": "Kontaktinformation", "confirm-the-new-password": "Neues Passwort bestätigen",
"contact-is-being-updated": "Kontakt wird aktualisiert ...", "contact": "Kontakt",
"contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe", "contact-added": "Kontakt wurde hinzugefügt",
"contacts": "Kontakte", "contact-deleted": "Kontakt gelöscht",
"contacts-are-being-loaded": "Kontakte werden geladen ...", "contact-information": "Kontaktinformation",
"copied-link-to-clipboard": "Link wurde in die Zwischenablage kopiert", "contact-is-being-added": "Kontakt wird erstellt...",
"copied-token-to-clipboard": "Token wurde in die Zwischenablage kopiert", "contact-is-being-updated": "Kontakt wird aktualisiert ...",
"count_organizations": "Organisationen (Anzahl)", "contact-is-not-a-member-in-any-group": "Kontakt gehört zu keiner Gruppe",
"count_teams": "Teams (Anzahl)", "contacts": "Kontakte",
"create": "Erstellen", "contacts-are-being-loaded": "Kontakte werden geladen ...",
"create-a-new": "Erstelle eine neue", "copied-link-to-clipboard": "Link wurde in die Zwischenablage kopiert",
"create-a-new-card": "Neue Läuferkarte erstellen", "copied-token-to-clipboard": "Token wurde in die Zwischenablage kopiert",
"create-a-new-contact": "Kontakt erstellen", "count_organizations": "Organisationen (Anzahl)",
"create-a-new-distance-donation": "Erstelle ein neues Sponsoring", "count_teams": "Teams (Anzahl)",
"create-a-new-donor": "Neue Sponsor:in erstellen", "create": "Erstellen",
"create-a-new-fixed-donation": "Erstelle eine neue Festbetragsspende", "create-a-new": "Erstelle eine neue",
"create-a-new-organization": "Neue Organisation anlegen", "create-a-new-card": "Neue Läuferkarte erstellen",
"create-a-new-runner": "Neue Läufer:in erstellen", "create-a-new-contact": "Kontakt erstellen",
"create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)", "create-a-new-distance-donation": "Erstelle ein neues Sponsoring",
"create-a-new-scanstation": "Neue Station erstellen", "create-a-new-donor": "Neue Sponsor:in erstellen",
"create-a-new-team": "Erstelle ein neues Team", "create-a-new-fixed-donation": "Erstelle eine neue Festbetragsspende",
"create-a-new-track": "Neuen Track erstellen", "create-a-new-organization": "Neue Organisation anlegen",
"create-a-new-user": "Neue Benutzer:in anlegen", "create-a-new-runner": "Neue Läufer:in erstellen",
"create-a-new-user-group": "Erstelle eine neue Gruppe", "create-a-new-scan-fixed-only": "Neuen Scan erstellen (nur mit Festdistanz)",
"create-and-generate-pdf": "Erstellen und PDF herunterladen", "create-a-new-scanstation": "Neue Station erstellen",
"create-bulk-blanco-cards": "Blankokarten erstellen", "create-a-new-team": "Erstelle ein neues Team",
"create-bulk-cards": "Blankokarten erstellen", "create-a-new-track": "Neuen Track erstellen",
"create-organization": "Organisation erstellen", "create-a-new-user": "Neue Benutzer:in anlegen",
"create-team": "Team erstellen", "create-a-new-user-group": "Erstelle eine neue Gruppe",
"create-track": "Track erstellen", "create-and-generate-pdf": "Erstellen und PDF herunterladen",
"create-user": "Benutzer anlegen", "create-bulk-blanco-cards": "Blankokarten erstellen",
"create-without-pdf": "Ohne PDF erstellen", "create-bulk-cards": "Blankokarten erstellen",
"created-blanco-cards": "Blankokarten wurden erstellt", "create-organization": "Organisation erstellen",
"creating-blanco-cards": "Erstelle Blankokarten", "create-team": "Team erstellen",
"credits": "Credits", "create-track": "Track erstellen",
"csv_import__class": "Klasse", "create-user": "Benutzer anlegen",
"csv_import__firstname": "Vorname", "create-without-pdf": "Ohne PDF erstellen",
"csv_import__lastname": "Nachname", "created-blanco-cards": "Blankokarten wurden erstellt",
"csv_import__middlename": "Mittelname", "creating-blanco-cards": "Erstelle Blankokarten",
"csv_import__team": "Team", "credits": "Credits",
"danger-zone": "Gefahrenzone", "csv_import__class": "Klasse",
"dashboard-greeting": "Hallo", "csv_import__firstname": "Vorname",
"dashboard-title": "Dashboard", "csv_import__lastname": "Nachname",
"datatable": { "csv_import__middlename": "Mittelname",
"search": "🔍 Suche ...", "csv_import__team": "Team",
"an_error_happened_while_fetching_the_data": "Beim Abrufen der Daten ist ein Fehler aufgetreten", "danger-zone": "Gefahrenzone",
"loading": "Wird geladen...", "dashboard-greeting": "Hallo",
"next": "Nächste", "dashboard-title": "Dashboard",
"of": "von", "datatable": {
"previous": "Vorherige", "search": "🔍 Suche ...",
"to": "bis", "an_error_happened_while_fetching_the_data": "Beim Abrufen der Daten ist ein Fehler aufgetreten",
"showing": "Zeige", "loading": "Wird geladen...",
"no_matching_records_found": "Keine passenden Einträge gefunden", "next": "Nächste",
"page": "Seite", "of": "von",
"records": "Einträge", "previous": "Vorherige",
"sort_column_ascending": "Spalte aufsteigend sortieren", "to": "bis",
"sort_column_descending": "Spalte absteigend sortieren" "showing": "Zeige",
}, "no_matching_records_found": "Keine passenden Einträge gefunden",
"delete": "Löschen", "page": "Seite",
"delete-contact": "Kontakt löschen", "records": "Einträge",
"delete-donation": "Sponsporing löschen", "sort_column_ascending": "Spalte aufsteigend sortieren",
"delete-donor": "Sponsor:in löschen", "sort_column_descending": "Spalte absteigend sortieren"
"delete-group": "Gruppe löschen", },
"delete-organization": "Organisation löschen", "delete": "Löschen",
"delete-profile": "Profil löschen", "delete-contact": "Kontakt löschen",
"delete-runner": "Läufer:in löschen", "delete-donation": "Sponsoring löschen",
"delete-scan": "Scan löschen", "delete-donor": "Sponsor:in löschen",
"delete-station": "Station löschen", "delete-group": "Gruppe löschen",
"delete-team": "Team Löschen", "delete-organization": "Organisation löschen",
"delete-user": "Benutzer:in löschen", "delete-profile": "Profil löschen",
"deleted-scan": "Scan wurde gelöscht", "delete-runner": "Läufer:in löschen",
"dependency_name": "Name", "delete-scan": "Scan löschen",
"description": "Beschreibung", "delete-station": "Station löschen",
"description-optional": "Beschreibung (optional)", "delete-team": "Team Löschen",
"deselect-all": "Alle abwählen", "delete-user": "Benutzer:in löschen",
"details": "Details", "deleted-scan": "Scan wurde gelöscht",
"disabled": "deaktiviert", "dependency_name": "Name",
"distance": "Distanz", "description": "Beschreibung",
"distance-donation": "Sponsoring", "description-optional": "Beschreibung (optional)",
"distance-in-km": "Distanz (in KM)", "deselect-all": "Alle abwählen",
"distance-track": "Distanz (+Track)", "details": "Details",
"do-you-really-want-to-delete-your-profile": "Möchtest du dein Profil wirklich löschen?", "disabled": "deaktiviert",
"do-you-want-to-delete-the-organization-delete_org-name": "Möchtest du die Organisation {orgname} löschen?", "distance": "Distanz",
"do-you-want-to-delete-the-team-delete_team-name": "Möchtest du das Team {teamname} löschen?", "distance-donation": "Sponsoring",
"do-you-want-to-delete-this-donor-with-all-related-donations": "Möchtest du diese Sponsor:in mit all ihren Sponsorings löschen?", "distance-in-km": "Distanz (in KM)",
"documentation": "Dokumentation", "distance-track": "Distanz (+Track)",
"donation-amount": "Sponsoringbetrag", "do-you-really-want-to-delete-your-profile": "Möchtest du dein Profil wirklich löschen?",
"donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.", "do-you-want-to-delete-the-organization-delete_org-name": "Möchtest du die Organisation {orgname} löschen?",
"donations": "Sponsorings", "do-you-want-to-delete-the-team-delete_team-name": "Möchtest du das Team {teamname} löschen?",
"donor": "Sponsor:in", "do-you-want-to-delete-this-donor-with-all-related-donations": "Möchtest du diese Sponsor:in mit all ihren Sponsorings löschen?",
"donor-added": "Sponsor:in hinzugefügt", "documentation": "Dokumentation",
"donor-deleted": "Sponsor:in gelöscht", "donation-amount": "Sponsoringbetrag",
"donor-has-no-associated-donations": "Zur Sponsor:in gibt es noch keine Sponsorings", "donation-amount-must-be-greater-that-0-00eur": "Der Sponsoringbetrag muss größer als 0.00€ sein.",
"donor-is-being-added": "Sponsor:in wird hinzugefügt...", "donation-deleted": "Sponsoring gelöscht",
"donor-is-being-updated": "Sponsor:in wird aktualisiert", "donation-updated": "Sponsoring wurde aktualisiert",
"donors": "Sponsor:innen", "donation_added": "Sponsoring hinzugefügt",
"donors-are-being-loaded": "Sponsor:innen werden geladen", "donations": "Sponsorings",
"dont-have-your-email-connected": "Deine E-Mail ist nicht verknüpft?", "donor": "Sponsor:in",
"dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌", "donor-added": "Sponsor:in hinzugefügt",
"e-mail-adress": "E-Mail-Adresse", "donor-deleted": "Sponsor:in gelöscht",
"edit": "Bearbeiten", "donor-has-no-associated-donations": "Zur Sponsor:in gibt es noch keine Sponsorings",
"edit-a-card": "Läuferkarte bearbeiten", "donor-is-being-added": "Sponsor:in wird hinzugefügt...",
"edit-permissions": "Berechtigungen bearbeiten", "donor-is-being-updated": "Sponsor:in wird aktualisiert",
"email_address_or_username": "E-Mail-Adresse/ Benutzername", "donors": "Sponsor:innen",
"enabled": "aktiviert", "donors-are-being-loaded": "Sponsor:innen werden geladen",
"enabled_large": "Aktiviert", "dont-have-your-email-connected": "Deine E-Mail ist nicht verknüpft?",
"english": "Englisch", "dont-panic-were-resetting-it": "Keine Panik, wir setzen es zurück ✌",
"error-during-import": "Fehler beim Importieren", "e-mail-adress": "E-Mail-Adresse",
"error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage", "edit": "Bearbeiten",
"error_on_login": "😢Fehler beim Login", "edit-a-card": "Läuferkarte bearbeiten",
"erteilte": "Direkt erteilte", "edit-permissions": "Berechtigungen bearbeiten",
"everything-concerning-your-profile": "Alles zu deinem Profil", "email_address_or_username": "E-Mail-Adresse/ Benutzername",
"everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃‍♂️🏃‍♀️🏃‍♂️", "enabled": "aktiviert",
"faq": "FAQ", "enabled_large": "Aktiviert",
"filter-by-organization-team": "Filtern nach Organisation / Team", "english": "Englisch",
"first-name": "Vorname", "enter-payment": "Zahlung eingeben",
"first-name-is-required": "Vorname muss angegeben werden", "error-during-import": "Fehler beim Importieren",
"first-scan-of-the-day": "Erster Scan des Tages", "error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage",
"fixed-donation": "Festbetragsspende", "error_on_login": "😢Fehler beim Login",
"forgot_password": "Passwort vergessen?", "erteilte": "Direkt erteilte",
"geerbte": "geerbte", "everything-concerning-your-profile": "Alles zu deinem Profil",
"general-stats": "Allgemeine Statistiken", "everything-is-more-fun-together": "Im Team macht's mehr Spaß 🏃‍♂️🏃‍♀️🏃‍♂️",
"general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten", "faq": "FAQ",
"generate-runnercards": "Läuferkarten generieren", "filename_sponsoringquittungsliste": "SponsoringQuittungsListe",
"generate-sponsoring-contract": "Sponsoringvertrag generieren", "filter-by-organization-team": "Filtern nach Organisation / Team",
"generate-sponsoring-contracts": "Sponsoringverträge generieren", "first-name": "Vorname",
"generating-pdf": "Pdf wird generiert...", "first-name-is-required": "Vorname muss angegeben werden",
"generating-pdfs": "PDFs werden generiert...", "first-scan-of-the-day": "Erster Scan des Tages",
"generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.", "fixed-donation": "Festbetragsspende",
"german": "Deutsch", "forgot_password": "Passwort vergessen?",
"go-to-login": "Zum Login", "geerbte": "geerbte",
"goback": "Zur Startseite", "general-stats": "Allgemeine Statistiken",
"granted": "Gewährt", "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten",
"group": "Gruppe", "generate-runner-certificate": "Urkunde generieren",
"group-added": "Gruppe hinzugefügt", "generate-runner-certificates": "Urkunden generieren",
"group-is-being-added": "Gruppe wird erstellt", "generate-runnercards": "Läuferkarten generieren",
"group-name-is-required": "Der Gruppenname muss angegeben werden.", "generate-sponsoring-contract": "Sponsoringvertrag generieren",
"group-updated": "Gruppe aktualisiert", "generate-sponsoring-contracts": "Sponsoringverträge generieren",
"groups": "Gruppen", "generating-pdf": "PDF wird generiert...",
"groups-are-being-loaded": "Gruppen werden geladen", "generating-pdfs": "PDFs werden generiert...",
"home": "Start", "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.",
"icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:", "german": "Deutsch",
"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.", "go-to-login": "Zum Login",
"import-finished": "Import abgeschlossen", "goback": "Zur Startseite",
"import-runners": "Läufer:innen importieren", "granted": "Gewährt",
"import__target-organization": "Ziel Organisation", "group": "Gruppe",
"imprint": "Impressum ", "group-added": "Gruppe hinzugefügt",
"imprint-loading": "Impressum lädt...", "group-is-being-added": "Gruppe wird erstellt",
"inactive": "Inaktiv", "group-name-is-required": "Der Gruppenname muss angegeben werden.",
"installed-version": "Installierte Version", "group-updated": "Gruppe aktualisiert",
"internal-error": "Interner Fehler", "groups": "Gruppen",
"invalid": "Ungültig", "groups-are-being-loaded": "Gruppen werden geladen",
"invalid-mail-reset": "Das ist keine gültige E-Mail", "home": "Start",
"just-enter-how-many-you-want-and-the-system-will-create-them": "Gebe einfach ein, wie viele Blankokarten das System erstellen soll.", "icon-image-credits": "Wir möchten uns außerdem für die verwendeten Icons und Bilder bedanken bei:",
"laeufer-hinzufuegen": "Läufer:in hinzufügen", "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.",
"laeufer-importieren": "Läufer:innen importieren", "import-finished": "Import abgeschlossen",
"laptime": "Rundenzeit", "import-runners": "Läufer:innen importieren",
"last-name": "Nachname", "import__target-organization": "Ziel Organisation",
"last-name-is-required": "Nachname muss angegeben werden", "imprint": "Impressum ",
"lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.", "imprint-loading": "Impressum lädt...",
"license": "Lizenz", "inactive": "Inaktiv",
"licenses-are-being-loaded": "Lizenzen werden geladen...", "installed-version": "Installierte Version",
"loading-cards": "Läuferkarten werden geladen", "internal-error": "Interner Fehler",
"loading-contact-details": "Kontaktdaten werden geladen ...", "invalid": "Ungültig",
"loading-donation-details": "Lade Sponsoringdetails", "invalid-mail-reset": "Das ist keine gültige E-Mail",
"loading-donor-details": "Lade Details", "just-enter-how-many-you-want-and-the-system-will-create-them": "Gebe einfach ein, wie viele Blankokarten das System erstellen soll.",
"loading-group-detail": "Lade Gruppendetails...", "laeufer-hinzufuegen": "Läufer:in hinzufügen",
"loading-profile-data": "Lade Profildaten", "laeufer-importieren": "Läufer:innen importieren",
"loading-runners": "Läufer:innen werden geladen...", "laptime": "Rundenzeit",
"loading-station-details": "Lade Scanstation-Details ...", "last-name": "Nachname",
"log_in": "Anmelden", "last-name-is-required": "Nachname muss angegeben werden",
"log_in_to_your_account": "Bitte melde dich an", "lfk-is-os": "Das \"Lauf für Kaya!\" Frontend ist (wie alle anderen Projekte für den \"LfK!\" auch) ein OpenSource Projekt.",
"login_is_checked": "Login wird überprüft", "license": "Lizenz",
"logout": "Abmelden", "licenses-are-being-loaded": "Lizenzen werden geladen...",
"mail-validation-in-progress": "E-Mail Verifizierung läuft... ", "loading-cards": "Läuferkarten werden geladen",
"manage-admin-users": "Nutzer verwalten", "loading-contact-details": "Kontaktdaten werden geladen ...",
"middle-name": "Mittelname", "loading-donation-details": "Lade Sponsoringdetails",
"minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)", "loading-donor-details": "Lade Details",
"minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein", "loading-group-detail": "Lade Gruppendetails...",
"must-be-at-least-10-characters-long": "Passwort muss mindestens 10 Zeichen lang sein!", "loading-profile-data": "Lade Profildaten",
"must-contain-a-lowercase-letter": "Passwort muss einen Großbuchstaben enthalten!", "loading-runners": "Läufer:innen werden geladen...",
"must-contain-a-number": "Passwort muss eine Zahl enthalten!", "loading-station-details": "Lade Scanstation-Details ...",
"must-contain-a-uppercase-letter": "Passwort muss einen Kleinbuchstaben enthalten!", "log_in": "Anmelden",
"name": "Name", "log_in_to_your_account": "Bitte melde dich an",
"name-is-required": "Der Gruppenname muss angegeben werden", "login_is_checked": "Login wird überprüft",
"new-password": "Neues Passwort", "logout": "Abmelden",
"no-contact-found": "Keine Kontakte gefunden", "mail-validation-in-progress": "E-Mail Verifizierung läuft... ",
"no-contact-selected": "Kein Kontakt ausgewählt", "manage-admin-users": "Nutzer verwalten",
"no-contact-specified": "Kein Kontakt angegeben", "middle-name": "Mittelname",
"no-donors-found": "Keine Spender:innen gefunden", "minimum-lap-time-in-s": "Minimale Rundenzeit (in Sekunden)",
"no-license-text-could-be-found": "Kein Lizenz-Text gefunden 😢", "minimum-lap-time-must-be-a-positive-number-or-0": "Die minimale Rundenzeit muss eine positive Zahl oder 0 sein",
"no-organization-or-team-found": "Keine Organisationen oder Teams gefunden", "must-be-at-least-10-characters-long": "Passwort muss mindestens 10 Zeichen lang sein!",
"no-organization-specified": "Keine Organisation angegeben", "must-contain-a-lowercase-letter": "Passwort muss einen Großbuchstaben enthalten!",
"no-organizations-found": "Keine Organisationen gefunden", "must-contain-a-number": "Passwort muss eine Zahl enthalten!",
"no-runners-found": "Keine Läufer:innen gefunden", "must-contain-a-uppercase-letter": "Passwort muss einen Kleinbuchstaben enthalten!",
"no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.", "name": "Name",
"non-blanko": "Keine/Blankokarte", "name-is-required": "Der Gruppenname muss angegeben werden",
"organization": "Organisation", "new-password": "Neues Passwort",
"organization-added": "Organisation hinzugefügt", "no-contact-found": "Keine Kontakte gefunden",
"organization-deleted": "Organisation gelöscht", "no-contact-selected": "Kein Kontakt ausgewählt",
"organization-detail-is-being-loaded": "Organisationsdetails werden geladen ...", "no-contact-specified": "Kein Kontakt angegeben",
"organization-is-being-added": "Organisation wird hinzugefügt ...", "no-donors-found": "Keine Spender:innen gefunden",
"organization-name-is-required": "Der Name muss angegeben werden", "no-license-text-could-be-found": "Kein Lizenz-Text gefunden 😢",
"organizations": "Organisationen", "no-organization-or-team-found": "Keine Organisationen oder Teams gefunden",
"organizations-are-being-loaded": "Organisationen werden geladen ...", "no-organization-specified": "Keine Organisation angegeben",
"orgs": "Organisationen", "no-organizations-found": "Keine Organisationen gefunden",
"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!", "no-runners-found": "Keine Läufer:innen gefunden",
"password": "Passwort", "no-tracks-added-yet": "Es wurden noch keine Tracks erstellt.",
"password-changed": "Passwort wurde aktualisiert!", "non-blanko": "Keine/Blankokarte",
"password-is-required": "Passwort muss angegeben werden", "open": "OFFEN",
"password-reset-failed": "Passwort zurücksetzen ist fehlgeschlagen!", "organization": "Organisation",
"password-reset-in-progress": "Passwort wird zurückgesetzt...", "organization-added": "Organisation hinzugefügt",
"password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.", "organization-deleted": "Organisation gescht",
"password-reset-successful": "Passwort erfolgreich zurückgesetzt!", "organization-detail-is-being-loaded": "Organisationsdetails werden geladen ...",
"passwords-dont-match": "Die Passwörter stimmen nicht überein!", "organization-is-being-added": "Organisation wird hinzugefügt ...",
"pdf-generation-failed": "PDF Generierung fehlgeschlagen!", "organization-name-is-required": "Der Name muss angegeben werden",
"pdf-successfully-generated": "PDF wurde erfolgreich generiert!", "organizations": "Organisationen",
"pdfs-successfully-generated": "Alle PDFs wurden generiert!", "organizations-are-being-loaded": "Organisationen werden geladen ...",
"per-kilometer": "pro Kilometer", "orgs": "Organisationen",
"permissions": "Berechtigungen", "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!",
"permissions-updated": "Berechtigungen aktualisiert!", "paid": "BEZAHLT",
"phone": "Telefon", "paid-amount": "Gezahlter Betrag",
"please-copy-the-token-and-store-it-somewhere-save": "Bitte kopiere dir den Token und bewahre ihn gut auf.", "password": "Passwort",
"please-provide-a-password": "Bitte gebe ein Passwort an...", "password-changed": "Passwort wurde aktualisiert!",
"please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen", "password-is-required": "Passwort muss angegeben werden",
"please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.", "password-reset-failed": "Passwort zurücksetzen ist fehlgeschlagen!",
"please-provide-the-nessecary-information-to-create-a-new-scan": "Bitte gebe alle notwendigen Informationen an, um einen neuen Scan zu erstellen.", "password-reset-in-progress": "Passwort wird zurückgesetzt...",
"please-provide-the-required-csv-xlsx-file": "Bitte eine CSV oder XLSX Datei hochladen.", "password-reset-mail-sent": "Passwort-Reset Mail wurde an \"{usersEmail}\" geschickt.",
"please-provide-the-required-information-for-creating-a-new-user-group": "Bitte gebe alle für eine neue Gruppe notwendigen Informationen an.", "password-reset-successful": "Passwort erfolgreich zurückgesetzt!",
"please-provide-the-required-information-to-add-a-new-contact": "Bitte gebe alle nötigen Informationen an, im den neuen Kontakt zu erstellen.", "passwords-dont-match": "Die Passwörter stimmen nicht überein!",
"please-provide-the-required-information-to-add-a-new-organization": "Bitte gebe alle nötigen Informationen an, im die neue Organisation zu erstellen.", "payment-amount-must-be-greater-than-0-00eur": "Der Zahlungsbetrag muss größer als 0.00€ sein!",
"please-provide-the-required-information-to-add-a-new-runner": "Bitte die benötigten Informationen angeben.", "pdf-generation-failed": "PDF Generierung fehlgeschlagen!",
"please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.", "pdf-successfully-generated": "PDF wurde erfolgreich generiert!",
"please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.", "pdfs-successfully-generated": "Alle PDFs wurden generiert!",
"please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.", "per-kilometer": "pro Kilometer",
"please-provide-the-required-information-to-create-a-new-scanstation": "Bitte gebe alle für eine Scannerstation notwendigen Informationen an", "permissions": "Berechtigungen",
"please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...", "permissions-updated": "Berechtigungen aktualisiert!",
"privacy": "Datenschutz", "phone": "Telefon",
"privacy-loading": "Datenschutzerklärung lädt...", "please-copy-the-token-and-store-it-somewhere-save": "Bitte kopiere dir den Token und bewahre ihn gut auf.",
"profile": "Profil", "please-provide-a-password": "Bitte gebe ein Passwort an...",
"profile-picture": "Profilbild", "please-provide-the-nessecary-information-to-add-a-new-donor": "Bitte mach die Notwendigen Angaben, um eine neue Sponsor:in zu erstellen",
"profile-updated": "Profil wurde aktualisiert!", "please-provide-the-nessecary-information-to-create-a-new-donation": "Bitte gebe alle für das Sponsoring notwendigen Daten an.",
"read-license": "Lizenz-Text lesen", "please-provide-the-nessecary-information-to-create-a-new-scan": "Bitte gebe alle notwendigen Informationen an, um einen neuen Scan zu erstellen.",
"receipt-needed": "Spendenquittung benötigt", "please-provide-the-required-csv-xlsx-file": "Bitte eine CSV oder XLSX Datei hochladen.",
"repo_link": "Link", "please-provide-the-required-information-for-creating-a-new-user-group": "Bitte gebe alle für eine neue Gruppe notwendigen Informationen an.",
"request-a-new-reset-mail": "Neue Reset-Mail anfordern", "please-provide-the-required-information-to-add-a-new-contact": "Bitte gebe alle nötigen Informationen an, im den neuen Kontakt zu erstellen.",
"reset-my-password": "Passwort zurücksetzen", "please-provide-the-required-information-to-add-a-new-organization": "Bitte gebe alle nötigen Informationen an, im die neue Organisation zu erstellen.",
"reset-password": "Passwort zurücksetzen", "please-provide-the-required-information-to-add-a-new-runner": "Bitte die benötigten Informationen angeben.",
"runner": "Läufer:in", "please-provide-the-required-information-to-add-a-new-team": "Bitte gebe alle nötigen Informationen an, im das neue Team zu erstellen.",
"runner-added": "Läufer:in hinzugefügt", "please-provide-the-required-information-to-add-a-new-track": "Bitte die benötigten Informationen angeben.",
"runner-import": "Läufer:innen Import", "please-provide-the-required-information-to-add-a-new-user": "Bitte gebe alle nötigen Informationen an, im die neue Benutzer:in zu erstellen.",
"runner-is-being-added": "Läufer:in wird hinzugefügt...", "please-provide-the-required-information-to-create-a-new-scanstation": "Bitte gebe alle für eine Scannerstation notwendigen Informationen an",
"runner-updated": "Läufer:in aktualisiert!", "please-request-a-new-reset-mail": "Bitte eine neue Passwortreset-Mail anfordern...",
"runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen", "please-wait-a-moment-your-login-is-still-being-processed": "Bitte warte einen Moment, deine Anmeldung wird verarbeitet",
"runners": "Läufer", "privacy": "Datenschutz",
"runners-are-being-imported": "Läufer:innen werden importiert ...", "privacy-loading": "Datenschutzerklärung lädt...",
"runners-are-being-loaded": "Läufer:innen werden geladen ...", "profile": "Profil",
"save": "Speichern", "profile-deleted": "Profil gelöscht!",
"save-changes": "Änderungen speichern", "profile-picture": "Profilbild",
"scan-added": "Scan hinzugefügt", "profile-updated": "Profil wurde aktualisiert!",
"scan-is-being-updated": "Scan wird aktualisiert", "read-license": "Lizenz-Text lesen",
"scan-with-fixed-distance": "Scan mit Festdistanz", "receipt-needed": "Spendenquittung benötigt",
"scans": "Scans", "repo_link": "Link",
"scans-are-being-loaded": "Scans werden geladen", "request-a-new-reset-mail": "Neue Reset-Mail anfordern",
"scanstation": "Scanner Station", "reset-my-password": "Passwort zurücksetzen",
"scanstation-added": "Station wurde erstellt", "reset-password": "Passwort zurücksetzen",
"scanstation-is-being-added": "Scannerstation wird angelegt...", "runner": "Läufer:in",
"scanstations": "Scanner Stationen", "runner-added": "Läufer:in hinzugefügt",
"scanstations-are-being-loaded": "Scannerstationen werden geladen...", "runner-import": "Läufer:innen Import",
"search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)", "runner-is-being-added": "Läufer:in wird hinzugefügt...",
"search-for-an-organization-or-team-by-name-or-id": "Suche eine Organisation oder ein Team (via Name oder Id)", "runner-updated": "Läufer:in aktualisiert!",
"search-for-donor-name-or-id": "Suche eine Spender:in (via Name oder Id)", "runnercards": "Laeuferkarten",
"search-for-permission": "Berechtigungen durchsuchen", "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen",
"search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)", "runners": "Läufer",
"select-all": "Alle auswählen", "runners-are-being-imported": "Läufer:innen werden importiert ...",
"select-language": "Sprache auswählen", "runners-are-being-loaded": "Läufer:innen werden geladen ...",
"selfservice-registration": "Selfservice Registrierung", "save": "Speichern",
"send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services", "save-changes": "Änderungen speichern",
"set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen", "scan-added": "Scan hinzugefügt",
"settings": "Einstellungen", "scan-is-being-updated": "Scan wird aktualisiert",
"settings-for-your-profile": "Die Einstellungen deines Accounts", "scan-with-fixed-distance": "Scan mit Festdistanz",
"something-about-the-group": "Infos zur Gruppe", "scans": "Scans",
"stats-are-being-loaded": "Die Statistiken werden geladen...", "scans-are-being-loaded": "Scans werden geladen",
"status": "Status", "scanstation": "Scanner Station",
"stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können", "scanstation-added": "Station wurde erstellt",
"successful-password-reset": "Passwort erfolgreich zurückgesetzt!", "scanstation-is-being-added": "Scannerstation wird angelegt...",
"team": "Team", "scanstations": "Scanner Stationen",
"team-detail-is-being-loaded": "Team wird geladen...", "scanstations-are-being-loaded": "Scannerstationen werden geladen...",
"team-name": "Teamname", "search-for-an-organization-by-name-or-id": "Suche eine Organisation (via Name oder Id)",
"team-name-is-required": "Teamname ist erforderlich", "search-for-an-organization-or-team-by-name-or-id": "Suche eine Organisation oder ein Team (via Name oder Id)",
"teams": "Teams", "search-for-donor-name-or-id": "Suche eine Spender:in (via Name oder Id)",
"teams-are-being-loaded": "Teams werden geladen ...", "search-for-permission": "Berechtigungen durchsuchen",
"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...", "search-for-runner-by-name-or-id": "Suche eine Läufer:in (via Name oder Id)",
"the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.", "select-all": "Alle auswählen",
"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!", "select-language": "Sprache auswählen",
"there-are-no-cards-yet": "Es gibt noch keine Läuferkarten.", "selfservice-registration": "Selfservice Registrierung",
"there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.", "send-a-mail-to-lfk-odit-services": "Sende eine Mail an lfk@odit.services",
"there-are-no-donations-yet": "Es gibt noch keine Sponsorings", "set-the-user-active-inactive": "Den Benutzer auf (in)aktiv setzen",
"there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen", "settings": "Einstellungen",
"there-are-no-groups-yet": "Es gibt noch keine Gruppen", "settings-for-your-profile": "Die Einstellungen deines Accounts",
"there-are-no-organizations-added-yet": "Es wurden noch keine Organisationen hinzugefügt.", "something-about-the-group": "Infos zur Gruppe",
"there-are-no-runners-added-yet": "Es wurden noch keine Läufer:innen hinzugefügt.", "sponsoring-quittungs-liste_herunterladen": "Sponsoring-Quittungs-Liste herunterladen",
"there-are-no-scans-yet": "Es gibt noch keine Scans", "sponsorings": "Sponsoringerklaerungen",
"there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.", "stats-are-being-loaded": "Die Statistiken werden geladen...",
"there-are-no-users-added-yet": "Es wurden noch keine Benutzer hinzugefügt.", "status": "Status",
"this-card-is": "Diese Karte ist", "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können",
"this-might-take-a-moment": "Das könnte einen kleinen Moment dauern", "successful-password-reset": "Passwort erfolgreich zurückgesetzt!",
"this-scanstation-is": "Diese Station ist", "team": "Team",
"token": "Token", "team-added": "Team wurde hinzugefügt",
"total-distance": "gelaufene Strecke", "team-deleted": "Team gelöscht",
"total-donation-amount": "Gesamtbetrag", "team-detail-is-being-loaded": "Team wird geladen...",
"total-donations": "Spendensumme", "team-is-being-added": "Team wird erstellt...",
"total-scans": "gesamte Scans", "team-name": "Teamname",
"track": "Track", "team-name-is-required": "Teamname ist erforderlich",
"track-added": "Track hinzugefügt", "teams": "Teams",
"track-data-is-being-loaded": "Trackdaten werden geladen", "teams-are-being-loaded": "Teams werden geladen ...",
"track-is-being-added": "Track wird hinzugefügt...", "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...",
"track-length-in-m": "Tracklänge (in Metern)", "the-scans-distance-must-be-greater-than-0m": "Die Distanz muss größer als 0m sein.",
"track-length-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein", "the-scanstations-api-token-will-only-get-displayed-once-you-wont-be-able-to-change-or-view-it-again": "Der Scannerstation Token wird nur einmal angezeigt - du kannst ihn nicht ändern oder ihn dir nochmal anzeigen lassen!",
"track-name": "Trackname", "there-are-no-cards-yet": "Es gibt noch keine Läuferkarten.",
"track-name-must-not-be-empty": "Der Name muss angegeben werden", "there-are-no-contacts-added-yet": "Es wurden noch keine Kontakte hinzugefügt.",
"tracks": "Tracks", "there-are-no-donations-yet": "Es gibt noch keine Sponsorings",
"update-password": "Passwort ändern", "there-are-no-donors-yet": "Es gibt noch keine Sponsor:innen",
"updated-contact": "Kontakt aktualisiert!", "there-are-no-groups-yet": "Es gibt noch keine Gruppen",
"updated-donor": "Sponsor:in wurde aktualisiert", "there-are-no-organizations-added-yet": "Es wurden noch keine Organisationen hinzugefügt.",
"updated-organization": "Organisation wurde aktualisiert", "there-are-no-runners-added-yet": "Es wurden noch keine Läufer:innen hinzugefügt.",
"updated-scan": "Scan wurde aktualisiert", "there-are-no-scans-yet": "Es gibt noch keine Scans",
"updateing-group": "Gruppe wird aktualisiert...", "there-are-no-teams-added-yet": "Es wurden noch keine Teams hinzugefügt.",
"updating-card": "Karte wird aktualisiert", "there-are-no-users-added-yet": "Es wurden noch keine Benutzer hinzugefügt.",
"updating-organization": "Organisation wird aktualisiert", "this-card-is": "Diese Karte ist",
"updating-permissions": "Berechtigungen werden aktualisiert...", "this-might-take-a-moment": "Das könnte einen kleinen Moment dauern",
"updating-runner": "Läufer:in wird aktualisiert.", "this-scanstation-is": "Diese Station ist",
"updating-user": "Benutzer:in wird aktualisiert...", "token": "Token",
"updating-your-profile": "Profil wird aktualisiert...", "total-distance": "gelaufene Strecke",
"user-added": "Benutzer hinzugefügt", "total-donation-amount": "Gesamtbetrag",
"user-groups": "Benutzergruppen", "total-donations": "Spendensumme",
"user-is-being-added": "Benutzer wird hinzugefügt ...", "total-paid-amount": "Gezahlter Gesamtbetrag",
"user-updated": "Benutzer:in wurde aktualisiert", "total-scans": "gesamte Scans",
"username": "Benutzername", "total_donation_amount_in_eur": "Gesamtbetrag in €",
"users": "Benutzer", "track": "Track",
"valid": "Gültig", "track-added": "Track hinzugefügt",
"valid-city-is-required": "Du musst eine Stadt angeben", "track-data-is-being-loaded": "Trackdaten werden geladen",
"valid-email-is-required": "Es wird eine valide E-Mail Adresse benötigt", "track-is-being-added": "Track wird hinzugefügt...",
"valid-international-phone-number-is-required": "Du musst eine Telefonnummer im internationalen Format angeben...", "track-is-being-updated": "Track wird aktualisiert...",
"valid-zipcode-postal-code-is-required": "Du musst eine valide Postleitzahl angeben", "track-length-in-m": "Tracklänge (in Metern)",
"verfuegbare": "Verfügbar", "track-length-must-be-greater-than-0": "Die Länge muss größer als 0 (Meter) sein",
"welcome_wavinghand": "Willkommen 👋", "track-name": "Trackname",
"yes-i-copied-the-token": "Ja, ich habe den Token kopiert", "track-name-must-not-be-empty": "Der Name muss angegeben werden",
"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!", "track-was-updated": "Track wurde aktualisiert",
"you-can-now-use-your-new-password-to-log-in-to-your-account": "Du kannst dich jetzt mit deinem neuen Passwort anmelden! 🎉", "tracks": "Tracks",
"you-can-provide-a-runner-but-you-dont-have-to": "Du kannst eine Läufer:in angeben, musst aber nicht.", "unpaid": "Offen",
"you-dont-have-any-scanstations-yet": "Es gibt noch keine Scannerstationen", "update-card": "Karte aktualisieren",
"you-have-to-provide-an-organization": "Du musst eine Organisation angeben", "update-password": "Passwort ändern",
"you-have-to-save-your-changes-to-generate-a-link": "Du musst deine Änderungen speichern, um einen Link zu generieren.", "updated-contact": "Kontakt aktualisiert!",
"you-must-create-at-least-one-card-or-cancel": "Du musst mindestens eine Blankokarte erstellen (oder abbrechen).", "updated-donor": "Sponsor:in wurde aktualisiert",
"zip-postal-code": "Postleitzahl", "updated-organization": "Organisation wurde aktualisiert",
"generate-runner-certificates": "Urkunden generieren" "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-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"
}

View File

@@ -1,429 +1,462 @@
{ {
"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-user-group": "Add User Group",
"add-your-first-contact": "Add your first contact", "add-your-first-card": "Add your first card",
"add-your-first-donor": "add your first donor", "add-your-first-contact": "Add your first contact",
"add-your-first-group": "Add your first group", "add-your-first-donor": "add your first donor",
"add-your-first-organization": "Add your first organization", "add-your-first-group": "Add your first group",
"add-your-first-runner": "Add your first runner", "add-your-first-organization": "Add your first organization",
"add-your-first-team": "Add your first team", "add-your-first-runner": "Add your first runner",
"add-your-first-track": "Add your first track.", "add-your-first-team": "Add your first team",
"add-your-first-user": "Add your first user", "add-your-first-track": "Add your first track.",
"add-your-fist-donation": "Add your fist donation", "add-your-first-user": "Add your first user",
"add-your-fist-scan": "Add your fist scan", "add-your-fist-donation": "Add your fist donation",
"adding-card": "Adding Card", "add-your-fist-scan": "Add your fist scan",
"adding-scan": "Adding Scan", "adding-card": "Adding Card",
"address": "Address", "adding-donation": "Adding donation...",
"address-is-required": "Address is required", "adding-scan": "Adding Scan",
"after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!", "address": "Address",
"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-is-required": "Address is required",
"all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well", "after-deletion-we-cant-restore-your-old-profile": "After deletion we can't restore your old profile!",
"all-associated-runners-will-be-deleted-too": "All associated 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.",
"all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!", "all-associated-donations-will-get-deleted-as-well": "All associated donations will get deleted as well",
"amount": "Amount", "all-associated-runners-will-be-deleted-too": "All associated runners will be deleted too!",
"amount-per-kilometer": "Amount per kilometer", "all-associated-teams-and-runners-will-be-deleted-too": "All associated teams and runners will be deleted too!",
"apartment-suite-etc": "Apartment, suite, etc.", "already-paid": "Already paid",
"application_name": "Lauf für Kaya! - Admin", "amount": "Amount",
"applying-changes": "Applying Changes", "amount-per-kilometer": "Amount per kilometer",
"attention": "Attention!", "apartment-suite-etc": "Apartment, suite, etc.",
"author": "Author", "application_name": "Lauf für Kaya! - Admin",
"bitte-bestaetige-diese-laeufer-fuer-den-import": "Please confirm these runners for import.", "applying-changes": "Applying Changes",
"by": "by", "attention": "Attention!",
"cancel": "Cancel", "author": "Author",
"cancel-delete": "Cancel Delete", "bitte-bestaetige-diese-laeufer-fuer-den-import": "Please confirm these runners for import.",
"cancel-keep-donor": "Cancel, keep donor", "by": "by",
"cancel-keep-my-profile": "Cancel, keep my profile", "cancel": "Cancel",
"cancel-keep-organization": "Cancel, keep organization", "cancel-delete": "Cancel Delete",
"cancel-keep-team": "Cancel, keep team", "cancel-keep-donor": "Cancel, keep donor",
"cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity", "cancel-keep-my-profile": "Cancel, keep my profile",
"card-added": "Card added", "cancel-keep-organization": "Cancel, keep organization",
"card-deleted": "Card deleted", "cancel-keep-team": "Cancel, keep team",
"card-updated": "Card updated", "cannot-reset-your-password-directly": "Bummer. We unfortunately cannot reset your password directly. Please send us a mail and confirm your identity",
"cards": "Cards", "card-added": "Card added",
"change-your-password-here": "Change your password here", "card-deleted": "Card deleted",
"changing-your-password": "Changing your password", "card-updated": "Card updated",
"city": "City", "cards": "Cards",
"click-to-copy-the-link-into-your-clipboard": "Click to copy the link into your clipboard", "certificates": "Certificates",
"click-to-copy-token-to-clipboard": "Click to copy the token to your clipboard", "change-your-password-here": "Change your password here",
"close": "Close", "changing-your-password": "Changing your password",
"code": "Code", "city": "City",
"configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times", "click-to-copy-the-link-into-your-clipboard": "Click to copy the link into your clipboard",
"confirm": "Confirm", "click-to-copy-token-to-clipboard": "Click to copy the token to your clipboard",
"confirm-delete": "Confirm Delete", "close": "Close",
"confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations", "code": "Code",
"confirm-delete-my-user-profile": "Confirm, delete my user profile", "configure-the-tracks-and-minimum-lap-times": "configure the tracks & minimum lap times",
"confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.", "confirm": "Confirm",
"confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.", "confirm-delete": "Confirm Delete",
"confirm-deletion": "Confirm Deletion", "confirm-delete-donor-with-all-donations": "Confirm, delete donor with all donations",
"confirm-the-new-password": "Confirm the new password", "confirm-delete-my-user-profile": "Confirm, delete my user profile",
"contact": "Contact", "confirm-delete-organization-and-associated-teams-runners": "Confirm, delete organization and associated teams+runners.",
"contact-deleted": "Contact deleted", "confirm-delete-team-and-associated-runners": "Confirm, delete team and associated runners.",
"contact-information": "Contact Information", "confirm-deletion": "Confirm Deletion",
"contact-is-being-updated": "Contact is being updated...", "confirm-the-new-password": "Confirm the new password",
"contact-is-not-a-member-in-any-group": "Contact is not a member in any group", "contact": "Contact",
"contacts": "Contacts", "contact-added": "Contact added",
"contacts-are-being-loaded": "contacts are being loaded...", "contact-deleted": "Contact deleted",
"copied-link-to-clipboard": "Copied link to clipboard", "contact-information": "Contact Information",
"copied-token-to-clipboard": "Copied token to clipboard", "contact-is-being-added": "Contact is being added...",
"count_organizations": "# Organizations", "contact-is-being-updated": "Contact is being updated...",
"count_teams": "# Teams", "contact-is-not-a-member-in-any-group": "Contact is not a member in any group",
"create": "Create", "contacts": "Contacts",
"create-a-new": "Create a new", "contacts-are-being-loaded": "contacts are being loaded...",
"create-a-new-card": "Create a new card", "copied-link-to-clipboard": "Copied link to clipboard",
"create-a-new-contact": "Create a new contact", "copied-token-to-clipboard": "Copied token to clipboard",
"create-a-new-distance-donation": "Create a new distance donation", "count_organizations": "# Organizations",
"create-a-new-donor": "Create a new donor", "count_teams": "# Teams",
"create-a-new-fixed-donation": "Create a new fixed donation", "create": "Create",
"create-a-new-organization": "Create a new Organization", "create-a-new": "Create a new",
"create-a-new-runner": "Create a new Runner", "create-a-new-card": "Create a new card",
"create-a-new-scan-fixed-only": "Create a new scan (fixed only)", "create-a-new-contact": "Create a new contact",
"create-a-new-scanstation": "Create a new station", "create-a-new-distance-donation": "Create a new distance donation",
"create-a-new-team": "Create a new team", "create-a-new-donor": "Create a new donor",
"create-a-new-track": "Create a new Track", "create-a-new-fixed-donation": "Create a new fixed donation",
"create-a-new-user": "Create a new User", "create-a-new-organization": "Create a new Organization",
"create-a-new-user-group": "Create a new user group", "create-a-new-runner": "Create a new Runner",
"create-and-generate-pdf": "Create and generate PDF", "create-a-new-scan-fixed-only": "Create a new scan (fixed only)",
"create-bulk-blanco-cards": "Create bulk blanco cards", "create-a-new-scanstation": "Create a new station",
"create-bulk-cards": "Add blanco cards", "create-a-new-team": "Create a new team",
"create-organization": "Create Organization", "create-a-new-track": "Create a new Track",
"create-team": "Create Team", "create-a-new-user": "Create a new User",
"create-track": "Create Track", "create-a-new-user-group": "Create a new user group",
"create-user": "Create User", "create-and-generate-pdf": "Create and generate PDF",
"create-without-pdf": "Create without PDF", "create-bulk-blanco-cards": "Create bulk blanco cards",
"created-blanco-cards": "Created blanco cards", "create-bulk-cards": "Add blanco cards",
"creating-blanco-cards": "Creating blanco cards", "create-organization": "Create Organization",
"credits": "Credits", "create-team": "Create Team",
"csv_import__class": "Class", "create-track": "Create Track",
"csv_import__firstname": "Firstname", "create-user": "Create User",
"csv_import__lastname": "Lastname", "create-without-pdf": "Create without PDF",
"csv_import__middlename": "Middlename", "created-blanco-cards": "Created blanco cards",
"csv_import__team": "Team", "creating-blanco-cards": "Creating blanco cards",
"danger-zone": "Danger zone", "credits": "Credits",
"dashboard-greeting": "Hello", "csv_import__class": "Class",
"dashboard-title": "Dashboard", "csv_import__firstname": "Firstname",
"datatable": { "csv_import__lastname": "Lastname",
"search": "🔍 Search...", "csv_import__middlename": "Middlename",
"sort_column_ascending": "Sort column ascending", "csv_import__team": "Team",
"sort_column_descending": "Sort column descending", "danger-zone": "Danger zone",
"previous": "Previous", "dashboard-greeting": "Hello",
"next": "Next", "dashboard-title": "Dashboard",
"page": "Page", "datatable": {
"showing": "Showing", "search": "🔍 Search...",
"records": "Records", "sort_column_ascending": "Sort column ascending",
"of": "of", "sort_column_descending": "Sort column descending",
"to": "to", "previous": "Previous",
"loading": "Loading...", "next": "Next",
"no_matching_records_found": "No matching records found", "page": "Page",
"an_error_happened_while_fetching_the_data": "An error happened while fetching the data" "showing": "Showing",
}, "records": "Records",
"delete": "Delete", "of": "of",
"delete-contact": "Delete Contact", "to": "to",
"delete-donation": "Delete Donation", "loading": "Loading...",
"delete-donor": "Delete donor", "no_matching_records_found": "No matching records found",
"delete-group": "Delete Group", "an_error_happened_while_fetching_the_data": "An error happened while fetching the data"
"delete-organization": "Delete Organization", },
"delete-profile": "Delete Profile", "delete": "Delete",
"delete-runner": "Delete Runner", "delete-contact": "Delete Contact",
"delete-scan": "Delete scan", "delete-donation": "Delete Donation",
"delete-station": "Delete station", "delete-donor": "Delete donor",
"delete-team": "Delete Team", "delete-group": "Delete Group",
"delete-user": "Delete User", "delete-organization": "Delete Organization",
"deleted-scan": "Deleted scan", "delete-profile": "Delete Profile",
"dependency_name": "Name", "delete-runner": "Delete Runner",
"description": "description", "delete-scan": "Delete scan",
"description-optional": "Description (optional)", "delete-station": "Delete station",
"deselect-all": "deselect all", "delete-team": "Delete Team",
"details": "Details", "delete-user": "Delete User",
"disabled": "disabled", "deleted-scan": "Deleted scan",
"distance": "Distance", "dependency_name": "Name",
"distance-donation": "distance donation", "description": "description",
"distance-in-km": "Distance in km", "description-optional": "Description (optional)",
"distance-track": "Distance (+Track)", "deselect-all": "deselect all",
"do-you-really-want-to-delete-your-profile": "Do you really want to delete your profile?", "details": "Details",
"do-you-want-to-delete-the-organization-delete_org-name": "Do you want to delete the organization {orgname}?", "disabled": "disabled",
"do-you-want-to-delete-the-team-delete_team-name": "Do you want to delete the team {teamname}?", "distance": "Distance",
"do-you-want-to-delete-this-donor-with-all-related-donations": "Do you want to delete this donor with all related donations", "distance-donation": "distance donation",
"documentation": "Documentation", "distance-in-km": "Distance in km",
"donation-amount": "Donation amount", "distance-track": "Distance (+Track)",
"donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€", "do-you-really-want-to-delete-your-profile": "Do you really want to delete your profile?",
"donations": "Donations", "do-you-want-to-delete-the-organization-delete_org-name": "Do you want to delete the organization {orgname}?",
"donor": "Donor", "do-you-want-to-delete-the-team-delete_team-name": "Do you want to delete the team {teamname}?",
"donor-added": "Donor added", "do-you-want-to-delete-this-donor-with-all-related-donations": "Do you want to delete this donor with all related donations",
"donor-deleted": "donor deleted", "documentation": "Documentation",
"donor-has-no-associated-donations": "Donor has no associated donations.", "donation-amount": "Donation amount",
"donor-is-being-added": "Donor is being added...", "donation-amount-must-be-greater-that-0-00eur": "Donation amount must be greater that 0.00€",
"donor-is-being-updated": "Donor is being updated", "donation-deleted": "Donation deleted",
"donors": "Donors", "donation-updated": "Donation updated",
"donors-are-being-loaded": "donors are being loaded", "donation_added": "Donation_added",
"dont-have-your-email-connected": "Don't have your email connected?", "donations": "Donations",
"dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌", "donor": "Donor",
"e-mail-adress": "E-Mail Adress", "donor-added": "Donor added",
"edit": "Edit", "donor-deleted": "donor deleted",
"edit-a-card": "Edit a card", "donor-has-no-associated-donations": "Donor has no associated donations.",
"edit-permissions": "edit permissions", "donor-is-being-added": "Donor is being added...",
"email_address_or_username": "Email / username", "donor-is-being-updated": "Donor is being updated",
"enabled": "enabled", "donors": "Donors",
"enabled_large": "Enabled", "donors-are-being-loaded": "donors are being loaded",
"english": "English", "dont-have-your-email-connected": "Don't have your email connected?",
"error-during-import": "Error during import", "dont-panic-were-resetting-it": "Don't panic, we're resetting it ✌",
"error-whyile-copying-to-clipboard": "Error while copying to clipboard", "e-mail-adress": "E-Mail Adress",
"error_on_login": "Error on login", "edit": "Edit",
"erteilte": "Directly granted", "edit-a-card": "Edit a card",
"everything-concerning-your-profile": "Everything concerning your profile", "edit-permissions": "edit permissions",
"everything-is-more-fun-together": "everything is more fun together 🏃‍♂️🏃‍♀️🏃‍♂️", "email_address_or_username": "Email / username",
"faq": "FAQ", "enabled": "enabled",
"filter-by-organization-team": "Filter by Organization/ Team", "enabled_large": "Enabled",
"first-name": "First name", "english": "English",
"first-name-is-required": "First Name is required", "enter-payment": "Enter payment",
"first-scan-of-the-day": "First scan of the day.", "error-during-import": "Error during import",
"fixed-donation": "fixed donation", "error-whyile-copying-to-clipboard": "Error while copying to clipboard",
"forgot_password": "Forgot your password?", "error_on_login": "Error on login",
"geerbte": "inherited", "erteilte": "Directly granted",
"general-stats": "General Stats", "everything-concerning-your-profile": "Everything concerning your profile",
"general_promise_error": "😢 Error", "everything-is-more-fun-together": "everything is more fun together 🏃‍♂️🏃‍♀️🏃‍♂️",
"generate-runnercards": "Generate Runnercards", "faq": "FAQ",
"generate-sponsoring-contract": "generate sponsoring contract", "filename_sponsoringquittungsliste": "DonorReceiptList",
"generate-sponsoring-contracts": "generate sponsoring contracts", "filter-by-organization-team": "Filter by Organization/ Team",
"generating-pdf": "generating PDF...", "first-name": "First name",
"generating-pdfs": "generating PDFs...", "first-name-is-required": "First Name is required",
"generic-ui-logic-error": "Something went wrong in the UI logic", "first-scan-of-the-day": "First scan of the day.",
"german": "German", "fixed-donation": "fixed donation",
"go-to-login": "Go To Login", "forgot_password": "Forgot your password?",
"goback": "Go Home", "geerbte": "inherited",
"granted": "granted", "general-stats": "General Stats",
"group": "Group", "general_promise_error": "😢 Error",
"group-added": "Group added", "generate-runner-certificate": "Generate runner certificate",
"group-is-being-added": "Group is being added...", "generate-runner-certificates": "Generate runner certificates",
"group-name-is-required": "Group name is required", "generate-runnercards": "Generate Runnercards",
"group-updated": "group updated", "generate-sponsoring-contract": "generate sponsoring contract",
"groups": "Groups", "generate-sponsoring-contracts": "generate sponsoring contracts",
"groups-are-being-loaded": "Groups are being loaded", "generating-pdf": "generating PDF...",
"home": "Home", "generating-pdfs": "generating PDFs...",
"icon-image-credits": "We also want to thank these projects for illustrations and icons:", "generic-ui-logic-error": "Something went wrong in the UI logic",
"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.", "german": "German",
"import-finished": "Import finished", "go-to-login": "Go To Login",
"import-runners": "Import runners", "goback": "Go Home",
"import__target-organization": "Target Organization", "granted": "granted",
"imprint": "Imprint", "group": "Group",
"imprint-loading": "Imprint loading...", "group-added": "Group added",
"inactive": "Inactive", "group-is-being-added": "Group is being added...",
"installed-version": "Installed version", "group-name-is-required": "Group name is required",
"internal-error": "Internal Error", "group-updated": "group updated",
"invalid": "Invalid", "groups": "Groups",
"invalid-mail-reset": "the provided email is invalid", "groups-are-being-loaded": "Groups are being loaded",
"just-enter-how-many-you-want-and-the-system-will-create-them": "Just enter how many you want and the system will create them", "home": "Home",
"laeufer-hinzufuegen": "Add runner", "icon-image-credits": "We also want to thank these projects for illustrations and icons:",
"laeufer-importieren": "Läufer importieren", "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.",
"laptime": "Laptime", "import-finished": "Import finished",
"last-name": "Last name", "import-runners": "Import runners",
"last-name-is-required": "Last Name is required", "import__target-organization": "Target Organization",
"lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.", "imprint": "Imprint",
"license": "License", "imprint-loading": "Imprint loading...",
"licenses-are-being-loaded": "Licenses are being loaded...", "inactive": "Inactive",
"loading-cards": "Loading cards", "installed-version": "Installed version",
"loading-contact-details": "Loading contact details...", "internal-error": "Internal Error",
"loading-donation-details": "Loading donation details", "invalid": "Invalid",
"loading-donor-details": "Loading donor details", "invalid-mail-reset": "the provided email is invalid",
"loading-group-detail": "Loading group detail...", "just-enter-how-many-you-want-and-the-system-will-create-them": "Just enter how many you want and the system will create them",
"loading-profile-data": "Loading profile data", "laeufer-hinzufuegen": "Add runner",
"loading-runners": "loading runners...", "laeufer-importieren": "Läufer importieren",
"loading-station-details": "Loading station details", "laptime": "Laptime",
"log_in": "Log in", "last-name": "Last name",
"log_in_to_your_account": "Log in to your account", "last-name-is-required": "Last Name is required",
"login_is_checked": "Login is being checked...", "lfk-is-os": "The \"Lauf für Kaya!\" Frontend is (like all other projects for the \"LfK!\" Also) an open source project.",
"logout": "Logout", "license": "License",
"mail-validation-in-progress": "mail validation in progress...", "licenses-are-being-loaded": "Licenses are being loaded...",
"manage-admin-users": "manage admin users", "loading-cards": "Loading cards",
"middle-name": "Middle name", "loading-contact-details": "Loading contact details...",
"minimum-lap-time-in-s": "minimum lap time in s", "loading-donation-details": "Loading donation details",
"minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0", "loading-donor-details": "Loading donor details",
"must-be-at-least-10-characters-long": "Must be at least 10 characters long!", "loading-group-detail": "Loading group detail...",
"must-contain-a-lowercase-letter": "Must contain a lowercase letter!", "loading-profile-data": "Loading profile data",
"must-contain-a-number": "Must contain a number!", "loading-runners": "loading runners...",
"must-contain-a-uppercase-letter": "Must contain a uppercase letter!", "loading-station-details": "Loading station details",
"name": "Name", "log_in": "Log in",
"name-is-required": "Name is required", "log_in_to_your_account": "Log in to your account",
"new-password": "New password", "login_is_checked": "Login is being checked...",
"no-contact-found": "No contacts found", "logout": "Logout",
"no-contact-selected": "No contact selected", "mail-validation-in-progress": "mail validation in progress...",
"no-contact-specified": "no contact specified", "manage-admin-users": "manage admin users",
"no-donors-found": "No donors found", "middle-name": "Middle name",
"no-license-text-could-be-found": "No license text could be found 😢", "minimum-lap-time-in-s": "minimum lap time in s",
"no-organization-or-team-found": "No organization or team found", "minimum-lap-time-must-be-a-positive-number-or-0": "minimum lap time must be a positive number or 0",
"no-organization-specified": "no organization specified", "must-be-at-least-10-characters-long": "Must be at least 10 characters long!",
"no-organizations-found": "No organizations found", "must-contain-a-lowercase-letter": "Must contain a lowercase letter!",
"no-runners-found": "No runners found", "must-contain-a-number": "Must contain a number!",
"no-tracks-added-yet": "there are no tracks added yet.", "must-contain-a-uppercase-letter": "Must contain a uppercase letter!",
"non-blanko": "Non/Blanko", "name": "Name",
"organization": "Organization", "name-is-required": "Name is required",
"organization-added": "Organization added", "new-password": "New password",
"organization-deleted": "Organization deleted", "no-contact-found": "No contacts found",
"organization-detail-is-being-loaded": "organization detail is being loaded...", "no-contact-selected": "No contact selected",
"organization-is-being-added": "Organization is being added...", "no-contact-specified": "no contact specified",
"organization-name-is-required": "Organization name is required", "no-donors-found": "No donors found",
"organizations": "Organizations", "no-license-text-could-be-found": "No license text could be found 😢",
"organizations-are-being-loaded": "organizations are being loaded...", "no-organization-or-team-found": "No organization or team found",
"orgs": "Organizations", "no-organization-specified": "no organization specified",
"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!", "no-organizations-found": "No organizations found",
"password": "Password", "no-runners-found": "No runners found",
"password-changed": "Password changed!", "no-tracks-added-yet": "there are no tracks added yet.",
"password-is-required": "Password is required", "non-blanko": "Non/Blanko",
"password-reset-failed": "Password reset failed!", "open": "OPEN",
"password-reset-in-progress": "Password Reset in Progress...", "organization": "Organization",
"password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".", "organization-added": "Organization added",
"password-reset-successful": "Password Reset successful!", "organization-deleted": "Organization deleted",
"passwords-dont-match": "Passwords don't match!", "organization-detail-is-being-loaded": "organization detail is being loaded...",
"pdf-generation-failed": "PDF generation failed!", "organization-is-being-added": "Organization is being added...",
"pdf-successfully-generated": "PDF successfully generated!", "organization-name-is-required": "Organization name is required",
"pdfs-successfully-generated": "PDFs successfully generated!", "organizations": "Organizations",
"per-kilometer": "per Kilometer", "organizations-are-being-loaded": "organizations are being loaded...",
"permissions": "Permissions", "orgs": "Organizations",
"permissions-updated": "Permissions updated!", "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!",
"phone": "Phone", "paid": "PAID",
"please-copy-the-token-and-store-it-somewhere-save": "Please copy the token and store it somewhere safe.", "paid-amount": "Paid amount",
"please-provide-a-password": "Please provide a password...", "password": "Password",
"please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor", "password-changed": "Password changed!",
"please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation", "password-is-required": "Password is required",
"please-provide-the-nessecary-information-to-create-a-new-scan": "Please provide the nessecary information to create a new scan.", "password-reset-failed": "Password reset failed!",
"please-provide-the-required-csv-xlsx-file": "Please provide the required csv/ xlsx file", "password-reset-in-progress": "Password Reset in Progress...",
"please-provide-the-required-information-for-creating-a-new-user-group": "Please provide the required information for creating a new user group.", "password-reset-mail-sent": "Password reset mail was sent to \"{usersEmail}\".",
"please-provide-the-required-information-to-add-a-new-contact": "Please provide the required information to add a new contact.", "password-reset-successful": "Password Reset successful!",
"please-provide-the-required-information-to-add-a-new-organization": "Please provide the required information to add a new organization.", "passwords-dont-match": "Passwords don't match!",
"please-provide-the-required-information-to-add-a-new-runner": "Please provide the required information to add a new runner.", "payment-amount-must-be-greater-than-0-00eur": "Payment amount must be greater than 0.00€!",
"please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.", "pdf-generation-failed": "PDF generation failed!",
"please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.", "pdf-successfully-generated": "PDF successfully generated!",
"please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.", "pdfs-successfully-generated": "PDFs successfully generated!",
"please-provide-the-required-information-to-create-a-new-scanstation": "Please provide the required information to create a new scanstation", "per-kilometer": "per Kilometer",
"please-request-a-new-reset-mail": "Please request a new reset mail...", "permissions": "Permissions",
"privacy": "Privacy", "permissions-updated": "Permissions updated!",
"privacy-loading": "Privacy loading...", "phone": "Phone",
"profile": "Profile", "please-copy-the-token-and-store-it-somewhere-save": "Please copy the token and store it somewhere safe.",
"profile-picture": "Profile Picture", "please-provide-a-password": "Please provide a password...",
"profile-updated": "Profile updated!", "please-provide-the-nessecary-information-to-add-a-new-donor": "Please provide the nessecary information to add a new donor",
"read-license": "Read License", "please-provide-the-nessecary-information-to-create-a-new-donation": "Please provide the nessecary information to create a new donation",
"receipt-needed": "Receipt needed", "please-provide-the-nessecary-information-to-create-a-new-scan": "Please provide the nessecary information to create a new scan.",
"repo_link": "Link", "please-provide-the-required-csv-xlsx-file": "Please provide the required csv/ xlsx file",
"request-a-new-reset-mail": "Request a new reset mail", "please-provide-the-required-information-for-creating-a-new-user-group": "Please provide the required information for creating a new user group.",
"reset-my-password": "Reset my password", "please-provide-the-required-information-to-add-a-new-contact": "Please provide the required information to add a new contact.",
"reset-password": "Reset your password", "please-provide-the-required-information-to-add-a-new-organization": "Please provide the required information to add a new organization.",
"runner": "Runner", "please-provide-the-required-information-to-add-a-new-runner": "Please provide the required information to add a new runner.",
"runner-added": "Runner added", "please-provide-the-required-information-to-add-a-new-team": "Please provide the required information to add a new team.",
"runner-import": "Runner Import", "please-provide-the-required-information-to-add-a-new-track": "Please provide the required information to add a new track.",
"runner-is-being-added": "Runner is being added...", "please-provide-the-required-information-to-add-a-new-user": "Please provide the required information to add a new user.",
"runner-updated": "Runner updated!", "please-provide-the-required-information-to-create-a-new-scanstation": "Please provide the required information to create a new scanstation",
"runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"", "please-request-a-new-reset-mail": "Please request a new reset mail...",
"runners": "Runners", "please-wait-a-moment-your-login-is-still-being-processed": "Please wait a moment, your login is still being processed",
"runners-are-being-imported": "Runners are being imported...", "privacy": "Privacy",
"runners-are-being-loaded": "runners are being loaded...", "privacy-loading": "Privacy loading...",
"save": "Save", "profile": "Profile",
"save-changes": "Save Changes", "profile-deleted": "Profile deleted!",
"scan-added": "Scan added", "profile-picture": "Profile Picture",
"scan-is-being-updated": "Scan is being updated", "profile-updated": "Profile updated!",
"scan-with-fixed-distance": "Scan with fixed distance", "read-license": "Read License",
"scans": "Scans", "receipt-needed": "Receipt needed",
"scans-are-being-loaded": "Scans are being loaded", "repo_link": "Link",
"scanstation": "Scanstation", "request-a-new-reset-mail": "Request a new reset mail",
"scanstation-added": "Scanstation added", "reset-my-password": "Reset my password",
"scanstation-is-being-added": "Adding scanstation...", "reset-password": "Reset your password",
"scanstations": "Scanstations", "runner": "Runner",
"scanstations-are-being-loaded": "Loading scanstations...", "runner-added": "Runner added",
"search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)", "runner-import": "Runner Import",
"search-for-an-organization-or-team-by-name-or-id": "Search for an organization or team (by name or id)", "runner-is-being-added": "Runner is being added...",
"search-for-donor-name-or-id": "Search for donor (by name or id)", "runner-updated": "Runner updated!",
"search-for-permission": "Search for permission", "runnercards": "Runnercards",
"search-for-runner-by-name-or-id": "Search for runner (by name or id)", "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"",
"select-all": "select all", "runners": "Runners",
"select-language": "Select language", "runners-are-being-imported": "Runners are being imported...",
"selfservice-registration": "Selfservice registration", "runners-are-being-loaded": "runners are being loaded...",
"send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services", "save": "Save",
"set-the-user-active-inactive": "set the user active/ inactive", "save-changes": "Save Changes",
"settings": "Settings", "scan-added": "Scan added",
"settings-for-your-profile": "Settings for your profile", "scan-is-being-updated": "Scan is being updated",
"something-about-the-group": "Something about the group...", "scan-with-fixed-distance": "Scan with fixed distance",
"stats-are-being-loaded": "stats are being loaded...", "scans": "Scans",
"status": "Status", "scans-are-being-loaded": "Scans are being loaded",
"stuff-that-could-harm-your-profile": "Stuff that could harm your profile", "scanstation": "Scanstation",
"successful-password-reset": "Successful password reset!", "scanstation-added": "Scanstation added",
"team": "Team", "scanstation-is-being-added": "Adding scanstation...",
"team-detail-is-being-loaded": "team detail is being loaded...", "scanstations": "Scanstations",
"team-name": "Team name", "scanstations-are-being-loaded": "Loading scanstations...",
"team-name-is-required": "team name is required", "search-for-an-organization-by-name-or-id": "Search for an organization (by name or id)",
"teams": "Teams", "search-for-an-organization-or-team-by-name-or-id": "Search for an organization or team (by name or id)",
"teams-are-being-loaded": "teams are being loaded...", "search-for-donor-name-or-id": "Search for donor (by name or id)",
"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...", "search-for-permission": "Search for permission",
"the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m", "search-for-runner-by-name-or-id": "Search for runner (by name or id)",
"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!", "select-all": "select all",
"there-are-no-cards-yet": "There are no cards yet.", "select-language": "Select language",
"there-are-no-contacts-added-yet": "There are no contacts added yet.", "selfservice-registration": "Selfservice registration",
"there-are-no-donations-yet": "There are no donations yet", "send-a-mail-to-lfk-odit-services": "send a mail to lfk@odit.services",
"there-are-no-donors-yet": "There are no donors yet", "set-the-user-active-inactive": "set the user active/ inactive",
"there-are-no-groups-yet": "There are no groups yet", "settings": "Settings",
"there-are-no-organizations-added-yet": "There are no organizations added yet.", "settings-for-your-profile": "Settings for your profile",
"there-are-no-runners-added-yet": "There are no runners added yet.", "something-about-the-group": "Something about the group...",
"there-are-no-scans-yet": "There are no scans yet", "sponsoring-quittungs-liste_herunterladen": "Download donor receipt list",
"there-are-no-teams-added-yet": "There are no teams added yet.", "sponsorings": "Sponsorings",
"there-are-no-users-added-yet": "There are no users added yet.", "stats-are-being-loaded": "stats are being loaded...",
"this-card-is": "This card is", "status": "Status",
"this-might-take-a-moment": "This might take a moment 👀", "stuff-that-could-harm-your-profile": "Stuff that could harm your profile",
"this-scanstation-is": "This scanstation is", "successful-password-reset": "Successful password reset!",
"token": "Token", "team": "Team",
"total-distance": "total distance", "team-added": "Team added",
"total-donation-amount": "total donation amount", "team-deleted": "Team deleted",
"total-donations": "total donations", "team-detail-is-being-loaded": "team detail is being loaded...",
"total-scans": "total scans", "team-is-being-added": "Team is being added...",
"track": "Track", "team-name": "Team name",
"track-added": "Track added", "team-name-is-required": "team name is required",
"track-data-is-being-loaded": "Track data is being loaded", "teams": "Teams",
"track-is-being-added": "Track is being added...", "teams-are-being-loaded": "teams are being loaded...",
"track-length-in-m": "Track Length in m", "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...",
"track-length-must-be-greater-than-0": "Track length must be greater than 0", "the-scans-distance-must-be-greater-than-0m": "The scan's distance must be greater than 0m",
"track-name": "Track 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!",
"track-name-must-not-be-empty": "Track name must not be empty", "there-are-no-cards-yet": "There are no cards yet.",
"tracks": "Tracks", "there-are-no-contacts-added-yet": "There are no contacts added yet.",
"update-card": "Update Card", "there-are-no-donations-yet": "There are no donations yet",
"update-password": "Update password", "there-are-no-donors-yet": "There are no donors yet",
"updated-contact": "Updated contact!", "there-are-no-groups-yet": "There are no groups yet",
"updated-donor": "updated donor", "there-are-no-organizations-added-yet": "There are no organizations added yet.",
"updated-organization": "updated organization", "there-are-no-runners-added-yet": "There are no runners added yet.",
"updated-scan": "updated scan", "there-are-no-scans-yet": "There are no scans yet",
"updateing-group": "updateing group...", "there-are-no-teams-added-yet": "There are no teams added yet.",
"updating-card": "Updating card", "there-are-no-users-added-yet": "There are no users added yet.",
"updating-organization": "updating organization", "this-card-is": "This card is",
"updating-permissions": "updating permissions...", "this-might-take-a-moment": "This might take a moment 👀",
"updating-runner": "Updating runner...", "this-scanstation-is": "This scanstation is",
"updating-user": "updating user...", "token": "Token",
"updating-your-profile": "Updating your profile...", "total-distance": "total distance",
"user-added": "User added", "total-donation-amount": "total donation amount",
"user-groups": "User Groups", "total-donations": "total donations",
"user-is-being-added": "User is being added...", "total-paid-amount": "Total paid amount",
"user-updated": "User updated", "total-scans": "total scans",
"username": "Username", "total_donation_amount_in_eur": "Total donation amount in €",
"users": "Users", "track": "Track",
"valid": "Valid", "track-added": "Track added",
"valid-city-is-required": "Valid city is required", "track-data-is-being-loaded": "Track data is being loaded",
"valid-email-is-required": "valid email is required", "track-is-being-added": "Track is being added...",
"valid-international-phone-number-is-required": "valid international phone number is required...", "track-is-being-updated": "Track is being updated...",
"valid-zipcode-postal-code-is-required": "Valid zipcode/ postal code is required", "track-length-in-m": "Track Length in m",
"verfuegbare": "availdable", "track-length-must-be-greater-than-0": "Track length must be greater than 0",
"welcome_wavinghand": "Welcome 👋", "track-name": "Track name",
"yes-i-copied-the-token": "Yes, I copied the token", "track-name-must-not-be-empty": "Track name must not be empty",
"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!", "track-was-updated": "Track was updated!",
"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! 🎉", "tracks": "Tracks",
"you-can-provide-a-runner-but-you-dont-have-to": "You can provide a runner, but you don't have to.", "unpaid": "Unpaid",
"you-dont-have-any-scanstations-yet": "You don't have any scanstations yet", "update-card": "Update Card",
"you-have-to-provide-an-organization": "You have to provide an organization", "update-password": "Update password",
"you-have-to-save-your-changes-to-generate-a-link": "You have to save your changes to generate a link.", "updated-contact": "Updated contact!",
"you-must-create-at-least-one-card-or-cancel": "You must create at least one card (or cancel).", "updated-donor": "updated donor",
"zip-postal-code": "ZIP/ postal code", "updated-organization": "updated organization",
"generate-runner-certificates": "Generate runner certificates" "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-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"
}

View File

@@ -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
View File

@@ -0,0 +1,3 @@
@tailwind base;
@tailwind components;
@tailwind utilities;

View File

@@ -1,4 +1,6 @@
module.exports = { module.exports = {
mode: 'jit',
purge: [ './src/**/*.svelte' ],
theme: { theme: {
extend: { extend: {
colors: { colors: {

View File

@@ -1,5 +1,4 @@
import svelte from '@sveltejs/vite-plugin-svelte'; import svelte from '@sveltejs/vite-plugin-svelte';
import windiCSS from 'vite-plugin-windicss';
import { minify } from 'html-minifier'; import { minify } from 'html-minifier';
import { defineConfig } from 'vite'; import { defineConfig } from 'vite';
// //
@@ -24,17 +23,6 @@ export default defineConfig(({ command, mode }) => {
minify: isProduction minify: isProduction
}, },
plugins: [ plugins: [
windiCSS({
//@ts-ignore
verbose: true,
silent: false,
debug: true,
config: 'tailwind.config.js', // tailwind config file path (optional)
compile: false, // false: interpretation mode; true: compilation mode
prefix: 'windi-', // set compilation mode style prefix
globalPreflight: true, // set preflight style is global or scoped
globalUtility: true // set utility style is global or scoped
}),
svelte({ svelte({
//@ts-ignore //@ts-ignore
hot: !isProduction, hot: !isProduction,