Compare commits
	
		
			731 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 102471eaaa | |||
| 90b0fec236 | |||
| 4883e179e7 | |||
| 13c6e96292 | |||
| f547c0cc81 | |||
| fbe38eede9 | |||
| 22551c379f | |||
| a102af5a78 | |||
| e9dffcea83 | |||
| b9563d75dd | |||
| 3a569422ad | |||
| 0ee43f80a6 | |||
| f4542adf3b | |||
| 9f0623d194 | |||
| 5bab95a942 | |||
| 831f36946d | |||
| a4fbabaf9a | |||
| 04897c7d2e | |||
| b7e6fdaeac | |||
| 481f6b686e | |||
| e7a69ebdca | |||
| 194c3c4886 | |||
| f5a46aa203 | |||
| 443371e2fd | |||
| d7ab9247cd | |||
| a2cd54fba4 | |||
| 9048f3df77 | |||
| fecf3b59a3 | |||
| 18a4623e71 | |||
| 968a7ccc0e | |||
| 6249502a88 | |||
| 8a78034079 | |||
| 7633b7b056 | |||
| b8e6b24bf3 | |||
| 97b7ca931f | |||
| 48dd9acde5 | |||
| 5147a20b3c | |||
| bb2319a78d | |||
| 7c10d95c1c | |||
| f734d1e3f6 | |||
| e567bb35c3 | |||
| 3ec18a6964 | |||
| 847fa288f1 | |||
| 824ecfab2e | |||
| 5f5d8277b9 | |||
| 0a6cf619b0 | |||
| 050a146ae0 | |||
| 1bc53146b9 | |||
| e82350df4a | |||
| 3d3ce2918b | |||
| 79e6a4212d | |||
| 37cdbba0a3 | |||
| c37fb98bed | |||
| 975f145444 | |||
| 391186d01f | |||
| ae056cd88c | |||
| 7f989b206b | |||
| 65ce02e777 | |||
| 878d9714cb | |||
| f99b7f4bb8 | |||
| e23098410c | |||
| 04494d2a2a | |||
| e2d6fbb513 | |||
| 477c650f3f | |||
| fc15c68cba | |||
| 32b5f5420b | |||
| ee87f82799 | |||
| 7c08f522aa | |||
| e211554579 | |||
| 7ba890dfd7 | |||
| 68b4309164 | |||
| d803f3d490 | |||
| 5468766d87 | |||
| e967d8d20c | |||
| ad4db882f0 | |||
| 84aa846b87 | |||
| 3532968b33 | |||
| 6bb49db4ee | |||
| 38fb111f7a | |||
| a50447f457 | |||
| b1a2044631 | |||
| c883920caa | |||
| 21453ef272 | |||
| 10182433f8 | |||
| b338f33a63 | |||
| cb82200481 | |||
| 9abf74d6d2 | |||
| 50e81a6cb5 | |||
| 8fae1fb6b3 | |||
| 372fa110ec | |||
| 35bec9fe58 | |||
| 93d67bdba9 | |||
| a5e72a18e3 | |||
| 91d2f46b93 | |||
| c60bae4533 | |||
| 43ac878d44 | |||
| ceabd06a43 | |||
| 9bfc0c5338 | |||
| fb8206ff13 | |||
| dceb0ef461 | |||
| 88bc1982ca | |||
| e741a9d7e7 | |||
| 65f1d22205 | |||
| d867c08aba | |||
| 6193eff38e | |||
| f1929e7cf9 | |||
| 373484c242 | |||
| f77460bb0c | |||
| 574e0dcb05 | |||
| 7b19a0aa08 | |||
| 08642d7618 | |||
| c3e9c27cd3 | |||
| 29a2854671 | |||
| 8e6786e722 | |||
| 6ad40564e3 | |||
| 776973bfe9 | |||
| 6025e43baa | |||
| d9a47f882c | |||
| 4235758a6d | |||
| 59fe2dfabb | |||
| 6364536dcd | |||
| a8a771114d | |||
| 4e0a2c8301 | |||
| b6fed92a17 | |||
| 97b57aeb0c | |||
| e25ed1fff9 | |||
| a2ff5b8a14 | |||
| 0284f18beb | |||
| 803d64c78c | |||
| dacb2f8ace | |||
| b7a53960e5 | |||
| 66f1e6b4fe | |||
| 33166bfafc | |||
| b2648645e8 | |||
| 53e3ddb751 | |||
| edc2dcab92 | |||
| d49f545d94 | |||
| 3b98c99b72 | |||
| 1da775a09b | |||
| f0475bd9a0 | |||
| 15d8afefbb | |||
| f3bcc01685 | |||
| 95238606d5 | |||
| bbf8170cb9 | |||
| 8c628f23dc | |||
| 811f5d5754 | |||
| 69ec7fc1fe | |||
| 064197d222 | |||
| e9cf2bc849 | |||
| 103ad57ddc | |||
| 2856c5c1b7 | |||
| a953349c14 | |||
| 175d86745f | |||
| 081a141218 | |||
| e904ab0b84 | |||
| a2f9dbbe01 | |||
| 8b922309b9 | |||
| 4c81e3c432 | |||
| 6c1a707166 | |||
| 7d8253618b | |||
| dbaf85799a | |||
| daeea24e0e | |||
| 0fca0352c5 | |||
| 8eaad8219a | |||
| 6bc92f4a08 | |||
| 8be40e2d80 | |||
| 01b415d4cb | |||
| 5e82638f35 | |||
| 46d076af9d | |||
| 38a665024e | |||
| d32eb8266b | |||
| bc4ac0f316 | |||
| 6952b8727f | |||
| 81d4da6550 | |||
| 6154ca7ddf | |||
| 8fb1e0ca0f | |||
| 763a01af09 | |||
| 663cb29ccd | |||
| 56c3365656 | |||
| e7b2c64798 | |||
| 7cb6b63eb9 | |||
| d6d88f5f60 | |||
| 2c208c4381 | |||
| 39bc6c4945 | |||
| b94e3b745f | |||
| 6f337aeee1 | |||
| 5d48060834 | |||
| c842c203e2 | |||
| 5bcfc8db75 | |||
| 27b4dde755 | |||
| 91ab199769 | |||
| e75be49be4 | |||
| 505fb8cb08 | |||
| e5c9265588 | |||
| 02003ec80e | |||
| 133470b6f2 | |||
| 4a6230c439 | |||
| fdc7d80bbf | |||
| 352551e168 | |||
| 7aec050419 | |||
| 4289034436 | |||
| 8f8858f100 | |||
| d98fb0d5b2 | |||
| 5014bf5bc5 | |||
| 0708cabc75 | |||
| 4fcb26cf93 | |||
| 269def20d1 | |||
| b8de9e0e42 | |||
| 7b2b598588 | |||
| e0b61486b0 | |||
| 3f86f7412f | |||
| 6454d960de | |||
| 37154c188b | |||
| 8da7578a0a | |||
| 2a64094006 | |||
| e9ce9644ff | |||
| 52439aa5bc | |||
| ccf865687b | |||
| cac34db1fd | |||
| faf3893180 | |||
| c33dfcfddd | |||
| 019e14ab1f | |||
| b5790196c6 | |||
| 94a64ca690 | |||
| a6ce04c903 | |||
| 165c154233 | |||
| 318547db46 | |||
| e60c09e19c | |||
| 4834d1484c | |||
| 4b6342727e | |||
| cb5fa52cd9 | |||
| 947d01cf7f | |||
| 3563394fb3 | |||
| 269d7a7def | |||
| e95f2333b0 | |||
| 950217e0a3 | |||
| 5e65fb3301 | |||
| 2a294cde04 | |||
| e95420d79c | |||
| cffbd17dc7 | |||
| 00de8c3d75 | |||
| 1f4711d07a | |||
| 30e3396897 | |||
| 5291e049a1 | |||
| 08fbb504c9 | |||
| e9ca1d3e5d | |||
| eb80406fdb | |||
| 9fe53b0b9c | |||
| 9ae5e62e5d | |||
| 1613ae7de6 | |||
| ed1caa7be7 | |||
| d88f3a5a27 | |||
| c98eb49ae3 | |||
| 6d9d8a4724 | |||
| 7d8c68a455 | |||
| 8f33640bec | |||
| f89023e24a | |||
| f5d14f2e18 | |||
| 9811ede3b2 | |||
| b2e51fea48 | |||
| 2a915620c9 | |||
| 526688935f | |||
| c7dcf7c66d | |||
| 06411dc147 | |||
| 195d182cc9 | |||
| 17217dae76 | |||
| f105cc0a41 | |||
| 2f62c7ae89 | |||
| f6c1fea17c | |||
| 178dc93319 | |||
| 8ffe8eff06 | |||
| bd4952ee57 | |||
| 4b171fd04f | |||
| 2c198cfde8 | |||
| 7c6d39b5fa | |||
| 9c13b2f9e9 | |||
| 1ec9556aa6 | |||
| adec38b50b | |||
| a35af6f020 | |||
| e74ff5e885 | |||
| c87561f63b | |||
| c681570134 | |||
| 53b945c72f | |||
| f6985daec7 | |||
| 5662c3b6da | |||
| 9def0b27c9 | |||
| 52a02c82d2 | |||
| c241961d0a | |||
| 8f50555a06 | |||
| b35375c929 | |||
| bf1e715261 | |||
| 0240e1dca2 | |||
| 7cec2a00c5 | |||
| 239f79fecb | |||
| 0265a59b82 | |||
| 57dce34fc5 | |||
| d8110580e9 | |||
| 03b7ada5ef | |||
| 1df505ea00 | |||
| 3e8dac3203 | |||
| 95707a71a9 | |||
| fc2c2907c4 | |||
| ebdd1c2c0c | |||
| 13254b24dd | |||
| e17eb64031 | |||
| a0727a0291 | |||
| f7f7926829 | |||
| 9b7dca341b | |||
| da3300562a | |||
| e2ddb5a14c | |||
| fb9645aed6 | |||
| a4ebc7e126 | |||
| fd5db7d68a | |||
| e7eddb4f08 | |||
| 94155845f0 | |||
| 3abf608b15 | |||
| d31fe2363b | |||
| 11a56f87e8 | |||
| 19793cdcd4 | |||
| 9363773fa1 | |||
| c7990882cf | |||
| d4ab76ea1b | |||
| 2c992a0e63 | |||
| 88f96acc3c | |||
| 245db06173 | |||
| 49c2cd5c4b | |||
| 64db553185 | |||
| a06a19ce9c | |||
| 592ddc1541 | |||
| cb5f2b73d0 | |||
| 2304b12c1c | |||
| 38d3e1912c | |||
| fbc14fd7b4 | |||
| 0283df22c8 | |||
| 845737ee8e | |||
| 6993511c67 | |||
| 9111ad147c | |||
| 333214aa8f | |||
| c0cde02fec | |||
| 070a20a2e5 | |||
| c5e8409079 | |||
| 67eff0eda9 | |||
| 3de7b632e0 | |||
| 842248e4c4 | |||
| c5d7ec25b5 | |||
| a9a965d698 | |||
| dc866dd540 | |||
| 89252619b1 | |||
| 2699b06d7c | |||
| fd0d45f721 | |||
| 5ecf838dd2 | |||
| 45a7a90cb8 | |||
| cac851f2b1 | |||
| 238082b657 | |||
| aecbabe522 | |||
| a9cdac4f74 | |||
| a59dbbe50e | |||
| 9bec95ede8 | |||
| 70307a9e82 | |||
| ef077b4e6a | |||
| dcabed4e93 | |||
| 1af047f66e | |||
| ee91748b3c | |||
| e5241d619b | |||
| d79608edbb | |||
| 4cbd26580e | |||
| fe62ad5539 | |||
| eb13f038a1 | |||
| 9505c2b030 | |||
| 008835c24f | |||
| 7083b3d8d2 | |||
| 754931b2f6 | |||
| 2dc8ffba32 | |||
| d0fe6a2e85 | |||
| 85705b6e68 | |||
| 3ea7a015a9 | |||
| 44329413ed | |||
| 46db68ab22 | |||
| dc9d7f22a2 | |||
| f917018fd9 | |||
| 7b420c430d | |||
| 00359d25c1 | |||
| d8a3063735 | |||
| 6491af19e3 | |||
| 61328d20ed | |||
| 0a6d92a1f3 | |||
| 3a576d1073 | |||
| b30b98b521 | |||
| 43d82a2af0 | |||
| 6a4495b813 | |||
| e8a0ad6647 | |||
| 92b89cc4d8 | |||
| 268b1b1d98 | |||
| 75bc89ca30 | |||
| 0625937068 | |||
| 32a9074963 | |||
| b869b5fd2a | |||
| 3a3e2f7157 | |||
| bea57aa03a | |||
| 30991d5364 | |||
| 5cc8b0811c | |||
| 2c73b9862d | |||
| 732b2f061e | |||
| 3680533eef | |||
| 1307d72c9d | |||
| 405dfa0c34 | |||
| 5c2d154ad1 | |||
| f2bf8d9bac | |||
| f9cfd6bd06 | |||
| 287f63fa52 | |||
| 5fe47634e8 | |||
| a6590910cf | |||
| ad454c386c | |||
| 0b2c296de0 | |||
| 0e85940cba | |||
| 8d479c32f8 | |||
| 549785cf7d | |||
| aafc4c8d62 | |||
| 47dedbdc73 | |||
| 6fe134afc8 | |||
| 63a50f92e7 | |||
| ca6da15ef7 | |||
| 8dfa19fa0f | |||
| 0feee0ae2f | |||
| 2a6a39916a | |||
| f0a2b2859f | |||
| 32ddb66fc8 | |||
| df63c2388d | |||
| 757655ea63 | |||
| 329c1cc037 | |||
| da6dd55d13 | |||
| 0e5490f1c8 | |||
| b82d638de1 | |||
| 224034dcc6 | |||
| 026d3d41c1 | |||
| fd1a06b359 | |||
| 452d010183 | |||
| eb1c17e3ac | |||
| a101873eb0 | |||
| 3d2acb692a | |||
| 0900c2691e | |||
| 1337676e08 | |||
| 2e075eafab | |||
| 14d64b6070 | |||
| 81b8fbf4e3 | |||
| 24d074752f | |||
| 08047a9307 | |||
| 1b0cd5b90b | |||
| 65e8998894 | |||
| 449948050b | |||
| cf97281592 | |||
| 75684efa1a | |||
| 2c4f27a943 | |||
| 53b7dec7cd | |||
| e0cbfb000b | |||
| 3a66f4c862 | |||
| 85ceaa464f | |||
| 976755338b | |||
| 1c980059cf | |||
| 2d8c4c1698 | |||
| 19a333d7bd | |||
| 96c55db63d | |||
| fecb07ee37 | |||
| e10c6480a5 | |||
| f3cc07c009 | |||
| 068076dd47 | |||
| 02158605be | |||
| 674e6a90ec | |||
| f679330466 | |||
| 93fc7c2e83 | |||
| f299617c60 | |||
| 28cbc5b98c | |||
| c28f1ee0bc | |||
| cff112d705 | |||
| 9fc4ad63c4 | |||
| 97054a71c1 | |||
| 2391668a25 | |||
| 717d33547c | |||
| 997be32679 | |||
| 134f00c40e | |||
| 47c898bdfd | |||
| e752ee12d1 | |||
| cc4515ff66 | |||
| f190292171 | |||
| b246f2b349 | |||
| 76b69d851a | |||
| 224f586368 | |||
| 9add6c8ff1 | |||
| 7a63d4eed1 | |||
| e54a4807f7 | |||
| cee04c1d6f | |||
| cbec78589d | |||
| a85db7cb3f | |||
| 2bd3779839 | |||
| 303e33cafb | |||
| b4e689dddf | |||
| 98a0b036c5 | |||
| fb3f30fb10 | |||
| 6213952007 | |||
| 07ac041d69 | |||
| 5c02028841 | |||
| c561b53670 | |||
| dcd0d5a362 | |||
| 18acac83bc | |||
| d7d44470bb | |||
| 0f0aae7ba4 | |||
| 4c0886a5d9 | |||
| 04a3038369 | |||
| bdcf5d3fc0 | |||
| c7a858eed7 | |||
| de5aa9237d | |||
| d015f97395 | |||
| 57618156b4 | |||
| 865254d646 | |||
| 1dbab03fe7 | |||
| a943aaf5fc | |||
| 6e6e8b2617 | |||
| 4c2c24af2c | |||
| 3d3a10aafb | |||
| 000fc97beb | |||
| 5645eeaafa | |||
| 961477d522 | |||
| a5f71015a6 | |||
| e42ea943b7 | |||
| 9c5fc6b61c | |||
| 302caf015f | |||
| e11296071a | |||
| 112eb29f93 | |||
| c6c97516b3 | |||
| 03676b2894 | |||
| 9ca57fac2e | |||
| 18f151c1fb | |||
| e90e56d8b2 | |||
| d241ca5698 | |||
| b512cf8667 | |||
| a24d2923c6 | |||
| 467808abef | |||
| 861f1f2216 | |||
| 509b22bea0 | |||
| 7447b2f4c1 | |||
| fef14b6e4f | |||
| 01d2a7e6aa | |||
| ac586fec5a | |||
| 5476808683 | |||
| 331d737796 | |||
| ef81b8adf9 | |||
| 8a7d635cef | |||
| 4c259c1eef | |||
| 5b4ede5e2f | |||
| d0ab3dda78 | |||
| d9cf51b4bb | |||
| aa17f24220 | |||
| cf60edf7d4 | |||
| ffbc243194 | |||
| b6b07cf30c | |||
| 495a6b22bd | |||
| 0acaffbdfa | |||
| 6043bc4517 | |||
| e6ed066e3f | |||
| ee4e8655b8 | |||
| 37970d2be6 | |||
| 1376788016 | |||
| 4cad86cf85 | |||
| 6304116edb | |||
| 834ff8fa63 | |||
| 1f428a535e | |||
| 0c40966970 | |||
| 9da071fe9b | |||
| 892a04f289 | |||
| 27cc9727f1 | |||
| f0738d451b | |||
| 9e6a8daf2c | |||
| bfacfec765 | |||
| 0bae5bf32b | |||
| 22b09d16d0 | |||
| 9c867e106e | |||
| 304f28a3c1 | |||
| d65d3793de | |||
| 3638d87bd2 | |||
| b97a92860d | |||
| 7c86a5eeb3 | |||
| d23dbaaf69 | |||
| e6ffc371e1 | |||
| 3177c6eaa3 | |||
| acd2f0519d | |||
| 18ec100c33 | |||
| fa55fce76e | |||
| f47d5e347d | |||
| 7488a8b597 | |||
| 2e3ac154be | |||
| 2472640755 | |||
| 7b685d6cad | |||
| 17f6f4e616 | |||
| 48cfc15cfb | |||
| bb9b779cee | |||
| af63ce67ae | |||
| 5cc4871ec4 | |||
| c8cfe669b8 | |||
| 8b74d6d759 | |||
| a9227768de | |||
| d966e1d4de | |||
| ceb2146c1b | |||
| 8d006d8c74 | |||
| 777304f259 | |||
| 12433f7c23 | |||
| 44b53da345 | |||
| ab45fc144e | |||
| e99e9e0708 | |||
| 467404bfc8 | |||
| ce50fa2a62 | |||
| 10a011d842 | |||
| 5352410d0c | |||
| c5d155396a | |||
| 93187099d3 | |||
| aa24b1dce5 | |||
| eb3ede9593 | |||
| d7fecfbd0b | |||
| b065b4ff21 | |||
| 87370d24be | |||
| 8f8b9988ad | |||
| f8ccf4f5d8 | |||
| 25d8b86efd | |||
| 0cd3e937d8 | |||
| 89bb9c215e | |||
| 2d18686ce7 | |||
| 1d999d4910 | |||
| 7dfaa7579a | |||
| 08cb079e97 | |||
| 450aa83592 | |||
| 0614c76e92 | |||
| 97e338f9d4 | |||
| 636f018daa | |||
| c8d639024a | |||
| 6be2ee626a | |||
| f7fc1967a5 | |||
| aedb8a765b | |||
| cf58bd15c3 | |||
| 34f4f68524 | |||
| 09b8144080 | |||
| f1e6fb4ce7 | |||
| 2ca63fd1f6 | |||
| a5d25e7d92 | |||
| 4167819e7a | |||
| 5bd3a463f0 | |||
| 79c447b4c6 | |||
| 540304f947 | |||
| 75d8f7331b | |||
| b2509e9e53 | |||
| 7862f44653 | |||
| 962dd0c1bb | |||
| 5d5f7c7f5c | |||
| 6aaf838451 | |||
| ad3bd312e9 | |||
| 5fa9939696 | |||
| 4956bb0e9c | |||
| c074c12be7 | |||
| ddbc293e9c | |||
| a3921b45c7 | |||
| 38e1c8c5a1 | |||
| c2d29ff233 | |||
| 2316baa898 | |||
| f185d559c0 | |||
| 73d95bc004 | |||
| fcd55f89d7 | |||
| f9fe793573 | |||
| bc36411993 | |||
| 48506236bf | |||
| ded9b589fe | |||
| 67c3732fad | |||
| 2932f4591e | |||
| df53c07450 | |||
| 40899e9d80 | |||
| f794af0950 | |||
| 1665a1a093 | |||
| 4a36fb6d95 | |||
| acf78a8822 | |||
| f5c1ec9939 | |||
| 4b3d38b05b | |||
| 23e0b53107 | |||
| c907486c4d | |||
| 6b5945add8 | |||
| 55693de934 | |||
| d467475b6d | |||
| 44bc14820f | |||
| ec447e2e36 | |||
| 0af2647965 | |||
| 08442154f4 | |||
| 9f7d2234fb | |||
| ddd82a71a7 | |||
| 014ba3bf67 | |||
| c87321f804 | |||
| 8b451b3c67 | |||
| 0cfc87fbe6 | |||
| ae9673070c | |||
| 008027db0e | |||
| aec5e3473e | |||
| 95c8fde72f | |||
| 0f32968fae | |||
| ae79e9fea1 | |||
| 1a52aaf8d1 | |||
| d6c315ab8e | |||
| 983ce56048 | |||
| de2fe0e9f1 | |||
| c3c95bf291 | |||
| d2050b5948 | |||
| 6b92405bae | |||
| 49e87ccb15 | |||
| 50fffef13b | |||
| 82b1811971 | |||
| aeadef60bb | |||
| a1ab65a0e9 | |||
| 17e0805fe6 | |||
| ddd9c396b6 | |||
| ef49e507c1 | |||
| fbe74a5d80 | |||
| 076893981f | |||
| fac059f02c | |||
| 0313f8cc49 | |||
| 7ad6b73574 | |||
| 3cd0468b19 | |||
| f46ccb610e | |||
| 8a32569a3b | |||
| 535b23ae91 | |||
| 4715978f81 | |||
| a516aa7775 | |||
| 77e9c205f9 | |||
| e852305400 | |||
| c6a15264b3 | |||
| 2d0beaaaad | |||
| 5c5ef95d2b | |||
| 9aa8e7edff | 
| @@ -1,3 +1,4 @@ | |||||||
| public/env.sample.js | public/env.sample.js | ||||||
| public/workbox-*.js | .pnpm-store | ||||||
| public/workbox-*.js.map | .yarn | ||||||
|  | .pnp.* | ||||||
							
								
								
									
										84
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										84
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,84 +0,0 @@ | |||||||
| --- |  | ||||||
| kind: secret |  | ||||||
| name: docker_username |  | ||||||
| get: |  | ||||||
|   path: odit-registry-builder |  | ||||||
|   name: username |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: secret |  | ||||||
| name: docker_password |  | ||||||
| get: |  | ||||||
|   path: odit-registry-builder |  | ||||||
|   name: password |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: secret |  | ||||||
| name: git_ssh |  | ||||||
| get: |  | ||||||
|   path: odit-git-bot |  | ||||||
|   name: sshkey |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: pipeline |  | ||||||
| type: kubernetes |  | ||||||
| name: build:dev |  | ||||||
|  |  | ||||||
| steps: |  | ||||||
|   - name: run full license export |  | ||||||
|     depends_on: ["clone"] |  | ||||||
|     image: node:alpine |  | ||||||
|     commands: |  | ||||||
|       - yarn |  | ||||||
|       - yarn licenses:export |  | ||||||
|   - name: push new licenses file to repo |  | ||||||
|     depends_on: ["run full license export"] |  | ||||||
|     image: appleboy/drone-git-push |  | ||||||
|     settings: |  | ||||||
|       branch: dev |  | ||||||
|       commit: true |  | ||||||
|       commit_message: new license file version [CI SKIP] |  | ||||||
|       author_email: bot@odit.services |  | ||||||
|       remote: git@git.odit.services:lfk/frontend.git |  | ||||||
|       ssh_key: |  | ||||||
|         from_secret: git_ssh |  | ||||||
|   - name: build dev |  | ||||||
|     image: plugins/docker |  | ||||||
|     depends_on: [clone] |  | ||||||
|     settings: |  | ||||||
|       username: |  | ||||||
|         from_secret: docker_username |  | ||||||
|       password: |  | ||||||
|         from_secret: docker_password |  | ||||||
|       repo: registry.odit.services/lfk/frontend |  | ||||||
|       tags: |  | ||||||
|         - dev |  | ||||||
|       registry: registry.odit.services |  | ||||||
|       mtu: 1000 |  | ||||||
| trigger: |  | ||||||
|   branch: |  | ||||||
|     - dev |  | ||||||
|   event: |  | ||||||
|     - push |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: pipeline |  | ||||||
| type: kubernetes |  | ||||||
| name: build:tags |  | ||||||
| steps: |  | ||||||
|   - name: build $DRONE_TAG |  | ||||||
|     image: plugins/docker |  | ||||||
|     depends_on: [clone] |  | ||||||
|     settings: |  | ||||||
|       username: |  | ||||||
|         from_secret: docker_username |  | ||||||
|       password: |  | ||||||
|         from_secret: docker_password |  | ||||||
|       repo: registry.odit.services/lfk/frontend |  | ||||||
|       tags: |  | ||||||
|         - '${DRONE_TAG}' |  | ||||||
|       registry: registry.odit.services |  | ||||||
|       mtu: 1000 |  | ||||||
| trigger: |  | ||||||
|   event: |  | ||||||
|   - tag |  | ||||||
							
								
								
									
										33
									
								
								.gitea/workflows/release.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								.gitea/workflows/release.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | name: Build release images | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - "*.*.*" | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build-container: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |       - name: Set up Node.js | ||||||
|  |         uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version: 19 | ||||||
|  |       - run: npm i -g pnpm@10.7 && pnpm i | ||||||
|  |       - run: pnpm licenses:export | ||||||
|  |       - name: Login to registry | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry: registry.odit.services | ||||||
|  |           username: ${{ vars.REGISTRY_USERNAME }} | ||||||
|  |           password: ${{ secrets.REGISTRY_PASSWORD }} | ||||||
|  |       - name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v3 | ||||||
|  |       - name: Build and push | ||||||
|  |         uses: docker/build-push-action@v6 | ||||||
|  |         with: | ||||||
|  |           push: true | ||||||
|  |           tags: | | ||||||
|  |             ${{ vars.REGISTRY }}/lfk/frontend:${{ github.ref_name }} | ||||||
|  |           platforms: linux/amd64,linux/arm64 | ||||||
							
								
								
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,8 @@ | |||||||
| node_modules | node_modules | ||||||
| build |  | ||||||
| package-lock.json |  | ||||||
| yarn.lock |  | ||||||
| *.map | *.map | ||||||
| public/env.js | public/env.js | ||||||
| public/sw.js |  | ||||||
| public/index.html |  | ||||||
| public/workbox-*.js |  | ||||||
| svelte.config.js |  | ||||||
| public/index.html | public/index.html | ||||||
|  | /dist | ||||||
|  | .pnpm-store | ||||||
|  | .yarn | ||||||
|  | .pnp.* | ||||||
							
								
								
									
										7
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,9 +5,8 @@ | |||||||
|     "remimarsal.prettier-now", |     "remimarsal.prettier-now", | ||||||
|     "svelte.svelte-vscode", |     "svelte.svelte-vscode", | ||||||
|     "lokalise.i18n-ally", |     "lokalise.i18n-ally", | ||||||
|         "fivethree.vscode-svelte-snippets" |     "fivethree.vscode-svelte-snippets", | ||||||
|  |     "voorjaar.windicss-intellisense" | ||||||
|   ], |   ], | ||||||
|     "unwantedRecommendations": [ |   "unwantedRecommendations": ["antfu.i18n-ally"] | ||||||
|         "antfu.i18n-ally" |  | ||||||
|     ] |  | ||||||
| } | } | ||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| { | { | ||||||
|   "i18n-ally.localesPaths": "src/locales", |   "i18n-ally.localesPaths": "src/locales", | ||||||
|     "i18n-ally.keystyle": "nested" |   "i18n-ally.keystyle": "nested", | ||||||
|  |   "windicss.enableCodeFolding": false | ||||||
| } | } | ||||||
							
								
								
									
										1055
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										1055
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										24
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,18 +1,16 @@ | |||||||
| FROM node:15.5.1-alpine3.12 | FROM registry.odit.services/hub/library/node:23.10.0-alpine3.21 AS build | ||||||
|  | ARG NPM_REGISTRY_URL=https://registry.npmjs.org | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| RUN npm i -g pnpm |  | ||||||
| COPY package.json ./ | COPY package.json pnpm-lock.yaml vite.config.js tailwind.config.cjs postcss.config.cjs index.html ./ | ||||||
| RUN pnpm i | RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@10.7 | ||||||
| COPY package.json *.config.js workbox-config.js template-copy.js index.template.html s-config.template.js ./ | RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i | ||||||
|  |  | ||||||
| COPY src ./src | COPY src ./src | ||||||
| COPY public ./public | COPY public ./public | ||||||
| RUN pnpm run build | RUN pnpm build | ||||||
|  |  | ||||||
| # final image | # final image | ||||||
| FROM alpine | FROM registry.odit.services/library/nginx-brotli:3.15 AS final | ||||||
| COPY --from=0 /app/build /app | COPY --from=build /app/dist /usr/share/nginx/html | ||||||
| RUN rm -rf /app/build/_dist_/components |  | ||||||
| RUN rm -rf /app/build/_dist_/locales |  | ||||||
| RUN rm -rf /app/build-manifest.json |  | ||||||
| FROM fholzer/nginx-brotli:v1.19.1 |  | ||||||
| COPY --from=1 /app /usr/share/nginx/html |  | ||||||
| COPY ./nginx.conf /etc/nginx/nginx.conf | COPY ./nginx.conf /etc/nginx/nginx.conf | ||||||
							
								
								
									
										27
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | # @odit/lfk-frontend | ||||||
|  |  | ||||||
|  | ## ✒️ Overview | ||||||
|  |  | ||||||
|  | This is an API client for [https://git.odit.services/lfk/backend](@lfk/backend) | ||||||
|  |  | ||||||
|  | This application is intended for use by admin users/ members only. | ||||||
|  |  | ||||||
|  | ## 🚀 Getting Started | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | pnpm i | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Development | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | pnpm dev | ||||||
|  | / | ||||||
|  | pnpm dev --open | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Build | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | pnpm build | ||||||
|  | ``` | ||||||
							
								
								
									
										22
									
								
								index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | <!doctype html> | ||||||
|  | <html lang="en"> | ||||||
|  |   <head> | ||||||
|  |     <meta charset="utf-8" /> | ||||||
|  |     <link rel="icon" href="/favicon.png" /> | ||||||
|  |     <link rel="manifest" href="/manifest.webmanifest" /> | ||||||
|  |     <link rel="apple-touch-icon" href="/lfk-logo.png" /> | ||||||
|  |     <meta name="theme-color" content="#FFFFFF" /> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||||
|  |     <meta name="description" content="Lauf Für Kaya! - Admin" /> | ||||||
|  |     <title>Lauf für Kaya! - Admin</title> | ||||||
|  |   </head> | ||||||
|  |  | ||||||
|  |   <body> | ||||||
|  |     <span style="display: none; visibility: hidden" id="buildinfo" | ||||||
|  |       >RELEASE_INFO-1.9.4-RELEASE_INFO</span | ||||||
|  |     > | ||||||
|  |     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||||
|  |     <script src="/env.js"></script> | ||||||
|  |     <script type="module" src="/src/main.js"></script> | ||||||
|  |   </body> | ||||||
|  | </html> | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html lang="en"> |  | ||||||
|  |  | ||||||
| <head> |  | ||||||
|   <meta charset="utf-8" /> |  | ||||||
|   <link rel="icon" href="/favicon.png" /> |  | ||||||
|   <link rel="manifest" href="/manifest.webmanifest"> |  | ||||||
|   <link rel="apple-touch-icon" href="/lfk-logo.png"> |  | ||||||
|   <meta name="theme-color" content="#FFFFFF"> |  | ||||||
|   <meta name="viewport" content="width=device-width, initial-scale=1" /> |  | ||||||
|   <meta name="description" content="Lauf Für Kaya! - Admin" /> |  | ||||||
|   <title>Lauf für Kaya! - Admin</title> |  | ||||||
|   __TAILWIND_INSERT__ |  | ||||||
| </head> |  | ||||||
|  |  | ||||||
| <body> |  | ||||||
|   <span style="display: none;visibility: hidden;" id="buildinfo">RELEASE_INFO-0.8.5-RELEASE_INFO</span> |  | ||||||
|   <noscript>You need to enable JavaScript to run this app.</noscript> |  | ||||||
|   <script src="/env.js"></script> |  | ||||||
|   <script defer type="module" src="/_dist_/index.js"></script> |  | ||||||
| </body> |  | ||||||
|  |  | ||||||
| </html> |  | ||||||
							
								
								
									
										14
									
								
								nginx.conf
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								nginx.conf
									
									
									
									
									
								
							| @@ -6,6 +6,20 @@ http { | |||||||
|     server { |     server { | ||||||
|         error_page 404 /index.html; |         error_page 404 /index.html; | ||||||
|         root /usr/share/nginx/html; |         root /usr/share/nginx/html; | ||||||
|  |         location /assets { | ||||||
|  |             expires 1y; | ||||||
|  |             log_not_found off; | ||||||
|  |             access_log off; | ||||||
|  |         } | ||||||
|  |         location = /index.html { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|  |         location = / { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|  |         location = /env.js { | ||||||
|  |             add_header Cache-Control 'no-store'; | ||||||
|  |         } | ||||||
|         location / { |         location / { | ||||||
|             try_files $uri $uri/ /index.html; |             try_files $uri $uri/ /index.html; | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								order.js
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								order.js
									
									
									
									
									
								
							| @@ -1,11 +1,13 @@ | |||||||
| const fs = require('fs'); | import fs from "fs"; | ||||||
| // get all language files | // get all language files | ||||||
| const files = fs.readdirSync('./src/locales/'); | const files = fs.readdirSync("./src/locales/"); | ||||||
| files.forEach((f) => { | files.forEach((f) => { | ||||||
|   // read file as object |   // read file as object | ||||||
|   const unordered = JSON.parse(fs.readFileSync(`src/locales/${f}`)); |   const unordered = JSON.parse(fs.readFileSync(`src/locales/${f}`)); | ||||||
|   // order object by keys alpabetically A-Z |   // order object by keys alpabetically A-Z | ||||||
| 	const ordered = Object.keys(unordered).sort().reduce((obj, key) => { |   const ordered = Object.keys(unordered) | ||||||
|  |     .sort() | ||||||
|  |     .reduce((obj, key) => { | ||||||
|       obj[key] = unordered[key]; |       obj[key] = unordered[key]; | ||||||
|       return obj; |       return obj; | ||||||
|     }, {}); |     }, {}); | ||||||
|   | |||||||
							
								
								
									
										81
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,61 +1,64 @@ | |||||||
| { | { | ||||||
|   "name": "@odit/lfk-frontend", |   "name": "@odit/lfk-frontend", | ||||||
| 	"version": "0.8.5", |   "version": "1.9.4", | ||||||
|  |   "type": "module", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "i18n-order": "node order.js", |     "i18n-order": "node order.js", | ||||||
| 		"dev:all": "yarn prebuild && snowpack dev", |     "dev": "vite", | ||||||
| 		"dev": "cross-env NODE_ENV_ODIT=development_fast node template-copy.js && yarn build:sw && snowpack dev", |     "format": "prettier --write --plugin-search-dir=. .", | ||||||
| 		"build": "yarn prebuild && snowpack build", |     "build": "vite build", | ||||||
| 		"prebuild": "cross-env NODE_ENV_ODIT=production node template-copy.js && yarn build:sw", |  | ||||||
| 		"build:sw": "workbox generateSW workbox-config.js", |  | ||||||
|     "release": "release-it", |     "release": "release-it", | ||||||
|     "licenses:export": "license-exporter --json -o public" |     "licenses:export": "license-exporter --json -o public" | ||||||
|   }, |   }, | ||||||
|   "license": "CC-BY-NC-SA-4.0", |   "license": "CC-BY-NC-SA-4.0", | ||||||
| 	"dependencies": { |  | ||||||
| 		"@odit/lfk-client-js": "0.6.4", |  | ||||||
| 		"csvtojson": "^2.0.10", |  | ||||||
| 		"gridjs": "3.3.0", |  | ||||||
| 		"localforage": "1.9.0", |  | ||||||
| 		"marked": "^2.0.1", |  | ||||||
| 		"svelte-focus-trap": "1.0.1", |  | ||||||
| 		"svelte-i18n": "3.3.7", |  | ||||||
| 		"svelte-select": "^3.17.0", |  | ||||||
| 		"tailwindcss": "2.0.3", |  | ||||||
| 		"tinro": "0.6.1", |  | ||||||
| 		"toastify-js": "1.9.3", |  | ||||||
| 		"validator": "13.5.2", |  | ||||||
| 		"xlsx": "^0.16.9" |  | ||||||
| 	}, |  | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
| 		"@odit/license-exporter": "^0.0.11", |     "@odit/license-exporter": "0.2.0", | ||||||
| 		"@snowpack/plugin-svelte": "3.5.2", |     "@sveltejs/vite-plugin-svelte": "2.1.1", | ||||||
| 		"auto-changelog": "^2.2.1", |     "auto-changelog": "2.5.0", | ||||||
| 		"autoprefixer": "10.2.5", |     "autoprefixer": "10.4.21", | ||||||
| 		"cross-env": "^7.0.3", |     "postcss": "8.5.3", | ||||||
| 		"postcss": "8.2.8", |     "prettier": "3.5.3", | ||||||
| 		"postcss-load-config": "3.0.1", |     "prettier-plugin-svelte": "3.3.3", | ||||||
| 		"release-it": "^14.4.1", |     "release-it": "17.10.0", | ||||||
| 		"snowpack": "3.0.13", |     "svelte-select": "3.17.0", | ||||||
| 		"svelte": "3.35.0", |     "tailwindcss": "3.4.15", | ||||||
| 		"svelte-preprocess": "4.6.9", |     "vite": "4.3.3" | ||||||
| 		"workbox-cli": "6.1.2" |  | ||||||
|   }, |   }, | ||||||
|   "release-it": { |   "release-it": { | ||||||
|     "git": { |     "git": { | ||||||
|       "commit": true, |       "commit": true, | ||||||
|       "requireCleanWorkingDir": false, |       "requireCleanWorkingDir": false, | ||||||
| 			"commitMessage": "🚀RELEASE v${version}", |       "commitMessage": "chore(release): ${version}", | ||||||
| 			"push": false, |       "push": true, | ||||||
|       "tag": true, |       "tag": true, | ||||||
| 			"tagName": null, |       "tagName": "${version}", | ||||||
| 			"tagAnnotation": "v${version}" |       "tagAnnotation": "${version}" | ||||||
|     }, |     }, | ||||||
|     "npm": { |     "npm": { | ||||||
|       "publish": false |       "publish": false | ||||||
|     }, |     }, | ||||||
|     "hooks": { |     "hooks": { | ||||||
| 			"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.template.html && node order.js  && git add src/locales" |       "after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.html && node order.js  && git add src/locales" | ||||||
| 		} |     } | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "@fontsource/athiti": "^5.2.5", | ||||||
|  |     "@odit/lfk-client-js": "1.2.0", | ||||||
|  |     "@paralleldrive/cuid2": "2.2.2", | ||||||
|  |     "@tanstack/svelte-table": "8.9.1", | ||||||
|  |     "bwip-js": "3.4.0", | ||||||
|  |     "check-password-strength": "2.0.10", | ||||||
|  |     "csvtojson": "2.0.10", | ||||||
|  |     "localforage": "1.10.0", | ||||||
|  |     "marked": "4.3.0", | ||||||
|  |     "svelte": "3.58.0", | ||||||
|  |     "svelte-french-toast": "1.2.0", | ||||||
|  |     "svelte-i18n": "3.6.0", | ||||||
|  |     "tinro": "0.6.12", | ||||||
|  |     "validator": "13.15.0", | ||||||
|  |     "xlsx": "0.18.5" | ||||||
|  |   }, | ||||||
|  |   "volta": { | ||||||
|  |     "node": "20.0.0" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										3743
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										3743
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | module.exports = { | ||||||
|  |   plugins: { | ||||||
|  |     tailwindcss: {}, | ||||||
|  |     autoprefixer: {}, | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -1,8 +1,12 @@ | |||||||
| const config = { | const config = { | ||||||
| 	baseurl: 'http://localhost:4010', |   baseurl: "http://localhost:4010", | ||||||
| 	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe', |   baseurl_selfservice: "http://localhost:5174", | ||||||
|  |   baseurl_documentserver: "http://localhost:4010/documents", | ||||||
|  |   documentserver_key: | ||||||
|  |     "NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe", | ||||||
|   // optional |   // optional | ||||||
| 	default_username: 'demo', |   default_username: "demo", | ||||||
| 	default_password: 'demo', |   default_password: "demo", | ||||||
| 	prefersHashRouting: true |   prefersHashRouting: true, | ||||||
| }; | }; | ||||||
|  | window.config = config; | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,4 +0,0 @@ | |||||||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 98.1 118"> |  | ||||||
|   <path fill="#ff3e00" d="M91.8 15.6C80.9-.1 59.2-4.7 43.6 5.2L16.1 22.8A31.25 31.25 0 001.9 43.9c-1.3 7.3-.2 14.8 3.3 21.3-2.4 3.6-4 7.6-4.7 11.8-1.6 8.9.5 18.1 5.7 25.4 11 15.7 32.6 20.3 48.2 10.4l27.5-17.5c7.5-4.7 12.7-12.4 14.2-21.1 1.3-7.3.2-14.8-3.3-21.3 2.4-3.6 4-7.6 4.7-11.8 1.7-9-.4-18.2-5.7-25.5"/> |  | ||||||
|   <path fill="#fff" d="M40.9 103.9a21.8 21.8 0 01-23.4-8.7c-3.2-4.4-4.4-9.9-3.5-15.3l.6-2.6.5-1.6 1.4 1c3.3 2.4 6.9 4.2 10.8 5.4l1 .3-.1 1c-.1 1.4.3 2.9 1.1 4.1a6.62 6.62 0 008.8 2L65.5 72c1.4-.9 2.3-2.2 2.6-3.8.3-1.6-.1-3.3-1-4.6a6.56 6.56 0 00-8.8-1.9l-10.5 6.7a18.6 18.6 0 01-5.6 2.4 21.8 21.8 0 01-23.4-8.7 20.2 20.2 0 01-3.4-15.3c.9-5.2 4.1-9.9 8.6-12.7l27.5-17.5c1.7-1.1 3.6-1.9 5.6-2.5a21.8 21.8 0 0123.4 8.7c3.2 4.4 4.4 9.9 3.5 15.3-.2.9-.4 1.7-.7 2.6l-.5 1.6-1.4-1c-3.3-2.4-6.9-4.2-10.8-5.4l-1-.3.1-1c.1-1.4-.3-2.9-1.1-4.1a6.56 6.56 0 00-8.8-1.9L32.4 46.1c-1.4.9-2.3 2.2-2.6 3.8s.1 3.3 1 4.6a6.56 6.56 0 008.8 1.9l10.5-6.7c1.7-1.1 3.6-1.9 5.6-2.5a21.8 21.8 0 0123.4 8.7c3.2 4.4 4.4 9.9 3.5 15.3-.9 5.2-4.1 9.9-8.6 12.7l-27.5 17.5c-1.7 1.1-3.6 1.9-5.6 2.5"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 1.1 KiB | 
| @@ -29,6 +29,11 @@ | |||||||
|       "sizes": "1540x144", |       "sizes": "1540x144", | ||||||
|       "type": "image/png" |       "type": "image/png" | ||||||
|     }, |     }, | ||||||
| 		{ "src": "/maskable_icon_x1.png", "sizes": "750x750", "type": "image/png", "purpose": "any maskable" } |     { | ||||||
|  |       "src": "/maskable_icon_x1.png", | ||||||
|  |       "sizes": "750x750", | ||||||
|  |       "type": "image/png", | ||||||
|  |       "purpose": "any maskable" | ||||||
|  |     } | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +0,0 @@ | |||||||
| const sveltePreprocess = require('svelte-preprocess'); |  | ||||||
| const preprocess = sveltePreprocess(__insert__); |  | ||||||
|  |  | ||||||
| module.exports = { |  | ||||||
| 	preprocess |  | ||||||
| }; |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| /** @type {import("snowpack").SnowpackUserConfig } */ |  | ||||||
| module.exports = { |  | ||||||
| 	mount: { |  | ||||||
| 		public: '/', |  | ||||||
| 		src: '/_dist_' |  | ||||||
| 	}, |  | ||||||
| 	plugins: [ '@snowpack/plugin-svelte' ], |  | ||||||
| 	routes: [ |  | ||||||
| 		/* Enable an SPA Fallback in development: */ |  | ||||||
| 		{ match: 'routes', src: '.*', dest: '/index.html' } |  | ||||||
| 	], |  | ||||||
| 	packageOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 		sourceMap: false |  | ||||||
| 	}, |  | ||||||
| 	devOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	buildOptions: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	alias: { |  | ||||||
| 		/* ... */ |  | ||||||
| 	}, |  | ||||||
| 	optimize: { bundle: true, minify: true } |  | ||||||
| }; |  | ||||||
| @@ -1,7 +1,4 @@ | |||||||
| <script> | <script> | ||||||
|   import "./TailwindStyles.svelte"; |  | ||||||
|   import "toastify-js/src/toastify.css"; |  | ||||||
|   import "gridjs/dist/theme/mermaid.css"; |  | ||||||
|   import { Route, router } from "tinro"; |   import { Route, router } from "tinro"; | ||||||
|   router.subscribe((routeInfo) => { |   router.subscribe((routeInfo) => { | ||||||
|     window.scrollTo(0, 0); |     window.scrollTo(0, 0); | ||||||
| @@ -25,11 +22,10 @@ | |||||||
|     name: "lfk_admin", |     name: "lfk_admin", | ||||||
|     version: 1.0, |     version: 1.0, | ||||||
|     storeName: "lfk_admin", |     storeName: "lfk_admin", | ||||||
|     description: "LfK! admin dashbaord", |     description: "LfK! admin dashboard", | ||||||
|   }); |   }); | ||||||
|   window.onunhandledrejection = (event) => { |   window.onunhandledrejection = (event) => { | ||||||
|     if (event.reason.toString() == "Error: Unauthorized") { |     if (event.reason.toString() == "Error: Unauthorized") { | ||||||
|       console.log("Found 1"); |  | ||||||
|       localForage.clear(); |       localForage.clear(); | ||||||
|       location.replace("/"); |       location.replace("/"); | ||||||
|     } |     } | ||||||
| @@ -53,7 +49,6 @@ | |||||||
|   import { OpenAPI } from "@odit/lfk-client-js"; |   import { OpenAPI } from "@odit/lfk-client-js"; | ||||||
|   import UserDetail from "./components/users/UserDetail.svelte"; |   import UserDetail from "./components/users/UserDetail.svelte"; | ||||||
|   OpenAPI.BASE = config.baseurl; |   OpenAPI.BASE = config.baseurl; | ||||||
|   import { register as registerSW } from "./swmodule"; |  | ||||||
|   import TeamDetail from "./components/teams/TeamDetail.svelte"; |   import TeamDetail from "./components/teams/TeamDetail.svelte"; | ||||||
|   import UserPermissions from "./components/users/UserPermissions.svelte"; |   import UserPermissions from "./components/users/UserPermissions.svelte"; | ||||||
|   import GroupPermissions from "./components/groups/GroupPermissions.svelte"; |   import GroupPermissions from "./components/groups/GroupPermissions.svelte"; | ||||||
| @@ -69,33 +64,34 @@ | |||||||
|   import Donations from "./components/donations/Donations.svelte"; |   import Donations from "./components/donations/Donations.svelte"; | ||||||
|   import DonationDetail from "./components/donations/DonationDetail.svelte"; |   import DonationDetail from "./components/donations/DonationDetail.svelte"; | ||||||
|   import GroupDetail from "./components/groups/GroupDetail.svelte"; |   import GroupDetail from "./components/groups/GroupDetail.svelte"; | ||||||
|   import ScanStationsOverview from "./components/scanstations/ScanStationsOverview.svelte"; |  | ||||||
|   import ScanStations from "./components/scanstations/ScanStations.svelte"; |   import ScanStations from "./components/scanstations/ScanStations.svelte"; | ||||||
|   import ScanStationDetail from "./components/scanstations/ScanStationDetail.svelte"; |   import ScanStationDetail from "./components/scanstations/ScanStationDetail.svelte"; | ||||||
|   import Scans from "./components/scans/Scans.svelte"; |   import Scans from "./components/scans/Scans.svelte"; | ||||||
|   import ScanDetail from "./components/scans/ScanDetail.svelte"; |   import ScanDetail from "./components/scans/ScanDetail.svelte"; | ||||||
|  |   import Cards from "./components/cards/Cards.svelte"; | ||||||
|  |   import StatsClients from "./components/statsclients/StatsClients.svelte"; | ||||||
|  |   import StatsClientDetail from "./components/statsclients/StatsClientDetail.svelte"; | ||||||
|   store.init(); |   store.init(); | ||||||
|   registerSW(); |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <Route> | <Route> | ||||||
|   {#if $router.path === '/forgot_password'} |   {#if $router.path === "/forgot_password"} | ||||||
|     <Route path="/forgot_password"> |     <Route path="/forgot_password"> | ||||||
|       <ForgotPassword /> |       <ForgotPassword /> | ||||||
|     </Route> |     </Route> | ||||||
|   {:else if $router.path.includes('/reset')} |   {:else if $router.path.includes("/reset")} | ||||||
|     <Route path="/reset/:resetkey" let:params> |     <Route path="/reset/:resetkey" let:params> | ||||||
|       <ResetPassword {params} /> |       <ResetPassword {params} /> | ||||||
|     </Route> |     </Route> | ||||||
|   {:else if $router.path === '/about'} |   {:else if $router.path === "/about"} | ||||||
|     <Route path="/about"> |     <Route path="/about"> | ||||||
|       <About /> |       <About /> | ||||||
|     </Route> |     </Route> | ||||||
|   {:else if $router.path === '/imprint'} |   {:else if $router.path === "/imprint"} | ||||||
|     <Route path="/imprint"> |     <Route path="/imprint"> | ||||||
|       <Imprint /> |       <Imprint /> | ||||||
|     </Route> |     </Route> | ||||||
|   {:else if $router.path === '/privacy'} |   {:else if $router.path === "/privacy"} | ||||||
|     <Route path="/privacy"> |     <Route path="/privacy"> | ||||||
|       <Privacy /> |       <Privacy /> | ||||||
|     </Route> |     </Route> | ||||||
| @@ -185,6 +181,14 @@ import ScanDetail from "./components/scans/ScanDetail.svelte"; | |||||||
|             <DonationDetail {params} /> |             <DonationDetail {params} /> | ||||||
|           </Route> |           </Route> | ||||||
|         </Route> |         </Route> | ||||||
|  |         <Route path="/cards/*"> | ||||||
|  |           <Route path="/"> | ||||||
|  |             <Cards /> | ||||||
|  |           </Route> | ||||||
|  |           <!-- <Route path="/:scanid" let:params> | ||||||
|  |             <ScanDetail {params} /> | ||||||
|  |           </Route> --> | ||||||
|  |         </Route> | ||||||
|         <Route path="/scans/*"> |         <Route path="/scans/*"> | ||||||
|           <Route path="/"> |           <Route path="/"> | ||||||
|             <Scans /> |             <Scans /> | ||||||
| @@ -201,6 +205,14 @@ import ScanDetail from "./components/scans/ScanDetail.svelte"; | |||||||
|             <ScanStationDetail {params} /> |             <ScanStationDetail {params} /> | ||||||
|           </Route> |           </Route> | ||||||
|         </Route> |         </Route> | ||||||
|  |         <Route path="/statsclients/*"> | ||||||
|  |           <Route path="/"> | ||||||
|  |             <StatsClients /> | ||||||
|  |           </Route> | ||||||
|  |           <Route path="/:clientid" let:params> | ||||||
|  |             <StatsClientDetail {params} /> | ||||||
|  |           </Route> | ||||||
|  |         </Route> | ||||||
|         <Route path="/about"> |         <Route path="/about"> | ||||||
|           <About /> |           <About /> | ||||||
|         </Route> |         </Route> | ||||||
|   | |||||||
| @@ -1,6 +0,0 @@ | |||||||
| <style global> |  | ||||||
|   /*! @import */ |  | ||||||
|   @tailwind base; |  | ||||||
|   @tailwind components; |  | ||||||
|   @tailwind utilities; |  | ||||||
| </style> |  | ||||||
| @@ -1,28 +1,22 @@ | |||||||
| <script> | <script> | ||||||
|   import { ApiError, AuthService } from "@odit/lfk-client-js"; |   import { AuthService } from "@odit/lfk-client-js"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import "toastify-js/src/toastify.css"; |  | ||||||
|   import isEmail from "validator/es/lib/isEmail"; |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|  |  | ||||||
|   let reset_mail_sent = false; |   let reset_mail_sent = false; | ||||||
|   let usersEmail = ""; |   let usersEmail = ""; | ||||||
|   function reset() { |   function reset() { | ||||||
|     if (isEmail(usersEmail)) { |     if (isEmail(usersEmail)) { | ||||||
|  |       toast.loading($_("mail-validation-in-progress")); | ||||||
|       AuthService.authControllerGetResetToken("de", { email: usersEmail }) |       AuthService.authControllerGetResetToken("de", { email: usersEmail }) | ||||||
|         .then((resp) => { |         .then((resp) => { | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_("mail-validation-in-progress"), |  | ||||||
|             duration: 3500, |  | ||||||
|           }).showToast(); |  | ||||||
|           reset_mail_sent = true; |           reset_mail_sent = true; | ||||||
|         }) |         }) | ||||||
|         .catch((err) => {}); |         .catch((err) => {}); | ||||||
|     } else { |     } else { | ||||||
|       Toastify({ |       toast($_("invalid-mail-reset")); | ||||||
|         text: $_("invalid-mail-reset"), |  | ||||||
|         duration: 3500, |  | ||||||
|       }).showToast(); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
| @@ -32,17 +26,18 @@ | |||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|       <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|         {$_('application_name')} |         {$_("application_name")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> | ||||||
|         {$_('password-reset-mail-sent', { values: { usersEmail: usersEmail } })} |         {$_("password-reset-mail-sent", { values: { usersEmail: usersEmail } })} | ||||||
|       </p> |       </p> | ||||||
|       <div class="mt-6"> |       <div class="mt-6"> | ||||||
|         <div class="mt-6"> |         <div class="mt-6"> | ||||||
|           <a |           <a | ||||||
|             href="/" |             href="/" | ||||||
|             class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm"> |             class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm" | ||||||
|             {$_('goback')} |           > | ||||||
|  |             {$_("goback")} | ||||||
|           </a> |           </a> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
| @@ -53,25 +48,26 @@ | |||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|       <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|         {$_('application_name')} |         {$_("application_name")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-6 text-sm text-center text-gray-900"> |       <p class="mt-6 text-sm text-center text-gray-900"> | ||||||
|         {$_('forgot_password')} |         {$_("forgot_password")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> | ||||||
|         {$_('dont-panic-were-resetting-it')} |         {$_("dont-panic-were-resetting-it")} | ||||||
|       </p> |       </p> | ||||||
|       <div> |       <div> | ||||||
|         <div class="rounded-md shadow-sm"> |         <div class="rounded-md shadow-sm"> | ||||||
|           <div> |           <div> | ||||||
|             <input |             <input | ||||||
|               aria-label={$_('e-mail-adress')} |               aria-label={$_("e-mail-adress")} | ||||||
|               name="email" |               name="email" | ||||||
|               type="email" |               type="email" | ||||||
|               required="" |               required="" | ||||||
|               class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border text-gray-900 rounded-t-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" |               class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border text-gray-900 rounded-t-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" | ||||||
|               placeholder={$_('e-mail-adress')} |               placeholder={$_("e-mail-adress")} | ||||||
|               bind:value={usersEmail} /> |               bind:value={usersEmail} | ||||||
|  |             /> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
| @@ -79,19 +75,22 @@ | |||||||
|           <button |           <button | ||||||
|             on:click={reset} |             on:click={reset} | ||||||
|             type="submit" |             type="submit" | ||||||
|             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm" | ||||||
|  |           > | ||||||
|             <span class="absolute left-0 inset-y pl-3"> |             <span class="absolute left-0 inset-y pl-3"> | ||||||
|               <svg |               <svg | ||||||
|                 class="h-5 w-5 text-gray-500" |                 class="h-5 w-5 text-gray-500" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 viewBox="0 0 20 20"> |                 viewBox="0 0 20 20" | ||||||
|  |               > | ||||||
|                 <path |                 <path | ||||||
|                   fill-rule="evenodd" |                   fill-rule="evenodd" | ||||||
|                   d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" |                   d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" | ||||||
|                   clip-rule="evenodd" /> |                   clip-rule="evenodd" | ||||||
|  |                 /> | ||||||
|               </svg> |               </svg> | ||||||
|             </span> |             </span> | ||||||
|             {$_('reset-my-password')} |             {$_("reset-my-password")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|         <div class="mt-6"> |         <div class="mt-6"> | ||||||
| @@ -100,24 +99,30 @@ | |||||||
|               <div class="w-full border-t border-gray-300" /> |               <div class="w-full border-t border-gray-300" /> | ||||||
|             </div> |             </div> | ||||||
|             <div class="relative flex justify-center text-sm"> |             <div class="relative flex justify-center text-sm"> | ||||||
|               <span |               <span class="px-2 bg-gray-100 text-gray-500" | ||||||
|                 class="px-2 bg-gray-100 text-gray-500">{$_('dont-have-your-email-connected')}</span> |                 >{$_("dont-have-your-email-connected")}</span | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|           <span |           <span | ||||||
|             class="mt-2 text-sm px-2 bg-gray-100 text-gray-500 justify-center relative flex">{$_('cannot-reset-your-password-directly')}</span> |             class="mt-2 text-sm px-2 bg-gray-100 text-gray-500 justify-center relative flex" | ||||||
|  |             >{$_("cannot-reset-your-password-directly")}</span | ||||||
|  |           > | ||||||
|  |  | ||||||
|           <div class="mt-6"> |           <div class="mt-6"> | ||||||
|             <a |             <a | ||||||
|               href="mailto:lfk@odit.services" |               href="mailto:lfk@odit.services" | ||||||
|               class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm"> |               class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm" | ||||||
|               {$_('send-a-mail-to-lfk-odit-services')} |             > | ||||||
|  |               {$_("send-a-mail-to-lfk-odit-services")} | ||||||
|             </a> |             </a> | ||||||
|           </div> |           </div> | ||||||
|           <div class="mt-6"> |           <div class="mt-6"> | ||||||
|             <a |             <a | ||||||
|               href="/" |               href="/" | ||||||
|               class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm">{$_('goback')}</a> |               class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md text-gray-900 font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm" | ||||||
|  |               >{$_("goback")}</a | ||||||
|  |             > | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -5,7 +5,8 @@ | |||||||
|   store.init(); |   store.init(); | ||||||
|   import { OpenAPI, AuthService } from "@odit/lfk-client-js"; |   import { OpenAPI, AuthService } from "@odit/lfk-client-js"; | ||||||
|   import Footer from "../general/Footer.svelte"; |   import Footer from "../general/Footer.svelte"; | ||||||
|   import Toastify from "toastify-js"; |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   // ------ |   // ------ | ||||||
|   let username = config.default_username || ""; |   let username = config.default_username || ""; | ||||||
|   let password = config.default_password || ""; |   let password = config.default_password || ""; | ||||||
| @@ -19,11 +20,6 @@ | |||||||
|         OpenAPI.TOKEN = value.access_token; |         OpenAPI.TOKEN = value.access_token; | ||||||
|         const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1])); |         const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1])); | ||||||
|         store.login(value, jwtinfo); |         store.login(value, jwtinfo); | ||||||
|         Toastify({ |  | ||||||
|           text: $_("welcome_wavinghand"), |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
| @@ -32,45 +28,38 @@ | |||||||
|     // prevent login button spamming |     // prevent login button spamming | ||||||
|     if (last_loginclick_processed && is_blocked_by_autologin === false) { |     if (last_loginclick_processed && is_blocked_by_autologin === false) { | ||||||
|       last_loginclick_processed = false; |       last_loginclick_processed = false; | ||||||
|       Toastify({ |       toast.loading($_("login_is_checked")); | ||||||
|         text: $_("login_is_checked"), |       let postdata = {}; | ||||||
|         duration: 500, |       if (isEmail(username)) { | ||||||
|       }).showToast(); |         postdata = { | ||||||
|       AuthService.authControllerLogin({ |           email: username, | ||||||
|  |           password, | ||||||
|  |         }; | ||||||
|  |       } else { | ||||||
|  |         postdata = { | ||||||
|           username, |           username, | ||||||
|           password, |           password, | ||||||
|       }) |         }; | ||||||
|  |       } | ||||||
|  |       AuthService.authControllerLogin(postdata) | ||||||
|         .then(async (result) => { |         .then(async (result) => { | ||||||
|           await localForage.setItem("logindata", result); |           await localForage.setItem("logindata", result); | ||||||
|           OpenAPI.TOKEN = result.access_token; |           OpenAPI.TOKEN = result.access_token; | ||||||
|           const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1])); |           const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1])); | ||||||
|           store.login(result.access_token, jwtinfo); |           store.login(result.access_token, jwtinfo); | ||||||
|           location.replace("/"); |           location.replace("/"); | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_("welcome_wavinghand"), |  | ||||||
|             duration: 500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_("error_on_login"), |           toast.error($_("error_on_login")); | ||||||
|             duration: 500, |  | ||||||
|             backgroundColor: |  | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|           }).showToast(); |  | ||||||
|         }) |         }) | ||||||
|         .finally(() => { |         .finally(() => { | ||||||
|           last_loginclick_processed = true; |           last_loginclick_processed = true; | ||||||
|         }); |         }); | ||||||
|       // last login was not processed yet |       // last login was not processed yet | ||||||
|     } else { |     } else { | ||||||
|       Toastify({ |       toast($_("please-wait-a-moment-your-login-is-still-being-processed")); | ||||||
|         text: "chill...", |  | ||||||
|         duration: 1500, |  | ||||||
|         backgroundColor: |  | ||||||
|           "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|       }).showToast(); |  | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|   function handleKeydown(e) { |   function handleKeydown(e) { | ||||||
| @@ -81,34 +70,37 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <div | <div | ||||||
|   class="min-h-screen flex items-center justify-center bg-gray-100 text-gray-900"> |   class="min-h-screen flex items-center justify-center bg-gray-100 text-gray-900" | ||||||
|  | > | ||||||
|   <div class="max-w-md w-full py-12 px-6" role="main"> |   <div class="max-w-md w-full py-12 px-6" role="main"> | ||||||
|     <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |     <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|     <p class="mt-6 text-lg text-center font-bold">{$_('application_name')}</p> |     <p class="mt-6 text-xl text-center font-bold">{$_("application_name")}</p> | ||||||
|     <p class="mt-6 text-sm text-center">{$_('log_in_to_your_account')}</p> |     <p class="mt-2 mb-6 text-sm text-center">{$_("log_in_to_your_account")}</p> | ||||||
|     <div> |     <div> | ||||||
|       <div class="rounded-md shadow-sm"> |       <div class="rounded-md shadow-sm"> | ||||||
|         <div> |         <div> | ||||||
|           <!-- svelte-ignore a11y-autofocus --> |           <!-- svelte-ignore a11y-autofocus --> | ||||||
|           <input |           <input | ||||||
|             autofocus |             autofocus | ||||||
|             aria-label={$_('email_address_or_username')} |             aria-label={$_("email_address_or_username")} | ||||||
|             type="text" |             type="text" | ||||||
|             required="" |             required="" | ||||||
|             class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border rounded-t-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" |             class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border rounded-t-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" | ||||||
|             on:keydown={handleKeydown} |             on:keydown={handleKeydown} | ||||||
|             placeholder={$_('email_address_or_username')} |             placeholder={$_("email_address_or_username")} | ||||||
|             bind:value={username} /> |             bind:value={username} | ||||||
|  |           /> | ||||||
|         </div> |         </div> | ||||||
|         <div class="-mt-px relative"> |         <div class="-mt-px relative"> | ||||||
|           <input |           <input | ||||||
|             aria-label={$_('password')} |             aria-label={$_("password")} | ||||||
|             type="password" |             type="password" | ||||||
|             required="" |             required="" | ||||||
|             bind:value={password} |             bind:value={password} | ||||||
|             class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border rounded-b-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" |             class="border-gray-300 placeholder-gray-500 appearance-none rounded-none relative block w-full px-3 py-2 border rounded-b-md focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" | ||||||
|             on:keydown={handleKeydown} |             on:keydown={handleKeydown} | ||||||
|             placeholder={$_('password')} /> |             placeholder={$_("password")} | ||||||
|  |           /> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
| @@ -116,29 +108,33 @@ | |||||||
|         <button |         <button | ||||||
|           on:click={login} |           on:click={login} | ||||||
|           type="submit" |           type="submit" | ||||||
|           class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |           class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm" | ||||||
|  |         > | ||||||
|           <span class="absolute left-0 inset-y pl-3"> |           <span class="absolute left-0 inset-y pl-3"> | ||||||
|             <svg |             <svg | ||||||
|               class="h-5 w-5 text-gray-500" |               class="h-5 w-5 text-gray-500" | ||||||
|               fill="currentColor" |               fill="currentColor" | ||||||
|               viewBox="0 0 20 20"> |               viewBox="0 0 20 20" | ||||||
|  |             > | ||||||
|               <path |               <path | ||||||
|                 fill-rule="evenodd" |                 fill-rule="evenodd" | ||||||
|                 d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" |                 d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" | ||||||
|                 clip-rule="evenodd" /> |                 clip-rule="evenodd" | ||||||
|  |               /> | ||||||
|             </svg> |             </svg> | ||||||
|           </span> |           </span> | ||||||
|           {$_('log_in')} |           {$_("log_in")} | ||||||
|         </button> |         </button> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <div class="mt-2"> |     <!-- <div class="mt-2"> | ||||||
|       <a |       <a | ||||||
|         href="/forgot_password" |         href="/forgot_password" | ||||||
|         class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm"> |         class="block w-full text-center py-2 px-3 border border-gray-300 rounded-md font-medium hover:border-gray-400 focus:outline-none focus:border-gray-400 sm:text-sm" | ||||||
|         {$_('forgot_password')} |       > | ||||||
|  |         {$_("forgot_password")} | ||||||
|       </a> |       </a> | ||||||
|     </div> |     </div> --> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
| <Footer /> | <Footer /> | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								src/components/auth/PasswordStrength.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/components/auth/PasswordStrength.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | <script context="module"> | ||||||
|  |   import { passwordStrength } from "check-password-strength"; | ||||||
|  |   export function password_strong_enough(password_change) { | ||||||
|  |     let strength = passwordStrength(password_change); | ||||||
|  |     return ( | ||||||
|  |       strength?.contains.includes("lowercase") && | ||||||
|  |       strength?.contains.includes("uppercase") && | ||||||
|  |       strength?.contains.includes("number") && | ||||||
|  |       strength?.length > 9 | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   export function password_strong_enough_and_equal( | ||||||
|  |     password_change, | ||||||
|  |     password_confirm | ||||||
|  |   ) { | ||||||
|  |     return ( | ||||||
|  |       password_strong_enough(password_change) && | ||||||
|  |       password_change === password_confirm | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  |   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | ||||||
|  |   import { passwordStrength as Strength } from "check-password-strength"; | ||||||
|  |   export let password_change; | ||||||
|  |   export let password_confirm; | ||||||
|  |  | ||||||
|  |   $: strength = Strength(password_change); | ||||||
|  |   $: passwords_match = | ||||||
|  |     !password_confirm || password_confirm === password_change; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="ml-4"> | ||||||
|  |   <ul class="list-disc font-medium tracking-wide text-red-500 text-xs"> | ||||||
|  |     {#if !strength.contains.includes("lowercase")} | ||||||
|  |       <li>{$_("must-contain-a-lowercase-letter")}</li> | ||||||
|  |     {/if} | ||||||
|  |     {#if !strength.contains.includes("uppercase")} | ||||||
|  |       <li>{$_("must-contain-a-uppercase-letter")}</li> | ||||||
|  |     {/if} | ||||||
|  |     {#if !strength.contains.includes("number")} | ||||||
|  |       <li>{$_("must-contain-a-number")}</li> | ||||||
|  |     {/if} | ||||||
|  |     {#if !(strength.length > 9)} | ||||||
|  |       <li>{$_("must-be-at-least-10-characters-long")}</li> | ||||||
|  |     {/if} | ||||||
|  |     {#if !(passwords_match == true)} | ||||||
|  |       <li>{$_("passwords-dont-match")}</li> | ||||||
|  |     {/if} | ||||||
|  |   </ul> | ||||||
|  | </div> | ||||||
| @@ -1,33 +1,30 @@ | |||||||
| <script> | <script> | ||||||
|   import { AuthService } from "@odit/lfk-client-js"; |   import { AuthService } from "@odit/lfk-client-js"; | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import Toastify from "toastify-js"; |   import toast from "svelte-french-toast"; | ||||||
|   import "toastify-js/src/toastify.css"; |   import PasswordStrength, { | ||||||
|  |     password_strong_enough, | ||||||
|  |   } from "../auth/PasswordStrength.svelte"; | ||||||
|   let state = "reset_in_progress"; |   let state = "reset_in_progress"; | ||||||
|   let password = ""; |   let password = ""; | ||||||
|   export let params; |   export let params; | ||||||
|   function set_new_password() { |   function set_new_password() { | ||||||
|     if (password.trim() !== "") { |     if (password.trim() !== "") { | ||||||
|         Toastify({ |       toast.loading($_("password-reset-in-progress")); | ||||||
|           text: $_('password-reset-in-progress'), |       AuthService.authControllerResetPassword(atob(params.resetkey), { | ||||||
|           duration: 3500, |         password, | ||||||
|         }).showToast(); |       }) | ||||||
|         AuthService.authControllerResetPassword(atob(params.resetkey),{ password }) |  | ||||||
|         .then((resp) => { |         .then((resp) => { | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_('password-reset-successful'), |           toast($_("password-reset-successful")); | ||||||
|             duration: 3500, |  | ||||||
|           }).showToast(); |  | ||||||
|           state = "reset_success"; |           state = "reset_success"; | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           state = "reset_error"; |           state = "reset_error"; | ||||||
|         }); |         }); | ||||||
|     } else { |     } else { | ||||||
|       Toastify({ |       toast.dismiss(); | ||||||
|         text: $_('please-provide-a-password'), |       toast.error($_("please-provide-a-password")); | ||||||
|         duration: 3500, |  | ||||||
|       }).showToast(); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
| @@ -37,20 +34,21 @@ | |||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|       <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|         {$_('application_name')} |         {$_("application_name")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> | ||||||
|         {$_('successful-password-reset')} |         {$_("successful-password-reset")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> | ||||||
|         {$_('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")} | ||||||
|       </p> |       </p> | ||||||
|       <div class="mt-6"> |       <div class="mt-6"> | ||||||
|         <div class="mt-6"> |         <div class="mt-6"> | ||||||
|           <a |           <a | ||||||
|             href="/login/" |             href="/login/" | ||||||
|             class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm" | ||||||
|             {$_('go-to-login')} |           > | ||||||
|  |             {$_("go-to-login")} | ||||||
|           </a> |           </a> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
| @@ -61,20 +59,21 @@ | |||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|       <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|       {$_('application_name')} |         {$_("application_name")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold"> | ||||||
|       {$_('password-reset-failed')} |         {$_("password-reset-failed")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> |       <p class="mt-2 mb-2 text-sm text-center text-gray-900"> | ||||||
|       {$_('please-request-a-new-reset-mail')} |         {$_("please-request-a-new-reset-mail")} | ||||||
|       </p> |       </p> | ||||||
|       <div class="mt-6"> |       <div class="mt-6"> | ||||||
|         <div class="mt-6"> |         <div class="mt-6"> | ||||||
|           <a |           <a | ||||||
|             href="/forgot_password/" |             href="/forgot_password/" | ||||||
|           class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm" | ||||||
|           {$_('request-a-new-reset-mail')} |           > | ||||||
|  |             {$_("request-a-new-reset-mail")} | ||||||
|           </a> |           </a> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
| @@ -85,42 +84,49 @@ | |||||||
|     <div class="max-w-md w-full py-12 px-6"> |     <div class="max-w-md w-full py-12 px-6"> | ||||||
|       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> |       <img style="height:10rem;" class="mx-auto" src="/lfk-logo.png" alt="" /> | ||||||
|       <p class="mt-6 text-lg text-center font-bold text-gray-900"> |       <p class="mt-6 text-lg text-center font-bold text-gray-900"> | ||||||
|         {$_('application_name')} |         {$_("application_name")} | ||||||
|       </p> |       </p> | ||||||
|       <p class="mt-2 mb-4 text-md text-center text-gray-900"> |       <p class="mt-2 mb-4 text-md text-center text-gray-900"> | ||||||
|         {$_('reset-password')} |         {$_("reset-password")} | ||||||
|       </p> |       </p> | ||||||
|       <div> |       <div> | ||||||
|         <div class="rounded-md shadow-sm"> |         <div class="rounded-md shadow-sm"> | ||||||
|           <div> |           <div> | ||||||
|             <input |             <input | ||||||
|               aria-label={$_('new-password')} |               aria-label={$_("new-password")} | ||||||
|               name="password" |               name="password" | ||||||
|               type="password" |               type="password" | ||||||
|               required="" |               required="" | ||||||
|               class="border-gray-300 placeholder-gray-500 appearance-none rounded-md relative block w-full px-3 py-2 border text-gray-900 focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" |               class="border-gray-300 placeholder-gray-500 appearance-none rounded-md relative block w-full px-3 py-2 border text-gray-900 focus:outline-none focus:shadow-outline-blue focus:border-blue-300 focus:z-10 sm:text-sm" | ||||||
|               placeholder={$_('new-password')} |               placeholder={$_("new-password")} | ||||||
|               bind:value={password} /> |               bind:value={password} | ||||||
|  |             /> | ||||||
|           </div> |           </div> | ||||||
|  |           <PasswordStrength bind:password_change={password} /> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <div class="mt-5"> |         <div class="mt-5"> | ||||||
|           <button |           <button | ||||||
|             on:click={set_new_password} |             on:click={set_new_password} | ||||||
|  |             disabled={!password_strong_enough(password)} | ||||||
|  |             class:opacity-50={!password_strong_enough(password)} | ||||||
|             type="submit" |             type="submit" | ||||||
|             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"> |             class="relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm" | ||||||
|  |           > | ||||||
|             <span class="absolute left-0 inset-y pl-3"> |             <span class="absolute left-0 inset-y pl-3"> | ||||||
|               <svg |               <svg | ||||||
|                 class="h-5 w-5 text-gray-500" |                 class="h-5 w-5 text-gray-500" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 viewBox="0 0 20 20"> |                 viewBox="0 0 20 20" | ||||||
|  |               > | ||||||
|                 <path |                 <path | ||||||
|                   fill-rule="evenodd" |                   fill-rule="evenodd" | ||||||
|                   d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" |                   d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" | ||||||
|                   clip-rule="evenodd" /> |                   clip-rule="evenodd" | ||||||
|  |                 /> | ||||||
|               </svg> |               </svg> | ||||||
|             </span> |             </span> | ||||||
|             {$_('reset-my-password')} |             {$_("reset-my-password")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,274 +0,0 @@ | |||||||
| <!-- |  | ||||||
|   This example requires Tailwind CSS v2.0+  |  | ||||||
|    |  | ||||||
|   This example requires some changes to your config: |  | ||||||
|    |  | ||||||
|   ``` |  | ||||||
|   // tailwind.config.js |  | ||||||
|   module.exports = { |  | ||||||
|     // ... |  | ||||||
|     plugins: [ |  | ||||||
|       // ... |  | ||||||
|       require('@tailwindcss/forms'), |  | ||||||
|     ] |  | ||||||
|   } |  | ||||||
|   ``` |  | ||||||
| --> |  | ||||||
| <div> |  | ||||||
|   <div class="md:grid md:grid-cols-3 md:gap-6"> |  | ||||||
|     <div class="md:col-span-1"> |  | ||||||
|       <div class="px-4 sm:px-0"> |  | ||||||
|         <h3 class="text-lg font-medium leading-6 text-gray-900">Profile</h3> |  | ||||||
|         <p class="mt-1 text-sm text-gray-600"> |  | ||||||
|           This information will be displayed publicly so be careful what you share. |  | ||||||
|         </p> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="mt-5 md:mt-0 md:col-span-2"> |  | ||||||
|       <form action="#" method="POST"> |  | ||||||
|         <div class="shadow sm:rounded-md sm:overflow-hidden"> |  | ||||||
|           <div class="px-4 py-5 bg-white space-y-6 sm:p-6"> |  | ||||||
|             <div class="grid grid-cols-3 gap-6"> |  | ||||||
|               <div class="col-span-3 sm:col-span-2"> |  | ||||||
|                 <label for="company_website" class="block text-sm font-medium text-gray-700"> |  | ||||||
|                   Website |  | ||||||
|                 </label> |  | ||||||
|                 <div class="mt-1 flex rounded-md shadow-sm"> |  | ||||||
|                   <span class="inline-flex items-center px-3 rounded-l-md border border-r-0 border-gray-300 bg-gray-50 text-gray-500 text-sm"> |  | ||||||
|                     http:// |  | ||||||
|                   </span> |  | ||||||
|                   <input type="text" name="company_website" id="company_website" class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-r-md sm:text-sm border-gray-300" placeholder="www.example.com"> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|  |  | ||||||
|             <div> |  | ||||||
|               <label for="about" class="block text-sm font-medium text-gray-700"> |  | ||||||
|                 About |  | ||||||
|               </label> |  | ||||||
|               <div class="mt-1"> |  | ||||||
|                 <textarea id="about" name="about" rows="3" class="shadow-sm focus:ring-indigo-500 focus:border-indigo-500 mt-1 block w-full sm:text-sm border-gray-300 rounded-md" placeholder="you@example.com"></textarea> |  | ||||||
|               </div> |  | ||||||
|               <p class="mt-2 text-sm text-gray-500"> |  | ||||||
|                 Brief description for your profile. URLs are hyperlinked. |  | ||||||
|               </p> |  | ||||||
|             </div> |  | ||||||
|  |  | ||||||
|             <div> |  | ||||||
|               <!-- svelte-ignore a11y-label-has-associated-control --> |  | ||||||
|               <label class="block text-sm font-medium text-gray-700"> |  | ||||||
|                 Photo |  | ||||||
|               </label> |  | ||||||
|               <div class="mt-2 flex items-center"> |  | ||||||
|                 <span class="inline-block h-12 w-12 rounded-full overflow-hidden bg-gray-100"> |  | ||||||
|                   <svg class="h-full w-full text-gray-300" fill="currentColor" viewBox="0 0 24 24"> |  | ||||||
|                     <path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z" /> |  | ||||||
|                   </svg> |  | ||||||
|                 </span> |  | ||||||
|                 <button type="button" class="ml-5 bg-white py-2 px-3 border border-gray-300 rounded-md shadow-sm text-sm leading-4 font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"> |  | ||||||
|                   Change |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|  |  | ||||||
|             <div> |  | ||||||
|               <!-- svelte-ignore a11y-label-has-associated-control --> |  | ||||||
|               <label class="block text-sm font-medium text-gray-700"> |  | ||||||
|                 Cover photo |  | ||||||
|               </label> |  | ||||||
|               <div class="mt-2 flex justify-center px-6 pt-5 pb-6 border-2 border-gray-300 border-dashed rounded-md"> |  | ||||||
|                 <div class="space-y-1 text-center"> |  | ||||||
|                   <svg class="mx-auto h-12 w-12 text-gray-400" stroke="currentColor" fill="none" viewBox="0 0 48 48" aria-hidden="true"> |  | ||||||
|                     <path d="M28 8H12a4 4 0 00-4 4v20m32-12v8m0 0v8a4 4 0 01-4 4H12a4 4 0 01-4-4v-4m32-4l-3.172-3.172a4 4 0 00-5.656 0L28 28M8 32l9.172-9.172a4 4 0 015.656 0L28 28m0 0l4 4m4-24h8m-4-4v8m-12 4h.02" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" /> |  | ||||||
|                   </svg> |  | ||||||
|                   <div class="flex text-sm text-gray-600"> |  | ||||||
|                     <label for="file-upload" class="relative cursor-pointer bg-white rounded-md font-medium text-indigo-600 hover:text-indigo-500 focus-within:outline-none focus-within:ring-2 focus-within:ring-offset-2 focus-within:ring-indigo-500"> |  | ||||||
|                       <span>Upload a file</span> |  | ||||||
|                       <input id="file-upload" name="file-upload" type="file" class="sr-only"> |  | ||||||
|                     </label> |  | ||||||
|                     <p class="pl-1">or drag and drop</p> |  | ||||||
|                   </div> |  | ||||||
|                   <p class="text-xs text-gray-500"> |  | ||||||
|                     PNG, JPG, GIF up to 10MB |  | ||||||
|                   </p> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|           <div class="px-4 py-3 bg-gray-50 text-right sm:px-6"> |  | ||||||
|             <button type="submit" class="inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"> |  | ||||||
|               Save |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="hidden sm:block" aria-hidden="true"> |  | ||||||
|   <div class="py-5"> |  | ||||||
|     <div class="border-t border-gray-200"></div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="mt-10 sm:mt-0"> |  | ||||||
|   <div class="md:grid md:grid-cols-3 md:gap-6"> |  | ||||||
|     <div class="md:col-span-1"> |  | ||||||
|       <div class="px-4 sm:px-0"> |  | ||||||
|         <h3 class="text-lg font-medium leading-6 text-gray-900">Personal Information</h3> |  | ||||||
|         <p class="mt-1 text-sm text-gray-600"> |  | ||||||
|           Use a permanent address where you can receive mail. |  | ||||||
|         </p> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="mt-5 md:mt-0 md:col-span-2"> |  | ||||||
|       <form action="#" method="POST"> |  | ||||||
|         <div class="shadow overflow-hidden sm:rounded-md"> |  | ||||||
|           <div class="px-4 py-5 bg-white sm:p-6"> |  | ||||||
|             <div class="grid grid-cols-6 gap-6"> |  | ||||||
|               <div class="col-span-6 sm:col-span-3"> |  | ||||||
|                 <label for="first_name" class="block text-sm font-medium text-gray-700">First name</label> |  | ||||||
|                 <input type="text" name="first_name" id="first_name" autocomplete="given-name" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-3"> |  | ||||||
|                 <label for="last_name" class="block text-sm font-medium text-gray-700">Last name</label> |  | ||||||
|                 <input type="text" name="last_name" id="last_name" autocomplete="family-name" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-4"> |  | ||||||
|                 <label for="email_address" class="block text-sm font-medium text-gray-700">Email address</label> |  | ||||||
|                 <input type="text" name="email_address" id="email_address" autocomplete="email" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-3"> |  | ||||||
|                 <label for="country" class="block text-sm font-medium text-gray-700">Country / Region</label> |  | ||||||
|                 <select id="country" name="country" autocomplete="country" class="mt-1 block w-full py-2 px-3 border border-gray-300 bg-white rounded-md shadow-sm focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm"> |  | ||||||
|                   <option>United States</option> |  | ||||||
|                   <option>Canada</option> |  | ||||||
|                   <option>Mexico</option> |  | ||||||
|                 </select> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6"> |  | ||||||
|                 <label for="street_address" class="block text-sm font-medium text-gray-700">Street address</label> |  | ||||||
|                 <input type="text" name="street_address" id="street_address" autocomplete="street-address" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-6 lg:col-span-2"> |  | ||||||
|                 <label for="city" class="block text-sm font-medium text-gray-700">City</label> |  | ||||||
|                 <input type="text" name="city" id="city" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-3 lg:col-span-2"> |  | ||||||
|                 <label for="state" class="block text-sm font-medium text-gray-700">State / Province</label> |  | ||||||
|                 <input type="text" name="state" id="state" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|  |  | ||||||
|               <div class="col-span-6 sm:col-span-3 lg:col-span-2"> |  | ||||||
|                 <label for="postal_code" class="block text-sm font-medium text-gray-700">ZIP / Postal</label> |  | ||||||
|                 <input type="text" name="postal_code" id="postal_code" autocomplete="postal-code" class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 rounded-md"> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|           <div class="px-4 py-3 bg-gray-50 text-right sm:px-6"> |  | ||||||
|             <button type="submit" class="inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"> |  | ||||||
|               Save |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="hidden sm:block" aria-hidden="true"> |  | ||||||
|   <div class="py-5"> |  | ||||||
|     <div class="border-t border-gray-200"></div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="mt-10 sm:mt-0"> |  | ||||||
|   <div class="md:grid md:grid-cols-3 md:gap-6"> |  | ||||||
|     <div class="md:col-span-1"> |  | ||||||
|       <div class="px-4 sm:px-0"> |  | ||||||
|         <h3 class="text-lg font-medium leading-6 text-gray-900">Notifications</h3> |  | ||||||
|         <p class="mt-1 text-sm text-gray-600"> |  | ||||||
|           Decide which communications you'd like to receive and how. |  | ||||||
|         </p> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="mt-5 md:mt-0 md:col-span-2"> |  | ||||||
|       <form action="#" method="POST"> |  | ||||||
|         <div class="shadow overflow-hidden sm:rounded-md"> |  | ||||||
|           <div class="px-4 py-5 bg-white space-y-6 sm:p-6"> |  | ||||||
|             <fieldset> |  | ||||||
|               <legend class="text-base font-medium text-gray-900">By Email</legend> |  | ||||||
|               <div class="mt-4 space-y-4"> |  | ||||||
|                 <div class="flex items-start"> |  | ||||||
|                   <div class="flex items-center h-5"> |  | ||||||
|                     <input id="comments" name="comments" type="checkbox" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"> |  | ||||||
|                   </div> |  | ||||||
|                   <div class="ml-3 text-sm"> |  | ||||||
|                     <label for="comments" class="font-medium text-gray-700">Comments</label> |  | ||||||
|                     <p class="text-gray-500">Get notified when someones posts a comment on a posting.</p> |  | ||||||
|                   </div> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="flex items-start"> |  | ||||||
|                   <div class="flex items-center h-5"> |  | ||||||
|                     <input id="candidates" name="candidates" type="checkbox" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"> |  | ||||||
|                   </div> |  | ||||||
|                   <div class="ml-3 text-sm"> |  | ||||||
|                     <label for="candidates" class="font-medium text-gray-700">Candidates</label> |  | ||||||
|                     <p class="text-gray-500">Get notified when a candidate applies for a job.</p> |  | ||||||
|                   </div> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="flex items-start"> |  | ||||||
|                   <div class="flex items-center h-5"> |  | ||||||
|                     <input id="offers" name="offers" type="checkbox" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"> |  | ||||||
|                   </div> |  | ||||||
|                   <div class="ml-3 text-sm"> |  | ||||||
|                     <label for="offers" class="font-medium text-gray-700">Offers</label> |  | ||||||
|                     <p class="text-gray-500">Get notified when a candidate accepts or rejects an offer.</p> |  | ||||||
|                   </div> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             </fieldset> |  | ||||||
|             <fieldset> |  | ||||||
|               <div> |  | ||||||
|                 <legend class="text-base font-medium text-gray-900">Push Notifications</legend> |  | ||||||
|                 <p class="text-sm text-gray-500">These are delivered via SMS to your mobile phone.</p> |  | ||||||
|               </div> |  | ||||||
|               <div class="mt-4 space-y-4"> |  | ||||||
|                 <div class="flex items-center"> |  | ||||||
|                   <input id="push_everything" name="push_notifications" type="radio" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300"> |  | ||||||
|                   <label for="push_everything" class="ml-3 block text-sm font-medium text-gray-700"> |  | ||||||
|                     Everything |  | ||||||
|                   </label> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="flex items-center"> |  | ||||||
|                   <input id="push_email" name="push_notifications" type="radio" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300"> |  | ||||||
|                   <label for="push_email" class="ml-3 block text-sm font-medium text-gray-700"> |  | ||||||
|                     Same as email |  | ||||||
|                   </label> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="flex items-center"> |  | ||||||
|                   <input id="push_nothing" name="push_notifications" type="radio" class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300"> |  | ||||||
|                   <label for="push_nothing" class="ml-3 block text-sm font-medium text-gray-700"> |  | ||||||
|                     No push notifications |  | ||||||
|                   </label> |  | ||||||
|                 </div> |  | ||||||
|               </div> |  | ||||||
|             </fieldset> |  | ||||||
|           </div> |  | ||||||
|           <div class="px-4 py-3 bg-gray-50 text-right sm:px-6"> |  | ||||||
|             <button type="submit" class="inline-flex justify-center py-2 px-4 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"> |  | ||||||
|               Save |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| @@ -7,16 +7,19 @@ | |||||||
|     <div class="w-full bg-white flex items-center justify-center"> |     <div class="w-full bg-white flex items-center justify-center"> | ||||||
|       <div class="max-w-sm m-8"> |       <div class="max-w-sm m-8"> | ||||||
|         <div class="text-black text-5xl md:text-15xl font-black"> |         <div class="text-black text-5xl md:text-15xl font-black"> | ||||||
|           {$_('internal-error')} |           {$_("internal-error")} | ||||||
|         </div> |         </div> | ||||||
|         <div class="w-16 h-1 bg-purple-light my-3 md:my-6" /> |         <div class="w-16 h-1 bg-purple-light my-3 md:my-6" /> | ||||||
|         <p |         <p | ||||||
|           class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal"> |           class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal" | ||||||
|           {$_('generic-ui-logic-error')} |         > | ||||||
|  |           {$_("generic-ui-logic-error")} | ||||||
|         </p> |         </p> | ||||||
|         <a |         <a | ||||||
|           href="/" |           href="/" | ||||||
|           class="bg-transparent text-grey-darkest font-bold uppercase tracking-wide py-3 px-6 border-2 border-grey-light hover:border-grey rounded-lg">{$_('goback')}</a> |           class="bg-transparent text-grey-darkest font-bold uppercase tracking-wide py-3 px-6 border-2 border-grey-light hover:border-grey rounded-lg" | ||||||
|  |           >{$_("goback")}</a | ||||||
|  |         > | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
|  |  | ||||||
| <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> | ||||||
|   | |||||||
| @@ -1,24 +1,25 @@ | |||||||
| export function getlang(langkeys) { | export function getlang(langkeys) { | ||||||
|   return { |   return { | ||||||
|     search: { |     search: { | ||||||
| 			placeholder: langkeys.search |       placeholder: langkeys.search, | ||||||
|     }, |     }, | ||||||
|     sort: { |     sort: { | ||||||
|       sortAsc: langkeys.sort_column_ascending, |       sortAsc: langkeys.sort_column_ascending, | ||||||
| 			sortDesc: langkeys.sort_column_descending |       sortDesc: langkeys.sort_column_descending, | ||||||
|     }, |     }, | ||||||
|     pagination: { |     pagination: { | ||||||
|       previous: langkeys.previous, |       previous: langkeys.previous, | ||||||
|       next: langkeys.next, |       next: langkeys.next, | ||||||
| 			navigate: (page, pages) => `${langkeys.page} ${page} ${langkeys.of} ${pages}`, |       navigate: (page, pages) => | ||||||
|  |         `${langkeys.page} ${page} ${langkeys.of} ${pages}`, | ||||||
|       page: (page) => `${langkeys.page} ${page}`, |       page: (page) => `${langkeys.page} ${page}`, | ||||||
|       showing: langkeys.showing, |       showing: langkeys.showing, | ||||||
|       of: langkeys.of, |       of: langkeys.of, | ||||||
|       to: langkeys.to, |       to: langkeys.to, | ||||||
| 			results: langkeys.records |       results: langkeys.records, | ||||||
|     }, |     }, | ||||||
|     loading: langkeys.loading, |     loading: langkeys.loading, | ||||||
|     noRecordsFound: langkeys.no_matching_records_found, |     noRecordsFound: langkeys.no_matching_records_found, | ||||||
| 		error: langkeys.an_error_happened_while_fetching_the_data |     error: langkeys.an_error_happened_while_fetching_the_data, | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								src/components/base/importfixes.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/components/base/importfixes.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | <!-- | ||||||
|  |     Temporary tailwind import fixes for classes that wouldn't be directly used otherwise.  | ||||||
|  |     Or as others may call it: Real big bullshit time. | ||||||
|  |     Issue: https://git.odit.services/lfk/frontend/issues/136 | ||||||
|  |  --> | ||||||
|  | <div class="opacity-50" /> | ||||||
| @@ -1,10 +1,10 @@ | |||||||
| /** Dispatch event on click outside of node */ | /** Dispatch event on click outside of node */ | ||||||
| export function clickOutside(node) { | export function clickOutside(node) { | ||||||
|   const handleClick = (event) => { |   const handleClick = (event) => { | ||||||
| 		if (event.target.getAttribute('data-id') === 'modal_backdrop') { |     if (event.target.getAttribute("data-id") === "modal_backdrop") { | ||||||
| 			node.dispatchEvent(new CustomEvent('click_outside', node)); |       node.dispatchEvent(new CustomEvent("click_outside", node)); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 	document.removeEventListener('click', handleClick, true); |   document.removeEventListener("click", handleClick, true); | ||||||
| 	document.addEventListener('click', handleClick, true); |   document.addEventListener("click", handleClick, true); | ||||||
| } | } | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										218
									
								
								src/components/cards/AddCardBulkModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										218
									
								
								src/components/cards/AddCardBulkModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,218 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |  | ||||||
|  |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|  |   import DocumentServer from "../pdf_generation/DocumentServer"; | ||||||
|  |   export let bulk_modal_open; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   const documentServer = new DocumentServer(config.baseurl_documentserver,config.documentserver_key); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   $: card_count = 0; | ||||||
|  |   $: is_card_count_valid = card_count > 0; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   $: createbtnenabled = is_card_count_valid; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         bulk_modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit_with_print(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit_without_print() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       toast.loading($_("creating-blanco-cards")); | ||||||
|  |       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true) | ||||||
|  |         .then((result) => { | ||||||
|  |           bulk_modal_open = false; | ||||||
|  |           // | ||||||
|  |           toast.dismiss(); | ||||||
|  |           toast.success($_("created-blanco-cards")); | ||||||
|  |           dispatch("created", { cards: result }); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function submit_with_print() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       toast.dismiss(); | ||||||
|  |       toast.loading($_("creating-blanco-cards")); | ||||||
|  |       RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true) | ||||||
|  |         .then((result) => { | ||||||
|  |           bulk_modal_open = false; | ||||||
|  |           // | ||||||
|  |           toast.dismiss(); | ||||||
|  |           toast.success($_("created-blanco-cards")); | ||||||
|  |           toast.loading($_("generating-pdf")); | ||||||
|  |           dispatch("created", { cards: result }); | ||||||
|  |           documentServer.generateCards(result, "de") | ||||||
|  |             .then((blob) => { | ||||||
|  |               const url = window.URL.createObjectURL(blob); | ||||||
|  |               let a = document.createElement("a"); | ||||||
|  |               a.href = url; | ||||||
|  |               a.download = "Bulkcards.pdf"; | ||||||
|  |               document.body.appendChild(a); | ||||||
|  |               a.click(); | ||||||
|  |               a.remove(); | ||||||
|  |               toast.dismiss(); | ||||||
|  |               toast.success($_("pdf-successfully-generated")); | ||||||
|  |             }) | ||||||
|  |             .catch((err) => { | ||||||
|  |               console.error(err); | ||||||
|  |             }); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if bulk_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       bulk_modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-2xl sm:w-full" | ||||||
|  |         role="dialog" | ||||||
|  |         aria-modal="true" | ||||||
|  |         aria-labelledby="modal-headline" | ||||||
|  |       > | ||||||
|  |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w- rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   fill="currentColor" | ||||||
|  |                   d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:mt-0 sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("create-bulk-blanco-cards")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_( | ||||||
|  |                     "just-enter-how-many-you-want-and-the-system-will-create-them" | ||||||
|  |                   )} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="amount" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("amount")}</label | ||||||
|  |                   > | ||||||
|  |                   <div class="mt-1 flex rounded-md shadow-sm"> | ||||||
|  |                     <input | ||||||
|  |                       autocomplete="off" | ||||||
|  |                       class:border-red-500={!is_card_count_valid} | ||||||
|  |                       class:focus:border-red-500={!is_card_count_valid} | ||||||
|  |                       class:focus:ring-red-500={!is_card_count_valid} | ||||||
|  |                       bind:value={card_count} | ||||||
|  |                       type="number" | ||||||
|  |                       step="1" | ||||||
|  |                       name="amount" | ||||||
|  |                       class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 p-2" | ||||||
|  |                       placeholder="400" | ||||||
|  |                     /> | ||||||
|  |                     <span | ||||||
|  |                       class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500 text-sm" | ||||||
|  |                       >{$_("cards")}</span | ||||||
|  |                     > | ||||||
|  |                   </div> | ||||||
|  |                   {#if !is_card_count_valid} | ||||||
|  |                     <span | ||||||
|  |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|  |                     > | ||||||
|  |                       {$_("you-must-create-at-least-one-card-or-cancel")} | ||||||
|  |                     </span> | ||||||
|  |                   {/if} | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit_with_print} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|  |           > | ||||||
|  |             {$_("create-and-generate-pdf")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             disabled={!createbtnenabled} | ||||||
|  |             class:opacity-50={!createbtnenabled} | ||||||
|  |             on:click={submit_without_print} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-400 text-base font-medium text-white hover:bg-gray-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|  |           > | ||||||
|  |             {$_("create-without-pdf")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               bulk_modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="mr-auto 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" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										191
									
								
								src/components/cards/AddCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								src/components/cards/AddCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,191 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |  | ||||||
|  |   import { RunnerCardService, RunnerService } from "@odit/lfk-client-js"; | ||||||
|  |   import Select from "svelte-select"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|  |   export let modal_open; | ||||||
|  |  | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   const getRunnerLabel = (option) => { | ||||||
|  |     if (option.middlename) { | ||||||
|  |       return option.firstname + " " + option.middlename + " " + option.lastname; | ||||||
|  |     } | ||||||
|  |     return option.firstname + " " + option.lastname; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const filterRunners = (label, filterText, option) => { | ||||||
|  |     if (filterText.startsWith("#")) { | ||||||
|  |       return option.value.id == parseInt(filterText.replace("#", "")); | ||||||
|  |     } | ||||||
|  |     return ( | ||||||
|  |       label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|  |       option.value.toString().startsWith(filterText.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: runner = 0; | ||||||
|  |   $: enabled = true; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |  | ||||||
|  |   let loading = true; | ||||||
|  |   let runners = []; | ||||||
|  |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|  |     runners = val.map((r) => { | ||||||
|  |       return { label: getRunnerLabel(r), value: r }; | ||||||
|  |     }); | ||||||
|  |     loading = false; | ||||||
|  |   }); | ||||||
|  |   $: createbtnenabled = true; | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       toast.loading($_("adding-card")); | ||||||
|  |       let postdata = { | ||||||
|  |         runner, | ||||||
|  |         enabled, | ||||||
|  |       }; | ||||||
|  |       RunnerCardService.runnerCardControllerPost(postdata) | ||||||
|  |         .then((result) => { | ||||||
|  |           runner = 0; | ||||||
|  |           modal_open = false; | ||||||
|  |           // | ||||||
|  |           toast.dismiss(); | ||||||
|  |           toast.success($_("card-added")); | ||||||
|  |           dispatch("created", { cards: [result] }); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <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:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("create-a-new-card")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_("you-can-provide-a-runner-but-you-dont-have-to")} | ||||||
|  |                   {$_( | ||||||
|  |                     "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button" | ||||||
|  |                   )} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="donor" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("runner")}</label | ||||||
|  |                   > | ||||||
|  |                   <Select | ||||||
|  |                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     itemFilter={(label, filterText, option) => | ||||||
|  |                       filterRunners(label, filterText, option)} | ||||||
|  |                     items={runners} | ||||||
|  |                     bind:loading | ||||||
|  |                     showChevron={!loading} | ||||||
|  |                     placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|  |                     noOptionsMessage={$_("no-runners-found")} | ||||||
|  |                     on:select={(selectedValue) => | ||||||
|  |                       (runner = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (runner = null)} | ||||||
|  |                   /> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <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" | ||||||
|  |           > | ||||||
|  |             {$_("create")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										200
									
								
								src/components/cards/CardDetailModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								src/components/cards/CardDetailModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,200 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |  | ||||||
|  |   import { RunnerCardService, RunnerService } from "@odit/lfk-client-js"; | ||||||
|  |   import Select from "svelte-select"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|  |   export let edit_modal_open; | ||||||
|  |   export let runner = {}; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   const getRunnerLabel = (option) => | ||||||
|  |     option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
|  |   const filterRunners = (label, filterText, option) => { | ||||||
|  |     if (filterText.startsWith("#")) { | ||||||
|  |       return option.value.id == parseInt(filterText.replace("#", "")); | ||||||
|  |     } | ||||||
|  |     return ( | ||||||
|  |       label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|  |       option.value.toString().startsWith(filterText.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   function focus(el) { | ||||||
|  |     el.focus(); | ||||||
|  |   } | ||||||
|  |   $: runners = []; | ||||||
|  |   $: enabled = true; | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   RunnerService.runnerControllerGetAll().then((val) => { | ||||||
|  |     runners = val.map((r) => { | ||||||
|  |       return { label: getRunnerLabel(r), value: r }; | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |   $: createbtnenabled = !( | ||||||
|  |     JSON.stringify(editable) === JSON.stringify(original_data) | ||||||
|  |   ); | ||||||
|  |   (() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         edit_modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   })(); | ||||||
|  |   function submit() { | ||||||
|  |     if (processed_last_submit === true) { | ||||||
|  |       processed_last_submit = false; | ||||||
|  |       toast.loading($_("updating-card")); | ||||||
|  |       RunnerCardService.runnerCardControllerPut(original_data.id, editable) | ||||||
|  |         .then((result) => { | ||||||
|  |           runner = {}; | ||||||
|  |           editable = {}; | ||||||
|  |           original_data = {}; | ||||||
|  |           edit_modal_open = false; | ||||||
|  |           // | ||||||
|  |           toast.dismiss(); | ||||||
|  |           toast.success($_("card-updated")); | ||||||
|  |           dispatch("dataUpdated", { card: result }); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           // | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           processed_last_submit = true; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if edit_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       edit_modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <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:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("edit-a-card")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_("you-can-provide-a-runner-but-you-dont-have-to")} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="runner" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("runner")}</label | ||||||
|  |                   > | ||||||
|  |                   <Select | ||||||
|  |                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     itemFilter={(label, filterText, option) => | ||||||
|  |                       filterRunners(label, filterText, option)} | ||||||
|  |                     items={runners} | ||||||
|  |                     showChevron={true} | ||||||
|  |                     placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|  |                     noOptionsMessage={$_("no-runners-found")} | ||||||
|  |                     bind:selectedValue={runner} | ||||||
|  |                     on:select={(selectedValue) => | ||||||
|  |                       (editable.runner = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (editable.runner = null)} | ||||||
|  |                   /> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <p class="text-gray-500"> | ||||||
|  |                     <input | ||||||
|  |                       id="enabled" | ||||||
|  |                       on:change={() => { | ||||||
|  |                         editable.enabled = !editable.enabled; | ||||||
|  |                       }} | ||||||
|  |                       name="enabled" | ||||||
|  |                       type="checkbox" | ||||||
|  |                       checked={editable.enabled} | ||||||
|  |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                     /> | ||||||
|  |                     {$_("this-card-is")} | ||||||
|  |                     {#if editable.enabled} | ||||||
|  |                       {$_("enabled")} | ||||||
|  |                     {:else}{$_("disabled")}{/if} | ||||||
|  |                   </p> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <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" | ||||||
|  |           > | ||||||
|  |             {$_("save-changes")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               edit_modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										16
									
								
								src/components/cards/CardRunner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/cards/CardRunner.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let runner; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !runner} | ||||||
|  |   {$_("non-blanko")} | ||||||
|  | {:else} | ||||||
|  |   <a href={`/runners/${runner.id}`}> | ||||||
|  |     {#if runner.middlename} | ||||||
|  |       {runner.firstname} {runner.middlename} {runner.lastname} | ||||||
|  |     {:else} | ||||||
|  |       {runner.firstname} {runner.lastname} | ||||||
|  |     {/if} | ||||||
|  |   </a> | ||||||
|  | {/if} | ||||||
							
								
								
									
										16
									
								
								src/components/cards/CardStatus.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/cards/CardStatus.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let enabled = false; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if enabled} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-green-100 text-green-800" | ||||||
|  |     >{$_("enabled")}</span | ||||||
|  |   > | ||||||
|  | {:else} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-red-100 text-red-800" | ||||||
|  |     >{$_("disabled")}</span | ||||||
|  |   > | ||||||
|  | {/if} | ||||||
							
								
								
									
										53
									
								
								src/components/cards/Cards.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/components/cards/Cards.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import AddCardBulkModal from "./AddCardBulkModal.svelte"; | ||||||
|  |   import AddCardModal from "./AddCardModal.svelte"; | ||||||
|  |   import CardsOverview from "./CardsOverview.svelte"; | ||||||
|  |   $: current_cards = []; | ||||||
|  |   export let modal_open = false; | ||||||
|  |   export let bulk_modal_open = false; | ||||||
|  |   let addCards; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <section class="container p-5"> | ||||||
|  |   <h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|  |     {$_("cards")} | ||||||
|  |   </h4> | ||||||
|  |   {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:CREATE")} | ||||||
|  |     <button | ||||||
|  |       on:click={() => { | ||||||
|  |         modal_open = true; | ||||||
|  |       }} | ||||||
|  |       type="button" | ||||||
|  |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm  mb-1 lg:mb-0" | ||||||
|  |     > | ||||||
|  |       {$_("add-card")} | ||||||
|  |     </button> | ||||||
|  |     <button | ||||||
|  |       on:click={() => { | ||||||
|  |         bulk_modal_open = true; | ||||||
|  |       }} | ||||||
|  |       type="button" | ||||||
|  |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm  mb-1 lg:mb-0" | ||||||
|  |     > | ||||||
|  |       {$_("create-bulk-cards")} | ||||||
|  |     </button> | ||||||
|  |   {/if} | ||||||
|  |   <CardsOverview bind:current_cards bind:addCards /> | ||||||
|  | </section> | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:CREATE")} | ||||||
|  |   <AddCardModal | ||||||
|  |     bind:modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addCards(event.detail.cards); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  |   <AddCardBulkModal | ||||||
|  |     bind:bulk_modal_open | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addCards(event.detail.cards); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  | {/if} | ||||||
							
								
								
									
										12
									
								
								src/components/cards/CardsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/components/cards/CardsEmptyState.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import cards_empty from "./cards.svg"; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="text-center items-center justify-center"> | ||||||
|  |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|  |     <img class="m-auto mt-2" style="height:15rem" src={cards_empty} alt="" /> | ||||||
|  |     <span class="font-bold">{$_("there-are-no-cards-yet")}</span><br /> | ||||||
|  |     <span>{$_("add-your-first-card")}</span> | ||||||
|  |   </p> | ||||||
|  | </div> | ||||||
							
								
								
									
										316
									
								
								src/components/cards/CardsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										316
									
								
								src/components/cards/CardsOverview.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,316 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { RunnerCardService } from "@odit/lfk-client-js"; | ||||||
|  |   import store from "../../store"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|  |   import CardsEmptyState from "./CardsEmptyState.svelte"; | ||||||
|  |   import CardDetailModal from "./CardDetailModal.svelte"; | ||||||
|  |   import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|  |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|  |   import CardStatus from "./CardStatus.svelte"; | ||||||
|  |   import CardRunner from "./CardRunner.svelte"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import { runnerFilter, statusFilter } from "../shared/tablefilters"; | ||||||
|  |   import DeleteCardModal from "./DeleteCardModal.svelte"; | ||||||
|  |  | ||||||
|  |   export let edit_modal_open = false; | ||||||
|  |   export let runner = {}; | ||||||
|  |   export let editable = {}; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   export let current_cards = []; | ||||||
|  |   export const addCards = (cards) => { | ||||||
|  |     current_cards = current_cards.concat(...cards); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_cards, | ||||||
|  |     })); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   $: dataLoaded = false; | ||||||
|  |   $: selected = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|  |   $: selectedCards = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.original) || []; | ||||||
|  |   $: active_delete = undefined; | ||||||
|  |   $: cards_show = generate_cards.length > 0; | ||||||
|  |   $: generate_cards = []; | ||||||
|  |  | ||||||
|  |   const columns = [ | ||||||
|  |     { | ||||||
|  |       accessorKey: "code", | ||||||
|  |       header: () => $_("code"), | ||||||
|  |       filterFn: `includesString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "runner", | ||||||
|  |       header: () => $_("runner"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(CardRunner, { runner: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `runner`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "enabled", | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(CardStatus, { enabled: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       header: () => $_("status"), | ||||||
|  |       filterFn: `status`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(TableActions, { | ||||||
|  |           detailsAction: () => { | ||||||
|  |             open_edit_modal( | ||||||
|  |               current_cards[ | ||||||
|  |                 current_cards.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ] | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_delete = | ||||||
|  |               current_cards[ | ||||||
|  |                 current_cards.findIndex((r) => r.id == info.row.original.id) | ||||||
|  |               ]; | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes("CARD:DELETE"), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |  | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     filterFns: { | ||||||
|  |       runner: runnerFilter, | ||||||
|  |       status: statusFilter, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const table = createSvelteTable(options); | ||||||
|  |  | ||||||
|  |   function open_edit_modal(card) { | ||||||
|  |     const getRunnerLabel = (option) => | ||||||
|  |       option.firstname + | ||||||
|  |       " " + | ||||||
|  |       (option.middlename || "") + | ||||||
|  |       " " + | ||||||
|  |       option.lastname; | ||||||
|  |     if (card.runner?.id) { | ||||||
|  |       runner = Object.assign( | ||||||
|  |         { runner }, | ||||||
|  |         { label: getRunnerLabel(card.runner), value: card.runner } | ||||||
|  |       ); | ||||||
|  |       card.runner = card.runner.id; | ||||||
|  |     } else { | ||||||
|  |       card.runner = null; | ||||||
|  |       runner = null; | ||||||
|  |     } | ||||||
|  |     editable = Object.assign(editable, card); | ||||||
|  |     original_data = Object.assign(original_data, card); | ||||||
|  |     edit_modal_open = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function deleteCard(delete_card_id) { | ||||||
|  |     await RunnerCardService.runnerCardControllerRemove(delete_card_id, true); | ||||||
|  |     current_cards = current_cards.filter((r) => r.id !== delete_card_id); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_cards, | ||||||
|  |     })); | ||||||
|  |     toast.success($_("card-deleted")); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   onMount(async () => { | ||||||
|  |     let page = 0; | ||||||
|  |     let pagesize = 500; | ||||||
|  |     while (page >= 0) { | ||||||
|  |       const cards = await RunnerCardService.runnerCardControllerGetAll( | ||||||
|  |         page, | ||||||
|  |         pagesize | ||||||
|  |       ); | ||||||
|  |       if (cards.length == 0) { | ||||||
|  |         page = -2; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       current_cards = current_cards.concat(...cards); | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_cards, | ||||||
|  |       })); | ||||||
|  |  | ||||||
|  |       dataLoaded = true; | ||||||
|  |       page++; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:UPDATE")} | ||||||
|  |   <CardDetailModal | ||||||
|  |     bind:edit_modal_open | ||||||
|  |     bind:runner | ||||||
|  |     bind:editable | ||||||
|  |     bind:original_data | ||||||
|  |     on:dataUpdated={(event) => { | ||||||
|  |       current_cards[ | ||||||
|  |         current_cards.findIndex((c) => c.id === event.detail.card.id) | ||||||
|  |       ] = event.detail.card; | ||||||
|  |       current_cards = current_cards; | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_cards, | ||||||
|  |       })); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")} | ||||||
|  |   <DeleteCardModal | ||||||
|  |     delete_card={active_delete} | ||||||
|  |     modal_open={active_delete != undefined} | ||||||
|  |     on:delete={(event) => { | ||||||
|  |       deleteCard(event.detail.id); | ||||||
|  |     }} | ||||||
|  |   /> | ||||||
|  |   {#if !dataLoaded} | ||||||
|  |     <div | ||||||
|  |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|  |       role="alert" | ||||||
|  |     > | ||||||
|  |       <p class="font-bold">{$_("loading-cards")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|  |     </div> | ||||||
|  |   {:else if current_cards.length === 0} | ||||||
|  |     <CardsEmptyState /> | ||||||
|  |   {:else} | ||||||
|  |     <div class="h-12 mt-1"> | ||||||
|  |       {#if selected.length > 0} | ||||||
|  |         <button | ||||||
|  |           type="button" | ||||||
|  |           class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm inline-flex" | ||||||
|  |           id="options-menu" | ||||||
|  |           on:click={async () => { | ||||||
|  |             const prom = []; | ||||||
|  |             for (const card of selectedCards) { | ||||||
|  |               prom.push( | ||||||
|  |                 await RunnerCardService.runnerCardControllerRemove( | ||||||
|  |                   card.id, | ||||||
|  |                   true | ||||||
|  |                 ) | ||||||
|  |               ); | ||||||
|  |             } | ||||||
|  |             await Promise.all(prom); | ||||||
|  |             for (const card of selectedCards) { | ||||||
|  |               current_cards = current_cards.filter((r) => r.id !== card.id); | ||||||
|  |             } | ||||||
|  |             options.update((options) => ({ | ||||||
|  |               ...options, | ||||||
|  |               data: current_cards, | ||||||
|  |             })); | ||||||
|  |             $table.resetRowSelection(); | ||||||
|  |           }} | ||||||
|  |         > | ||||||
|  |           {$_("delete-cards")} | ||||||
|  |           <svg | ||||||
|  |             xmlns="http://www.w3.org/2000/svg" | ||||||
|  |             fill="none" | ||||||
|  |             viewBox="0 0 24 24" | ||||||
|  |             stroke-width="1.5" | ||||||
|  |             stroke="currentColor" | ||||||
|  |             class="w-5 h-5" | ||||||
|  |           > | ||||||
|  |             <path | ||||||
|  |               stroke-linecap="round" | ||||||
|  |               stroke-linejoin="round" | ||||||
|  |               d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0" | ||||||
|  |             /> | ||||||
|  |           </svg> | ||||||
|  |         </button> | ||||||
|  |       {/if} | ||||||
|  |       <GenerateRunnerCards | ||||||
|  |         cards_show={selected.length > 0} | ||||||
|  |         bind:generate_cards={selectedCards} | ||||||
|  |       /> | ||||||
|  |     </div> | ||||||
|  |     <div class="overflow-x-auto"> | ||||||
|  |       <table class="w-full"> | ||||||
|  |         <thead class="border-b border-gray-400"> | ||||||
|  |           {#each $table.getHeaderGroups() as headerGroup} | ||||||
|  |             <tr class="select-none"> | ||||||
|  |               <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|  |                 <InputElement | ||||||
|  |                   type="checkbox" | ||||||
|  |                   checked={$table.getIsAllRowsSelected()} | ||||||
|  |                   indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|  |                   on:change={() => $table.toggleAllRowsSelected()} | ||||||
|  |                 /> | ||||||
|  |               </th> | ||||||
|  |               {#each headerGroup.headers as header} | ||||||
|  |                 <TableHeader {header} /> | ||||||
|  |               {/each} | ||||||
|  |             </tr> | ||||||
|  |           {/each} | ||||||
|  |         </thead> | ||||||
|  |         <tbody> | ||||||
|  |           {#each $table.getRowModel().rows as row} | ||||||
|  |             <tr class="odd:bg-white even:bg-gray-100"> | ||||||
|  |               <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|  |                 <InputElement | ||||||
|  |                   type="checkbox" | ||||||
|  |                   checked={row.getIsSelected()} | ||||||
|  |                   on:change={() => row.toggleSelected()} | ||||||
|  |                 /> | ||||||
|  |               </td> | ||||||
|  |               {#each row.getVisibleCells() as cell} | ||||||
|  |                 <td> | ||||||
|  |                   <svelte:component | ||||||
|  |                     this={flexRender( | ||||||
|  |                       cell.column.columnDef.cell, | ||||||
|  |                       cell.getContext() | ||||||
|  |                     )} | ||||||
|  |                   /> | ||||||
|  |                 </td> | ||||||
|  |               {/each} | ||||||
|  |             </tr> | ||||||
|  |           {/each} | ||||||
|  |         </tbody> | ||||||
|  |       </table> | ||||||
|  |     </div> | ||||||
|  |     <TableBottom {table} {selected} /> | ||||||
|  |   {/if} | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   table tbody tr td:nth-child(2) { | ||||||
|  |     font-family: monospace; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
							
								
								
									
										123
									
								
								src/components/cards/DeleteCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								src/components/cards/DeleteCardModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,123 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_card = { | ||||||
|  |     id: 0, | ||||||
|  |     code: "", | ||||||
|  |     runner: { | ||||||
|  |       firstname: "", | ||||||
|  |       lastname: "", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   onMount(() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   async function submit() { | ||||||
|  |     dispatch("delete", { id: delete_card.id }); | ||||||
|  |     modal_open = false; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <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:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("please-confirm-the-deletion-of-card")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="w-full"> | ||||||
|  |                 {$_("card")} #{delete_card.code}<br /> | ||||||
|  |                 <span class="inline-block"> | ||||||
|  |                   {$_("runner")}: | ||||||
|  |                   {#if delete_card.runner} | ||||||
|  |                     <span class="inline-block" | ||||||
|  |                       >{delete_card.runner.firstname} | ||||||
|  |                       {delete_card.runner.lastname}</span | ||||||
|  |                     > | ||||||
|  |                   {:else} | ||||||
|  |                     {$_("non-blanko")} | ||||||
|  |                   {/if}</span | ||||||
|  |                 > | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <button | ||||||
|  |             on:click={submit} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-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" | ||||||
|  |           > | ||||||
|  |             {$_("delete")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										1
									
								
								src/components/cards/cards.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/components/cards/cards.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <svg data-name="Layer 1" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 653.9 247.6"><path d="M272 211l-53 12s-11-2-1-17l4-4 27-14v-2l-6-41-2-16 17-17 4-3 44 41v1l-19 33z" fill="#ffb7b7"/><path d="M253 198l-54 13a6 6 0 01-5-7 16 16 0 012-5 48 48 0 016-9l28-14-4-24-1-12-2-8-2-16 21-19 22 21 22 20-3 5-3 7-17 30-3 6z" fill="#ffb7b7"/><path d="M346 190s-20-1-28-15a24 24 0 01-3-14l-8-17-11-23-30-4-2 1-21 19-7 6-10 9-49 44a37 37 0 01-7 9 50 50 0 01-9 7c-10 5-24 9-44 7L10 248 0 176l89-29 131-86 89 23 41 58z" fill="#ffb7b7"/><path d="M648 0H275a5 5 0 00-5 5v221a5 5 0 005 6h373a5 5 0 006-6V5a5 5 0 00-6-5z" fill="#fff"/><path d="M648 0H275a5 5 0 00-5 5v221a5 5 0 005 6h373a5 5 0 006-6V5a5 5 0 00-6-5zm4 226a4 4 0 01-4 4H275a4 4 0 01-3-4V5a4 4 0 013-3h373a4 4 0 014 3z" fill="#3f3d56"/><path d="M312 30a9 9 0 119-9 9 9 0 01-9 9zm0-17a8 8 0 107 8 8 8 0 00-7-8z" fill="#6c63ff"/><path d="M297 21a8 8 0 016-8 8 8 0 100 16 8 8 0 01-6-8zM349 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM368 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM386 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM415 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM434 130a7 7 0 01-7-7v-20a7 7 0 0113 0v20a7 7 0 01-6 7zM452 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM481 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM499 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM518 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM546 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM565 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7zM583 130a7 7 0 01-7-7v-20a7 7 0 0114 0v20a7 7 0 01-7 7z" fill="#6c63ff"/><path d="M396 208h-99a5 5 0 110-10h99a5 5 0 010 10zM364 188h-35a5 5 0 110-10h35a5 5 0 110 10z" fill="#e6e6e6"/><path fill="#3f3d56" d="M271 46h381v2H271z"/><path opacity=".1" d="M228 203l-1-2 33-15 8-27-12-10 1-1 13 10-8 30-34 15zM196 199l-9 4-17 2a50 50 0 01-9 7l-16-1-26-1 88-74 18 4-29 59z"/><path d="M318 175l-8 1-47 4-29 1-38 18-9 4-70 8 95-81 11 2 20 5 22 5 18 1 24 1 20 1a13 13 0 0112 13c0 7-5 14-21 17z" fill="#ffb7b7"/><path d="M325 170s-7-2-9-9c-2-4-1-9 3-15l1 1c-3 6-4 10-3 14 2 5 9 7 9 7zM197 197l34-16v2l-33 16zM218 135l48-19v2l-41 16 35 6v2l-42-7z" opacity=".1"/></svg> | ||||||
| After Width: | Height: | Size: 2.1 KiB | 
| @@ -1,7 +1,7 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { |   import { | ||||||
|     GroupContactService, |     GroupContactService, | ||||||
|     RunnerTeamService, |     RunnerTeamService, | ||||||
| @@ -9,7 +9,7 @@ | |||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|   import isEmail from "validator/es/lib/isEmail"; |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|   import isMobilePhone from "validator/es/lib/isMobilePhone"; |   import isMobilePhone from "validator/es/lib/isMobilePhone"; | ||||||
|   import Toastify from "toastify-js"; |   import toast from "svelte-french-toast"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_contacts; |   export let current_contacts; | ||||||
|   $: selected_team = []; |   $: selected_team = []; | ||||||
| @@ -43,7 +43,7 @@ | |||||||
|   $: address_zipcode_value = ""; |   $: address_zipcode_value = ""; | ||||||
|   $: address_city_value = ""; |   $: address_city_value = ""; | ||||||
|   $: processed_last_submit = true; |   $: processed_last_submit = true; | ||||||
|   $: address_checked = true; |   $: address_checked = false; | ||||||
|   $: isPhoneValidOrEmpty = |   $: isPhoneValidOrEmpty = | ||||||
|     (phone_input_value.includes("+") && |     (phone_input_value.includes("+") && | ||||||
|       isMobilePhone( |       isMobilePhone( | ||||||
| @@ -85,10 +85,7 @@ | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       toast.loading($_("contact-is-being-added")); | ||||||
|         text: "Contact is being added...", |  | ||||||
|         duration: -1, |  | ||||||
|       }).showToast(); |  | ||||||
|       let address = {}; |       let address = {}; | ||||||
|       if (address_checked === true) { |       if (address_checked === true) { | ||||||
|         address = { |         address = { | ||||||
| @@ -122,11 +119,8 @@ | |||||||
|           email_input_value = ""; |           email_input_value = ""; | ||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           // |           // | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: "Contact added", |           toast.success($_("contact-added")); | ||||||
|             duration: 500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
|           current_contacts.push(result); |           current_contacts.push(result); | ||||||
|           current_contacts = current_contacts; |           current_contacts = current_contacts; | ||||||
|         }) |         }) | ||||||
| @@ -135,8 +129,6 @@ | |||||||
|         }) |         }) | ||||||
|         .finally(() => { |         .finally(() => { | ||||||
|           processed_last_submit = true; |           processed_last_submit = true; | ||||||
|           // |  | ||||||
|           toast.hideToast(); |  | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -144,59 +136,71 @@ | |||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|     }}> |     }} | ||||||
|  |   > | ||||||
|     <div |     <div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|         role="dialog" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |       > | ||||||
|           <div class="sm:flex sm:items-start"> |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|             <div |             <div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|               <svg |               <svg | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |                 viewBox="0 0 24 24" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   d="M2 22a8 8 0 1 1 16 0H2zm8-9c-3.315 0-6-2.685-6-6s2.685-6 6-6 6 2.685 6 6-2.685 6-6 6zm10 4h4v2h-4v-2zm-3-5h7v2h-7v-2zm2-5h5v2h-5V7z" /></svg> |                   d="M2 22a8 8 0 1 1 16 0H2zm8-9c-3.315 0-6-2.685-6-6s2.685-6 6-6 6 2.685 6 6-2.685 6-6 6zm10 4h4v2h-4v-2zm-3-5h7v2h-7v-2zm2-5h5v2h-5V7z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> |             <div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('create-a-new-contact')} |                 {$_("create-a-new-contact")} | ||||||
|               </h3> |               </h3> | ||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <p class="text-sm text-gray-500"> |                 <p class="text-sm text-gray-500"> | ||||||
|                   {$_('please-provide-the-required-information-to-add-a-new-contact')} |                   {$_( | ||||||
|  |                     "please-provide-the-required-information-to-add-a-new-contact" | ||||||
|  |                   )} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <div class="grid grid-cols-6 gap-6"> |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="firstname" |                     for="firstname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('first-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("first-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     use:focus |                     use:focus | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('first-name')} |                     placeholder={$_("first-name")} | ||||||
|                     class:border-red-500={!isFirstnameValid} |                     class:border-red-500={!isFirstnameValid} | ||||||
|                     class:focus:border-red-500={!isFirstnameValid} |                     class:focus:border-red-500={!isFirstnameValid} | ||||||
|                     class:focus:ring-red-500={!isFirstnameValid} |                     class:focus:ring-red-500={!isFirstnameValid} | ||||||
| @@ -204,34 +208,41 @@ | |||||||
|                     bind:this={firstname_input} |                     bind:this={firstname_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="firstname" |                     name="firstname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isFirstnameValid} |                   {#if !isFirstnameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('first-name-is-required')} |                     > | ||||||
|  |                       {$_("first-name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="trackname" |                     for="trackname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('middle-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("middle-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('middle-name')} |                     placeholder={$_("middle-name")} | ||||||
|                     bind:value={middlename_input_value} |                     bind:value={middlename_input_value} | ||||||
|                     bind:this={middlename_input} |                     bind:this={middlename_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="trackname" |                     name="trackname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="lastname" |                     for="lastname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('last-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("last-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder="{$_('last-name')}" |                     placeholder={$_("last-name")} | ||||||
|                     class:border-red-500={!isLastnameValid} |                     class:border-red-500={!isLastnameValid} | ||||||
|                     class:focus:border-red-500={!isLastnameValid} |                     class:focus:border-red-500={!isLastnameValid} | ||||||
|                     class:focus:ring-red-500={!isLastnameValid} |                     class:focus:ring-red-500={!isLastnameValid} | ||||||
| @@ -239,23 +250,28 @@ | |||||||
|                     bind:this={lastname_input} |                     bind:this={lastname_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="lastname" |                     name="lastname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isLastnameValid} |                   {#if !isLastnameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('last-name-is-required')} |                     > | ||||||
|  |                       {$_("last-name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="team" |                     for="team" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('teams')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("teams")}</label | ||||||
|  |                   > | ||||||
|                   <select |                   <select | ||||||
|                     name="team" |                     name="team" | ||||||
|                     multiple |                     multiple | ||||||
|                     bind:value={selected_team} |                     bind:value={selected_team} | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2"> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   > | ||||||
|                     {#each teams as team} |                     {#each teams as team} | ||||||
|                       <option value={team.id}> |                       <option value={team.id}> | ||||||
|                         {team.parentGroup.name} |                         {team.parentGroup.name} | ||||||
| @@ -271,10 +287,12 @@ | |||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="phone" |                     for="phone" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('phone')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("phone")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('phone')} |                     placeholder={$_("phone")} | ||||||
|                     class:border-red-500={!isPhoneValidOrEmpty} |                     class:border-red-500={!isPhoneValidOrEmpty} | ||||||
|                     class:focus:border-red-500={!isPhoneValidOrEmpty} |                     class:focus:border-red-500={!isPhoneValidOrEmpty} | ||||||
|                     class:focus:ring-red-500={!isPhoneValidOrEmpty} |                     class:focus:ring-red-500={!isPhoneValidOrEmpty} | ||||||
| @@ -282,21 +300,27 @@ | |||||||
|                     bind:this={phone_input} |                     bind:this={phone_input} | ||||||
|                     type="tel" |                     type="tel" | ||||||
|                     name="phone" |                     name="phone" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isPhoneValidOrEmpty} |                   {#if !isPhoneValidOrEmpty} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {@html $_('the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number')} |                     > | ||||||
|  |                       {@html $_( | ||||||
|  |                         "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number" | ||||||
|  |                       )} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="email" |                     for="email" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('e-mail-adress')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("e-mail-adress")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('e-mail-adress')} |                     placeholder={$_("e-mail-adress")} | ||||||
|                     class:border-red-500={!isEmailValidOrEmpty} |                     class:border-red-500={!isEmailValidOrEmpty} | ||||||
|                     class:focus:border-red-500={!isEmailValidOrEmpty} |                     class:focus:border-red-500={!isEmailValidOrEmpty} | ||||||
|                     class:focus:ring-red-500={!isEmailValidOrEmpty} |                     class:focus:ring-red-500={!isEmailValidOrEmpty} | ||||||
| @@ -304,11 +328,13 @@ | |||||||
|                     bind:this={email_input} |                     bind:this={email_input} | ||||||
|                     type="email" |                     type="email" | ||||||
|                     name="email" |                     name="email" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isEmailValidOrEmpty} |                   {#if !isEmailValidOrEmpty} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('valid-email-is-required')} |                     > | ||||||
|  |                       {$_("valid-email-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
| @@ -319,22 +345,25 @@ | |||||||
|                       id="comments" |                       id="comments" | ||||||
|                       name="comments" |                       name="comments" | ||||||
|                       type="checkbox" |                       type="checkbox" | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="ml-3 text-sm"> |                   <div class="ml-3 text-sm"> | ||||||
|                     <label |                     <label for="comments" class="font-semibold text-gray-700" | ||||||
|                       for="comments" |                       >{$_("address")}</label | ||||||
|                       class="font-medium text-gray-700">{$_('address')}</label> |                     > | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|                 {#if address_checked === true} |                 {#if address_checked === true} | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address1" |                       for="address1" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('address')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("address")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder="{$_('address')}" |                       placeholder={$_("address")} | ||||||
|                       class:border-red-500={!isAddress1Valid} |                       class:border-red-500={!isAddress1Valid} | ||||||
|                       class:focus:border-red-500={!isAddress1Valid} |                       class:focus:border-red-500={!isAddress1Valid} | ||||||
|                       class:focus:ring-red-500={!isAddress1Valid} |                       class:focus:ring-red-500={!isAddress1Valid} | ||||||
| @@ -342,34 +371,41 @@ | |||||||
|                       bind:this={address_input1} |                       bind:this={address_input1} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address1" |                       name="address1" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !isAddress1Valid} |                     {#if !isAddress1Valid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('address-is-required')} |                       > | ||||||
|  |                         {$_("address-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address2" |                       for="address2" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("apartment-suite-etc")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder={$_('apartment-suite-etc')} |                       placeholder={$_("apartment-suite-etc")} | ||||||
|                       bind:value={address_input2_value} |                       bind:value={address_input2_value} | ||||||
|                       bind:this={address_input2} |                       bind:this={address_input2} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address2" |                       name="address2" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="zipcode" |                       for="zipcode" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("zip-postal-code")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder={$_('zip-postal-code')} |                       placeholder={$_("zip-postal-code")} | ||||||
|                       class:border-red-500={!iszipcodevalid} |                       class:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:border-red-500={!iszipcodevalid} |                       class:focus:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:ring-red-500={!iszipcodevalid} |                       class:focus:ring-red-500={!iszipcodevalid} | ||||||
| @@ -377,21 +413,25 @@ | |||||||
|                       bind:this={address_zipcode} |                       bind:this={address_zipcode} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="zipcode" |                       name="zipcode" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iszipcodevalid} |                     {#if !iszipcodevalid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('valid-zipcode-postal-code-is-required')} |                       > | ||||||
|  |                         {$_("valid-zipcode-postal-code-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="city" |                       for="city" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('city')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("city")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder="{$_('city')}" |                       placeholder={$_("city")} | ||||||
|                       class:border-red-500={!iscityvalid} |                       class:border-red-500={!iscityvalid} | ||||||
|                       class:focus:border-red-500={!iscityvalid} |                       class:focus:border-red-500={!iscityvalid} | ||||||
|                       class:focus:ring-red-500={!iscityvalid} |                       class:focus:ring-red-500={!iscityvalid} | ||||||
| @@ -399,11 +439,13 @@ | |||||||
|                       bind:this={address_city} |                       bind:this={address_city} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="city" |                       name="city" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iscityvalid} |                     {#if !iscityvalid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('valid-city-is-required')} |                       > | ||||||
|  |                         {$_("valid-city-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
| @@ -412,22 +454,24 @@ | |||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|           <button |           <button | ||||||
|             disabled={!createbtnenabled} |             disabled={!createbtnenabled} | ||||||
|             class:opacity-50={!createbtnenabled} |             class:opacity-50={!createbtnenabled} | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             type="button" |             type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|             {$_('create')} |           > | ||||||
|  |             {$_("create")} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               modal_open = false; |               modal_open = false; | ||||||
|             }} |             }} | ||||||
|             type="button" |             type="button" | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full 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 hidden lg:block" | ||||||
|             {$_('cancel')} |           > | ||||||
|  |             {$_("cancel")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -6,9 +6,9 @@ | |||||||
| 		RunnerTeamService, | 		RunnerTeamService, | ||||||
| 		RunnerOrganizationService, | 		RunnerOrganizationService, | ||||||
| 	} from "@odit/lfk-client-js"; | 	} from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
| 	import PromiseError from "../base/PromiseError.svelte"; | 	import PromiseError from "../base/PromiseError.svelte"; | ||||||
| 	import isEmail from "validator/es/lib/isEmail"; | 	import isEmail from "validator/es/lib/isEmail"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
| 	let data_loaded = false; | 	let data_loaded = false; | ||||||
| 	let orgs = []; | 	let orgs = []; | ||||||
| 	let teams = []; | 	let teams = []; | ||||||
| @@ -48,8 +48,8 @@ | |||||||
| 				address2: "", | 				address2: "", | ||||||
| 				city: "", | 				city: "", | ||||||
| 				postalcode: "", | 				postalcode: "", | ||||||
|         country: "" | 				country: "", | ||||||
|       } | 			}; | ||||||
| 		} | 		} | ||||||
| 	}); | 	}); | ||||||
| 	RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { | 	RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => { | ||||||
| @@ -67,10 +67,7 @@ | |||||||
| 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | ||||||
| 	function submit() { | 	function submit() { | ||||||
| 		if (data_loaded === true && save_enabled) { | 		if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ | 			toast.loading($_("contact-is-being-updated")); | ||||||
|         text: $_("contact-is-being-updated"), |  | ||||||
|         duration: 2500, |  | ||||||
|       }).showToast(); |  | ||||||
| 			editable.address.country = "DE"; | 			editable.address.country = "DE"; | ||||||
| 			if (editable.address_checked === false) { | 			if (editable.address_checked === false) { | ||||||
| 				editable.address = null; | 				editable.address = null; | ||||||
| @@ -82,11 +79,8 @@ | |||||||
| 				.then((resp) => { | 				.then((resp) => { | ||||||
| 					Object.assign(original_data, editable); | 					Object.assign(original_data, editable); | ||||||
| 					original_data = original_data; | 					original_data = original_data; | ||||||
|           Toastify({ | 					toast.dismiss(); | ||||||
|             text: $_("updated-contact"), | 					toast.success($_("updated-contact")); | ||||||
|             duration: 2500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
| 				}) | 				}) | ||||||
| 				.catch((err) => {}); | 				.catch((err) => {}); | ||||||
| 		} else { | 		} else { | ||||||
| @@ -102,7 +96,7 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#await promise} | {#await promise} | ||||||
|   {$_('loading-contact-details')} | 	{$_("loading-contact-details")} | ||||||
| {:then} | {:then} | ||||||
| 	<section class="container p-5 select-none"> | 	<section class="container p-5 select-none"> | ||||||
| 		<div class="flex flex-row mb-4"> | 		<div class="flex flex-row mb-4"> | ||||||
| @@ -110,57 +104,46 @@ | |||||||
| 				<nav class="w-full flex"> | 				<nav class="w-full flex"> | ||||||
| 					<ol class="list-none flex flex-row items-center justify-start"> | 					<ol class="list-none flex flex-row items-center justify-start"> | ||||||
| 						<li class="flex items-center"> | 						<li class="flex items-center"> | ||||||
|               <svg | 							<a class="mr-2" href="./" | ||||||
|                 fill="currentColor" | 								><svg | ||||||
| 									xmlns="http://www.w3.org/2000/svg" | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
| 									width="24" | 									width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | 									height="24" | ||||||
|                 <path |  | ||||||
|                   d="M2 22a8 8 0 1 1 16 0H2zm8-9c-3.315 0-6-2.685-6-6s2.685-6 6-6 6 2.685 6 6-2.685 6-6 6zm10 4h4v2h-4v-2zm-3-5h7v2h-7v-2zm2-5h5v2h-5V7z" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center ml-2"> |  | ||||||
|               <a class="mr-2" href="./">{$_('contacts')}</a><svg |  | ||||||
|                 stroke="currentColor" |  | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
| 									viewBox="0 0 24 24" | 									viewBox="0 0 24 24" | ||||||
|  | 									fill="none" | ||||||
|  | 									stroke="currentColor" | ||||||
|  | 									stroke-width="2" | ||||||
| 									stroke-linecap="round" | 									stroke-linecap="round" | ||||||
| 									stroke-linejoin="round" | 									stroke-linejoin="round" | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" | 									class="inline-block" | ||||||
|                 height="1em" | 									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg | ||||||
|                 width="1em" | 								> | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line | 								{$_("contacts")}</a | ||||||
|                   x1="5" | 							> | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <span class="mr-2">{original_data.firstname} |  | ||||||
|                 {original_data.middlename || ''} |  | ||||||
|                 {original_data.lastname}</span> |  | ||||||
| 						</li> | 						</li> | ||||||
| 					</ol> | 					</ol> | ||||||
| 				</nav> | 				</nav> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> | 		<div class="mb-4 text-3xl font-extrabold leading-tight"> | ||||||
| 			{original_data.firstname} | 			{original_data.firstname} | ||||||
|       {original_data.middlename || ''} | 			{original_data.middlename || ""} | ||||||
| 			{original_data.lastname} | 			{original_data.lastname} | ||||||
|       <span data-id="contact_actions_${editable.id}"> | 			<div data-id="contact_actions_${editable.id}"> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:DELETE')} | 				{#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:DELETE")} | ||||||
| 					{#if delete_triggered} | 					{#if delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| 							on:click={deleteContact} | 							on:click={deleteContact} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('confirm-deletion')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("confirm-deletion")}</button | ||||||
|  | 						> | ||||||
| 						<button | 						<button | ||||||
| 							on:click={() => { | 							on:click={() => { | ||||||
| 								delete_triggered = !delete_triggered; | 								delete_triggered = !delete_triggered; | ||||||
| 							}} | 							}} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm">{$_('cancel')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("cancel")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 					{#if !delete_triggered} | 					{#if !delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| @@ -168,7 +151,9 @@ | |||||||
| 								delete_triggered = true; | 								delete_triggered = true; | ||||||
| 							}} | 							}} | ||||||
| 							type="button" | 							type="button" | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('delete-contact')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("delete-contact")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 				{/if} | 				{/if} | ||||||
| 				{#if !delete_triggered} | 				{#if !delete_triggered} | ||||||
| @@ -177,112 +162,124 @@ | |||||||
| 						class:opacity-50={!save_enabled} | 						class:opacity-50={!save_enabled} | ||||||
| 						type="button" | 						type="button" | ||||||
| 						on:click={submit} | 						on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('save-changes')}</button> | 						class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 						>{$_("save-changes")}</button | ||||||
|  | 					> | ||||||
| 				{/if} | 				{/if} | ||||||
|       </span> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<!--  --> | 		<!--  --> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="firstname" class="font-semibold text-gray-700" | ||||||
|         for="firstname" | 				>{$_("first-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('first-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('first-name')} | 				placeholder={$_("first-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				class:border-red-500={!isFirstnameValid} | 				class:border-red-500={!isFirstnameValid} | ||||||
| 				class:focus:border-red-500={!isFirstnameValid} | 				class:focus:border-red-500={!isFirstnameValid} | ||||||
| 				class:focus:ring-red-500={!isFirstnameValid} | 				class:focus:ring-red-500={!isFirstnameValid} | ||||||
| 				bind:value={editable.firstname} | 				bind:value={editable.firstname} | ||||||
| 				name="firstname" | 				name="firstname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isFirstnameValid} | 			{#if !isFirstnameValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('first-name-is-required')} | 				> | ||||||
|  | 					{$_("first-name-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="middlename" class="font-semibold text-gray-700" | ||||||
|         for="middlename" | 				>{$_("middle-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('middle-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('middle-name')} | 				placeholder={$_("middle-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.middlename} | 				bind:value={editable.middlename} | ||||||
| 				name="middlename" | 				name="middlename" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="lastname" class="font-semibold text-gray-700" | ||||||
|         for="lastname" | 				>{$_("last-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('last-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('last-name')} | 				placeholder={$_("last-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.lastname} | 				bind:value={editable.lastname} | ||||||
| 				class:border-red-500={!isLastnameValid} | 				class:border-red-500={!isLastnameValid} | ||||||
| 				class:focus:border-red-500={!isLastnameValid} | 				class:focus:border-red-500={!isLastnameValid} | ||||||
| 				class:focus:ring-red-500={!isLastnameValid} | 				class:focus:ring-red-500={!isLastnameValid} | ||||||
| 				name="lastname" | 				name="lastname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isLastnameValid} | 			{#if !isLastnameValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('last-name-is-required')} | 				> | ||||||
|  | 					{$_("last-name-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="email" class="font-semibold text-gray-700" | ||||||
|         for="email" | 				>{$_("e-mail-adress")}</label | ||||||
|         class="font-medium text-gray-700">{$_('e-mail-adress')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('e-mail-adress')} | 				placeholder={$_("e-mail-adress")} | ||||||
| 				type="email" | 				type="email" | ||||||
| 				bind:value={editable.email} | 				bind:value={editable.email} | ||||||
| 				class:border-red-500={!isEmailValid} | 				class:border-red-500={!isEmailValid} | ||||||
| 				class:focus:border-red-500={!isEmailValid} | 				class:focus:border-red-500={!isEmailValid} | ||||||
| 				class:focus:ring-red-500={!isEmailValid} | 				class:focus:ring-red-500={!isEmailValid} | ||||||
| 				name="email" | 				name="email" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isEmailValid} | 			{#if !isEmailValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('valid-email-is-required')} | 				> | ||||||
|  | 					{$_("valid-email-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label for="phone" class="font-medium text-gray-700">{$_('phone')}</label> | 			<label for="phone" class="font-semibold text-gray-700">{$_("phone")}</label> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('phone')} | 				placeholder={$_("phone")} | ||||||
| 				type="tel" | 				type="tel" | ||||||
| 				class:border-red-500={!isPhoneValidOrEmpty} | 				class:border-red-500={!isPhoneValidOrEmpty} | ||||||
| 				class:focus:border-red-500={!isPhoneValidOrEmpty} | 				class:focus:border-red-500={!isPhoneValidOrEmpty} | ||||||
| 				class:focus:ring-red-500={!isPhoneValidOrEmpty} | 				class:focus:ring-red-500={!isPhoneValidOrEmpty} | ||||||
| 				bind:value={editable.phone} | 				bind:value={editable.phone} | ||||||
| 				name="phone" | 				name="phone" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isPhoneValidOrEmpty} | 			{#if !isPhoneValidOrEmpty} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('valid-international-phone-number-is-required')} | 				> | ||||||
|  | 					{$_("valid-international-phone-number-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <span class="font-medium text-gray-700">{$_('groups')}</span> | 			<span class="font-semibold text-gray-700">{$_("groups")}</span> | ||||||
| 			<select | 			<select | ||||||
| 				bind:value={editable.groups} | 				bind:value={editable.groups} | ||||||
| 				name="team" | 				name="team" | ||||||
| 				multiple | 				multiple | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2"> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			> | ||||||
| 				{#each teams as team} | 				{#each teams as team} | ||||||
| 					<option value={team.id}> | 					<option value={team.id}> | ||||||
| 						{team.parentGroup.name} | 						{team.parentGroup.name} | ||||||
| @@ -303,19 +300,20 @@ | |||||||
| 					id="comments" | 					id="comments" | ||||||
| 					name="comments" | 					name="comments" | ||||||
| 					type="checkbox" | 					type="checkbox" | ||||||
|           class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | 					class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="ml-3 text-sm"> | 			<div class="ml-3 text-sm"> | ||||||
|         <label | 				<label for="comments" class="font-semibold text-gray-700" | ||||||
|           for="comments" | 					>{$_("address")}</label | ||||||
|           class="font-medium text-gray-700">{$_('address')}</label> | 				> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		{#if editable.address_checked === true} | 		{#if editable.address_checked === true} | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="address1" class="block text-sm font-medium text-gray-700" | ||||||
|           for="address1" | 					>{$_("address")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('address')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
| 					placeholder="Address" | 					placeholder="Address" | ||||||
| @@ -325,65 +323,72 @@ | |||||||
| 					bind:value={editable.address.address1} | 					bind:value={editable.address.address1} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="address1" | 					name="address1" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !isAddress1Valid} | 				{#if !isAddress1Valid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('address-is-required')} | 					> | ||||||
|  | 						{$_("address-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="address2" class="block text-sm font-medium text-gray-700" | ||||||
|           for="address2" | 					>{$_("apartment-suite-etc")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('apartment-suite-etc')} | 					placeholder={$_("apartment-suite-etc")} | ||||||
| 					bind:value={editable.address.address2} | 					bind:value={editable.address.address2} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="address2" | 					name="address2" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="zipcode" class="block text-sm font-medium text-gray-700" | ||||||
|           for="zipcode" | 					>{$_("zip-postal-code")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('zip-postal-code')} | 					placeholder={$_("zip-postal-code")} | ||||||
| 					class:border-red-500={!iszipcodevalid} | 					class:border-red-500={!iszipcodevalid} | ||||||
| 					class:focus:border-red-500={!iszipcodevalid} | 					class:focus:border-red-500={!iszipcodevalid} | ||||||
| 					class:focus:ring-red-500={!iszipcodevalid} | 					class:focus:ring-red-500={!iszipcodevalid} | ||||||
| 					bind:value={editable.address.postalcode} | 					bind:value={editable.address.postalcode} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="zipcode" | 					name="zipcode" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !iszipcodevalid} | 				{#if !iszipcodevalid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-zipcode-postal-code-is-required')} | 					> | ||||||
|  | 						{$_("valid-zipcode-postal-code-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="city" class="block text-sm font-medium text-gray-700" | ||||||
|           for="city" | 					>{$_("city")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('city')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('city')} | 					placeholder={$_("city")} | ||||||
| 					class:border-red-500={!iscityvalid} | 					class:border-red-500={!iscityvalid} | ||||||
| 					class:focus:border-red-500={!iscityvalid} | 					class:focus:border-red-500={!iscityvalid} | ||||||
| 					class:focus:ring-red-500={!iscityvalid} | 					class:focus:ring-red-500={!iscityvalid} | ||||||
| 					bind:value={editable.address.city} | 					bind:value={editable.address.city} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="city" | 					name="city" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !iscityvalid} | 				{#if !iscityvalid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-city-is-required')} | 					> | ||||||
|  | 						{$_("valid-city-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
|   | |||||||
| @@ -8,22 +8,23 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> |   <h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('contacts')} |     {$_("contacts")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:CREATE')} |   </h4> | ||||||
|  |   {#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:CREATE")} | ||||||
|     <button |     <button | ||||||
|       on:click={() => { |       on:click={() => { | ||||||
|         modal_open = true; |         modal_open = true; | ||||||
|       }} |       }} | ||||||
|       type="button" |       type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm" | ||||||
|         {$_('create-a-new-contact')} |     > | ||||||
|  |       {$_("create-a-new-contact")} | ||||||
|     </button> |     </button> | ||||||
|   {/if} |   {/if} | ||||||
|   </span> |  | ||||||
|   <ContactsOverview bind:current_contacts /> |   <ContactsOverview bind:current_contacts /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:CREATE")} | ||||||
|   <AddContactModal bind:current_contacts bind:modal_open /> |   <AddContactModal bind:current_contacts bind:modal_open /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -9,8 +9,8 @@ | |||||||
| <div class="text-center items-center justify-center"> | <div class="text-center items-center justify-center"> | ||||||
|   <p class="mb-16 text-lg text-gray-500"> |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|     <img class="w-full h-44" src={team_empty} alt="" /> |     <img class="w-full h-44" src={team_empty} alt="" /> | ||||||
|     <span class="font-bold">{$_('there-are-no-contacts-added-yet')}</span><br /> |     <span class="font-bold">{$_("there-are-no-contacts-added-yet")}</span><br /> | ||||||
|     <span>{$_('add-your-first-contact')}</span> |     <span>{$_("add-your-first-contact")}</span> | ||||||
|   </p> |   </p> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,5 @@ | |||||||
| <script> | <script> | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
| 	import { GroupContactService } from "@odit/lfk-client-js"; | 	import { GroupContactService } from "@odit/lfk-client-js"; | ||||||
| 	const promise = GroupContactService.groupContactControllerGetAll().then( | 	const promise = GroupContactService.groupContactControllerGetAll().then( | ||||||
| 		(result) => { | 		(result) => { | ||||||
| @@ -9,18 +8,20 @@ | |||||||
| 	); | 	); | ||||||
| 	import store from "../../store"; | 	import store from "../../store"; | ||||||
| 	import ContactsEmptyState from "./ContactsEmptyState.svelte"; | 	import ContactsEmptyState from "./ContactsEmptyState.svelte"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
| 	$: searchvalue = ""; | 	$: searchvalue = ""; | ||||||
| 	$: active_deletes = []; | 	$: active_deletes = []; | ||||||
| 	export let current_contacts = []; | 	export let current_contacts = []; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes("TEAM:GET")} | ||||||
| 	{#await promise} | 	{#await 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">{$_('contacts-are-being-loaded')}</p> | 		> | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> | 			<p class="font-bold">{$_("contacts-are-being-loaded")}</p> | ||||||
|  | 			<p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
| 		</div> | 		</div> | ||||||
| 	{:then} | 	{:then} | ||||||
| 		{#if current_contacts.length === 0} | 		{#if current_contacts.length === 0} | ||||||
| @@ -29,31 +30,36 @@ | |||||||
| 			<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="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border" | ||||||
|  | 			/> | ||||||
| 			<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 class="odd:bg-white even:bg-gray-100"> | ||||||
| 							<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" | ||||||
|                 {$_('groups')} | 							> | ||||||
|  | 								{$_("groups")} | ||||||
| 							</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" | ||||||
|                 {$_('address')} | 							> | ||||||
|  | 								{$_("address")} | ||||||
| 							</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> | ||||||
| @@ -63,13 +69,16 @@ | |||||||
| 								.toString() | 								.toString() | ||||||
| 								.toLowerCase() | 								.toLowerCase() | ||||||
| 								.includes(searchvalue)} | 								.includes(searchvalue)} | ||||||
|                 <tr data-rowid="team_{t.id}"> | 								<tr | ||||||
|  | 									class="odd:bg-white even:bg-gray-100" | ||||||
|  | 									data-rowid="team_{t.id}" | ||||||
|  | 								> | ||||||
| 									<td class="px-6 py-4 whitespace-nowrap"> | 									<td class="px-6 py-4 whitespace-nowrap"> | ||||||
| 										<div class="flex items-center"> | 										<div class="flex items-center"> | ||||||
| 											<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.firstname} | 													{t.firstname} | ||||||
|                           {t.middlename || ''} | 													{t.middlename || ""} | ||||||
| 													{t.lastname} | 													{t.lastname} | ||||||
| 												</div> | 												</div> | ||||||
| 											</div> | 											</div> | ||||||
| @@ -77,28 +86,32 @@ | |||||||
| 									</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 | ||||||
|                         <div class="text-sm font-medium text-gray-900"> | 												class="text-sm font-medium text-gray-900 gap-0.5 flex flex-wrap" | ||||||
|  | 											> | ||||||
| 												{#if t.groups.length > 0} | 												{#if t.groups.length > 0} | ||||||
| 													{#each t.groups as g} | 													{#each t.groups as g} | ||||||
|                               {#if g.responseType === 'RUNNERORGANIZATION'} | 														{#if g.responseType === "RUNNERORGANIZATION"} | ||||||
| 															<a | 															<a | ||||||
| 																href="../orgs/{g.id}" | 																href="../orgs/{g.id}" | ||||||
|                                   class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{g.name}</a> | 																class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800 border border-current" | ||||||
|  | 																>{g.name}</a | ||||||
|  | 															> | ||||||
| 														{:else} | 														{:else} | ||||||
| 															<a | 															<a | ||||||
| 																href="../teams/{g.id}" | 																href="../teams/{g.id}" | ||||||
|                                   class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{g.parentGroup.name} | 																class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800 border border-current" | ||||||
|  | 																>{g.parentGroup.name} | ||||||
| 																> | 																> | ||||||
|                                   {g.name}</a> | 																{g.name}</a | ||||||
|  | 															> | ||||||
| 														{/if} | 														{/if} | ||||||
| 													{/each} | 													{/each} | ||||||
| 												{:else} | 												{:else} | ||||||
|                             {$_('contact-is-not-a-member-in-any-group')} | 													{$_("contact-is-not-a-member-in-any-group")} | ||||||
| 												{/if} | 												{/if} | ||||||
| 											</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"> | ||||||
| @@ -106,7 +119,7 @@ | |||||||
| 												<div class="text-sm font-medium text-gray-900"> | 												<div class="text-sm font-medium text-gray-900"> | ||||||
| 													{#if t.address.address1 !== null} | 													{#if t.address.address1 !== null} | ||||||
| 														{t.address.address1}<br /> | 														{t.address.address1}<br /> | ||||||
|                             {t.address.address2 || ''}<br /> | 														{t.address.address2 || ""}<br /> | ||||||
| 														{t.address.postalcode} | 														{t.address.postalcode} | ||||||
| 														{t.address.city} | 														{t.address.city} | ||||||
| 														{t.address.country} | 														{t.address.country} | ||||||
| @@ -117,45 +130,53 @@ | |||||||
| 									</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-delete')}</button> | 												class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer" | ||||||
|  | 												>{$_("cancel-delete")}</button | ||||||
|  | 											> | ||||||
| 											<button | 											<button | ||||||
| 												on:click={() => { | 												on:click={() => { | ||||||
|                           GroupContactService.groupContactControllerRemove(t.id, false).then( | 													toast.loading($_("deleting-contact")); | ||||||
|                             (resp) => { | 													GroupContactService.groupContactControllerRemove( | ||||||
|  | 														t.id, | ||||||
|  | 														false | ||||||
|  | 													).then((resp) => { | ||||||
| 														current_contacts = current_contacts.filter( | 														current_contacts = current_contacts.filter( | ||||||
| 															(obj) => obj.id !== t.id | 															(obj) => obj.id !== t.id | ||||||
| 														); | 														); | ||||||
|                               Toastify({ | 														toast.dismiss(); | ||||||
|                                 text: $_('contact-deleted'), | 														toast.success($_("contact-deleted")); | ||||||
|                                 duration: 500, | 													}); | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             } |  | ||||||
|                           ); |  | ||||||
| 												}} | 												}} | ||||||
| 												tabindex="0" | 												tabindex="0" | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> | 												class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  | 												>{$_("confirm-delete")}</button | ||||||
|  | 											> | ||||||
| 										</td> | 										</td> | ||||||
| 									{:else} | 									{:else} | ||||||
| 										<td | 										<td | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | 											class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium" | ||||||
|  | 										> | ||||||
| 											<a | 											<a | ||||||
| 												href="./{t.id}" | 												href="./{t.id}" | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> | 												class="text-indigo-600 hover:text-indigo-900" | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')} | 												>{$_("details")}</a | ||||||
|  | 											> | ||||||
|  | 											{#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} | ||||||
| @@ -169,7 +190,7 @@ | |||||||
| 	{: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> | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| 	import { router } from "tinro"; | 	import { router } from "tinro"; | ||||||
| 	import NoComponentLoaded from "../base/NoComponentLoaded.svelte"; | 	import NoComponentLoaded from "../base/NoComponentLoaded.svelte"; | ||||||
| 	import { AuthService } from "@odit/lfk-client-js"; | 	import { AuthService } from "@odit/lfk-client-js"; | ||||||
|  | 	import { Toaster } from "svelte-french-toast"; | ||||||
| 	$: navOpen = false; | 	$: navOpen = false; | ||||||
| 	function logout() { | 	function logout() { | ||||||
| 		localForage.clear(); | 		localForage.clear(); | ||||||
| @@ -13,296 +14,426 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <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} | 		style="z-index:11;" | ||||||
|     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"> | 		class="select-none fixed top-0 left-0 h-full pb-10 overflow-x-hidden overflow-y-auto transition origin-left transform border-r w-60 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 font-bold">LfK!Admin</h3> | ||||||
| 		</a> | 		</a> | ||||||
| 		<nav class="text-sm font-medium text-gray-600" aria-label="Main Navigation"> | 		<nav class="text-sm font-medium text-gray-600" aria-label="Main Navigation"> | ||||||
| 			<a | 			<a | ||||||
|         class:bg-gray-100={$router.path === '/'} | 				class:activenav={$router.path === "/"} | ||||||
|         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 				class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|         href="/"> | 				href="/" | ||||||
|  | 			> | ||||||
| 				<svg | 				<svg | ||||||
|           class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 					xmlns="http://www.w3.org/2000/svg" | 					xmlns="http://www.w3.org/2000/svg" | ||||||
| 					viewBox="0 0 20 20" | 					viewBox="0 0 20 20" | ||||||
|           fill="currentColor"> | 					fill="currentColor" | ||||||
|  | 				> | ||||||
| 					<path | 					<path | ||||||
|             d="M10.707 2.293a1 1 0 00-1.414 0l-7 7a1 1 0 001.414 1.414L4 10.414V17a1 1 0 001 1h2a1 1 0 001-1v-2a1 1 0 011-1h2a1 1 0 011 1v2a1 1 0 001 1h2a1 1 0 001-1v-6.586l.293.293a1 1 0 001.414-1.414l-7-7z" /> | 						d="M10.707 2.293a1 1 0 00-1.414 0l-7 7a1 1 0 001.414 1.414L4 10.414V17a1 1 0 001 1h2a1 1 0 001-1v-2a1 1 0 011-1h2a1 1 0 011 1v2a1 1 0 001 1h2a1 1 0 001-1v-6.586l.293.293a1 1 0 001.414-1.414l-7-7z" | ||||||
|  | 					/> | ||||||
| 				</svg> | 				</svg> | ||||||
|         <span>{$_('dashboard-title')}</span> | 				<span>{$_("dashboard-title")}</span> | ||||||
| 			</a> | 			</a> | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path.includes('/orgs/')} | 					class:activenav={$router.path.includes("/runners/")} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/orgs/"> | 					href="/runners/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						viewBox="0 0 24 24" | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 						<path | ||||||
|  | 							d="M9.83 8.79L8 9.456V13H6V8.05h.015l5.268-1.918c.244-.093.51-.14.782-.131a2.616 2.616 0 0 1 2.427 1.82c.186.583.356.977.51 1.182A4.992 4.992 0 0 0 19 11v2a6.986 6.986 0 0 1-5.402-2.547l-.581 3.297L15 15.67V23h-2v-5.986l-2.05-1.987-.947 4.298-6.894-1.215.348-1.97 4.924.868L9.83 8.79zM13.5 5.5a2 2 0 1 1 0-4 2 2 0 0 1 0 4z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("runners")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("TEAM:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/teams/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/teams/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						viewBox="0 0 640 512" | ||||||
|  | 						><path | ||||||
|  | 							fill="currentColor" | ||||||
|  | 							d="M96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm448 0c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm32 32h-64c-17.6 0-33.5 7.1-45.1 18.6 40.3 22.1 68.9 62 75.1 109.4h66c17.7 0 32-14.3 32-32v-32c0-35.3-28.7-64-64-64zm-256 0c61.9 0 112-50.1 112-112S381.9 32 320 32 208 82.1 208 144s50.1 112 112 112zm76.8 32h-8.3c-20.8 10-43.9 16-68.5 16s-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h288c26.5 0 48-21.5 48-48v-28.8c0-63.6-51.6-115.2-115.2-115.2zm-223.7-13.4C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("teams")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/orgs/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/orgs/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
| 						viewBox="0 0 24 24" | 						viewBox="0 0 24 24" | ||||||
| 						width="24" | 						width="24" | ||||||
|             height="24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
|             <path |  | ||||||
|               d="M17 19h2v-8h-6v8h2v-6h2v6zM3 19V4a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1v5h2v10h1v2H2v-2h1zm4-8v2h2v-2H7zm0 4v2h2v-2H7zm0-8v2h2V7H7z" /></svg> |  | ||||||
|           <span>{$_('orgs')}</span> |  | ||||||
|         </a> |  | ||||||
|       {/if} |  | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('USER:GET')} |  | ||||||
|         <a |  | ||||||
|           class:bg-gray-100={$router.path === '/users/'} |  | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" |  | ||||||
|           href="/users/"> |  | ||||||
|           <svg |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             width="24" |  | ||||||
| 						height="24" | 						height="24" | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|             fill="currentColor" |  | ||||||
|             viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
| 						<path | 						<path | ||||||
|               d="M12 14v8H4a8 8 0 018-8zm0-1a6 6 0 110-12 6 6 0 010 12zm2.6 5.81a3.51 3.51 0 010-1.62l-1-.57 1-1.74 1 .58a3.5 3.5 0 011.4-.82V13.5h2v1.15a3.5 3.5 0 011.4.8l1-.57 1 1.74-1 .57a3.51 3.51 0 010 1.62l1 .57-1 1.74-1-.58a3.5 3.5 0 01-1.4.82v1.14h-2v-1.15a3.5 3.5 0 01-1.4-.8l-1 .57-1-1.74 1-.57zM18 17a1 1 0 100 2 1 1 0 000-2z" /></svg> | 							d="M17 19h2v-8h-6v8h2v-6h2v6zM3 19V4a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1v5h2v10h1v2H2v-2h1zm4-8v2h2v-2H7zm0 4v2h2v-2H7zm0-8v2h2V7H7z" | ||||||
|           <span>{$_('users')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("orgs")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path === '/groups/'} | 					class:activenav={$router.path.includes("/donors/")} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/groups/"> | 					href="/donors/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|             fill="currentColor" |  | ||||||
|             width="24" |  | ||||||
|             height="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 640 512"><path |  | ||||||
|               fill="currentColor" |  | ||||||
|               d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" /></svg> |  | ||||||
|           <span>{$_('user-groups')}</span> |  | ||||||
|         </a> |  | ||||||
|       {/if} |  | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:GET')} |  | ||||||
|         <a |  | ||||||
|           class:bg-gray-100={$router.path === '/runners/'} |  | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" |  | ||||||
|           href="/runners/"> |  | ||||||
|           <svg |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24" |  | ||||||
|             height="24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
|             <path |  | ||||||
|               d="M9.83 8.79L8 9.456V13H6V8.05h.015l5.268-1.918c.244-.093.51-.14.782-.131a2.616 2.616 0 0 1 2.427 1.82c.186.583.356.977.51 1.182A4.992 4.992 0 0 0 19 11v2a6.986 6.986 0 0 1-5.402-2.547l-.581 3.297L15 15.67V23h-2v-5.986l-2.05-1.987-.947 4.298-6.894-1.215.348-1.97 4.924.868L9.83 8.79zM13.5 5.5a2 2 0 1 1 0-4 2 2 0 0 1 0 4z" /></svg> |  | ||||||
|           <span>{$_('runners')}</span> |  | ||||||
|         </a> |  | ||||||
|       {/if} |  | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')} |  | ||||||
|         <a |  | ||||||
|           class:bg-gray-100={$router.path === '/teams/'} |  | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" |  | ||||||
|           href="/teams/"> |  | ||||||
|           <svg |  | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24" |  | ||||||
|             height="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 640 512"><path |  | ||||||
|               fill="currentColor" |  | ||||||
|               d="M96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm448 0c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm32 32h-64c-17.6 0-33.5 7.1-45.1 18.6 40.3 22.1 68.9 62 75.1 109.4h66c17.7 0 32-14.3 32-32v-32c0-35.3-28.7-64-64-64zm-256 0c61.9 0 112-50.1 112-112S381.9 32 320 32 208 82.1 208 144s50.1 112 112 112zm76.8 32h-8.3c-20.8 10-43.9 16-68.5 16s-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h288c26.5 0 48-21.5 48-48v-28.8c0-63.6-51.6-115.2-115.2-115.2zm-223.7-13.4C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" /></svg> |  | ||||||
|           <span>{$_('teams')}</span> |  | ||||||
|         </a> |  | ||||||
|       {/if} |  | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:GET')} |  | ||||||
|         <a |  | ||||||
|           class:bg-gray-100={$router.path.includes('/donors/')} |  | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" |  | ||||||
|           href="/donors/"> |  | ||||||
|           <svg |  | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" |  | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
| 						viewBox="0 0 24 24" | 						viewBox="0 0 24 24" | ||||||
| 						width="24" | 						width="24" | ||||||
|             height="24"><path fill="none" d="M0 0h24v24H0z" /> | 						height="24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 						<path | 						<path | ||||||
|               d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" /></svg> | 							d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" | ||||||
|           <span>{$_('donors')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("donors")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path.includes('/donations/')} | 					class:activenav={$router.path.includes("/donations/")} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/donations/"> | 					href="/donations/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
| 						viewBox="0 0 24 24" | 						viewBox="0 0 24 24" | ||||||
| 						width="24" | 						width="24" | ||||||
|             height="24"><path fill="none" d="M0 0h24v24H0z" /> | 						height="24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 						<path | 						<path | ||||||
|               d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" /></svg> | 							d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" | ||||||
|           <span>{$_('donations')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("donations")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('TRACK:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("TRACK:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path === '/tracks/'} | 					class:activenav={$router.path === "/tracks/"} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/tracks/"> | 					href="/tracks/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						width="24" | 						width="24" | ||||||
| 						height="24" | 						height="24" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|             viewBox="0 0 640 512"><path | 						viewBox="0 0 640 512" | ||||||
|  | 						><path | ||||||
| 							fill="currentColor" | 							fill="currentColor" | ||||||
|               d="M635.7 167.2L556.1 31.7c-8.8-15-28.3-20.1-43.5-11.5l-69 39.1L503.3 161c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L416 75l-55.2 31.3 27.9 47.4c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L333.2 122 278 153.3 337.8 255c2.2 3.7.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9l-59.7-101.7-55.2 31.3 27.9 47.4c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9l-27.9-47.5-55.2 31.3 59.7 101.7c2.2 3.7.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L84.9 262.9l-69 39.1C.7 310.7-4.6 329.8 4.2 344.8l79.6 135.6c8.8 15 28.3 20.1 43.5 11.5L624.1 210c15.2-8.6 20.4-27.8 11.6-42.8z" /></svg> | 							d="M635.7 167.2L556.1 31.7c-8.8-15-28.3-20.1-43.5-11.5l-69 39.1L503.3 161c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L416 75l-55.2 31.3 27.9 47.4c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L333.2 122 278 153.3 337.8 255c2.2 3.7.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9l-59.7-101.7-55.2 31.3 27.9 47.4c2.2 3.8.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9l-27.9-47.5-55.2 31.3 59.7 101.7c2.2 3.7.9 8.5-2.9 10.7l-13.8 7.8c-3.8 2.2-8.7.9-10.9-2.9L84.9 262.9l-69 39.1C.7 310.7-4.6 329.8 4.2 344.8l79.6 135.6c8.8 15 28.3 20.1 43.5 11.5L624.1 210c15.2-8.6 20.4-27.8 11.6-42.8z" | ||||||
|           <span>{$_('tracks')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("tracks")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path === '/scans/'} | 					class:activenav={$router.path === "/cards/"} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/scans/"> | 					href="/cards/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						width="24" | 						width="24" | ||||||
| 						height="24" | 						height="24" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|             viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> | 						viewBox="0 0 24 24" | ||||||
|  | 					> | ||||||
|  | 						<path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 						<path | 						<path | ||||||
| 							fill="currentColor" | 							fill="currentColor" | ||||||
|               d="M2 4h2v16H2V4zm4 0h1v16H6V4zm2 0h2v16H8V4zm3 0h2v16h-2V4zm3 0h2v16h-2V4zm3 0h1v16h-1V4zm2 0h3v16h-3V4z" /></svg> | 							d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("cards")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("SCAN:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/scans/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/scans/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						viewBox="0 0 24 24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 						<path | ||||||
|  | 							fill="currentColor" | ||||||
|  | 							d="M2 4h2v16H2V4zm4 0h1v16H6V4zm2 0h2v16H8V4zm3 0h2v16h-2V4zm3 0h2v16h-2V4zm3 0h1v16h-1V4zm2 0h3v16h-3V4z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
| 					<span>Scans</span> | 					<span>Scans</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path === '/contacts/'} | 					class:activenav={$router.path.includes("/contacts/")} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/contacts/"> | 					href="/contacts/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						xmlns="http://www.w3.org/2000/svg" | 						xmlns="http://www.w3.org/2000/svg" | ||||||
| 						viewBox="0 0 24 24" | 						viewBox="0 0 24 24" | ||||||
| 						width="24" | 						width="24" | ||||||
|             height="24"><path fill="none" d="M0 0h24v24H0z" /> | 						height="24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 						<path | 						<path | ||||||
|               d="M2 22a8 8 0 1 1 16 0H2zm8-9c-3.315 0-6-2.685-6-6s2.685-6 6-6 6 2.685 6 6-2.685 6-6 6zm10 4h4v2h-4v-2zm-3-5h7v2h-7v-2zm2-5h5v2h-5V7z" /></svg> | 							d="M2 22a8 8 0 1 1 16 0H2zm8-9c-3.315 0-6-2.685-6-6s2.685-6 6-6 6 2.685 6 6-2.685 6-6 6zm10 4h4v2h-4v-2zm-3-5h7v2h-7v-2zm2-5h5v2h-5V7z" | ||||||
|           <span>{$_('contacts')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("contacts")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
|       {#if store.state.jwtinfo.userdetails.permissions.includes('STATION:GET')} | 			{#if store.state.jwtinfo.userdetails.permissions.includes("STATION:GET")} | ||||||
| 				<a | 				<a | ||||||
|           class:bg-gray-100={$router.path === '/scanstations/'} | 					class:activenav={$router.path.includes("/scanstations/")} | ||||||
|           class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|           href="/scanstations/"> | 					href="/scanstations/" | ||||||
|  | 				> | ||||||
| 					<svg | 					<svg | ||||||
|             class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 						fill="currentColor" | 						fill="currentColor" | ||||||
| 						width="24" | 						width="24" | ||||||
| 						height="24" | 						height="24" | ||||||
| 						viewBox="0 0 24 24" | 						viewBox="0 0 24 24" | ||||||
|             xmlns="http://www.w3.org/2000/svg"><path | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|               fill="none" | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|               d="M0 0h24v24H0z" /> |  | ||||||
| 						<path | 						<path | ||||||
| 							fill="currentColor" | 							fill="currentColor" | ||||||
|               d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg> | 							d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" | ||||||
|           <span>{$_('scanstations')}</span> | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("scanstations")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("STATSCLIENT:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/statsclients/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/statsclients/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						viewBox="0 0 24 24" | ||||||
|  | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 						<path | ||||||
|  | 							fill="currentColor" | ||||||
|  | 							d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("statsclients")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("USER:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/users/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/users/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						viewBox="0 0 24 24" | ||||||
|  | 						><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 						<path | ||||||
|  | 							d="M12 14v8H4a8 8 0 018-8zm0-1a6 6 0 110-12 6 6 0 010 12zm2.6 5.81a3.51 3.51 0 010-1.62l-1-.57 1-1.74 1 .58a3.5 3.5 0 011.4-.82V13.5h2v1.15a3.5 3.5 0 011.4.8l1-.57 1 1.74-1 .57a3.51 3.51 0 010 1.62l1 .57-1 1.74-1-.58a3.5 3.5 0 01-1.4.82v1.14h-2v-1.15a3.5 3.5 0 01-1.4-.8l-1 .57-1-1.74 1-.57zM18 17a1 1 0 100 2 1 1 0 000-2z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("users")}</span> | ||||||
|  | 				</a> | ||||||
|  | 			{/if} | ||||||
|  | 			{#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:GET")} | ||||||
|  | 				<a | ||||||
|  | 					class:activenav={$router.path.includes("/groups/")} | ||||||
|  | 					class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|  | 					href="/groups/" | ||||||
|  | 				> | ||||||
|  | 					<svg | ||||||
|  | 						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						width="24" | ||||||
|  | 						height="24" | ||||||
|  | 						xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 						viewBox="0 0 640 512" | ||||||
|  | 						><path | ||||||
|  | 							fill="currentColor" | ||||||
|  | 							d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" | ||||||
|  | 						/></svg | ||||||
|  | 					> | ||||||
|  | 					<span>{$_("user-groups")}</span> | ||||||
| 				</a> | 				</a> | ||||||
| 			{/if} | 			{/if} | ||||||
| 			<a | 			<a | ||||||
|         class:bg-gray-100={$router.path === '/settings/'} | 				class:activenav={$router.path === "/settings/"} | ||||||
|         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 				class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|         href="/settings/"> | 				href="/settings/" | ||||||
|  | 			> | ||||||
| 				<svg | 				<svg | ||||||
|           class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 					xmlns="http://www.w3.org/2000/svg" | 					xmlns="http://www.w3.org/2000/svg" | ||||||
| 					viewBox="0 0 20 20" | 					viewBox="0 0 20 20" | ||||||
|           fill="currentColor"> | 					fill="currentColor" | ||||||
|  | 				> | ||||||
| 					<path | 					<path | ||||||
| 						fill-rule="evenodd" | 						fill-rule="evenodd" | ||||||
| 						d="M11.49 3.17c-.38-1.56-2.6-1.56-2.98 0a1.532 1.532 0 01-2.286.948c-1.372-.836-2.942.734-2.106 2.106.54.886.061 2.042-.947 2.287-1.561.379-1.561 2.6 0 2.978a1.532 1.532 0 01.947 2.287c-.836 1.372.734 2.942 2.106 2.106a1.532 1.532 0 012.287.947c.379 1.561 2.6 1.561 2.978 0a1.533 1.533 0 012.287-.947c1.372.836 2.942-.734 2.106-2.106a1.533 1.533 0 01.947-2.287c1.561-.379 1.561-2.6 0-2.978a1.532 1.532 0 01-.947-2.287c.836-1.372-.734-2.942-2.106-2.106a1.532 1.532 0 01-2.287-.947zM10 13a3 3 0 100-6 3 3 0 000 6z" | 						d="M11.49 3.17c-.38-1.56-2.6-1.56-2.98 0a1.532 1.532 0 01-2.286.948c-1.372-.836-2.942.734-2.106 2.106.54.886.061 2.042-.947 2.287-1.561.379-1.561 2.6 0 2.978a1.532 1.532 0 01.947 2.287c-.836 1.372.734 2.942 2.106 2.106a1.532 1.532 0 012.287.947c.379 1.561 2.6 1.561 2.978 0a1.533 1.533 0 012.287-.947c1.372.836 2.942-.734 2.106-2.106a1.533 1.533 0 01.947-2.287c1.561-.379 1.561-2.6 0-2.978a1.532 1.532 0 01-.947-2.287c.836-1.372-.734-2.942-2.106-2.106a1.532 1.532 0 01-2.287-.947zM10 13a3 3 0 100-6 3 3 0 000 6z" | ||||||
|             clip-rule="evenodd" /> | 						clip-rule="evenodd" | ||||||
|  | 					/> | ||||||
| 				</svg> | 				</svg> | ||||||
|         <span>{$_('settings')}</span> | 				<span>{$_("settings")}</span> | ||||||
| 			</a> | 			</a> | ||||||
| 			<a | 			<a | ||||||
|         class:bg-gray-100={$router.path === '/about/'} | 				class:activenav={$router.path === "/about/"} | ||||||
|         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 				class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
|         href="/about/"> | 				href="/about/" | ||||||
|  | 			> | ||||||
| 				<svg | 				<svg | ||||||
|           class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 					xmlns="http://www.w3.org/2000/svg" | 					xmlns="http://www.w3.org/2000/svg" | ||||||
| 					fill="none" | 					fill="none" | ||||||
| 					stroke="currentColor" | 					stroke="currentColor" | ||||||
| 					stroke-width="2" | 					stroke-width="2" | ||||||
| 					stroke-linecap="round" | 					stroke-linecap="round" | ||||||
| 					stroke-linejoin="round" | 					stroke-linejoin="round" | ||||||
|           viewBox="0 0 24 24"><circle cx="12" cy="12" r="10" /> | 					viewBox="0 0 24 24" | ||||||
|           <path d="M12 16v-4M12 8h.01" /></svg> | 					><circle cx="12" cy="12" r="10" /> | ||||||
|         <span>{$_('about')}</span> | 					<path d="M12 16v-4M12 8h.01" /></svg | ||||||
|  | 				> | ||||||
|  | 				<span>{$_("about")}</span> | ||||||
| 			</a> | 			</a> | ||||||
|       <span | 			<button | ||||||
| 				tabindex="0" | 				tabindex="0" | ||||||
|         class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900" | 				class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-200 hover:text-gray-900 w-full font-semibold" | ||||||
| 				on:click={() => { | 				on:click={() => { | ||||||
| 					AuthService.authControllerLogout(); | 					AuthService.authControllerLogout(); | ||||||
| 					logout(); | 					logout(); | ||||||
|         }}> | 				}} | ||||||
|  | 			> | ||||||
| 				<svg | 				<svg | ||||||
|           class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600" | 					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600" | ||||||
| 					fill="currentColor" | 					fill="currentColor" | ||||||
| 					width="24" | 					width="24" | ||||||
| 					height="24" | 					height="24" | ||||||
| 					xmlns="http://www.w3.org/2000/svg" | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|           viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> | 					viewBox="0 0 24 24" | ||||||
|  | 					><path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 					<path | 					<path | ||||||
|             d="M12 22C6.477 22 2 17.523 2 12S6.477 2 12 2a9.985 9.985 0 0 1 8 4h-2.71a8 8 0 1 0 .001 12h2.71A9.985 9.985 0 0 1 12 22zm7-6v-3h-8v-2h8V8l5 4-5 4z" /></svg> | 						d="M12 22C6.477 22 2 17.523 2 12S6.477 2 12 2a9.985 9.985 0 0 1 8 4h-2.71a8 8 0 1 0 .001 12h2.71A9.985 9.985 0 0 1 12 22zm7-6v-3h-8v-2h8V8l5 4-5 4z" | ||||||
|         <span>{$_('logout')}</span> | 					/></svg | ||||||
|       </span> | 				> | ||||||
|     </nav> | 				<span>{$_("logout")}</span> | ||||||
|  | 			</button> | ||||||
| 		</nav> | 		</nav> | ||||||
|  | 	</div> | ||||||
| 	<div class="ml-0 transition md:ml-60"> | 	<div class="ml-0 transition md:ml-60"> | ||||||
| 		<header | 		<header | ||||||
|  | 			class="flex items-center w-full px-4 bg-white border-b h-14 md:hidden" | ||||||
|  | 		> | ||||||
|  | 			<button | ||||||
| 				on:click={() => { | 				on:click={() => { | ||||||
| 					navOpen = true; | 					navOpen = true; | ||||||
| 				}} | 				}} | ||||||
|       class="flex items-center justify-between w-full px-4 bg-white border-b h-14 md:hidden"> | 				class="block btn btn-light md:hidden" | ||||||
|       <button 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" |  | ||||||
| 					xmlns="http://www.w3.org/2000/svg" | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|           viewBox="0 0 20 20" | 					fill="none" | ||||||
|           fill="currentcolor"><path | 					viewBox="0 0 24 24" | ||||||
|             fill-rule="evenodd" | 					stroke-width="1.5" | ||||||
|             d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4A1 1 0 013 5zm0 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zm0 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1z" | 					stroke="currentColor" | ||||||
|             clip-rule="evenodd" /></svg></button> | 					class="size-6" | ||||||
|  | 				> | ||||||
|  | 					<path | ||||||
|  | 						stroke-linecap="round" | ||||||
|  | 						stroke-linejoin="round" | ||||||
|  | 						d="M3.75 6.75h16.5M3.75 12h16.5m-16.5 5.25h16.5" | ||||||
|  | 					/> | ||||||
|  | 				</svg> | ||||||
|  | 			</button> | ||||||
|  | 			<span class="inline-block"> | ||||||
|  | 				<img src="/lfk-logo.png" alt="Logo" class="h-8 inline-block" /> | ||||||
|  | 				<span class="text-lg font-bold">LfK!Admin</span> | ||||||
|  | 			</span> | ||||||
| 		</header> | 		</header> | ||||||
|  | 		<Toaster position="top-right" /> | ||||||
| 		<slot> | 		<slot> | ||||||
| 			<NoComponentLoaded /> | 			<NoComponentLoaded /> | ||||||
| 		</slot> | 		</slot> | ||||||
| 	</div> | 	</div> | ||||||
|   <div | 	{#if navOpen === true} | ||||||
|  | 		<button | ||||||
| 			on:click={() => { | 			on:click={() => { | ||||||
| 				navOpen = false; | 				navOpen = false; | ||||||
| 			}} | 			}} | ||||||
| 			class:hidden={!navOpen} | 			class:hidden={!navOpen} | ||||||
|     class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden" /> | 			class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden" | ||||||
|  | 		/> | ||||||
|  | 	{/if} | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  | 	.collapsed_navigation { | ||||||
|  | 		transform: translateX(-100%); | ||||||
|  | 	} | ||||||
|  | 	@media (min-width: 768px) { | ||||||
|  | 		.collapsed_navigation { | ||||||
|  | 			transform: translateX(0px); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | </style> | ||||||
|   | |||||||
| @@ -1,22 +1,246 @@ | |||||||
| <script> | <script> | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
|   import StatCards from "./StatCards.svelte"; | 	import { StatsService } from "@odit/lfk-client-js"; | ||||||
| 	import store from "../../store"; | 	import store from "../../store"; | ||||||
|   let navOpen = false; | 	import StatCard from "./StatCard.svelte"; | ||||||
|  | 	const stats_promise = StatsService.statsControllerGet(); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | <div class="p-2 md:p-5 overflow-x-hidden"> | ||||||
|  | 	<h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|  | 		{$_("dashboard-greeting")} | ||||||
|  | 		<span class="text-blue-500" | ||||||
|  | 			>{store.state.jwtinfo.userdetails.firstname} | ||||||
|  | 			{store.state.jwtinfo.userdetails.lastname}</span | ||||||
|  | 		> | ||||||
|  | 	</h4> | ||||||
|  | 	{#await stats_promise} | ||||||
| 		<div | 		<div | ||||||
|   class="p-5 overflow-x-hidden" | 			class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|   on:click={() => { | 			role="alert" | ||||||
|     navOpen = false; | 		> | ||||||
|   }}> | 			<p class="font-bold">{$_("stats-are-being-loaded")}</p> | ||||||
|   <h1 class="text-3xl leading-tight"> | 			<p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     <span class="font-extrabold">{$_('dashboard-title')}</span> | 		</div> | ||||||
|     <span> | 	{:then stats} | ||||||
|       - | 		<div | ||||||
|       {$_('dashboard-greeting')}, | 			class="grid gap-1 grid-cols-2 lg:grid-cols-3 xl:grid-cols-5 2xl:grid-cols-6 sm:gap-4" | ||||||
|       <span | 		> | ||||||
|         class="text-blue-500">{store.state.jwtinfo.userdetails.firstname} {store.state.jwtinfo.userdetails.lastname}</span></span> | 			<StatCard | ||||||
|   </h1> | 				title={$_("runners")} | ||||||
|   <StatCards /> | 				value={stats.total_runners} | ||||||
|  | 				href="/runners/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					height="24" | ||||||
|  | 					width="24" | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("total-scans")} | ||||||
|  | 				value={stats.total_scans} | ||||||
|  | 				href="/scans/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					width="24" | ||||||
|  | 					height="24" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 					<path | ||||||
|  | 						fill="currentColor" | ||||||
|  | 						d="M2 4h2v16H2V4zm4 0h1v16H6V4zm2 0h2v16H8V4zm3 0h2v16h-2V4zm3 0h2v16h-2V4zm3 0h1v16h-1V4zm2 0h3v16h-3V4z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("total-donors")} | ||||||
|  | 				value={stats.total_donors} | ||||||
|  | 				href="/donors/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					width="24" | ||||||
|  | 					height="24" | ||||||
|  | 					><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("total-donation-count")} | ||||||
|  | 				value={stats.total_donations} | ||||||
|  | 				href="/donations/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					width="24" | ||||||
|  | 					height="24" | ||||||
|  | 					><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("average-donation")} | ||||||
|  | 				value={`${parseFloat(stats.average_donation / 100).toLocaleString( | ||||||
|  | 					undefined, | ||||||
|  | 					{ | ||||||
|  | 						minimumFractionDigits: 2, | ||||||
|  | 						maximumFractionDigits: 2, | ||||||
|  | 					} | ||||||
|  | 				)}`} | ||||||
|  | 				href="/donations/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					height="24" | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					width="24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M15 18.5A6.48 6.48 0 019.24 15H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24A6.491 6.491 0 0115 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3A8.955 8.955 0 0015 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06a8.262 8.262 0 000 2H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("total-donations")} | ||||||
|  | 				value={`${parseFloat(stats.total_donation / 100).toLocaleString( | ||||||
|  | 					undefined, | ||||||
|  | 					{ | ||||||
|  | 						minimumFractionDigits: 2, | ||||||
|  | 						maximumFractionDigits: 2, | ||||||
|  | 					} | ||||||
|  | 				)}`} | ||||||
|  | 				href="/donations/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					height="24" | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					width="24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M15 18.5A6.48 6.48 0 019.24 15H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24A6.491 6.491 0 0115 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3A8.955 8.955 0 0015 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06a8.262 8.262 0 000 2H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("total-distance")} | ||||||
|  | 				value={`${stats.total_distance / 1000}km`} | ||||||
|  | 				href="/scans/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					height="24" | ||||||
|  | 					width="24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("average-distance")} | ||||||
|  | 				value={`${parseFloat(stats.average_distance / 1000).toLocaleString( | ||||||
|  | 					undefined, | ||||||
|  | 					{ | ||||||
|  | 						minimumFractionDigits: 2, | ||||||
|  | 						maximumFractionDigits: 2, | ||||||
|  | 					} | ||||||
|  | 				)}km`} | ||||||
|  | 				href="/scans/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					height="24" | ||||||
|  | 					width="24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("count_teams")} | ||||||
|  | 				value={stats.total_teams} | ||||||
|  | 				href="/teams/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					stroke="currentColor" | ||||||
|  | 					fill="none" | ||||||
|  | 					stroke-width="2" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					stroke-linecap="round" | ||||||
|  | 					stroke-linejoin="round" | ||||||
|  | 					size="24" | ||||||
|  | 					class="stroke-current text-grey-500" | ||||||
|  | 					height="24" | ||||||
|  | 					width="24" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					><path d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2" /> | ||||||
|  | 					<circle cx="9" cy="7" r="4" /> | ||||||
|  | 					<path d="M23 21v-2a4 4 0 0 0-3-3.87" /> | ||||||
|  | 					<path d="M16 3.13a4 4 0 0 1 0 7.75" /></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("count_organizations")} | ||||||
|  | 				value={stats.total_orgs} | ||||||
|  | 				href="/orgs/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					height="24" | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					width="24" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M17 11V3H7v4H3v14h8v-4h2v4h8V11h-4zM7 19H5v-2h2v2zm0-4H5v-2h2v2zm0-4H5V9h2v2zm4 4H9v-2h2v2zm0-4H9V9h2v2zm0-4H9V5h2v2zm4 8h-2v-2h2v2zm0-4h-2V9h2v2zm0-4h-2V5h2v2zm4 12h-2v-2h2v2zm0-4h-2v-2h2v2z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 			<StatCard | ||||||
|  | 				title={$_("runner_via_selfservice")} | ||||||
|  | 				value={stats.runnersViaSelfservice} | ||||||
|  | 				href="/runners/" | ||||||
|  | 			> | ||||||
|  | 				<svg | ||||||
|  | 					height="24" | ||||||
|  | 					width="24" | ||||||
|  | 					fill="currentColor" | ||||||
|  | 					xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 					viewBox="0 0 24 24" | ||||||
|  | 					><path d="M0 0h24v24H0z" fill="none" /> | ||||||
|  | 					<path | ||||||
|  | 						d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z" | ||||||
|  | 					/></svg | ||||||
|  | 				> | ||||||
|  | 			</StatCard> | ||||||
|  | 		</div> | ||||||
|  | 	{:catch error} | ||||||
|  | 		<div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | ||||||
|  | 			<span class="inline-block align-middle mr-8"> | ||||||
|  | 				<b class="capitalize">{$_("general_promise_error")}</b> | ||||||
|  | 				{error} | ||||||
|  | 			</span> | ||||||
|  | 		</div> | ||||||
|  | 	{/await} | ||||||
| </div> | </div> | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								src/components/dashboard/StatCard.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/components/dashboard/StatCard.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |  | ||||||
|  |   export let href = "#"; | ||||||
|  |   export let title = ""; | ||||||
|  |   export let value = ""; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <a {href}> | ||||||
|  |   <div class="p-3 py-4 sm:p-4 rounded-lg bg-white border border-grey-100"> | ||||||
|  |     <div class="flex flex-row items-center justify-between"> | ||||||
|  |       <div class="flex flex-col"> | ||||||
|  |         <div class="text-md sm:text-xs uppercase font-normal text-grey-500"> | ||||||
|  |           {title} | ||||||
|  |         </div> | ||||||
|  |         <div class="text-2xl sm:text-xl font-bold font-mono">{value}</div> | ||||||
|  |       </div> | ||||||
|  |       <slot /> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </a> | ||||||
| @@ -1,165 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import { StatsService } from "@odit/lfk-client-js"; |  | ||||||
|   import { _ } from "svelte-i18n"; |  | ||||||
|   const stats_promise = StatsService.statsControllerGet(); |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <!--  --> |  | ||||||
| <h1>{$_('general-stats')}</h1> |  | ||||||
| {#await stats_promise} |  | ||||||
|   <div |  | ||||||
|     class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |  | ||||||
|     role="alert"> |  | ||||||
|     <p class="font-bold">{$_('stats-are-being-loaded')}</p> |  | ||||||
|     <p class="text-sm">{$_('this-might-take-a-moment')}</p> |  | ||||||
|   </div> |  | ||||||
| {:then stats} |  | ||||||
|   <div |  | ||||||
|     class="flex flex-col lg:flex-row w-full lg:space-x-2 space-y-2 lg:space-y-0 mb-2 lg:mb-4"> |  | ||||||
|     <a href="/runners/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('runners')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_runners}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 24 24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-scans')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_scans}</div> |  | ||||||
|           </div><svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="currentColor" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             size="24" |  | ||||||
|             class="stroke-current text-grey-500" |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><polyline |  | ||||||
|               points="22 12 18 12 15 21 9 3 6 12 2 12" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-donations')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_donation} €</div> |  | ||||||
|           </div><svg |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             height="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M15 18.5A6.48 6.48 0 019.24 15H15v-2H8.58c-.05-.33-.08-.66-.08-1s.03-.67.08-1H15V9H9.24A6.491 6.491 0 0115 5.5c1.61 0 3.09.59 4.23 1.57L21 5.3A8.955 8.955 0 0015 3c-3.92 0-7.24 2.51-8.48 6H3v2h3.06a8.262 8.262 0 000 2H3v2h3.52c1.24 3.49 4.56 6 8.48 6 2.31 0 4.41-.87 6-2.3l-1.78-1.77c-1.13.98-2.6 1.57-4.22 1.57z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('total-distance')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold"> |  | ||||||
|               {stats.total_distance / 1000} |  | ||||||
|               km |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             fill="currentColor" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             height="24" |  | ||||||
|             width="24"><path d="M0 0h24v24H0z" fill="none" /> |  | ||||||
|             <path |  | ||||||
|               d="M21 6H3c-1.1 0-2 .9-2 2v8c0 1.1.9 2 2 2h18c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2zm0 10H3V8h2v4h2V8h2v4h2V8h2v4h2V8h2v4h2V8h2v8z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <a href="/teams/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('count_teams')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_teams}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="none" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             size="24" |  | ||||||
|             class="stroke-current text-grey-500" |  | ||||||
|             height="24" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><path |  | ||||||
|               d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2" /> |  | ||||||
|             <circle cx="9" cy="7" r="4" /> |  | ||||||
|             <path d="M23 21v-2a4 4 0 0 0-3-3.87" /> |  | ||||||
|             <path d="M16 3.13a4 4 0 0 1 0 7.75" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|     <a href="/orgs/" class="w-full lg:w-1/4"> |  | ||||||
|       <div |  | ||||||
|         class="widget w-full p-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|         <div class="flex flex-row items-center justify-between"> |  | ||||||
|           <div class="flex flex-col"> |  | ||||||
|             <div class="text-xs uppercase font-light text-grey-500"> |  | ||||||
|               {$_('count_organizations')} |  | ||||||
|             </div> |  | ||||||
|             <div class="text-xl font-bold">{stats.total_orgs}</div> |  | ||||||
|           </div> |  | ||||||
|           <svg |  | ||||||
|             height="24" |  | ||||||
|             fill="currentColor" |  | ||||||
|             width="24" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg" |  | ||||||
|             viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
|             <path |  | ||||||
|               d="M17 11V3H7v4H3v14h8v-4h2v4h8V11h-4zM7 19H5v-2h2v2zm0-4H5v-2h2v2zm0-4H5V9h2v2zm4 4H9v-2h2v2zm0-4H9V9h2v2zm0-4H9V5h2v2zm4 8h-2v-2h2v2zm0-4h-2V9h2v2zm0-4h-2V5h2v2zm4 12h-2v-2h2v2zm0-4h-2v-2h2v2z" /></svg> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </a> |  | ||||||
|   </div> |  | ||||||
| {:catch error} |  | ||||||
|   <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |  | ||||||
|     <span class="inline-block align-middle mr-8"> |  | ||||||
|       <b class="capitalize">{$_('general_promise_error')}</b> |  | ||||||
|       {error} |  | ||||||
|     </span> |  | ||||||
|   </div> |  | ||||||
| {/await} |  | ||||||
| @@ -1,39 +1,28 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { |   import { | ||||||
|     DonationService, |     DonationService, | ||||||
|     DonorService, |     DonorService, | ||||||
|     RunnerService, |     RunnerService, | ||||||
|   } 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 { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_donations; |   const dispatch = createEventDispatcher(); | ||||||
|   const getDonorLabel = (option) => |   const getDonorLabel = (option) => | ||||||
|     option.firstname + " " + (option.middlename || "") + " " + option.lastname; |     option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
|   const filterDonors = (label, filterText, option) => |   const filterDonors = (label, filterText, option) => | ||||||
|     label.toLowerCase().includes(filterText.toLowerCase()) || |     label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|     option.value.id.toString().startsWith(filterText.toLowerCase()); |     option.value.id.toString().startsWith(filterText.toLowerCase()); | ||||||
|   function focus(el) { |  | ||||||
|     el.focus(); |  | ||||||
|   } |  | ||||||
|   $: donor = 0; |   $: donor = 0; | ||||||
|   $: runner = 0; |   $: runner = 0; | ||||||
|   $: donors = []; |   $: donors = []; | ||||||
|   $: runners = []; |   $: runners = []; | ||||||
|   $: is_fixed = false; |   $: is_fixed = false; | ||||||
|   DonorService.donorControllerGetAll().then((val) => { |   $: is_paid = false; | ||||||
|     donors = val.map((r) => { |  | ||||||
|       return { label: getDonorLabel(r), value: r }; |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|   RunnerService.runnerControllerGetAll().then((val) => { |  | ||||||
|     runners = val.map((r) => { |  | ||||||
|       return { label: getDonorLabel(r), value: r }; |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|   $: amount_input = 0; |   $: amount_input = 0; | ||||||
|   $: processed_last_submit = true; |   $: processed_last_submit = true; | ||||||
|   $: is_amount_valid = amount_input > 0; |   $: is_amount_valid = amount_input > 0; | ||||||
| @@ -56,15 +45,16 @@ | |||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       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({ |       toast.loading($_("adding-donation")); | ||||||
|         text: "adding donation", |  | ||||||
|         duration: -1, |  | ||||||
|       }).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; | ||||||
| @@ -72,21 +62,15 @@ | |||||||
|             amount_input = 0; |             amount_input = 0; | ||||||
|             modal_open = false; |             modal_open = false; | ||||||
|             // |             // | ||||||
|             Toastify({ |             toast.dismiss(); | ||||||
|               text: "donation_added", |             toast.success($_("donation_added")); | ||||||
|               duration: 500, |             dispatch("created", { donations: [result] }); | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
|             current_donations.push(result); |  | ||||||
|             current_donations = current_donations; |  | ||||||
|           }) |           }) | ||||||
|           .catch((err) => { |           .catch((err) => { | ||||||
|             // |             // | ||||||
|           }) |           }) | ||||||
|           .finally(() => { |           .finally(() => { | ||||||
|             processed_last_submit = true; |             processed_last_submit = true; | ||||||
|             // |  | ||||||
|             toast.hideToast(); |  | ||||||
|           }); |           }); | ||||||
|       } else { |       } else { | ||||||
|         let postdata = { |         let postdata = { | ||||||
| @@ -101,29 +85,237 @@ | |||||||
|             amount_input = 0; |             amount_input = 0; | ||||||
|             modal_open = false; |             modal_open = false; | ||||||
|             // |             // | ||||||
|             Toastify({ |             toast.dismiss(); | ||||||
|               text: "donation_added", |             toast.success($_("donation_added")); | ||||||
|               duration: 500, |             dispatch("created", { donations: [result] }); | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
|             current_donations.push(result); |  | ||||||
|             current_donations = current_donations; |  | ||||||
|           }) |           }) | ||||||
|           .catch((err) => { |           .catch((err) => { | ||||||
|             // |             // | ||||||
|           }) |           }) | ||||||
|           .finally(() => { |           .finally(() => { | ||||||
|             processed_last_submit = true; |             processed_last_submit = true; | ||||||
|             // |  | ||||||
|             toast.hideToast(); |  | ||||||
|           }); |           }); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   onMount(async () => { | ||||||
|  |     donors = (await DonorService.donorControllerGetAll()).map( | ||||||
|  |       (r) => { | ||||||
|  |         return { label: getDonorLabel(r), value: r }; | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |     runners = (await RunnerService.runnerControllerGetAll()).map( | ||||||
|  |       (r) => { | ||||||
|  |         return { label: getDonorLabel(r), value: r }; | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   }); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:text-left"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {#if is_fixed} | ||||||
|  |                   {$_("create-a-new-fixed-donation")} | ||||||
|  |                 {:else}{$_("create-a-new-distance-donation")}{/if} | ||||||
|  |               </h3> | ||||||
|  |               <label class="content-center align-middle object-center"> | ||||||
|  |                 <span class="text-base" class:text-gray-300={is_fixed} | ||||||
|  |                   >{$_("distance-donation")}</span | ||||||
|  |                 > | ||||||
|  |                 <input | ||||||
|  |                   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" | ||||||
|  |                   bind:checked={is_fixed} | ||||||
|  |                 /> | ||||||
|  |                 <span class="ml-2 text-base" class:text-gray-300={!is_fixed} | ||||||
|  |                   >{$_("fixed-donation")}</span | ||||||
|  |                 > | ||||||
|  |               </label> | ||||||
|  |               <div class="mt-2 mb-6"> | ||||||
|  |                 <p class="text-sm text-gray-500"> | ||||||
|  |                   {$_( | ||||||
|  |                     "please-provide-the-nessecary-information-to-create-a-new-donation" | ||||||
|  |                   )} | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="donor" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("donor")}</label | ||||||
|  |                   > | ||||||
|  |                   <Select | ||||||
|  |                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     itemFilter={(label, filterText, option) => | ||||||
|  |                       filterDonors(label, filterText, option)} | ||||||
|  |                     items={donors} | ||||||
|  |                     showChevron={true} | ||||||
|  |                     placeholder={$_("search-for-donor-name-or-id")} | ||||||
|  |                     noOptionsMessage={$_("no-donors-found")} | ||||||
|  |                     on:select={(selectedValue) => | ||||||
|  |                       (donor = selectedValue.detail.value.id)} | ||||||
|  |                     on:clear={() => (donors = null)} | ||||||
|  |                   /> | ||||||
|  |                 </div> | ||||||
|  |                 {#if !is_fixed} | ||||||
|  |                   <div class="col-span-6"> | ||||||
|  |                     <label | ||||||
|  |                       for="donor" | ||||||
|  |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("runner")}</label | ||||||
|  |                     > | ||||||
|  |                     <Select | ||||||
|  |                       containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                       itemFilter={(label, filterText, option) => | ||||||
|  |                         filterDonors(label, filterText, option)} | ||||||
|  |                       items={runners} | ||||||
|  |                       showChevron={true} | ||||||
|  |                       placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|  |                       noOptionsMessage={$_("no-runners-found")} | ||||||
|  |                       on:select={(selectedValue) => | ||||||
|  |                         (runner = selectedValue.detail.value.id)} | ||||||
|  |                       on:clear={() => (runner = null)} | ||||||
|  |                     /> | ||||||
|  |                   </div> | ||||||
|  |                 {/if} | ||||||
|  |                 <div class="col-span-6"> | ||||||
|  |                   <label | ||||||
|  |                     for="donation_amount_eur" | ||||||
|  |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                   > | ||||||
|  |                     {#if !is_fixed} | ||||||
|  |                       {$_("amount-per-kilometer")} | ||||||
|  |                     {:else}{$_("donation-amount")}{/if}</label | ||||||
|  |                   > | ||||||
|  |                   <div class="mt-1 flex rounded-md shadow-sm"> | ||||||
|  |                     <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={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 border-gray-300 border bg-gray-50 text-neutral-800 p-2" | ||||||
|  |                       placeholder="2.00" | ||||||
|  |                     /> | ||||||
|  |                     <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_amount_valid} | ||||||
|  |                     <span | ||||||
|  |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|  |                     > | ||||||
|  |                       {$_("donation-amount-must-be-greater-that-0-00eur")} | ||||||
|  |                     </span> | ||||||
|  |                   {/if} | ||||||
|  |                 </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 class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <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" | ||||||
|  |           > | ||||||
|  |             {$_("create")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
|  |  | ||||||
| <style> | <style> | ||||||
|   input:before { |   .toggle:before { | ||||||
|     content: ""; |     content: ""; | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     width: 1.25rem; |     width: 1.25rem; | ||||||
| @@ -137,159 +329,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> | ||||||
|  |  | ||||||
| {#if modal_open} |  | ||||||
|   <div |  | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |  | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |  | ||||||
|     on:click_outside={() => { |  | ||||||
|       modal_open = false; |  | ||||||
|     }}> |  | ||||||
|     <div |  | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |  | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |  | ||||||
|         <div |  | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |  | ||||||
|           data-id="modal_backdrop" /> |  | ||||||
|       </div> |  | ||||||
|       <span |  | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |  | ||||||
|         aria-hidden="true">​</span> |  | ||||||
|       <div |  | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |  | ||||||
|         role="dialog" |  | ||||||
|         aria-modal="true" |  | ||||||
|         aria-labelledby="modal-headline"> |  | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |  | ||||||
|           <div class="sm:flex sm:items-start"> |  | ||||||
|             <div |  | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |  | ||||||
|               <svg |  | ||||||
|                 class="h-6 w-6 text-blue-600" |  | ||||||
|                 fill="currentColor" |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 width="24" |  | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |  | ||||||
|                 <path |  | ||||||
|                   d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 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"> |  | ||||||
|                 {#if is_fixed} |  | ||||||
|                   {$_('create-a-new-fixed-donation')} |  | ||||||
|                 {:else}{$_('create-a-new-distance-donation')}{/if} |  | ||||||
|               </h3> |  | ||||||
|               <label class="content-center align-middle object-center"> |  | ||||||
|                 <span |  | ||||||
|                   class="ml-2 text-base" |  | ||||||
|                   class:text-gray-300={is_fixed}>{$_('distance-donation')}</span> |  | ||||||
|                 <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" |  | ||||||
|                   type="checkbox" |  | ||||||
|                   bind:checked={is_fixed} /> |  | ||||||
|                 <span |  | ||||||
|                   class="ml-2 text-base	" |  | ||||||
|                   class:text-gray-300={!is_fixed}>{$_('fixed-donation')}</span> |  | ||||||
|               </label> |  | ||||||
|               <div class="mt-2 mb-6"> |  | ||||||
|                 <p class="text-sm text-gray-500"> |  | ||||||
|                   {$_('please-provide-the-nessecary-information-to-create-a-new-donation')} |  | ||||||
|                 </p> |  | ||||||
|               </div> |  | ||||||
|               <div class="grid grid-cols-6 gap-6"> |  | ||||||
|                 <div class="col-span-6"> |  | ||||||
|                   <label |  | ||||||
|                     for="donor" |  | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('donor')}</label> |  | ||||||
|                   <Select |  | ||||||
|                     containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" |  | ||||||
|                     itemFilter={(label, filterText, option) => filterDonors(label, filterText, option)} |  | ||||||
|                     items={donors} |  | ||||||
|                     showChevron={true} |  | ||||||
|                     placeholder={$_('search-for-donor-name-or-id')} |  | ||||||
|                     noOptionsMessage={$_('no-donors-found')} |  | ||||||
|                     on:select={(selectedValue) => (donor = selectedValue.detail.value.id)} |  | ||||||
|                     on:clear={() => (donors = null)} /> |  | ||||||
|                 </div> |  | ||||||
|                 {#if !is_fixed} |  | ||||||
|                   <div class="col-span-6"> |  | ||||||
|                     <label |  | ||||||
|                       for="donor" |  | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('runner')}</label> |  | ||||||
|                     <Select |  | ||||||
|                       containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" |  | ||||||
|                       itemFilter={(label, filterText, option) => filterDonors(label, filterText, option)} |  | ||||||
|                       items={runners} |  | ||||||
|                       showChevron={true} |  | ||||||
|                       placeholder={$_('search-for-runner-by-name-or-id')} |  | ||||||
|                       noOptionsMessage={$_('no-runners-found')} |  | ||||||
|                       on:select={(selectedValue) => (runner = selectedValue.detail.value.id)} |  | ||||||
|                       on:clear={() => (runner = null)} /> |  | ||||||
|                   </div> |  | ||||||
|                 {/if} |  | ||||||
|                 <div class="col-span-6"> |  | ||||||
|                   <label |  | ||||||
|                     for="donation_amount_eur" |  | ||||||
|                     class="block text-sm font-medium text-gray-700"> |  | ||||||
|                     {#if !is_fixed} |  | ||||||
|                       {$_('amount-per-kilometer')} |  | ||||||
|                     {:else}{$_('donation-amount')}{/if}</label> |  | ||||||
|                   <div class="mt-1 flex rounded-md shadow-sm"> |  | ||||||
|                     <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={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 border-gray-300 border bg-gray-50 text-gray-500 p-2" |  | ||||||
|                       placeholder="2.00" /> |  | ||||||
|                     <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_amount_valid} |  | ||||||
|                     <span |  | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |  | ||||||
|                       {$_('donation-amount-must-be-greater-that-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"> |  | ||||||
|             {$_('create')} |  | ||||||
|           </button> |  | ||||||
|           <button |  | ||||||
|             on:click={() => { |  | ||||||
|               modal_open = false; |  | ||||||
|             }} |  | ||||||
|             type="button" |  | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |  | ||||||
|             {$_('cancel')} |  | ||||||
|           </button> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| {/if} |  | ||||||
|   | |||||||
							
								
								
									
										205
									
								
								src/components/donations/AddDonationPaymentModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								src/components/donations/AddDonationPaymentModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,205 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { DonationService } from "@odit/lfk-client-js"; | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|  |   export let payment_modal_open = false; | ||||||
|  |   export let original_data = {}; | ||||||
|  |   export let paid_amount_input = 0; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   $: processed_last_submit = true; | ||||||
|  |   $: 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; | ||||||
|  |       toast.loading($_("updating-donation")); | ||||||
|  |       const editable = Object.assign({}, original_data); | ||||||
|  |       editable.donor = editable.donor.id; | ||||||
|  |       editable.paidAmount = Math.round(paid_amount_input * 100); | ||||||
|  |       if (editable.responseType == "DISTANCEDONATION" || editable.runner) { | ||||||
|  |         editable.runner = editable.runner.id; | ||||||
|  |         DonationService.donationControllerPutDistance( | ||||||
|  |           original_data.id, | ||||||
|  |           editable | ||||||
|  |         ) | ||||||
|  |           .then((result) => { | ||||||
|  |             payment_modal_open = false; | ||||||
|  |             // | ||||||
|  |             toast.dismiss(); | ||||||
|  |  | ||||||
|  |             toast.success($_("donation-updated")); | ||||||
|  |             dispatch("created", { donation: result }); | ||||||
|  |           }) | ||||||
|  |           .catch((err) => { | ||||||
|  |             // | ||||||
|  |           }) | ||||||
|  |           .finally(() => { | ||||||
|  |             processed_last_submit = true; | ||||||
|  |           }); | ||||||
|  |       } else { | ||||||
|  |         DonationService.donationControllerPutFixed(original_data.id, editable) | ||||||
|  |           .then((result) => { | ||||||
|  |             payment_modal_open = false; | ||||||
|  |             // | ||||||
|  |             toast.dismiss(); | ||||||
|  |             toast.success($_("donation-updated")); | ||||||
|  |             dispatch("created", { donation: result }); | ||||||
|  |           }) | ||||||
|  |           .catch((err) => { | ||||||
|  |             // | ||||||
|  |           }) | ||||||
|  |           .finally(() => { | ||||||
|  |             processed_last_submit = true; | ||||||
|  |           }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if payment_modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       payment_modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <div | ||||||
|  |               class="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-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 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <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" | ||||||
|  |           > | ||||||
|  |             {$_("save-changes")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               payment_modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										117
									
								
								src/components/donations/DeleteDonationModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								src/components/donations/DeleteDonationModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import { clickOutside } from "../base/outsideclick"; | ||||||
|  |   import { createEventDispatcher, onMount } from "svelte"; | ||||||
|  |   export let modal_open; | ||||||
|  |   export let delete_donation = { | ||||||
|  |     id: 0, | ||||||
|  |     runner: { | ||||||
|  |       firstname: "", | ||||||
|  |       lastname: "", | ||||||
|  |     }, | ||||||
|  |     donor: { | ||||||
|  |       firstname: "", | ||||||
|  |       lastname: "", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|  |   onMount(() => { | ||||||
|  |     document.onkeydown = (e) => { | ||||||
|  |       e = e || window.event; | ||||||
|  |       if (e.key === "Escape") { | ||||||
|  |         modal_open = false; | ||||||
|  |       } | ||||||
|  |       if (e.keyCode === 13) { | ||||||
|  |         if (createbtnenabled === true) { | ||||||
|  |           createbtnenabled = false; | ||||||
|  |           submit(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |   async function submit() { | ||||||
|  |     dispatch("delete", { id: delete_donation.id }); | ||||||
|  |     modal_open = false; | ||||||
|  |   } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  |   <div | ||||||
|  |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  |     use:clickOutside | ||||||
|  |     on:click_outside={() => { | ||||||
|  |       modal_open = false; | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|  |     <div | ||||||
|  |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|  |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  |         <div | ||||||
|  |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|  |       </div> | ||||||
|  |       <span | ||||||
|  |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|  |       <div | ||||||
|  |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  |         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 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|  |             <div | ||||||
|  |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|  |               <svg | ||||||
|  |                 class="h-6 w-6 text-blue-600" | ||||||
|  |                 fill="currentColor" | ||||||
|  |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                 viewBox="0 0 24 24" | ||||||
|  |                 width="24" | ||||||
|  |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  |                 <path | ||||||
|  |                   d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|  |             </div> | ||||||
|  |             <div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|  |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  |                 {$_("please-confirm-the-deletion-of-donation")} | ||||||
|  |               </h3> | ||||||
|  |               <div class="w-full"> | ||||||
|  |                 <span class="inline-block" | ||||||
|  |                   ><b>{$_("donor")}</b>: {delete_donation.donor.firstname} | ||||||
|  |                   {delete_donation.donor.lastname}</span | ||||||
|  |                 > | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  |           <button | ||||||
|  |             on:click={submit} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-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" | ||||||
|  |           > | ||||||
|  |             {$_("delete")} | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             on:click={() => { | ||||||
|  |               modal_open = false; | ||||||
|  |             }} | ||||||
|  |             type="button" | ||||||
|  |             class="w-full 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 hidden lg:block" | ||||||
|  |           > | ||||||
|  |             {$_("cancel")} | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
| @@ -6,7 +6,8 @@ | |||||||
| 		DonorService, | 		DonorService, | ||||||
| 		RunnerService, | 		RunnerService, | ||||||
| 	} from "@odit/lfk-client-js"; | 	} from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; | 	import toast from "svelte-french-toast"; | ||||||
|  |  | ||||||
| 	import PromiseError from "../base/PromiseError.svelte"; | 	import PromiseError from "../base/PromiseError.svelte"; | ||||||
| 	import Select from "svelte-select"; | 	import Select from "svelte-select"; | ||||||
| 	let data_loaded = false; | 	let data_loaded = false; | ||||||
| @@ -20,6 +21,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 +33,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) => { | ||||||
| @@ -65,11 +70,9 @@ | |||||||
|  |  | ||||||
| 	function submit() { | 	function submit() { | ||||||
| 		if (data_loaded === true && save_enabled) { | 		if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ | 			toast($_("updating-donation")); | ||||||
|         text: "Donation is being updated", |  | ||||||
|         duration: 2500, |  | ||||||
|       }).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); | ||||||
| @@ -82,11 +85,7 @@ | |||||||
| 					.then((resp) => { | 					.then((resp) => { | ||||||
| 						Object.assign(original_data, editable); | 						Object.assign(original_data, editable); | ||||||
| 						original_data = original_data; | 						original_data = original_data; | ||||||
|             Toastify({ | 						toast.success($_("donation-updated")); | ||||||
|               text: "updated donation", |  | ||||||
|               duration: 2500, |  | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
| 					}) | 					}) | ||||||
| 					.catch((err) => {}); | 					.catch((err) => {}); | ||||||
| 			} else { | 			} else { | ||||||
| @@ -97,11 +96,7 @@ | |||||||
| 					.then((resp) => { | 					.then((resp) => { | ||||||
| 						Object.assign(original_data, editable); | 						Object.assign(original_data, editable); | ||||||
| 						original_data = original_data; | 						original_data = original_data; | ||||||
|             Toastify({ | 						toast.success($_("donation-updated")); | ||||||
|               text: "updated donation", |  | ||||||
|               duration: 2500, |  | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
| 					}) | 					}) | ||||||
| 					.catch((err) => {}); | 					.catch((err) => {}); | ||||||
| 			} | 			} | ||||||
| @@ -111,11 +106,7 @@ | |||||||
| 	function deleteDonation() { | 	function deleteDonation() { | ||||||
| 		DonationService.donationControllerRemove(original_data.id, false) | 		DonationService.donationControllerRemove(original_data.id, false) | ||||||
| 			.then((resp) => { | 			.then((resp) => { | ||||||
|         Toastify({ | 				toast.success($_("donation-deleted")); | ||||||
|           text: "Donation delete", |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
| 				location.replace("./"); | 				location.replace("./"); | ||||||
| 			}) | 			}) | ||||||
| 			.catch((err) => { | 			.catch((err) => { | ||||||
| @@ -126,72 +117,64 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#await promise} | {#await promise} | ||||||
|   {$_('loading-donation-details')} | 	{$_("loading-donation-details")} | ||||||
| {:then} | {:then} | ||||||
| 	<section class="container p-5 select-none"> | 	<section class="container p-5 select-none"> | ||||||
| 		<div class="flex flex-row mb-4"> | 		<div class="flex flex-row mb-4"> | ||||||
|       <div class="w-full"> | 			<div class="mt-2 w-full"> | ||||||
| 				<nav class="w-full flex"> | 				<nav class="w-full flex"> | ||||||
| 					<ol class="list-none flex flex-row items-center justify-start"> | 					<ol class="list-none flex flex-row items-center justify-start"> | ||||||
| 						<li class="flex items-center"> | 						<li class="flex items-center"> | ||||||
|               <svg | 							<a class="mr-2" href="./" | ||||||
|                 fill="currentColor" | 								><svg | ||||||
| 									xmlns="http://www.w3.org/2000/svg" | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
| 									width="24" | 									width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | 									height="24" | ||||||
|                 <path |  | ||||||
|                   d="M14 2a8 8 0 013.3 15.3A8 8 0 116.7 6.7 8 8 0 0114 2zm-3 7H9v1a2.5 2.5 0 00-.16 5h2.25a.5.5 0 010 1H7v2h2v1h2v-1a2.5 2.5 0 00.16-5H8.91a.5.5 0 010-1H13v-2h-2V9zm3-5a5.99 5.99 0 00-4.48 2.01 8 8 0 018.47 8.47A6 6 0 0014 4z" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center ml-2"> |  | ||||||
|               <a class="mr-2" href="./">{$_('donations')}</a><svg |  | ||||||
|                 stroke="currentColor" |  | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
| 									viewBox="0 0 24 24" | 									viewBox="0 0 24 24" | ||||||
|  | 									fill="none" | ||||||
|  | 									stroke="currentColor" | ||||||
|  | 									stroke-width="2" | ||||||
| 									stroke-linecap="round" | 									stroke-linecap="round" | ||||||
| 									stroke-linejoin="round" | 									stroke-linejoin="round" | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" | 									class="inline-block" | ||||||
|                 height="1em" | 									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg | ||||||
|                 width="1em" | 								> | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line | 								{$_("donations")}</a | ||||||
|                   x1="5" | 							> | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <span class="mr-2">{original_data.id}</span> |  | ||||||
| 						</li> | 						</li> | ||||||
| 					</ol> | 					</ol> | ||||||
| 				</nav> | 				</nav> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> | 		<div class="mb-4 text-3xl font-extrabold leading-tight"> | ||||||
| 			{original_data.donor.firstname} | 			{original_data.donor.firstname} | ||||||
|       {original_data.donor.middlename || ''} | 			{original_data.donor.middlename || ""} | ||||||
| 			{original_data.donor.lastname} | 			{original_data.donor.lastname} | ||||||
| 			> | 			> | ||||||
|       {#if original_data.responseType == 'DISTANCEDONATION'} | 			{#if original_data.responseType == "DISTANCEDONATION"} | ||||||
| 				{original_data.runner.firstname} | 				{original_data.runner.firstname} | ||||||
|         {original_data.runner.middlename || ''} | 				{original_data.runner.middlename || ""} | ||||||
| 				{original_data.runner.lastname} | 				{original_data.runner.lastname} | ||||||
| 			{:else} | 			{:else} | ||||||
|         {$_('fixed-donation')}: | 				{$_("fixed-donation")}: | ||||||
|         {amount_input.toFixed(2).toLocaleString('de-DE', { valute: 'EUR' })}€ | 				{amount_input.toFixed(2).toLocaleString("de-DE", { valute: "EUR" })}€ | ||||||
| 			{/if} | 			{/if} | ||||||
|       <span data-id="donation_actions_${original_data.id}"> | 			[#{original_data.id}] | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:DELETE')} | 			<div data-id="donation_actions_${original_data.id}"> | ||||||
|  | 				{#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:DELETE")} | ||||||
| 					{#if delete_triggered} | 					{#if delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| 							on:click={deleteDonation} | 							on:click={deleteDonation} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:">{$_('confirm-deletion')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("confirm-deletion")}</button | ||||||
|  | 						> | ||||||
| 						<button | 						<button | ||||||
| 							on:click={() => { | 							on:click={() => { | ||||||
| 								delete_triggered = !delete_triggered; | 								delete_triggered = !delete_triggered; | ||||||
| 							}} | 							}} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:">{$_('cancel')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("cancel")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 					{#if !delete_triggered} | 					{#if !delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| @@ -199,7 +182,9 @@ | |||||||
| 								delete_triggered = true; | 								delete_triggered = true; | ||||||
| 							}} | 							}} | ||||||
| 							type="button" | 							type="button" | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:">{$_('delete-donation')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("delete-donation")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 				{/if} | 				{/if} | ||||||
| 				{#if !delete_triggered} | 				{#if !delete_triggered} | ||||||
| @@ -208,55 +193,91 @@ | |||||||
| 						class:opacity-50={!save_enabled} | 						class:opacity-50={!save_enabled} | ||||||
| 						type="button" | 						type="button" | ||||||
| 						on:click={submit} | 						on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:">{$_('save-changes')}</button> | 						class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 						>{$_("save-changes")}</button | ||||||
|  | 					> | ||||||
| 				{/if} | 				{/if} | ||||||
|       </span> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<!--  --> | 		<!--  --> | ||||||
| 		<div> | 		<div> | ||||||
|  | 			<span class="font-semibold text-gray-700" | ||||||
|  | 				>{$_("total-donation-amount")}:</span | ||||||
|  | 			> | ||||||
| 			<span | 			<span | ||||||
|         class="font-medium text-gray-700">{$_('total-donation-amount')}:</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-semibold text-gray-700">{$_("paid-amount")}:</span> | ||||||
|  | 			<span | ||||||
|  | 				>{(editable.paidAmount / 100) | ||||||
|  | 					.toFixed(2) | ||||||
|  | 					.toLocaleString("de-DE", { valute: "EUR" })}€</span | ||||||
|  | 			> | ||||||
|  | 			| | ||||||
|  | 			<span class="font-semibold text-gray-700">{$_("status")}:</span> | ||||||
|  | 			{#if editable.status == "PAID"} | ||||||
|  | 				<span | ||||||
|  | 					class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-green-100 text-green-800" | ||||||
|  | 					>{$_("paid")}</span | ||||||
|  | 				> | ||||||
|  | 			{:else} | ||||||
|  | 				<span | ||||||
|  | 					class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-red-100 text-red-800" | ||||||
|  | 					>{$_("open")}</span | ||||||
|  | 				> | ||||||
|  | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<br /> | ||||||
|       <label | 		<div class=" mt-2 w-full"> | ||||||
|         for="donor" | 			<label for="donor" class="block font-semibold text-gray-700" | ||||||
|         class="block  font-medium text-gray-700">{$_('donor')}</label> | 				>{$_("donor")}</label | ||||||
|  | 			> | ||||||
| 			<Select | 			<Select | ||||||
|         containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | 				containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|         itemFilter={(label, filterText, option) => filterDonors(label, filterText, option)} | 				itemFilter={(label, filterText, option) => | ||||||
|  | 					filterDonors(label, filterText, option)} | ||||||
| 				items={current_donors} | 				items={current_donors} | ||||||
| 				showChevron={true} | 				showChevron={true} | ||||||
|         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) => { | ||||||
|         on:clear={() => (editable.donor = null)} /> | 					editable.donor = selectedValue.detail.value; | ||||||
|  | 					editable.donor.donationAmount = original_data.donor.donationAmount; | ||||||
|  | 					editable.donor.paidDonationAmount = | ||||||
|  | 						original_data.donor.paidDonationAmount; | ||||||
|  | 				}} | ||||||
|  | 				on:clear={() => (editable.donor = null)} | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|     {#if original_data.responseType == 'DISTANCEDONATION'} | 		{#if original_data.responseType == "DISTANCEDONATION"} | ||||||
|       <div class=" w-full"> | 			<div class=" mt-2 w-full"> | ||||||
|         <label | 				<label for="donor" class="block font-semibold text-gray-700" | ||||||
|           for="donor" | 					>{$_("runner")}</label | ||||||
|           class="block  font-medium text-gray-700">{$_('runner')}</label> | 				> | ||||||
| 				<Select | 				<Select | ||||||
|           containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | 					containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|           itemFilter={(label, filterText, option) => filterDonors(label, filterText, option)} | 					itemFilter={(label, filterText, option) => | ||||||
|  | 						filterDonors(label, filterText, option)} | ||||||
| 					items={current_runners} | 					items={current_runners} | ||||||
| 					showChevron={true} | 					showChevron={true} | ||||||
|           placeholder={$_('search-for-runner-by-name-or-id')} | 					placeholder={$_("search-for-runner-by-name-or-id")} | ||||||
|           noOptionsMessage={$_('no-runners-found')} | 					noOptionsMessage={$_("no-runners-found")} | ||||||
| 					bind:selectedValue={runner} | 					bind:selectedValue={runner} | ||||||
|           on:select={(selectedValue) => (editable.runner = selectedValue.detail.value)} | 					on:select={(selectedValue) => | ||||||
|           on:clear={() => (editable.runner = null)} /> | 						(editable.runner = selectedValue.detail.value)} | ||||||
|  | 					on:clear={() => (editable.runner = null)} | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 		{/if} | 		{/if} | ||||||
|     <div class=" w-full"> | 		<div class=" mt-2 w-full"> | ||||||
|       <label for="lastname" class="font-medium text-gray-700"> | 			<label for="lastname" class="font-semibold text-gray-700"> | ||||||
|         {#if original_data.responseType == 'DISTANCEDONATION'} | 				{#if original_data.responseType == "DISTANCEDONATION"} | ||||||
|           {$_('amount-per-kilometer')} | 					{$_("amount-per-kilometer")} | ||||||
|         {:else}{$_('donation-amount')}{/if} | 				{:else}{$_("donation-amount")}{/if} | ||||||
| 			</label> | 			</label> | ||||||
| 			<div class="mt-1 flex rounded-md shadow-sm"> | 			<div class="mt-1 flex rounded-md shadow-sm"> | ||||||
| 				<input | 				<input | ||||||
| @@ -268,15 +289,60 @@ | |||||||
| 					type="number" | 					type="number" | ||||||
| 					step="0.01" | 					step="0.01" | ||||||
| 					name="donation_amount_eur" | 					name="donation_amount_eur" | ||||||
|           class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 p-2" | 					class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 p-2" | ||||||
|           placeholder="2.00" /> | 					placeholder="2.00" | ||||||
|  | 				/> | ||||||
| 				<span | 				<span | ||||||
|           class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500 ">€</span> | 					class="inline-flex items-center px-3 rounded-r-md border border-gray-300 bg-gray-50 text-gray-500" | ||||||
|  | 					>€</span | ||||||
|  | 				> | ||||||
| 			</div> | 			</div> | ||||||
| 			{#if !is_amount_valid} | 			{#if !is_amount_valid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('donation-amount-must-be-greater-that-0-00eur')} | 				> | ||||||
|  | 					{$_("donation-amount-must-be-greater-that-0-00eur")} | ||||||
|  | 				</span> | ||||||
|  | 			{/if} | ||||||
|  | 		</div> | ||||||
|  | 		<div class="mt-2 w-full"> | ||||||
|  | 			<label for="token" class="block font-semibold 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> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								src/components/donations/DonationDonor.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/components/donations/DonationDonor.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let donor; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !donor || donor.firstname == 0} | ||||||
|  |   {$_("donor-has-no-associated-donations")} | ||||||
|  | {:else} | ||||||
|  |   <div class="flex items-center"> | ||||||
|  |     <a | ||||||
|  |       href="../donors/{donor.id}" | ||||||
|  |       class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800 border border-current" | ||||||
|  |       >{donor.firstname} | ||||||
|  |       {#if donor.middlename}{donor.middlename}{/if} | ||||||
|  |       {donor.lastname}</a | ||||||
|  |     > | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										18
									
								
								src/components/donations/DonationRunner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/components/donations/DonationRunner.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let runner; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !runner || runner.firstname == 0} | ||||||
|  |   {$_("fixed-donation")} | ||||||
|  | {:else} | ||||||
|  |   <div class="text-sm font-medium text-gray-900"> | ||||||
|  |     <a | ||||||
|  |       href="../runners/{runner.id}" | ||||||
|  |       class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800 border border-current" | ||||||
|  |       >{runner.firstname} | ||||||
|  |       {#if runner.middlename}{runner.middlename}{/if} | ||||||
|  |       {runner.lastname}</a | ||||||
|  |     > | ||||||
|  |   </div> | ||||||
|  | {/if} | ||||||
							
								
								
									
										16
									
								
								src/components/donations/DonationStatus.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/components/donations/DonationStatus.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let status; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if status == "PAID"} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-green-100 text-green-800" | ||||||
|  |     >{$_("paid")}</span | ||||||
|  |   > | ||||||
|  | {:else} | ||||||
|  |   <span | ||||||
|  |     class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-red-100 text-red-800" | ||||||
|  |     >{$_("open")}</span | ||||||
|  |   > | ||||||
|  | {/if} | ||||||
							
								
								
									
										21
									
								
								src/components/donations/DonationTableAction.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/components/donations/DonationTableAction.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |  | ||||||
|  |   export let detailsLink; | ||||||
|  |   export let detailsAction; | ||||||
|  |   export let deleteEnabled; | ||||||
|  |   export let deleteAction; | ||||||
|  |   export let paymentAction; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <button | ||||||
|  |   on:click={paymentAction} | ||||||
|  |   class="text-[#025a21] hover:text-green-900 mr-4">{$_("enter-payment")}</button | ||||||
|  | > | ||||||
|  | <TableActions | ||||||
|  |   bind:detailsAction | ||||||
|  |   bind:detailsLink | ||||||
|  |   bind:deleteAction | ||||||
|  |   bind:deleteEnabled | ||||||
|  | /> | ||||||
| @@ -5,25 +5,32 @@ | |||||||
|   import DonationsOverview from "./DonationsOverview.svelte"; |   import DonationsOverview from "./DonationsOverview.svelte"; | ||||||
|   $: current_donations = []; |   $: current_donations = []; | ||||||
|   export let modal_open = false; |   export let modal_open = false; | ||||||
|  |   let addDonations; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> |   <h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('donations')} |     {$_("donations")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:CREATE')} |   </h4> | ||||||
|  |   {#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:CREATE")} | ||||||
|     <button |     <button | ||||||
|       on:click={() => { |       on:click={() => { | ||||||
|         modal_open = true; |         modal_open = true; | ||||||
|       }} |       }} | ||||||
|       type="button" |       type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm" | ||||||
|         {$_('add-donation')} |     > | ||||||
|  |       {$_("add-donation")} | ||||||
|     </button> |     </button> | ||||||
|   {/if} |   {/if} | ||||||
|   </span> |   <DonationsOverview bind:current_donations bind:addDonations /> | ||||||
|   <DonationsOverview bind:current_donations /> |  | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:CREATE")} | ||||||
|   <AddDonationModal bind:current_donations bind:modal_open /> |   <AddDonationModal | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addDonations(event.detail.donations); | ||||||
|  |     }} | ||||||
|  |     bind:modal_open | ||||||
|  |   /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -5,8 +5,8 @@ | |||||||
|  |  | ||||||
| <div class="text-center items-center justify-center"> | <div class="text-center items-center justify-center"> | ||||||
|   <p class="mb-16 text-lg text-gray-500"> |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|     <img class="m-auto" style="height:15rem" src={donations_empty} alt="" /> |     <img class="m-auto mt-2" style="height:15rem" src={donations_empty} alt="" /> | ||||||
|     <span class="font-bold">{$_('there-are-no-donations-yet')}</span><br /> |     <span class="font-bold">{$_("there-are-no-donations-yet")}</span><br /> | ||||||
|     <span>{$_('add-your-fist-donation')}</span> |     <span>{$_("add-your-fist-donation")}</span> | ||||||
|   </p> |   </p> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,202 +1,291 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { DonationService, DonorService } from "@odit/lfk-client-js"; |   import { DonationService } from "@odit/lfk-client-js"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import DonationsEmptyState from "./DonationsEmptyState.svelte"; |   import DonationsEmptyState from "./DonationsEmptyState.svelte"; | ||||||
|  |   import AddDonationPaymentModal from "./AddDonationPaymentModal.svelte"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|  |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|  |   import DonationDonor from "./DonationDonor.svelte"; | ||||||
|  |   import DonationRunner from "./DonationRunner.svelte"; | ||||||
|  |   import DonationStatus from "./DonationStatus.svelte"; | ||||||
|  |   import DonationTableAction from "./DonationTableAction.svelte"; | ||||||
|  |   import DeleteDonationModal from "./DeleteDonationModal.svelte"; | ||||||
|  |   import { | ||||||
|  |     donationDonorFilter, | ||||||
|  |     donationRunnerFilter, | ||||||
|  |   } from "../shared/tablefilters"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|  |   $: active_edits = []; | ||||||
|  |   $: selected = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|  |   $: dataLoaded = false; | ||||||
|  |  | ||||||
|   export let current_donations = []; |   export let current_donations = []; | ||||||
|   const donations_promise = DonationService.donationControllerGetAll().then( |   export const addDonations = (donations) => { | ||||||
|     (val) => { |     current_donations = current_donations.concat(...donations); | ||||||
|       current_donations = val; |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_donations, | ||||||
|  |     })); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   //Section table | ||||||
|  |   const columns = [ | ||||||
|  |     { | ||||||
|  |       accessorKey: "id", | ||||||
|  |       header: () => "id", | ||||||
|  |       filterFn: `equalsString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "donor", | ||||||
|  |       header: () => $_("donor"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonationDonor, { donor: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `donor`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "runner", | ||||||
|  |       header: () => $_("runner"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonationRunner, { runner: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `runner`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "amountPerDistance", | ||||||
|  |       header: () => $_("amount-per-kilometer"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         if (!info.getValue()) { | ||||||
|  |           return $_("fixed-donation"); | ||||||
|         } |         } | ||||||
|  |         return `${(info.getValue() / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })} €`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "amount", | ||||||
|  |       header: () => $_("donation-amount"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return `${(info.getValue() / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })} €`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "paidAmount", | ||||||
|  |       header: () => $_("total-paid-amount"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return `${(info.getValue() / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })} €`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "status", | ||||||
|  |       header: () => $_("status"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonationStatus, { status: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonationTableAction, { | ||||||
|  |           detailsLink: `./${info.row.original.id}`, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_deletes = current_donations.filter( | ||||||
|  |               (r) => r.id == info.row.original.id | ||||||
|             ); |             ); | ||||||
|   function should_display_based_on_id(id) { |           }, | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |           paymentAction: () => { | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |             active_edits = current_donations.filter( | ||||||
|  |               (r) => r.id == info.row.original.id | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes( | ||||||
|  |               "DONATION:DELETE" | ||||||
|  |             ), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     filterFns: { | ||||||
|  |       donor: donationDonorFilter, | ||||||
|  |       runner: donationRunnerFilter, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|  |   }); | ||||||
|  |   const table = createSvelteTable(options); | ||||||
|  |  | ||||||
|  |   async function deleteDonation(delete_donation_id) { | ||||||
|  |     await DonationService.donationControllerRemove(delete_donation_id, true); | ||||||
|  |     current_donations = current_donations.filter( | ||||||
|  |       (r) => r.id !== delete_donation_id | ||||||
|  |     ); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_donations, | ||||||
|  |     })); | ||||||
|  |     toast.success($_("donation-deleted")); | ||||||
|   } |   } | ||||||
|     return id.toString() === searchvalue; |  | ||||||
|  |   onMount(async () => { | ||||||
|  |     let page = 0; | ||||||
|  |     let pagesize = 300; | ||||||
|  |     while (page >= 0) { | ||||||
|  |       const donations = await DonationService.donationControllerGetAll( | ||||||
|  |         page, | ||||||
|  |         pagesize | ||||||
|  |       ); | ||||||
|  |       if (donations.length == 0) { | ||||||
|  |         page = -2; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       current_donations = current_donations.concat(...donations); | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_donations, | ||||||
|  |       })); | ||||||
|  |  | ||||||
|  |       dataLoaded = true; | ||||||
|  |       page++; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')} | <AddDonationPaymentModal | ||||||
|   {#await donations_promise} |   original_data={active_edits[0]} | ||||||
|  |   payment_modal_open={active_edits.length > 0} | ||||||
|  |   paid_amount_input={(active_edits[0]?.paidAmount || 0) / 100} | ||||||
|  |   on:created={(event) => { | ||||||
|  |     current_donations = current_donations.map((d)=>{ | ||||||
|  |       if(d.id === event.detail.donation.id){ | ||||||
|  |         d.paidAmount = event.detail.donation.paidAmount; | ||||||
|  |       } | ||||||
|  |       return d; | ||||||
|  |     }) | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_donations, | ||||||
|  |     })); | ||||||
|  |   }} | ||||||
|  | /> | ||||||
|  | <DeleteDonationModal | ||||||
|  |   delete_donation={active_deletes[0]} | ||||||
|  |   modal_open={active_deletes.length > 0} | ||||||
|  |   on:delete={(event) => { | ||||||
|  |     deleteDonation(event.detail.id); | ||||||
|  |   }} | ||||||
|  | /> | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:GET")} | ||||||
|  |   {#if !dataLoaded} | ||||||
|     <div |     <div | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|       role="alert"> |       role="alert" | ||||||
|       <p class="font-bold">donations are being loaded</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("donations-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:else if current_donations.length === 0} | ||||||
|     {#if current_donations.length === 0} |  | ||||||
|     <DonationsEmptyState /> |     <DonationsEmptyState /> | ||||||
|   {: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="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border" | ||||||
|  |     /> | ||||||
|     <div |     <div | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |       class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll" | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |     > | ||||||
|           <thead class="bg-gray-50"> |       <table class="w-full"> | ||||||
|             <tr> |         <thead class="border-b border-gray-400"> | ||||||
|               <th |           {#each $table.getHeaderGroups() as headerGroup} | ||||||
|                 scope="col" |             <tr class="select-none"> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|                 {$_('donor')} |                 <InputElement | ||||||
|               </th> |                   type="checkbox" | ||||||
|               <th |                   checked={$table.getIsAllRowsSelected()} | ||||||
|                 scope="col" |                   indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   on:change={() => $table.toggleAllRowsSelected()} | ||||||
|                 {$_('runner')} |                 /> | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('amount-per-kilometer')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('donation-amount')} |  | ||||||
|               </th> |  | ||||||
|               <th scope="col" class="relative px-6 py-3"> |  | ||||||
|                 <span class="sr-only">{$_('action')}</span> |  | ||||||
|               </th> |               </th> | ||||||
|  |               {#each headerGroup.headers as header} | ||||||
|  |                 <TableHeader {header} /> | ||||||
|  |               {/each} | ||||||
|             </tr> |             </tr> | ||||||
|  |           {/each} | ||||||
|         </thead> |         </thead> | ||||||
|           <tbody class="divide-y divide-gray-200"> |         <tbody> | ||||||
|             {#each current_donations as donation} |           {#each $table.getRowModel().rows as row} | ||||||
|               {#if donation.donor.firstname |             <tr class="odd:bg-white even:bg-gray-100"> | ||||||
|                 .toLowerCase() |               <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|                 .includes( |                 <InputElement | ||||||
|                   searchvalue.toLowerCase() |                   type="checkbox" | ||||||
|                 ) || donation.donor.middlename |                   checked={row.getIsSelected()} | ||||||
|                   .toLowerCase() |                   on:change={() => row.toggleSelected()} | ||||||
|                   .includes( |                 /> | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.donor.lastname |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.runner?.firstname |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.runner?.middlename |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donation.runner?.lastname |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || should_display_based_on_id(donation.id)} |  | ||||||
|                 <tr data-rowid="donation_{donation.id}"> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       <a |  | ||||||
|                         href="../donors/{donation.donor.id}" |  | ||||||
|                         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{donation.donor.firstname} |  | ||||||
|                         {donation.donor.middlename || ''} |  | ||||||
|                         {donation.donor.lastname}</a> |  | ||||||
|                     </div> |  | ||||||
|               </td> |               </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |               {#each row.getVisibleCells() as cell} | ||||||
|                     {#if donation.runner} |                 <td> | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |                   <svelte:component | ||||||
|                         <a |                     this={flexRender( | ||||||
|                           href="../runners/{donation.runner.id}" |                       cell.column.columnDef.cell, | ||||||
|                           class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{donation.runner.firstname} |                       cell.getContext() | ||||||
|                           {donation.runner.middlename || ''} |                     )} | ||||||
|                           {donation.runner.lastname}</a> |                   /> | ||||||
|                       </div> |  | ||||||
|                     {:else} |  | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                         {$_('fixed-donation')} |  | ||||||
|                       </div> |  | ||||||
|                     {/if} |  | ||||||
|                 </td> |                 </td> | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |               {/each} | ||||||
|                     {#if donation.amountPerDistance} |  | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                         {(donation.amountPerDistance / 100) |  | ||||||
|                           .toFixed(2) |  | ||||||
|                           .toLocaleString('de-DE', { valute: 'EUR' })}€ |  | ||||||
|                       </div> |  | ||||||
|                     {:else} |  | ||||||
|                       <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                         {$_('fixed-donation')} |  | ||||||
|                       </div> |  | ||||||
|                     {/if} |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                       {(donation.amount / 100) |  | ||||||
|                         .toFixed(2) |  | ||||||
|                         .toLocaleString('de-DE', { valute: 'EUR' })}€ |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   {#if active_deletes[donation.id] === true} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           active_deletes[donation.id] = false; |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           DonationService.donationControllerRemove(donation.id, false).then( |  | ||||||
|                             (resp) => { |  | ||||||
|                               current_donations = current_donations.filter( |  | ||||||
|                                 (obj) => obj.id !== donation.id |  | ||||||
|                               ); |  | ||||||
|                               Toastify({ |  | ||||||
|                                 text: 'Donation deleted', |  | ||||||
|                                 duration: 500, |  | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             } |  | ||||||
|                           ); |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |  | ||||||
|                     </td> |  | ||||||
|                   {:else} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <a |  | ||||||
|                         href="./{donation.id}" |  | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |  | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:DELETE')} |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[donation.id] = true; |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                   {/if} |  | ||||||
|             </tr> |             </tr> | ||||||
|               {/if} |  | ||||||
|           {/each} |           {/each} | ||||||
|         </tbody> |         </tbody> | ||||||
|       </table> |       </table> | ||||||
|     </div> |     </div> | ||||||
|  |     <div class="h-2" /> | ||||||
|  |     <TableBottom {table} {selected} /> | ||||||
|   {/if} |   {/if} | ||||||
|   {:catch error} |  | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |  | ||||||
|       <span class="inline-block align-middle mr-8"> |  | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |  | ||||||
|         {error} |  | ||||||
|       </span> |  | ||||||
|     </div> |  | ||||||
|   {/await} |  | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   table tbody tr td:nth-child(2) { | ||||||
|  |     font-family: monospace; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|   | |||||||
| @@ -1,15 +1,14 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { |   import { DonorService } from "@odit/lfk-client-js"; | ||||||
|     DonorService |  | ||||||
|   } from "@odit/lfk-client-js"; |  | ||||||
|   import isEmail from "validator/es/lib/isEmail"; |   import isEmail from "validator/es/lib/isEmail"; | ||||||
|   import isMobilePhone from "validator/es/lib/isMobilePhone"; |   import isMobilePhone from "validator/es/lib/isMobilePhone"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|  |   import { createEventDispatcher } from "svelte"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_donors; |  | ||||||
|   let firstname_input; |   let firstname_input; | ||||||
|   let lastname_input; |   let lastname_input; | ||||||
|   let middlename_input; |   let middlename_input; | ||||||
| @@ -19,6 +18,7 @@ | |||||||
|   let address_input2; |   let address_input2; | ||||||
|   let address_zipcode; |   let address_zipcode; | ||||||
|   let address_city; |   let address_city; | ||||||
|  |   const dispatch = createEventDispatcher(); | ||||||
|   function focus(el) { |   function focus(el) { | ||||||
|     el.focus(); |     el.focus(); | ||||||
|   } |   } | ||||||
| @@ -74,10 +74,7 @@ | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       toast.loading($_("donor-is-being-added")); | ||||||
|         text: $_('donor-is-being-added'), |  | ||||||
|         duration: -1, |  | ||||||
|       }).showToast(); |  | ||||||
|       let address = {}; |       let address = {}; | ||||||
|       if (address_checked === true) { |       if (address_checked === true) { | ||||||
|         address = { |         address = { | ||||||
| @@ -92,7 +89,7 @@ | |||||||
|         firstname: firstname_input_value, |         firstname: firstname_input_value, | ||||||
|         lastname: lastname_input_value, |         lastname: lastname_input_value, | ||||||
|         address, |         address, | ||||||
|         receiptNeeded: address_checked |         receiptNeeded: address_checked, | ||||||
|       }; |       }; | ||||||
|       if (middlename_input_value) { |       if (middlename_input_value) { | ||||||
|         postdata.middlename = middlename_input_value; |         postdata.middlename = middlename_input_value; | ||||||
| @@ -111,21 +108,15 @@ | |||||||
|           email_input_value = ""; |           email_input_value = ""; | ||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           // |           // | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_('donor-added'), |           toast.success($_("donor-added")); | ||||||
|             duration: 500, |           dispatch("created", { donors: [result] }); | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
|           current_donors.push(result); |  | ||||||
|           current_donors = current_donors; |  | ||||||
|         }) |         }) | ||||||
|         .catch((err) => { |         .catch((err) => { | ||||||
|           // |           // | ||||||
|         }) |         }) | ||||||
|         .finally(() => { |         .finally(() => { | ||||||
|           processed_last_submit = true; |           processed_last_submit = true; | ||||||
|           // |  | ||||||
|           toast.hideToast(); |  | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -133,59 +124,71 @@ | |||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|     }}> |     }} | ||||||
|  |   > | ||||||
|     <div |     <div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|         role="dialog" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |       > | ||||||
|           <div class="sm:flex sm:items-start"> |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|             <div |             <div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|               <svg |               <svg | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |                 viewBox="0 0 24 24" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|         d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" /></svg> |                   d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> |             <div class="mt-3 text-center sm:text-left"> | ||||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('create-a-new-donor')} |                 {$_("create-a-new-donor")} | ||||||
|               </h3> |               </h3> | ||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <p class="text-sm text-gray-500"> |                 <p class="text-sm text-gray-500"> | ||||||
|                   {$_('please-provide-the-nessecary-information-to-add-a-new-donor')} |                   {$_( | ||||||
|  |                     "please-provide-the-nessecary-information-to-add-a-new-donor" | ||||||
|  |                   )} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <div class="grid grid-cols-6 gap-6"> |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="firstname" |                     for="firstname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('first-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("first-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     use:focus |                     use:focus | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('first-name')} |                     placeholder={$_("first-name")} | ||||||
|                     class:border-red-500={!isFirstnameValid} |                     class:border-red-500={!isFirstnameValid} | ||||||
|                     class:focus:border-red-500={!isFirstnameValid} |                     class:focus:border-red-500={!isFirstnameValid} | ||||||
|                     class:focus:ring-red-500={!isFirstnameValid} |                     class:focus:ring-red-500={!isFirstnameValid} | ||||||
| @@ -193,34 +196,41 @@ | |||||||
|                     bind:this={firstname_input} |                     bind:this={firstname_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="firstname" |                     name="firstname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isFirstnameValid} |                   {#if !isFirstnameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('first-name-is-required')} |                     > | ||||||
|  |                       {$_("first-name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="trackname" |                     for="trackname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('middle-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("middle-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('middle-name')} |                     placeholder={$_("middle-name")} | ||||||
|                     bind:value={middlename_input_value} |                     bind:value={middlename_input_value} | ||||||
|                     bind:this={middlename_input} |                     bind:this={middlename_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="trackname" |                     name="trackname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="lastname" |                     for="lastname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('last-name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("last-name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder="{$_('last-name')}" |                     placeholder={$_("last-name")} | ||||||
|                     class:border-red-500={!isLastnameValid} |                     class:border-red-500={!isLastnameValid} | ||||||
|                     class:focus:border-red-500={!isLastnameValid} |                     class:focus:border-red-500={!isLastnameValid} | ||||||
|                     class:focus:ring-red-500={!isLastnameValid} |                     class:focus:ring-red-500={!isLastnameValid} | ||||||
| @@ -228,21 +238,25 @@ | |||||||
|                     bind:this={lastname_input} |                     bind:this={lastname_input} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="lastname" |                     name="lastname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isLastnameValid} |                   {#if !isLastnameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('last-name-is-required')} |                     > | ||||||
|  |                       {$_("last-name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="phone" |                     for="phone" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('phone')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("phone")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('phone')} |                     placeholder={$_("phone")} | ||||||
|                     class:border-red-500={!isPhoneValidOrEmpty} |                     class:border-red-500={!isPhoneValidOrEmpty} | ||||||
|                     class:focus:border-red-500={!isPhoneValidOrEmpty} |                     class:focus:border-red-500={!isPhoneValidOrEmpty} | ||||||
|                     class:focus:ring-red-500={!isPhoneValidOrEmpty} |                     class:focus:ring-red-500={!isPhoneValidOrEmpty} | ||||||
| @@ -250,21 +264,27 @@ | |||||||
|                     bind:this={phone_input} |                     bind:this={phone_input} | ||||||
|                     type="tel" |                     type="tel" | ||||||
|                     name="phone" |                     name="phone" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isPhoneValidOrEmpty} |                   {#if !isPhoneValidOrEmpty} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {@html $_('the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number')} |                     > | ||||||
|  |                       {@html $_( | ||||||
|  |                         "the-provided-phone-number-is-invalid-less-than-br-greater-than-please-enter-a-valid-international-number" | ||||||
|  |                       )} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="email" |                     for="email" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('e-mail-adress')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("e-mail-adress")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('e-mail-adress')} |                     placeholder={$_("e-mail-adress")} | ||||||
|                     class:border-red-500={!isEmailValidOrEmpty} |                     class:border-red-500={!isEmailValidOrEmpty} | ||||||
|                     class:focus:border-red-500={!isEmailValidOrEmpty} |                     class:focus:border-red-500={!isEmailValidOrEmpty} | ||||||
|                     class:focus:ring-red-500={!isEmailValidOrEmpty} |                     class:focus:ring-red-500={!isEmailValidOrEmpty} | ||||||
| @@ -272,11 +292,13 @@ | |||||||
|                     bind:this={email_input} |                     bind:this={email_input} | ||||||
|                     type="email" |                     type="email" | ||||||
|                     name="email" |                     name="email" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isEmailValidOrEmpty} |                   {#if !isEmailValidOrEmpty} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('valid-email-is-required')} |                     > | ||||||
|  |                       {$_("valid-email-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
| @@ -287,19 +309,22 @@ | |||||||
|                       id="comments" |                       id="comments" | ||||||
|                       name="comments" |                       name="comments" | ||||||
|                       type="checkbox" |                       type="checkbox" | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="ml-3 text-sm"> |                   <div class="ml-3 text-sm"> | ||||||
|                     <label |                     <label for="comments" class="font-semibold text-gray-700" | ||||||
|                       for="comments" |                       >{$_("receipt-needed")}</label | ||||||
|                       class="font-medium text-gray-700">{$_('receipt-needed')}</label> |                     > | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|                 {#if address_checked === true} |                 {#if address_checked === true} | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address1" |                       for="address1" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('address')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("address")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder="Address" |                       placeholder="Address" | ||||||
| @@ -310,34 +335,41 @@ | |||||||
|                       bind:this={address_input1} |                       bind:this={address_input1} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address1" |                       name="address1" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !isAddress1Valid} |                     {#if !isAddress1Valid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('address-is-required')} |                       > | ||||||
|  |                         {$_("address-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address2" |                       for="address2" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("apartment-suite-etc")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder={$_('apartment-suite-etc')} |                       placeholder={$_("apartment-suite-etc")} | ||||||
|                       bind:value={address_input2_value} |                       bind:value={address_input2_value} | ||||||
|                       bind:this={address_input2} |                       bind:this={address_input2} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address2" |                       name="address2" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="zipcode" |                       for="zipcode" | ||||||
|                       class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("zip-postal-code")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder={$_('zip-postal-code')} |                       placeholder={$_("zip-postal-code")} | ||||||
|                       class:border-red-500={!iszipcodevalid} |                       class:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:border-red-500={!iszipcodevalid} |                       class:focus:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:ring-red-500={!iszipcodevalid} |                       class:focus:ring-red-500={!iszipcodevalid} | ||||||
| @@ -345,18 +377,22 @@ | |||||||
|                       bind:this={address_zipcode} |                       bind:this={address_zipcode} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="zipcode" |                       name="zipcode" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iszipcodevalid} |                     {#if !iszipcodevalid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('valid-zipcode-postal-code-is-required')} |                       > | ||||||
|  |                         {$_("valid-zipcode-postal-code-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="city" |                       for="city" | ||||||
|                       class="block text-sm font-medium text-gray-700">City</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >City</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                       placeholder="City" |                       placeholder="City" | ||||||
| @@ -367,11 +403,13 @@ | |||||||
|                       bind:this={address_city} |                       bind:this={address_city} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="city" |                       name="city" | ||||||
|                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iscityvalid} |                     {#if !iscityvalid} | ||||||
|                       <span |                       <span | ||||||
|                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                         {$_('valid-city-is-required')} |                       > | ||||||
|  |                         {$_("valid-city-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
| @@ -380,22 +418,24 @@ | |||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|           <button |           <button | ||||||
|             disabled={!createbtnenabled} |             disabled={!createbtnenabled} | ||||||
|             class:opacity-50={!createbtnenabled} |             class:opacity-50={!createbtnenabled} | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             type="button" |             type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|             {$_('create')} |           > | ||||||
|  |             {$_("create")} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               modal_open = false; |               modal_open = false; | ||||||
|             }} |             }} | ||||||
|             type="button" |             type="button" | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full 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 hidden lg:block" | ||||||
|             {$_('cancel')} |           > | ||||||
|  |             {$_("cancel")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,9 +1,6 @@ | |||||||
| <script> | <script> | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
| 	import { clickOutside } from "../base/outsideclick"; | 	import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { DonorService } from "@odit/lfk-client-js"; |  | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
| 	import { createEventDispatcher } from "svelte"; | 	import { createEventDispatcher } from "svelte"; | ||||||
| 	export let modal_open; | 	export let modal_open; | ||||||
| 	export let delete_donor; | 	export let delete_donor; | ||||||
| @@ -13,77 +10,78 @@ | |||||||
| 		dispatch("cancelDelete", { id: delete_donor.id }); | 		dispatch("cancelDelete", { id: delete_donor.id }); | ||||||
| 	} | 	} | ||||||
| 	function deleteDonor() { | 	function deleteDonor() { | ||||||
|     DonorService.donorControllerRemove( | 		dispatch("delete", { id: delete_donor.id }); | ||||||
|       delete_donor.id, |  | ||||||
|       true |  | ||||||
|     ) |  | ||||||
|       .then((resp) => { |  | ||||||
|         Toastify({ |  | ||||||
|           text: "Donor deleted", |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|         location.replace("./"); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => {}); |  | ||||||
| 	} | 	} | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
| 	<div | 	<div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" | 		class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
| 		use:clickOutside | 		use:clickOutside | ||||||
|     on:click_outside={cancelDelete}> | 		on:click_outside={cancelDelete} | ||||||
|  | 	> | ||||||
| 		<div | 		<div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | 			class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  | 		> | ||||||
| 			<div class="fixed inset-0 transition-opacity" aria-hidden="true"> | 			<div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
| 				<div | 				<div | ||||||
| 					class="absolute inset-0 bg-gray-500 opacity-75" | 					class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> | 					data-id="modal_backdrop" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<span | 			<span | ||||||
| 				class="hidden sm:inline-block sm:align-middle sm:h-screen" | 				class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> | 				aria-hidden="true">​</span | ||||||
|  | 			> | ||||||
| 			<div | 			<div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | 				class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
| 				role="dialog" | 				role="dialog" | ||||||
| 				aria-modal="true" | 				aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> | 				aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | 			> | ||||||
|           <div class="sm:flex sm:items-start"> | 				<div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  | 					<div class=""> | ||||||
| 						<div | 						<div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | 							class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|               <svg class="h-6 w-6 text-blue-600" fill="currentColor" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M9.33 11.5h2.17A4.5 4.5 0 0116 16H9v1h8v-1a5.58 5.58 0 00-.89-3H19a5 5 0 014.52 2.85A13.15 13.15 0 0113 21c-2.76 0-5.1-.59-7-1.63v-9.3a6.97 6.97 0 013.33 1.43zM5 19a1 1 0 01-1 1H2a1 1 0 01-1-1v-9a1 1 0 011-1h2a1 1 0 011 1v9zM18 5a3 3 0 110 6 3 3 0 010-6zm-7-3a3 3 0 110 6 3 3 0 010-6z"/></svg> | 						> | ||||||
|  | 							<svg | ||||||
|  | 								class="h-6 w-6 text-blue-600" | ||||||
|  | 								fill="currentColor" | ||||||
|  | 								xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 								viewBox="0 0 24 24" | ||||||
|  | 								><path fill="none" d="M0 0h24v24H0z" /><path | ||||||
|  | 									d="M9.33 11.5h2.17A4.5 4.5 0 0116 16H9v1h8v-1a5.58 5.58 0 00-.89-3H19a5 5 0 014.52 2.85A13.15 13.15 0 0113 21c-2.76 0-5.1-.59-7-1.63v-9.3a6.97 6.97 0 013.33 1.43zM5 19a1 1 0 01-1 1H2a1 1 0 01-1-1v-9a1 1 0 011-1h2a1 1 0 011 1v9zM18 5a3 3 0 110 6 3 3 0 010-6zm-7-3a3 3 0 110 6 3 3 0 010-6z" | ||||||
|  | 								/></svg | ||||||
|  | 							> | ||||||
| 						</div> | 						</div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | 						<div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
| 							<h3 class="text-lg leading-6 font-medium text-gray-900"> | 							<h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('attention')} | 								{$_( | ||||||
|  | 									"do-you-want-to-delete-this-donor-with-all-related-donations" | ||||||
|  | 								)} | ||||||
| 							</h3> | 							</h3> | ||||||
| 							<div class="mt-2 mb-6"> | 							<div class="mt-2 mb-6"> | ||||||
| 								<p class="text-sm text-gray-500"> | 								<p class="text-sm text-gray-500"> | ||||||
|                   {$_( | 									{$_("all-associated-donations-will-get-deleted-as-well")} | ||||||
|                     'do-you-want-to-delete-this-donor-with-all-related-donations' |  | ||||||
|                   )} |  | ||||||
|                   <br />  |  | ||||||
|                   {$_('all-associated-donations-will-get-deleted-as-well')} |  | ||||||
| 								</p> | 								</p> | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
| 				</div> | 				</div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | 				<div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
| 					<button | 					<button | ||||||
| 						on:click={deleteDonor} | 						on:click={deleteDonor} | ||||||
| 						type="button" | 						type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm"> | 						class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500" | ||||||
|             {$_('confirm-delete-donor-with-all-donations')} | 					> | ||||||
|  | 						{$_("confirm-delete-donor-with-all-donations")} | ||||||
| 					</button> | 					</button> | ||||||
| 					<button | 					<button | ||||||
| 						on:click={cancelDelete} | 						on:click={cancelDelete} | ||||||
| 						type="button" | 						type="button" | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> | 						class="w-full 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 hidden lg:block" | ||||||
|             {$_('cancel-keep-donor')} | 					> | ||||||
|  | 						{$_("cancel-keep-donor")} | ||||||
| 					</button> | 					</button> | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								src/components/donors/DonorAddress.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/components/donors/DonorAddress.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let address; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !address || !address.address1} | ||||||
|  |   {$_("no-address")} | ||||||
|  | {:else} | ||||||
|  |   {address.address1}<br /> | ||||||
|  |   <!-- {address.address2 || ''}<br /> --> | ||||||
|  |   {address.postalcode} | ||||||
|  |   {address.city} | ||||||
|  |   {address.country} | ||||||
|  | {/if} | ||||||
| @@ -1,17 +1,15 @@ | |||||||
| <script> | <script> | ||||||
|  | 	import { DonorService } from "@odit/lfk-client-js"; | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
| 	import store from "../../store"; | 	import store from "../../store"; | ||||||
|   import { DonorService, DonationService } from "@odit/lfk-client-js"; | 	import toast from "svelte-french-toast"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import PromiseError from "../base/PromiseError.svelte"; |  | ||||||
| 	import isEmail from "validator/es/lib/isEmail"; | 	import isEmail from "validator/es/lib/isEmail"; | ||||||
|   import ConfirmDonorDeletion from "./ConfirmDonorDeletion.svelte"; | 	import PromiseError from "../base/PromiseError.svelte"; | ||||||
| 	let data_loaded = false; | 	let data_loaded = false; | ||||||
| 	export let params; | 	export let params; | ||||||
| 	$: delete_triggered = false; | 	$: delete_triggered = false; | ||||||
| 	$: original_data = {}; | 	$: original_data = {}; | ||||||
| 	$: editable = {}; | 	$: editable = {}; | ||||||
|   $: current_donations = []; |  | ||||||
| 	$: changes_performed = !( | 	$: changes_performed = !( | ||||||
| 		JSON.stringify(original_data) === JSON.stringify(editable) | 		JSON.stringify(original_data) === JSON.stringify(editable) | ||||||
| 	); | 	); | ||||||
| @@ -28,11 +26,6 @@ | |||||||
| 		isPhoneValidOrEmpty && | 		isPhoneValidOrEmpty && | ||||||
| 		((isAddress1Valid && iszipcodevalid && iscityvalid) || | 		((isAddress1Valid && iszipcodevalid && iscityvalid) || | ||||||
| 			editable.address_checked === false); | 			editable.address_checked === false); | ||||||
|   const donation_promise = DonationService.donationControllerGetAll().then( |  | ||||||
|     (val) => { |  | ||||||
|       current_donations = val; |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
| 	const promise = DonorService.donorControllerGetOne(params.donorid).then( | 	const promise = DonorService.donorControllerGetOne(params.donorid).then( | ||||||
| 		(data) => { | 		(data) => { | ||||||
| 			data_loaded = true; | 			data_loaded = true; | ||||||
| @@ -58,56 +51,43 @@ | |||||||
| 	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0; | 	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0; | ||||||
| 	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; | 	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; | ||||||
| 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | ||||||
|   let modal_open = false; |  | ||||||
|   let delete_donor = {}; |  | ||||||
| 	function submit() { | 	function submit() { | ||||||
| 		if (data_loaded === true && save_enabled) { | 		if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ | 			toast($_("donor-is-being-updated")); | ||||||
|         text: $_("donor-is-being-updated"), |  | ||||||
|         duration: 2500, |  | ||||||
|       }).showToast(); |  | ||||||
| 			editable.address.country = "DE"; | 			editable.address.country = "DE"; | ||||||
| 			if (editable.address_checked === false) { | 			if (editable.address_checked === false) { | ||||||
| 				editable.address = null; | 				editable.address = null; | ||||||
| 			} | 			} | ||||||
| 			if (editable.email) editable.email = editable.email; | 			if (editable.email) editable.email = editable.email; | ||||||
|  | 			else editable.email = null; | ||||||
| 			if (editable.phone) editable.phone = editable.phone; | 			if (editable.phone) editable.phone = editable.phone; | ||||||
|  | 			else editable.phone = null; | ||||||
| 			if (editable.middlename) editable.middlename = editable.middlename; | 			if (editable.middlename) editable.middlename = editable.middlename; | ||||||
| 			editable.receiptNeeded = editable.address_checked; | 			editable.receiptNeeded = editable.address_checked; | ||||||
| 			DonorService.donorControllerPut(original_data.id, editable) | 			DonorService.donorControllerPut(original_data.id, editable) | ||||||
| 				.then((resp) => { | 				.then((resp) => { | ||||||
| 					Object.assign(original_data, editable); | 					Object.assign(original_data, editable); | ||||||
| 					original_data = original_data; | 					original_data = original_data; | ||||||
|           Toastify({ | 					toast.success($_("updated-donor")); | ||||||
|             text: $_("updated-donor"), |  | ||||||
|             duration: 2500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
| 				}) | 				}) | ||||||
| 				.catch((err) => {}); | 				.catch((err) => {}); | ||||||
| 		} else { | 		} else { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	function deleteDonor() { | 	function deleteDonor() { | ||||||
|     DonorService.donorControllerRemove(original_data.id, false) | 		DonorService.donorControllerRemove(original_data.id, true) | ||||||
| 			.then((resp) => { | 			.then((resp) => { | ||||||
|         Toastify({ | 				toast.success($_("donor-deleted")); | ||||||
|           text: $_("donor-deleted"), |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
| 				location.replace("./"); | 				location.replace("./"); | ||||||
| 			}) | 			}) | ||||||
| 			.catch((err) => { | 			.catch((err) => { | ||||||
|         modal_open = true; | 				console.log(err); | ||||||
|         delete_donor = original_data; |  | ||||||
| 			}); | 			}); | ||||||
| 	} | 	} | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <ConfirmDonorDeletion bind:modal_open bind:delete_donor /> | {#await promise} | ||||||
| {#await promise && donation_promise} | 	{$_("loading-donor-details")} | ||||||
|   {$_('loading-donor-details')} |  | ||||||
| {:then} | {:then} | ||||||
| 	<section class="container p-5 select-none"> | 	<section class="container p-5 select-none"> | ||||||
| 		<div class="flex flex-row mb-4"> | 		<div class="flex flex-row mb-4"> | ||||||
| @@ -115,57 +95,46 @@ | |||||||
| 				<nav class="w-full flex"> | 				<nav class="w-full flex"> | ||||||
| 					<ol class="list-none flex flex-row items-center justify-start"> | 					<ol class="list-none flex flex-row items-center justify-start"> | ||||||
| 						<li class="flex items-center"> | 						<li class="flex items-center"> | ||||||
|               <svg | 							<a class="mr-2" href="./" | ||||||
|                 fill="currentColor" | 								><svg | ||||||
| 									xmlns="http://www.w3.org/2000/svg" | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
| 									width="24" | 									width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | 									height="24" | ||||||
|                 <path |  | ||||||
|                   d="M9.33 11.5h2.17A4.5 4.5 0 0 1 16 16H8.999L9 17h8v-1a5.578 5.578 0 0 0-.886-3H19a5 5 0 0 1 4.516 2.851C21.151 18.972 17.322 21 13 21c-2.761 0-5.1-.59-7-1.625L6 10.071A6.967 6.967 0 0 1 9.33 11.5zM5 19a1 1 0 0 1-1 1H2a1 1 0 0 1-1-1v-9a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1v9zM18 5a3 3 0 1 1 0 6 3 3 0 0 1 0-6zm-7-3a3 3 0 1 1 0 6 3 3 0 0 1 0-6z" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center ml-2"> |  | ||||||
|               <a class="mr-2" href="./">{$_('donors')}</a><svg |  | ||||||
|                 stroke="currentColor" |  | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
| 									viewBox="0 0 24 24" | 									viewBox="0 0 24 24" | ||||||
|  | 									fill="none" | ||||||
|  | 									stroke="currentColor" | ||||||
|  | 									stroke-width="2" | ||||||
| 									stroke-linecap="round" | 									stroke-linecap="round" | ||||||
| 									stroke-linejoin="round" | 									stroke-linejoin="round" | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" | 									class="inline-block" | ||||||
|                 height="1em" | 									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg | ||||||
|                 width="1em" | 								> | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line | 								{$_("donors")}</a | ||||||
|                   x1="5" | 							> | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <span class="mr-2">{original_data.firstname} |  | ||||||
|                 {original_data.middlename || ''} |  | ||||||
|                 {original_data.lastname}</span> |  | ||||||
| 						</li> | 						</li> | ||||||
| 					</ol> | 					</ol> | ||||||
| 				</nav> | 				</nav> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> | 		<div class="mb-4 text-3xl font-extrabold leading-tight"> | ||||||
| 			{original_data.firstname} | 			{original_data.firstname} | ||||||
|       {original_data.middlename || ''} | 			{original_data.middlename || ""} | ||||||
| 			{original_data.lastname} | 			{original_data.lastname} | ||||||
|       <span data-id="donor_actions_${editable.id}"> | 			<div data-id="donor_actions_${editable.id}"> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:DELETE')} | 				{#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:DELETE")} | ||||||
| 					{#if delete_triggered} | 					{#if delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| 							on:click={deleteDonor} | 							on:click={deleteDonor} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:">{$_('confirm-deletion')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("confirm-deletion")}</button | ||||||
|  | 						> | ||||||
| 						<button | 						<button | ||||||
| 							on:click={() => { | 							on:click={() => { | ||||||
| 								delete_triggered = !delete_triggered; | 								delete_triggered = !delete_triggered; | ||||||
| 							}} | 							}} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:">{$_('cancel')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("cancel")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 					{#if !delete_triggered} | 					{#if !delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| @@ -173,7 +142,9 @@ | |||||||
| 								delete_triggered = true; | 								delete_triggered = true; | ||||||
| 							}} | 							}} | ||||||
| 							type="button" | 							type="button" | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:">{$_('delete-donor')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("delete-donor")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 				{/if} | 				{/if} | ||||||
| 				{#if !delete_triggered} | 				{#if !delete_triggered} | ||||||
| @@ -182,129 +153,157 @@ | |||||||
| 						class:opacity-50={!save_enabled} | 						class:opacity-50={!save_enabled} | ||||||
| 						type="button" | 						type="button" | ||||||
| 						on:click={submit} | 						on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:">{$_('save-changes')}</button> | 						class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 						>{$_("save-changes")}</button | ||||||
|  | 					> | ||||||
| 				{/if} | 				{/if} | ||||||
|       </span> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<!--  --> | 		<!--  --> | ||||||
| 		<div> | 		<div> | ||||||
|  | 			<span class="font-semibold text-gray-700" | ||||||
|  | 				>{$_("total-donation-amount")}:</span | ||||||
|  | 			> | ||||||
| 			<span | 			<span | ||||||
|         class="font-medium text-gray-700">{$_('total-donation-amount')}:</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-semibold 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-semibold text-gray-700">{$_("donations")}:</span> | ||||||
|       {#if current_donations.filter((d) => d.donor.id == editable.id).length > 0} | 			{#if original_data.donations.length > 0} | ||||||
|         {#each current_donations.filter((o) => o.donor.id == editable.id) as d} | 				{#each original_data.donations as d} | ||||||
|           {#if d.responseType === 'DISTANCEDONATION'} | 					{#if d.responseType === "DISTANCEDONATION"} | ||||||
| 						<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.middlename} | 							>{d.runner.firstname} | ||||||
|               {d.runner.lastname}</a> | 							{d.runner.middlename || ""} | ||||||
|  | 							{d.runner.lastname}</a | ||||||
|  | 						> | ||||||
| 					{:else} | 					{:else} | ||||||
| 						<a | 						<a | ||||||
| 							href="../donations/{d.id}" | 							href="../donations/{d.id}" | ||||||
|               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-700 text-white mr-1">{$_('fixed-donation')}: | 							class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-green-700 text-white mr-1" | ||||||
|  | 							>{$_("fixed-donation")}: | ||||||
| 							{(d.amount / 100) | 							{(d.amount / 100) | ||||||
| 								.toFixed(2) | 								.toFixed(2) | ||||||
|                 .toLocaleString('de-DE', { valute: 'EUR' })}€</a> | 								.toLocaleString("de-DE", { valute: "EUR" })}€</a | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 				{/each} | 				{/each} | ||||||
|       {:else}{$_('donor-has-no-associated-donations')}{/if} | 			{:else}{$_("donor-has-no-associated-donations")}{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<div class="mt-2 w-full"> | ||||||
|       <label | 			<label for="firstname" class="font-semibold text-gray-700" | ||||||
|         for="firstname" | 				>{$_("first-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('first-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('first-name')} | 				placeholder={$_("first-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				class:border-red-500={!isFirstnameValid} | 				class:border-red-500={!isFirstnameValid} | ||||||
| 				class:focus:border-red-500={!isFirstnameValid} | 				class:focus:border-red-500={!isFirstnameValid} | ||||||
| 				class:focus:ring-red-500={!isFirstnameValid} | 				class:focus:ring-red-500={!isFirstnameValid} | ||||||
| 				bind:value={editable.firstname} | 				bind:value={editable.firstname} | ||||||
| 				name="firstname" | 				name="firstname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isFirstnameValid} | 			{#if !isFirstnameValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('first-name-is-required')} | 				> | ||||||
|  | 					{$_("first-name-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<div class="mt-2 w-full"> | ||||||
|       <label | 			<label for="middlename" class="font-semibold text-gray-700" | ||||||
|         for="middlename" | 				>{$_("middle-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('middle-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('middle-name')} | 				placeholder={$_("middle-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.middlename} | 				bind:value={editable.middlename} | ||||||
| 				name="middlename" | 				name="middlename" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<div class="mt-2 w-full"> | ||||||
|       <label | 			<label for="lastname" class="font-semibold text-gray-700" | ||||||
|         for="lastname" | 				>{$_("last-name")}</label | ||||||
|         class="font-medium text-gray-700">{$_('last-name')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('last-name')} | 				placeholder={$_("last-name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.lastname} | 				bind:value={editable.lastname} | ||||||
| 				class:border-red-500={!isLastnameValid} | 				class:border-red-500={!isLastnameValid} | ||||||
| 				class:focus:border-red-500={!isLastnameValid} | 				class:focus:border-red-500={!isLastnameValid} | ||||||
| 				class:focus:ring-red-500={!isLastnameValid} | 				class:focus:ring-red-500={!isLastnameValid} | ||||||
| 				name="lastname" | 				name="lastname" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isLastnameValid} | 			{#if !isLastnameValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('last-name-is-required')} | 				> | ||||||
|  | 					{$_("last-name-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<div class="mt-2 w-full"> | ||||||
|       <label | 			<label for="email" class="font-semibold text-gray-700" | ||||||
|         for="email" | 				>{$_("e-mail-adress")}</label | ||||||
|         class="font-medium text-gray-700">{$_('e-mail-adress')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('e-mail-adress')} | 				placeholder={$_("e-mail-adress")} | ||||||
| 				type="email" | 				type="email" | ||||||
| 				bind:value={editable.email} | 				bind:value={editable.email} | ||||||
| 				class:border-red-500={!isEmailValid} | 				class:border-red-500={!isEmailValid} | ||||||
| 				class:focus:border-red-500={!isEmailValid} | 				class:focus:border-red-500={!isEmailValid} | ||||||
| 				class:focus:ring-red-500={!isEmailValid} | 				class:focus:ring-red-500={!isEmailValid} | ||||||
| 				name="email" | 				name="email" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isEmailValid} | 			{#if !isEmailValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('valid-email-is-required')} | 				> | ||||||
|  | 					{$_("valid-email-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class=" w-full"> | 		<div class="mt-2 w-full"> | ||||||
|       <label for="phone" class="font-medium text-gray-700">{$_('phone')}</label> | 			<label for="phone" class="font-semibold text-gray-700" | ||||||
|  | 				>{$_("phone")}</label | ||||||
|  | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('phone')} | 				placeholder={$_("phone")} | ||||||
| 				type="tel" | 				type="tel" | ||||||
| 				class:border-red-500={!isPhoneValidOrEmpty} | 				class:border-red-500={!isPhoneValidOrEmpty} | ||||||
| 				class:focus:border-red-500={!isPhoneValidOrEmpty} | 				class:focus:border-red-500={!isPhoneValidOrEmpty} | ||||||
| 				class:focus:ring-red-500={!isPhoneValidOrEmpty} | 				class:focus:ring-red-500={!isPhoneValidOrEmpty} | ||||||
| 				bind:value={editable.phone} | 				bind:value={editable.phone} | ||||||
| 				name="phone" | 				name="phone" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isPhoneValidOrEmpty} | 			{#if !isPhoneValidOrEmpty} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('valid-international-phone-number-is-required')} | 				> | ||||||
|  | 					{$_("valid-international-phone-number-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
| @@ -315,19 +314,20 @@ | |||||||
| 					id="comments" | 					id="comments" | ||||||
| 					name="comments" | 					name="comments" | ||||||
| 					type="checkbox" | 					type="checkbox" | ||||||
|           class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | 					class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="ml-3"> | 			<div class="ml-3"> | ||||||
|         <label | 				<label for="comments" class="font-semibold text-gray-700" | ||||||
|           for="comments" | 					>{$_("receipt-needed")}</label | ||||||
|           class="font-medium text-gray-700">{$_('receipt-needed')}</label> | 				> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		{#if editable.address_checked === true} | 		{#if editable.address_checked === true} | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="address1" class="block font-medium text-gray-700" | ||||||
|           for="address1" | 					>{$_("address")}</label | ||||||
|           class="block  font-medium text-gray-700">{$_('address')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
| 					placeholder="Address" | 					placeholder="Address" | ||||||
| @@ -337,65 +337,72 @@ | |||||||
| 					bind:value={editable.address.address1} | 					bind:value={editable.address.address1} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="address1" | 					name="address1" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !isAddress1Valid} | 				{#if !isAddress1Valid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('address-is-required')} | 					> | ||||||
|  | 						{$_("address-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="address2" class="block font-medium text-gray-700" | ||||||
|           for="address2" | 					>{$_("apartment-suite-etc")}</label | ||||||
|           class="block  font-medium text-gray-700">{$_('apartment-suite-etc')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('apartment-suite-etc')} | 					placeholder={$_("apartment-suite-etc")} | ||||||
| 					bind:value={editable.address.address2} | 					bind:value={editable.address.address2} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="address2" | 					name="address2" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="zipcode" class="block font-medium text-gray-700" | ||||||
|           for="zipcode" | 					>{$_("zip-postal-code")}</label | ||||||
|           class="block  font-medium text-gray-700">{$_('zip-postal-code')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('zip-postal-code')} | 					placeholder={$_("zip-postal-code")} | ||||||
| 					class:border-red-500={!iszipcodevalid} | 					class:border-red-500={!iszipcodevalid} | ||||||
| 					class:focus:border-red-500={!iszipcodevalid} | 					class:focus:border-red-500={!iszipcodevalid} | ||||||
| 					class:focus:ring-red-500={!iszipcodevalid} | 					class:focus:ring-red-500={!iszipcodevalid} | ||||||
| 					bind:value={editable.address.postalcode} | 					bind:value={editable.address.postalcode} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="zipcode" | 					name="zipcode" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !iszipcodevalid} | 				{#if !iszipcodevalid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-zipcode-postal-code-is-required')} | 					> | ||||||
|  | 						{$_("valid-zipcode-postal-code-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
| 			<div class="col-span-6"> | 			<div class="col-span-6"> | ||||||
|         <label | 				<label for="city" class="block font-medium text-gray-700" | ||||||
|           for="city" | 					>{$_("city")}</label | ||||||
|           class="block  font-medium text-gray-700">{$_('city')}</label> | 				> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder={$_('city')} | 					placeholder={$_("city")} | ||||||
| 					class:border-red-500={!iscityvalid} | 					class:border-red-500={!iscityvalid} | ||||||
| 					class:focus:border-red-500={!iscityvalid} | 					class:focus:border-red-500={!iscityvalid} | ||||||
| 					class:focus:ring-red-500={!iscityvalid} | 					class:focus:ring-red-500={!iscityvalid} | ||||||
| 					bind:value={editable.address.city} | 					bind:value={editable.address.city} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					name="city" | 					name="city" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 					class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 				{#if !iscityvalid} | 				{#if !iscityvalid} | ||||||
| 					<span | 					<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-city-is-required')} | 					> | ||||||
|  | 						{$_("valid-city-is-required")} | ||||||
| 					</span> | 					</span> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			</div> | 			</div> | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								src/components/donors/DonorDonations.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								src/components/donors/DonorDonations.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | <script> | ||||||
|  |   import { _ } from "svelte-i18n"; | ||||||
|  |   export let donations; | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if !donations || donations.length == 0} | ||||||
|  |   {$_("donor-has-no-associated-donations")} | ||||||
|  | {:else} | ||||||
|  |   {#each donations as donation} | ||||||
|  |     {#if donation.responseType === "DISTANCEDONATION"} | ||||||
|  |       <a | ||||||
|  |         href="../donations/{donation.id}" | ||||||
|  |         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-600 text-white mr-1" | ||||||
|  |         >{donation.runner.firstname} | ||||||
|  |         {donation.runner.middlename || ""} | ||||||
|  |         {donation.runner.lastname}</a | ||||||
|  |       > | ||||||
|  |     {:else} | ||||||
|  |       <a | ||||||
|  |         href="../donations/{donation.id}" | ||||||
|  |         class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full border border-current bg-green-700 text-white mr-1" | ||||||
|  |         >{$_("fixed-donation")}: | ||||||
|  |         {(donation.amount / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })}€</a | ||||||
|  |       > | ||||||
|  |     {/if} | ||||||
|  |   {/each} | ||||||
|  | {/if} | ||||||
| @@ -5,25 +5,73 @@ | |||||||
|   import DonorsOverview from "./DonorsOverview.svelte"; |   import DonorsOverview from "./DonorsOverview.svelte"; | ||||||
|   $: current_donors = []; |   $: current_donors = []; | ||||||
|   export let modal_open = false; |   export let modal_open = false; | ||||||
|  |   let addDonors; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> |   <h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('donors')} |     {$_("donors")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:CREATE')} |   </h4> | ||||||
|  |   {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:CREATE")} | ||||||
|     <button |     <button | ||||||
|       on:click={() => { |       on:click={() => { | ||||||
|         modal_open = true; |         modal_open = true; | ||||||
|       }} |       }} | ||||||
|       type="button" |       type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm  mb-1 lg:mb-0" | ||||||
|         {$_('add-donor')} |     > | ||||||
|  |       {$_("add-donor")} | ||||||
|     </button> |     </button> | ||||||
|   {/if} |   {/if} | ||||||
|   </span> |   {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:GET")} | ||||||
|   <DonorsOverview bind:current_donors /> |     <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/100).toFixed(2), | ||||||
|  |               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:w-auto sm:text-sm  mb-1 lg:mb-0" | ||||||
|  |     > | ||||||
|  |       {$_("sponsoring-quittungs-liste_herunterladen")} | ||||||
|  |     </button> | ||||||
|  |   {/if} | ||||||
|  |   <DonorsOverview bind:current_donors bind:addDonors /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:CREATE")} | ||||||
|   <AddDonorModal bind:current_donors bind:modal_open /> |   <AddDonorModal | ||||||
|  |     on:created={(event) => { | ||||||
|  |       addDonors(event.detail.donors); | ||||||
|  |     }} | ||||||
|  |     bind:modal_open | ||||||
|  |   /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
| <div class="text-center items-center justify-center"> | <div class="text-center items-center justify-center"> | ||||||
|   <p class="mb-16 text-lg text-gray-500"> |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|     <img class="w-full" style="height:15rem" src={donors_empty} alt="" /> |     <img class="w-full" style="height:15rem" src={donors_empty} alt="" /> | ||||||
|     <span class="font-bold">{$_('there-are-no-donors-yet')}</span><br /> |     <span class="font-bold">{$_("there-are-no-donors-yet")}</span><br /> | ||||||
|     <span>{$_('add-your-first-donor')}</span> |     <span>{$_("add-your-first-donor")}</span> | ||||||
|   </p> |   </p> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,212 +1,262 @@ | |||||||
| <script> | <script> | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { DonationService, DonorService } from "@odit/lfk-client-js"; |   import { DonorService } from "@odit/lfk-client-js"; | ||||||
|   import store from "../../store"; |   import store from "../../store"; | ||||||
|   import DonorsEmptyState from "./DonorsEmptyState.svelte"; |   import DonorsEmptyState from "./DonorsEmptyState.svelte"; | ||||||
|   import ConfirmDonorDeletion from "./ConfirmDonorDeletion.svelte"; |   import ConfirmDonorDeletion from "./ConfirmDonorDeletion.svelte"; | ||||||
|   import Toastify from "toastify-js"; |   import TableBottom from "../shared/TableBottom.svelte"; | ||||||
|  |   import { | ||||||
|  |     createSvelteTable, | ||||||
|  |     flexRender, | ||||||
|  |     getCoreRowModel, | ||||||
|  |     getFilteredRowModel, | ||||||
|  |     getPaginationRowModel, | ||||||
|  |     getSortedRowModel, | ||||||
|  |     renderComponent, | ||||||
|  |   } from "@tanstack/svelte-table"; | ||||||
|  |   import { writable } from "svelte/store"; | ||||||
|  |   import { onMount } from "svelte"; | ||||||
|  |   import InputElement from "../shared/InputElement.svelte"; | ||||||
|  |   import TableHeader from "../shared/TableHeader.svelte"; | ||||||
|  |   import TableActions from "../shared/TableActions.svelte"; | ||||||
|  |   import DonorAddress from "./DonorAddress.svelte"; | ||||||
|  |   import DonorDonations from "./DonorDonations.svelte"; | ||||||
|  |   import { filterAddress, filterName } from "../shared/tablefilters"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   $: searchvalue = ""; |   $: searchvalue = ""; | ||||||
|   $: active_deletes = []; |   $: active_deletes = []; | ||||||
|   $: current_donations = []; |   $: selectedDonors = | ||||||
|   let modal_open = false; |     $table?.getSelectedRowModel().rows.map((row) => row.original) || []; | ||||||
|   let delete_donor = {}; |   $: selected = | ||||||
|  |     $table?.getSelectedRowModel().rows.map((row) => row.index) || []; | ||||||
|  |  | ||||||
|  |   $: dataLoaded = false; | ||||||
|  |  | ||||||
|   export let current_donors = []; |   export let current_donors = []; | ||||||
|   const donors_promise = DonorService.donorControllerGetAll().then((val) => { |   export const addDonors = (donors) => { | ||||||
|     current_donors = val; |     current_donors = current_donors.concat(...donors); | ||||||
|   }); |     options.update((options) => ({ | ||||||
|   const donation_promise = DonationService.donationControllerGetAll().then( |       ...options, | ||||||
|     (val) => { |       data: current_donors, | ||||||
|       current_donations = val; |     })); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   //Section table | ||||||
|  |   const columns = [ | ||||||
|  |     { | ||||||
|  |       accessorKey: "id", | ||||||
|  |       header: () => "id", | ||||||
|  |       filterFn: `equalsString`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "name", | ||||||
|  |       header: () => $_("name"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         const d = info.row.original; | ||||||
|  |         if (d.middlename) { | ||||||
|  |           return `${d.firstname} ${d.middlename} ${d.lastname}`; | ||||||
|  |         } else { | ||||||
|  |           return `${d.firstname} ${d.lastname}`; | ||||||
|         } |         } | ||||||
|  |       }, | ||||||
|  |       filterFn: `name`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "address", | ||||||
|  |       header: () => $_("contact-information"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonorAddress, { address: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       filterFn: `address`, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "donations", | ||||||
|  |       header: () => $_("sponsorings"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(DonorDonations, { donations: info.getValue() }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "donationAmount", | ||||||
|  |       header: () => $_("total-donation-amount"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return `${(info.getValue() / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })}€`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "paidDonationAmount", | ||||||
|  |       header: () => $_("total-paid-amount"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return `${(info.getValue() / 100) | ||||||
|  |           .toFixed(2) | ||||||
|  |           .toLocaleString("de-DE", { valute: "EUR" })}€`; | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       accessorKey: "actions", | ||||||
|  |       header: () => $_("action"), | ||||||
|  |       cell: (info) => { | ||||||
|  |         return renderComponent(TableActions, { | ||||||
|  |           detailsLink: `./${info.row.original.id}`, | ||||||
|  |           deleteAction: () => { | ||||||
|  |             active_deletes = current_donors.filter( | ||||||
|  |               (r) => r.id == info.row.original.id | ||||||
|             ); |             ); | ||||||
|  |           }, | ||||||
|  |           deleteEnabled: | ||||||
|  |             store.state.jwtinfo.userdetails.permissions.includes( | ||||||
|  |               "DONOR:DELETE" | ||||||
|  |             ), | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       enableColumnFilter: false, | ||||||
|  |       enableSorting: false, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  |   const options = writable({ | ||||||
|  |     data: [], | ||||||
|  |     columns: columns, | ||||||
|  |     initialState: { | ||||||
|  |       pagination: { | ||||||
|  |         pageSize: 50, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     filterFns: { | ||||||
|  |       name: filterName, | ||||||
|  |       address: filterAddress, | ||||||
|  |     }, | ||||||
|  |     enableRowSelection: true, | ||||||
|  |     getCoreRowModel: getCoreRowModel(), | ||||||
|  |     getFilteredRowModel: getFilteredRowModel(), | ||||||
|  |     getPaginationRowModel: getPaginationRowModel(), | ||||||
|  |     getSortedRowModel: getSortedRowModel(), | ||||||
|  |   }); | ||||||
|  |   const table = createSvelteTable(options); | ||||||
|  |  | ||||||
|   function should_display_based_on_id(id) { |   function should_display_based_on_id(id) { | ||||||
|     if (searchvalue.toString().slice(-1) === "*") { |     if (searchvalue.toString().slice(-1) === "*") { | ||||||
|       return id.toString().startsWith(searchvalue.replace("*", "")); |       return id.toString().startsWith(searchvalue.replace("*", "")); | ||||||
|     } |     } | ||||||
|     return id.toString() === searchvalue; |     return id.toString() === searchvalue; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   onMount(async () => { | ||||||
|  |     let page = 0; | ||||||
|  |     let pagesize = 300; | ||||||
|  |     while (page >= 0) { | ||||||
|  |       const donors = await DonorService.donorControllerGetAll(page, pagesize); | ||||||
|  |       if (donors.length == 0) { | ||||||
|  |         page = -2; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       current_donors = current_donors.concat(...donors); | ||||||
|  |       options.update((options) => ({ | ||||||
|  |         ...options, | ||||||
|  |         data: current_donors, | ||||||
|  |       })); | ||||||
|  |  | ||||||
|  |       dataLoaded = true; | ||||||
|  |       page++; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <ConfirmDonorDeletion | <ConfirmDonorDeletion | ||||||
|   on:cancelDelete={(event) => { |   on:cancelDelete={(event) => { | ||||||
|     modal_open = false; |     active_deletes = active_deletes.filter((a) => a.id !== event.detail.id); | ||||||
|     active_deletes[event.detail.id] = false; |  | ||||||
|   }} |   }} | ||||||
|   bind:modal_open |   on:delete={async (event) => { | ||||||
|   bind:delete_donor /> |     toast.loading($_("deleting-donor")); | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:GET')} |     await DonorService.donorControllerRemove(event.detail.id, true); | ||||||
|   {#await donors_promise && donation_promise} |     toast.dismiss(); | ||||||
|  |     toast.success($_("donor-deleted")); | ||||||
|  |     current_donors = current_donors.filter((d) => d.id !== event.detail.id); | ||||||
|  |     active_deletes = active_deletes.filter((a) => a.id !== event.detail.id); | ||||||
|  |     options.update((options) => ({ | ||||||
|  |       ...options, | ||||||
|  |       data: current_donors, | ||||||
|  |     })); | ||||||
|  |   }} | ||||||
|  |   modal_open={active_deletes.length > 0} | ||||||
|  |   delete_donor={active_deletes[0]} | ||||||
|  | /> | ||||||
|  | {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:GET")} | ||||||
|  |   {#if !dataLoaded} | ||||||
|     <div |     <div | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|       role="alert"> |       role="alert" | ||||||
|       <p class="font-bold">{$_('donors-are-being-loaded')}</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("donors-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:else if current_donors.length === 0} | ||||||
|     {#if current_donors.length === 0} |  | ||||||
|     <DonorsEmptyState /> |     <DonorsEmptyState /> | ||||||
|   {: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="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border" | ||||||
|  |     /> | ||||||
|     <div |     <div | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |       class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll" | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |     > | ||||||
|           <thead class="bg-gray-50"> |       <table class="w-full"> | ||||||
|             <tr> |         <thead class="border-b border-gray-400"> | ||||||
|               <th |           {#each $table.getHeaderGroups() as headerGroup} | ||||||
|                 scope="col" |             <tr class="select-none"> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               <th class="inset-y-0 left-0 px-4 py-2 text-left w-px"> | ||||||
|                 {$_('name')} |                 <InputElement | ||||||
|  |                   type="checkbox" | ||||||
|  |                   checked={$table.getIsAllRowsSelected()} | ||||||
|  |                   indeterminate={$table.getIsSomeRowsSelected()} | ||||||
|  |                   on:change={() => $table.toggleAllRowsSelected()} | ||||||
|  |                 /> | ||||||
|               </th> |               </th> | ||||||
|               <th |               {#each headerGroup.headers as header} | ||||||
|                 scope="col" |                 <TableHeader {header} /> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |               {/each} | ||||||
|                 {$_('contact-information')} |             </tr> | ||||||
|               </th> |           {/each} | ||||||
|               <th |         </thead> | ||||||
|                 scope="col" |         <tbody> | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |           {#each $table.getRowModel().rows as row} | ||||||
|                 {$_('donations')} |             <tr class="odd:bg-white even:bg-gray-100"> | ||||||
|               </th> |               <td class="inset-y-0 left-0 px-4 py-2 text-center w-px"> | ||||||
|               <th |                 <InputElement | ||||||
|                 scope="col" |                   type="checkbox" | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |                   checked={row.getIsSelected()} | ||||||
|                 {$_('total-donation-amount')} |                   on:change={() => row.toggleSelected()} | ||||||
|               </th> |                 /> | ||||||
|               <th scope="col" class="relative px-6 py-3"> |               </td> | ||||||
|                 <span class="sr-only">{$_('action')}</span> |               {#each row.getVisibleCells() as cell} | ||||||
|               </th> |                 <td> | ||||||
|             </tr> |                   <svelte:component | ||||||
|           </thead> |                     this={flexRender( | ||||||
|           <tbody class="divide-y divide-gray-200"> |                       cell.column.columnDef.cell, | ||||||
|             {#each current_donors as donor} |                       cell.getContext() | ||||||
|               {#if donor.firstname |                     )} | ||||||
|                 .toLowerCase() |                   /> | ||||||
|                 .includes( |                 </td> | ||||||
|                   searchvalue.toLowerCase() |  | ||||||
|                 ) || donor.middlename |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || donor.lastname |  | ||||||
|                   .toLowerCase() |  | ||||||
|                   .includes( |  | ||||||
|                     searchvalue.toLowerCase() |  | ||||||
|                   ) || should_display_based_on_id(donor.id)} |  | ||||||
|                 <tr data-rowid="donor_{donor.id}"> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       <div class="ml-4"> |  | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                           {donor.firstname} |  | ||||||
|                           {donor.middlename || ''} |  | ||||||
|                           {donor.lastname} |  | ||||||
|                         </div> |  | ||||||
|                       </div> |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     {#if donor.email} |  | ||||||
|                       <div class="text-sm text-gray-500">{donor.email}</div> |  | ||||||
|                     {/if} |  | ||||||
|                     {#if donor.phone} |  | ||||||
|                       <div class="text-sm text-gray-500">{donor.phone}</div> |  | ||||||
|                     {/if} |  | ||||||
|                     {#if donor.address.address1 !== null} |  | ||||||
|                       {donor.address.address1}<br /> |  | ||||||
|                       {donor.address.address2 || ''}<br /> |  | ||||||
|                       {donor.address.postalcode} |  | ||||||
|                       {donor.address.city} |  | ||||||
|                       {donor.address.country} |  | ||||||
|                     {/if} |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     {#if current_donations.filter((d) => d.donor.id == donor.id).length > 0} |  | ||||||
|                       {#each current_donations.filter((o) => o.donor.id == donor.id) as d} |  | ||||||
|                         {#if d.responseType === 'DISTANCEDONATION'} |  | ||||||
|                           <a |  | ||||||
|                             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} |  | ||||||
|                             {d.runner.middlename} |  | ||||||
|                             {d.runner.lastname}</a> |  | ||||||
|                         {:else} |  | ||||||
|                           <a |  | ||||||
|                             href="../donations/{d.id}" |  | ||||||
|                             class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-700 text-white mr-1">{$_('fixed-donation')}: |  | ||||||
|                             {(d.amount / 100) |  | ||||||
|                               .toFixed(2) |  | ||||||
|                               .toLocaleString('de-DE', { valute: 'EUR' })}€</a> |  | ||||||
|                         {/if} |  | ||||||
|               {/each} |               {/each} | ||||||
|                     {:else}{$_('donor-has-no-associated-donations')}{/if} |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     {(donor.donationAmount / 100) |  | ||||||
|                       .toFixed(2) |  | ||||||
|                       .toLocaleString('de-DE', { valute: 'EUR' })}€ |  | ||||||
|                   </td> |  | ||||||
|                   {#if active_deletes[donor.id] === true} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           active_deletes[donor.id] = false; |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           DonorService.donorControllerRemove(donor.id, false) |  | ||||||
|                             .then((resp) => { |  | ||||||
|                               current_donors = current_donors.filter((obj) => obj.id !== donor.id); |  | ||||||
|                               Toastify({ |  | ||||||
|                                 text: 'Donor deleted', |  | ||||||
|                                 duration: 500, |  | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             }) |  | ||||||
|                             .catch((err) => { |  | ||||||
|                               modal_open = true; |  | ||||||
|                               delete_donor = donor; |  | ||||||
|                             }); |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |  | ||||||
|                     </td> |  | ||||||
|                   {:else} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <a |  | ||||||
|                         href="./{donor.id}" |  | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |  | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:DELETE')} |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[donor.id] = true; |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                   {/if} |  | ||||||
|             </tr> |             </tr> | ||||||
|               {/if} |  | ||||||
|           {/each} |           {/each} | ||||||
|         </tbody> |         </tbody> | ||||||
|       </table> |       </table> | ||||||
|     </div> |     </div> | ||||||
|  |     <div class="h-2" /> | ||||||
|  |     <TableBottom {table} {selected} /> | ||||||
|   {/if} |   {/if} | ||||||
|   {:catch error} |  | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |  | ||||||
|       <span class="inline-block align-middle mr-8"> |  | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |  | ||||||
|         {error} |  | ||||||
|       </span> |  | ||||||
|     </div> |  | ||||||
|   {/await} |  | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   table tbody tr td:nth-child(2) { | ||||||
|  |     font-family: monospace; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| <script> | <script> | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
| 	import { clickOutside } from "../base/outsideclick"; | 	import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   export let modal_open; | 	let modal_open = false; | ||||||
| 	(function () { | 	(function () { | ||||||
| 		document.onkeydown = function (e) { | 		document.onkeydown = function (e) { | ||||||
| 			e = e || window.event; | 			e = e || window.event; | ||||||
| @@ -24,44 +24,52 @@ | |||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
| 	<div | 	<div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" | 		class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
| 		use:clickOutside | 		use:clickOutside | ||||||
| 		on:click_outside={() => { | 		on:click_outside={() => { | ||||||
| 			modal_open = false; | 			modal_open = false; | ||||||
|     }}> | 		}} | ||||||
|  | 	> | ||||||
| 		<div | 		<div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> | 			class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 sm:block sm:p-0" | ||||||
|  | 		> | ||||||
| 			<div class="fixed inset-0 transition-opacity" aria-hidden="true"> | 			<div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
| 				<div | 				<div | ||||||
| 					class="absolute inset-0 bg-gray-500 opacity-75" | 					class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> | 					data-id="modal_backdrop" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			<span | 			<span | ||||||
| 				class="hidden sm:inline-block sm:align-middle sm:h-screen" | 				class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> | 				aria-hidden="true">​</span | ||||||
|  | 			> | ||||||
| 			<div | 			<div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" | 				class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
| 				role="dialog" | 				role="dialog" | ||||||
| 				aria-modal="true" | 				aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> | 				aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> | 			> | ||||||
|           <div class="sm:flex sm:items-start"> | 				<div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  | 					<div class=""> | ||||||
| 						<div | 						<div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> | 							class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  | 						> | ||||||
| 							<svg | 							<svg | ||||||
| 								fill="currentColor" | 								fill="currentColor" | ||||||
| 								class="h-6 w-6 text-blue-600" | 								class="h-6 w-6 text-blue-600" | ||||||
| 								xmlns="http://www.w3.org/2000/svg" | 								xmlns="http://www.w3.org/2000/svg" | ||||||
| 								viewBox="0 0 24 24" | 								viewBox="0 0 24 24" | ||||||
| 								width="24" | 								width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | 								height="24" | ||||||
|  | 								><path fill="none" d="M0 0h24v24H0z" /> | ||||||
| 								<path | 								<path | ||||||
|                   d="M14 20v2H2v-2h12zM14.586.686l7.778 7.778L20.95 9.88l-1.06-.354L17.413 12l5.657 5.657-1.414 1.414L16 13.414l-2.404 2.404.283 1.132-1.415 1.414-7.778-7.778 1.415-1.414 1.13.282 6.294-6.293-.353-1.06L14.586.686z" /></svg> | 									d="M14 20v2H2v-2h12zM14.586.686l7.778 7.778L20.95 9.88l-1.06-.354L17.413 12l5.657 5.657-1.414 1.414L16 13.414l-2.404 2.404.283 1.132-1.415 1.414-7.778-7.778 1.415-1.414 1.13.282 6.294-6.293-.353-1.06L14.586.686z" | ||||||
|  | 								/></svg | ||||||
|  | 							> | ||||||
| 						</div> | 						</div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> | 						<div class="mt-3 sm:mt-0 sm:ml-4 sm:text-left"> | ||||||
| 							<h3 class="text-lg leading-6 font-medium"> | 							<h3 class="text-lg leading-6 font-medium"> | ||||||
|                 {$_('read-license')} | 								{$_("read-license")} | ||||||
| 							</h3> | 							</h3> | ||||||
| 							<div class="mt-2 mb-6"> | 							<div class="mt-2 mb-6"> | ||||||
| 								<p class="text-sm text-gray-500">{currentlicense}</p> | 								<p class="text-sm text-gray-500">{currentlicense}</p> | ||||||
| @@ -72,14 +80,15 @@ | |||||||
| 						</div> | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
| 				</div> | 				</div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> | 				<div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
| 					<button | 					<button | ||||||
| 						on:click={() => { | 						on:click={() => { | ||||||
| 							modal_open = false; | 							modal_open = false; | ||||||
| 						}} | 						}} | ||||||
| 						type="button" | 						type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | 						class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm" | ||||||
|             {$_('close')} | 					> | ||||||
|  | 						{$_("close")} | ||||||
| 					</button> | 					</button> | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
| @@ -87,107 +96,101 @@ | |||||||
| 	</div> | 	</div> | ||||||
| {/if} | {/if} | ||||||
| <!-- /// --> | <!-- /// --> | ||||||
| <div class="pt-12 px-4 sm:px-6 lg:px-8 lg:pt-20 bg-gray-900 pb-12"> | <section class="container p-5"> | ||||||
|   <div class="text-center mb-8"> | 	<h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     <h1 | 		{$_("about")} | ||||||
|       class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl"> | 	</h4> | ||||||
|       {$_('about')} | 	<p class="mt-2 mb-2"> | ||||||
|       🧾 |  | ||||||
|     </h1> |  | ||||||
|     <p |  | ||||||
|       class="mt-2 max-w-xl mx-auto text-xl lg:max-w-3xl lg:text-2xl text-gray-300"> |  | ||||||
| 		Lauf für Kaya! | 		Lauf für Kaya! | ||||||
|       <strong class="text-white font-medium"> | 		<strong class="font-medium"> | ||||||
|         {$_('by')} | 			{$_("by")} | ||||||
| 			<a href="https://odit.services" class="underline">ODIT.Services</a> | 			<a href="https://odit.services" class="underline">ODIT.Services</a> | ||||||
| 		</strong> | 		</strong> | ||||||
| 		<br /> | 		<br /> | ||||||
|       <span class="text-lg">{$_('lfk-is-os')}</span> | 		<span>{$_("lfk-is-os")}</span> | ||||||
| 	</p> | 	</p> | ||||||
|   </div> | 	<h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
| </div> | 		{$_("credits")} | ||||||
|  | 	</h4> | ||||||
| <div class="pt-0 pb-16 overflow-hidden lg:pt-12 lg:py-24"> | 	<p class="text-left">{$_("oss_credit_description")}</p> | ||||||
|   <div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8"> | 	<div class="mt-5 overflow-x-auto"> | ||||||
|     <h2 class="text-4xl font-display font-semibold md:text-5xl"> |  | ||||||
|       {$_('credits')} |  | ||||||
|     </h2> |  | ||||||
|     <div class="max-w-3xl mx-auto text-xl leading-8 font-medium mt-8"> |  | ||||||
|       <p class="text-center">{$_('oss_credit_description')}</p> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-screen leading-8 pl-5 mt-5"> |  | ||||||
| 		{#await license_promise} | 		{#await license_promise} | ||||||
|         <p class="text-center w-full">{$_('licenses-are-being-loaded')}</p> | 			<p>{$_("licenses-are-being-loaded")}</p> | ||||||
| 		{:then} | 		{:then} | ||||||
|         <table> | 			<table class="font-mono"> | ||||||
|           <thead> | 				<thead class="border-b border-gray-400"> | ||||||
|             <tr> | 					<tr class="odd:bg-white even:bg-gray-100"> | ||||||
|               <th>{$_('dependency_name')}</th> | 						<th>{$_("dependency_name")}</th> | ||||||
|               <th>{$_('license')}</th> | 						<th>{$_("license")}</th> | ||||||
|               <th>{$_('repo_link')}</th> | 						<th>{$_("repo_link")}</th> | ||||||
|               <th>{$_('installed-version')}</th> | 						<th>{$_("installed-version")}</th> | ||||||
|               <th>{$_('author')}</th> | 						<th>{$_("author")}</th> | ||||||
| 					</tr> | 					</tr> | ||||||
| 				</thead> | 				</thead> | ||||||
| 				<tbody> | 				<tbody> | ||||||
| 					{#each licenses as l} | 					{#each licenses as l} | ||||||
|               <tr> | 						<tr class="odd:bg-white even:bg-gray-100 *:p-2"> | ||||||
| 							<td>{l.name}</td> | 							<td>{l.name}</td> | ||||||
| 							<td> | 							<td> | ||||||
|                   {l.license || '?'}<br /><span | 								<button | ||||||
| 									class="underline cursor-pointer" | 									class="underline cursor-pointer" | ||||||
| 									on:click={() => { | 									on:click={() => { | ||||||
| 										modal_open = true; | 										modal_open = true; | ||||||
|                       currentlicense = l.name + '@' + l.version; | 										currentlicense = l.name + "@" + l.version; | ||||||
|                       licensetext = l.licensetext || $_('no-license-text-could-be-found'); | 										licensetext = | ||||||
|                     }}>{$_('read-license')}</span> | 											l.licensetext || $_("no-license-text-could-be-found"); | ||||||
|  | 									}}>{l.license || "?"}</button | ||||||
|  | 								> | ||||||
| 							</td> | 							</td> | ||||||
| 							<td> | 							<td> | ||||||
| 								{(l.repo?.url || l.repo) | 								{(l.repo?.url || l.repo) | ||||||
|                     .replace('git+', '') | 									.replace("git+", "") | ||||||
|                     .replace('git://', '')} | 									.replace("git://", "")} | ||||||
| 							</td> | 							</td> | ||||||
|                 <td>{l.version || '?'}</td> | 							<td>{l.version || "?"}</td> | ||||||
|                 <td>{l.author?.name || l.author || '?'}</td> | 							<td>{l.author?.name || l.author || "?"}</td> | ||||||
| 						</tr> | 						</tr> | ||||||
| 					{/each} | 					{/each} | ||||||
| 				</tbody> | 				</tbody> | ||||||
| 			</table> | 			</table> | ||||||
| 		{:catch error} | 		{:catch error} | ||||||
| 			<div | 			<div | ||||||
|           class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> | 				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} | ||||||
| 	</div> | 	</div> | ||||||
|     <div class="w-full leading-8 mt-8"> | 	<div class="w-full mt-8"> | ||||||
|       <p class="text-xl font-medium">{$_('icon-image-credits')}</p> | 		<p class="font-medium">{$_("icon-image-credits")}</p> | ||||||
|       <ul class="list-disc"> | 		<ul class="list-disc ml-6"> | ||||||
| 			<li> | 			<li> | ||||||
| 				<a | 				<a | ||||||
| 					class="underline" | 					class="underline" | ||||||
| 					target="_blank" | 					target="_blank" | ||||||
| 					rel="noopener noreferrer" | 					rel="noopener noreferrer" | ||||||
|             href="https://storyset.com">https://storyset.com</a> | 					href="https://storyset.com">https://storyset.com</a | ||||||
|  | 				> | ||||||
| 			</li> | 			</li> | ||||||
| 			<li> | 			<li> | ||||||
| 				<a | 				<a | ||||||
| 					class="underline" | 					class="underline" | ||||||
| 					target="_blank" | 					target="_blank" | ||||||
| 					rel="noopener noreferrer" | 					rel="noopener noreferrer" | ||||||
|             href="https://undraw.co">https://undraw.co</a> | 					href="https://undraw.co">https://undraw.co</a | ||||||
|  | 				> | ||||||
| 			</li> | 			</li> | ||||||
| 			<li> | 			<li> | ||||||
| 				<a | 				<a | ||||||
| 					class="underline" | 					class="underline" | ||||||
| 					target="_blank" | 					target="_blank" | ||||||
| 					rel="noopener noreferrer" | 					rel="noopener noreferrer" | ||||||
|             href="https://remixicon.com">https://remixicon.com</a> | 					href="https://remixicon.com">https://remixicon.com</a | ||||||
|  | 				> | ||||||
| 			</li> | 			</li> | ||||||
| 		</ul> | 		</ul> | ||||||
| 	</div> | 	</div> | ||||||
|   </div> | </section> | ||||||
| </div> |  | ||||||
|   | |||||||
| @@ -20,23 +20,32 @@ | |||||||
|       class="underline" |       class="underline" | ||||||
|       href="https://odit.services" |       href="https://odit.services" | ||||||
|       rel="noopener,noreferrer" |       rel="noopener,noreferrer" | ||||||
|       target="_blank">ODIT.Services</a> |       target="_blank">ODIT.Services</a | ||||||
|  |     > | ||||||
|   </p> |   </p> | ||||||
|   <p class="text-sm text-gray-500 mt-4"> |   <p class="text-sm text-gray-500 mt-4"> | ||||||
|     <a |     <a | ||||||
|       class="underline" |       class="underline" | ||||||
|       target="_blank" |       target="_blank" | ||||||
|       rel="noopener, noreferrer" |       rel="noopener, noreferrer" | ||||||
|       href="https://git.odit.services/lfk/frontend/">LfK!Frontend</a>@<a |       href="https://git.odit.services/lfk/frontend/">LfK!Frontend</a | ||||||
|  |     >@<a | ||||||
|       class="underline" |       class="underline" | ||||||
|       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> | ||||||
|     - |     - | ||||||
|     <a class="underline" href="/imprint">{$_('imprint')}</a> |     <a class="underline" href="/imprint">{$_("imprint")}</a> | ||||||
|   </p> |   </p> | ||||||
| </footer> | </footer> | ||||||
|   | |||||||
| @@ -1,20 +1,20 @@ | |||||||
| <script> | <script> | ||||||
|   import { _, getLocaleFromNavigator } from "svelte-i18n"; |   import { _, getLocaleFromNavigator } from "svelte-i18n"; | ||||||
|   import marked from "marked"; |   import { parse } from "marked"; | ||||||
|   import Footer from "./Footer.svelte"; |   import Footer from "./Footer.svelte"; | ||||||
|   import * as css from "../base/simple.css"; |   // import * as css from "../base/simple.css"; | ||||||
|   let html = ""; |   let html = ""; | ||||||
|   async function load() { |   async function load() { | ||||||
|     let md = await fetch("/imprint_" + getLocaleFromNavigator() + ".md"); |     let md = await fetch("/imprint_" + getLocaleFromNavigator() + ".md"); | ||||||
|     let text = (await md.text()).toString(); |     let text = (await md.text()).toString(); | ||||||
|     if (text.includes("<meta")) { |     if (text.includes("<meta")) { | ||||||
|       md.ok=false |       md.ok = false; | ||||||
|     } |     } | ||||||
|     if (!md.ok) { |     if (!md.ok) { | ||||||
|       md = await fetch("/imprint_en.md"); |       md = await fetch("/imprint_en.md"); | ||||||
|       text = await md.text(); |       text = await md.text(); | ||||||
|     } |     } | ||||||
|     html = marked(text); |     html = parse(text); | ||||||
|   } |   } | ||||||
|   const promise = load(); |   const promise = load(); | ||||||
| </script> | </script> | ||||||
| @@ -22,8 +22,9 @@ | |||||||
| <div class="pt-12 px-4 sm:px-6 lg:px-8 lg:pt-20 bg-gray-900 pb-12"> | <div class="pt-12 px-4 sm:px-6 lg:px-8 lg:pt-20 bg-gray-900 pb-12"> | ||||||
|   <div class="text-center mb-8"> |   <div class="text-center mb-8"> | ||||||
|     <h1 |     <h1 | ||||||
|       class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl"> |       class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl" | ||||||
|       {$_('imprint')} |     > | ||||||
|  |       {$_("imprint")} | ||||||
|     </h1> |     </h1> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
| @@ -31,16 +32,17 @@ | |||||||
| <div class="pt-0 pb-16 overflow-hidden lg:pt-12 lg:py-24"> | <div class="pt-0 pb-16 overflow-hidden lg:pt-12 lg:py-24"> | ||||||
|   <div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8"> |   <div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8"> | ||||||
|     {#await promise} |     {#await promise} | ||||||
|       <p class="text-center w-full">{$_('imprint-loading')}</p> |       <p class="text-center w-full">{$_("imprint-loading")}</p> | ||||||
|     {:then} |     {:then} | ||||||
|       <div class="simplecontent"> |       <div class="simplecontent"> | ||||||
|         {@html html} |         {@html html} | ||||||
|       </div> |       </div> | ||||||
|     {:catch error} |     {:catch error} | ||||||
|       <div |       <div | ||||||
|         class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |         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> | ||||||
|   | |||||||
| @@ -7,16 +7,19 @@ | |||||||
|     <div class="w-full bg-white flex items-center justify-center"> |     <div class="w-full bg-white flex items-center justify-center"> | ||||||
|       <div class="max-w-sm m-8"> |       <div class="max-w-sm m-8"> | ||||||
|         <div class="text-black text-5xl md:text-15xl font-black"> |         <div class="text-black text-5xl md:text-15xl font-black"> | ||||||
|           {$_('404title')} |           {$_("404title")} | ||||||
|         </div> |         </div> | ||||||
|         <div class="w-16 h-1 bg-purple-light my-3 md:my-6" /> |         <div class="w-16 h-1 bg-purple-light my-3 md:my-6" /> | ||||||
|         <p |         <p | ||||||
|           class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal"> |           class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal" | ||||||
|           {$_('404message')} |         > | ||||||
|  |           {$_("404message")} | ||||||
|         </p> |         </p> | ||||||
|         <a |         <a | ||||||
|           href="/" |           href="/" | ||||||
|           class="bg-transparent text-grey-darkest font-bold uppercase tracking-wide py-3 px-6 border-2 border-grey-light hover:border-grey rounded-lg">{$_('goback')}</a> |           class="bg-transparent text-grey-darkest font-bold uppercase tracking-wide py-3 px-6 border-2 border-grey-light hover:border-grey rounded-lg" | ||||||
|  |           >{$_("goback")}</a | ||||||
|  |         > | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|   | |||||||
| @@ -1,20 +1,20 @@ | |||||||
| <script> | <script> | ||||||
|   import { _, getLocaleFromNavigator } from "svelte-i18n"; |   import { _, getLocaleFromNavigator } from "svelte-i18n"; | ||||||
|   import marked from "marked"; |   import { parse } from "marked"; | ||||||
|   import Footer from "./Footer.svelte"; |   import Footer from "./Footer.svelte"; | ||||||
|   import * as css from "../base/simple.css"; |   // import * as css from "../base/simple.css?inline"; | ||||||
|   let html = ""; |   let html = ""; | ||||||
|   async function load() { |   async function load() { | ||||||
|     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); |     let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md"); | ||||||
|     let text = (await md.text()).toString(); |     let text = (await md.text()).toString(); | ||||||
|     if (text.includes("<meta")) { |     if (text.includes("<meta")) { | ||||||
|       md.ok=false |       md.ok = false; | ||||||
|     } |     } | ||||||
|     if (!md.ok) { |     if (!md.ok) { | ||||||
|       md = await fetch("/privacy_en.md"); |       md = await fetch("/privacy_en.md"); | ||||||
|       text = await md.text(); |       text = await md.text(); | ||||||
|     } |     } | ||||||
|     html = marked(text); |     html = parse(text); | ||||||
|   } |   } | ||||||
|   const promise = load(); |   const promise = load(); | ||||||
| </script> | </script> | ||||||
| @@ -22,8 +22,9 @@ | |||||||
| <div class="pt-12 px-4 sm:px-6 lg:px-8 lg:pt-20 bg-gray-900 pb-12"> | <div class="pt-12 px-4 sm:px-6 lg:px-8 lg:pt-20 bg-gray-900 pb-12"> | ||||||
|   <div class="text-center mb-8"> |   <div class="text-center mb-8"> | ||||||
|     <h1 |     <h1 | ||||||
|       class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl"> |       class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl" | ||||||
|       {$_('privacy')} |     > | ||||||
|  |       {$_("privacy")} | ||||||
|     </h1> |     </h1> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
| @@ -31,16 +32,17 @@ | |||||||
| <div class="pt-0 pb-16 overflow-hidden lg:pt-12 lg:py-24"> | <div class="pt-0 pb-16 overflow-hidden lg:pt-12 lg:py-24"> | ||||||
|   <div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8"> |   <div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8"> | ||||||
|     {#await promise} |     {#await promise} | ||||||
|       <p class="text-center w-full">{$_('privacy-loading')}</p> |       <p class="text-center w-full">{$_("privacy-loading")}</p> | ||||||
|     {:then} |     {:then} | ||||||
|       <div class="simplecontent"> |       <div class="simplecontent"> | ||||||
|         {@html html} |         {@html html} | ||||||
|       </div> |       </div> | ||||||
|     {:catch error} |     {:catch error} | ||||||
|       <div |       <div | ||||||
|         class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |         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> | ||||||
|   | |||||||
| @@ -4,26 +4,33 @@ | |||||||
|  |  | ||||||
| <div class="md:flex flex-col md:flex-row h-screen w-full"> | <div class="md:flex flex-col md:flex-row h-screen w-full"> | ||||||
|   <div |   <div | ||||||
|     class="flex flex-col w-full md:w-64 text-gray-700 bg-white dark-mode:text-gray-200 dark-mode:bg-gray-800 flex-shrink-0"> |     class="flex flex-col w-full md:w-64 text-gray-700 bg-white dark-mode:text-gray-200 dark-mode:bg-gray-800 flex-shrink-0" | ||||||
|  |   > | ||||||
|     <div |     <div | ||||||
|       class="flex-shrink-0 px-8 py-4 flex flex-row items-center justify-between"> |       class="flex-shrink-0 px-8 py-4 flex flex-row items-center justify-between" | ||||||
|  |     > | ||||||
|       <a |       <a | ||||||
|         href="/#/test" |         href="/#/test" | ||||||
|         class="text-lg font-semibold tracking-widest text-gray-900 uppercase rounded-lg dark-mode:text-white focus:outline-none focus:shadow-outline">Sidebar</a> |         class="text-lg font-semibold tracking-widest text-gray-900 uppercase rounded-lg dark-mode:text-white focus:outline-none focus:shadow-outline" | ||||||
|  |         >Sidebar</a | ||||||
|  |       > | ||||||
|       <button |       <button | ||||||
|         class="rounded-lg md:hidden focus:outline-none focus:shadow-outline"> |         class="rounded-lg md:hidden focus:outline-none focus:shadow-outline" | ||||||
|  |       > | ||||||
|         <svg fill="currentColor" viewBox="0 0 20 20" class="w-6 h-6"> |         <svg fill="currentColor" viewBox="0 0 20 20" class="w-6 h-6"> | ||||||
|           {#if open} |           {#if open} | ||||||
|             <path |             <path | ||||||
|               fill-rule="evenodd" |               fill-rule="evenodd" | ||||||
|               d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" |               d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" | ||||||
|               clip-rule="evenodd" /> |               clip-rule="evenodd" | ||||||
|  |             /> | ||||||
|           {/if} |           {/if} | ||||||
|           {#if !open} |           {#if !open} | ||||||
|             <path |             <path | ||||||
|               fill-rule="evenodd" |               fill-rule="evenodd" | ||||||
|               d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 10a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM9 15a1 1 0 011-1h6a1 1 0 110 2h-6a1 1 0 01-1-1z" |               d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 10a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM9 15a1 1 0 011-1h6a1 1 0 110 2h-6a1 1 0 01-1-1z" | ||||||
|               clip-rule="evenodd" /> |               clip-rule="evenodd" | ||||||
|  |             /> | ||||||
|           {/if} |           {/if} | ||||||
|         </svg> |         </svg> | ||||||
|       </button> |       </button> | ||||||
| @@ -31,49 +38,63 @@ | |||||||
|     <nav |     <nav | ||||||
|       :class:block={open} |       :class:block={open} | ||||||
|       :class:hidden={!open} |       :class:hidden={!open} | ||||||
|       class="flex-grow md:block px-4 pb-4 md:pb-0 md:overflow-y-auto"> |       class="flex-grow md:block px-4 pb-4 md:pb-0 md:overflow-y-auto" | ||||||
|  |     > | ||||||
|       <a |       <a | ||||||
|         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-gray-200 rounded-lg dark-mode:bg-gray-700 dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-gray-200 rounded-lg dark-mode:bg-gray-700 dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|         href="#">Blog</a> |         href="#">Blog</a | ||||||
|  |       > | ||||||
|       <a |       <a | ||||||
|         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|         href="#">Portfolio</a> |         href="#">Portfolio</a | ||||||
|  |       > | ||||||
|       <a |       <a | ||||||
|         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|         href="#">About</a> |         href="#">About</a | ||||||
|  |       > | ||||||
|       <a |       <a | ||||||
|         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |         class="block px-4 py-2 mt-2 text-sm font-semibold text-gray-900 bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|         href="#">Contact</a> |         href="#">Contact</a | ||||||
|  |       > | ||||||
|       <div class="relative"> |       <div class="relative"> | ||||||
|         <button |         <button | ||||||
|           on:click={() => { |           on:click={() => { | ||||||
|             open = !open; |             open = !open; | ||||||
|           }} |           }} | ||||||
|           class="flex flex-row items-center w-full px-4 py-2 mt-2 text-sm font-semibold text-left bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:focus:bg-gray-600 dark-mode:hover:bg-gray-600 md:block hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline"> |           class="flex flex-row items-center w-full px-4 py-2 mt-2 text-sm font-semibold text-left bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:focus:bg-gray-600 dark-mode:hover:bg-gray-600 md:block hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|  |         > | ||||||
|           <span>Dropdown</span> |           <span>Dropdown</span> | ||||||
|           <svg |           <svg | ||||||
|             fill="currentColor" |             fill="currentColor" | ||||||
|             viewBox="0 0 20 20" |             viewBox="0 0 20 20" | ||||||
|             class="inline w-4 h-4 mt-1 ml-1 transition-transform duration-200 transform md:-mt-1"><path |             class="inline w-4 h-4 mt-1 ml-1 transition-transform duration-200 transform md:-mt-1" | ||||||
|  |             ><path | ||||||
|               fill-rule="evenodd" |               fill-rule="evenodd" | ||||||
|               d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z" |               d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z" | ||||||
|               clip-rule="evenodd" /></svg> |               clip-rule="evenodd" | ||||||
|  |             /></svg | ||||||
|  |           > | ||||||
|         </button> |         </button> | ||||||
|         <div |         <div | ||||||
|           class:block={open} |           class:block={open} | ||||||
|           class:hidden={!open} |           class:hidden={!open} | ||||||
|           class="absolute right-0 w-full mt-2 origin-top-right rounded-md shadow-lg"> |           class="absolute right-0 w-full mt-2 origin-top-right rounded-md shadow-lg" | ||||||
|  |         > | ||||||
|           <div |           <div | ||||||
|             class="px-2 py-2 bg-white rounded-md shadow dark-mode:bg-gray-800"> |             class="px-2 py-2 bg-white rounded-md shadow dark-mode:bg-gray-800" | ||||||
|  |           > | ||||||
|             <a |             <a | ||||||
|               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|               href="#">Link #1</a> |               href="#">Link #1</a | ||||||
|  |             > | ||||||
|             <a |             <a | ||||||
|               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|               href="#">Link #2</a> |               href="#">Link #2</a | ||||||
|  |             > | ||||||
|             <a |             <a | ||||||
|               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" |               class="block px-4 py-2 mt-2 text-sm font-semibold bg-transparent rounded-lg dark-mode:bg-transparent dark-mode:hover:bg-gray-600 dark-mode:focus:bg-gray-600 dark-mode:focus:text-white dark-mode:hover:text-white dark-mode:text-gray-200 md:mt-0 hover:text-gray-900 focus:text-gray-900 hover:bg-gray-200 focus:bg-gray-200 focus:outline-none focus:shadow-outline" | ||||||
|               href="#">Link #3</a> |               href="#">Link #3</a | ||||||
|  |             > | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import { UserGroupService } from "@odit/lfk-client-js"; |   import { UserGroupService } from "@odit/lfk-client-js"; | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_groups; |   export let current_groups; | ||||||
|   let description_input_value; |   let description_input_value; | ||||||
| @@ -32,10 +31,7 @@ | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       toast.loading($_("group-is-being-added")); | ||||||
|         text: $_('group-is-being-added'), |  | ||||||
|         duration: -1, |  | ||||||
|       }).showToast(); |  | ||||||
|       let postdata = { |       let postdata = { | ||||||
|         name: name_input_value, |         name: name_input_value, | ||||||
|         description: description_input_value, |         description: description_input_value, | ||||||
| @@ -46,11 +42,8 @@ | |||||||
|           description_input_value = ""; |           description_input_value = ""; | ||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           // |           // | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_('group-added'), |           toast.success($_("group-added")); | ||||||
|             duration: 500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
|           current_groups.push(result); |           current_groups.push(result); | ||||||
|           current_groups = current_groups; |           current_groups = current_groups; | ||||||
|         }) |         }) | ||||||
| @@ -59,8 +52,6 @@ | |||||||
|         }) |         }) | ||||||
|         .finally(() => { |         .finally(() => { | ||||||
|           processed_last_submit = true; |           processed_last_submit = true; | ||||||
|           // |  | ||||||
|           toast.hideToast(); |  | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -68,105 +59,124 @@ | |||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|     }}> |     }} | ||||||
|  |   > | ||||||
|     <div |     <div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|         role="dialog" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |       > | ||||||
|           <div class="sm:flex sm:items-start"> |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|             <div |             <div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|               <svg |               <svg | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 640 512" |                 viewBox="0 0 640 512" | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> |                 height="24" | ||||||
|  |                 ><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|                 <path |                 <path | ||||||
|                   d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" /></svg> |                   d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> |             <div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('create-a-new-user-group')} |                 {$_("create-a-new-user-group")} | ||||||
|               </h3> |               </h3> | ||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <p class="text-sm text-gray-500"> |                 <p class="text-sm text-gray-500"> | ||||||
|                   {$_('please-provide-the-required-information-for-creating-a-new-user-group')} |                   {$_( | ||||||
|  |                     "please-provide-the-required-information-for-creating-a-new-user-group" | ||||||
|  |                   )} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <div class="grid grid-cols-6 gap-6"> |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="firstname" |                     for="firstname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     use:focus |                     use:focus | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder="{$_('name')}" |                     placeholder={$_("name")} | ||||||
|                     class:border-red-500={!isNameValid} |                     class:border-red-500={!isNameValid} | ||||||
|                     class:focus:border-red-500={!isNameValid} |                     class:focus:border-red-500={!isNameValid} | ||||||
|                     class:focus:ring-red-500={!isNameValid} |                     class:focus:ring-red-500={!isNameValid} | ||||||
|                     bind:value={name_input_value} |                     bind:value={name_input_value} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="firstname" |                     name="firstname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isNameValid} |                   {#if !isNameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('name-is-required')} |                     > | ||||||
|  |                       {$_("name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="trackname" |                     for="trackname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('description-optional')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("description-optional")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder="{$_('something-about-the-group')}" |                     placeholder={$_("something-about-the-group")} | ||||||
|                     bind:value={description_input_value} |                     bind:value={description_input_value} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="trackname" |                     name="trackname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|           <button |           <button | ||||||
|             disabled={!createbtnenabled} |             disabled={!createbtnenabled} | ||||||
|             class:opacity-50={!createbtnenabled} |             class:opacity-50={!createbtnenabled} | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             type="button" |             type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|             {$_('create')} |           > | ||||||
|  |             {$_("create")} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               modal_open = false; |               modal_open = false; | ||||||
|             }} |             }} | ||||||
|             type="button" |             type="button" | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full 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 hidden lg:block" | ||||||
|             {$_('cancel')} |           > | ||||||
|  |             {$_("cancel")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
| <script> | <script> | ||||||
| 	import { _ } from "svelte-i18n"; | 	import { _ } from "svelte-i18n"; | ||||||
| 	import store from "../../store"; | 	import store from "../../store"; | ||||||
|   import { | 	import { UserGroupService } from "@odit/lfk-client-js"; | ||||||
|     UserGroupService | 	import toast from "svelte-french-toast"; | ||||||
|   } from "@odit/lfk-client-js"; |  | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
| 	import PromiseError from "../base/PromiseError.svelte"; | 	import PromiseError from "../base/PromiseError.svelte"; | ||||||
| 	let data_loaded = false; | 	let data_loaded = false; | ||||||
| 	export let params; | 	export let params; | ||||||
| @@ -31,10 +30,11 @@ | |||||||
| 	$: search_permission = ""; | 	$: search_permission = ""; | ||||||
| 	$: original_data = {}; | 	$: original_data = {}; | ||||||
| 	$: editable = {}; | 	$: editable = {}; | ||||||
|   $: changes_performed = !(JSON.stringify(original_data) == JSON.stringify(editable)); | 	$: changes_performed = !( | ||||||
|  | 		JSON.stringify(original_data) == JSON.stringify(editable) | ||||||
|  | 	); | ||||||
| 	$: isGroupnameValid = editable.name !== ""; | 	$: isGroupnameValid = editable.name !== ""; | ||||||
|   $: save_enabled = | 	$: save_enabled = changes_performed && isGroupnameValid; | ||||||
|     changes_performed && isGroupnameValid  |  | ||||||
| 	promise.then((data) => { | 	promise.then((data) => { | ||||||
| 		let current_target = ""; | 		let current_target = ""; | ||||||
| 		let colorindex = -1; | 		let colorindex = -1; | ||||||
| @@ -59,20 +59,13 @@ | |||||||
| 	}); | 	}); | ||||||
| 	function submit() { | 	function submit() { | ||||||
| 		if (data_loaded === true && save_enabled) { | 		if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ | 			toast($_("updating-group")); | ||||||
|         text: $_('updateing-group'), |  | ||||||
|         duration: 2500, |  | ||||||
|       }).showToast(); |  | ||||||
| 			UserGroupService.userGroupControllerPut(original_data.id, editable) | 			UserGroupService.userGroupControllerPut(original_data.id, editable) | ||||||
| 				.then((resp) => { | 				.then((resp) => { | ||||||
| 					Object.assign(original_data, editable); | 					Object.assign(original_data, editable); | ||||||
| 					original_data = editable; | 					original_data = editable; | ||||||
| 					Object.assign(original_data, editable); | 					Object.assign(original_data, editable); | ||||||
|           Toastify({ | 					toast.success($_("group-updated")); | ||||||
|             text: $_('group-updated'), |  | ||||||
|             duration: 2500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
| 				}) | 				}) | ||||||
| 				.catch((err) => {}); | 				.catch((err) => {}); | ||||||
| 		} else { | 		} else { | ||||||
| @@ -88,54 +81,53 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#await promise} | {#await promise} | ||||||
|   {$_('loading-group-detail')} | 	{$_("loading-group-detail")} | ||||||
| {:then} | {:then} | ||||||
| 	<section class="container p-5 select-none"> | 	<section class="container p-5 select-none"> | ||||||
| 		<div class="flex flex-row mb-4"> | 		<div class="flex flex-row mb-4"> | ||||||
| 			<div class="w-full"> | 			<div class="w-full"> | ||||||
| 				<nav class="w-full flex"> | 				<nav class="w-full flex"> | ||||||
| 					<ol class="list-none flex flex-row items-center justify-start"> | 					<ol class="list-none flex flex-row items-center justify-start"> | ||||||
|  | 						<li class="flex items-center"></li> | ||||||
| 						<li class="flex items-center"> | 						<li class="flex items-center"> | ||||||
|               <svg class="flex-shrink-0 w-5 h-5 mr-2" fill="currentColor" width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 512"><path fill="currentColor" d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z"></path></svg> | 							<a class="mr-2" href="../" | ||||||
|             </li> | 								><svg | ||||||
|             <li class="flex items-center"> | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|               <a class="mr-2" href="../">{$_('groups')}</a><svg | 									width="24" | ||||||
|                 stroke="currentColor" | 									height="24" | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
| 									viewBox="0 0 24 24" | 									viewBox="0 0 24 24" | ||||||
|  | 									fill="none" | ||||||
|  | 									stroke="currentColor" | ||||||
|  | 									stroke-width="2" | ||||||
| 									stroke-linecap="round" | 									stroke-linecap="round" | ||||||
| 									stroke-linejoin="round" | 									stroke-linejoin="round" | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" | 									class="inline-block" | ||||||
|                 height="1em" | 									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg | ||||||
|                 width="1em" | 								> | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line | 								{$_("groups")}</a | ||||||
|                   x1="5" | 							> | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <span class="mr-2">{editable.name}</span> |  | ||||||
| 						</li> | 						</li> | ||||||
| 					</ol> | 					</ol> | ||||||
| 				</nav> | 				</nav> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> | 		<div class="mb-4 text-3xl font-extrabold leading-tight"> | ||||||
|       {original_data.name} | 			{editable.name} | ||||||
|       <span data-id="group_actions_${editable.id}"> | 			<div data-id="group_actions_${editable.id}"> | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:DELETE')} | 				{#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:DELETE")} | ||||||
| 					{#if delete_triggered} | 					{#if delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| 							on:click={deleteGroup} | 							on:click={deleteGroup} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('confirm-deletion')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("confirm-deletion")}</button | ||||||
|  | 						> | ||||||
| 						<button | 						<button | ||||||
| 							on:click={() => { | 							on:click={() => { | ||||||
| 								delete_triggered = !delete_triggered; | 								delete_triggered = !delete_triggered; | ||||||
| 							}} | 							}} | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm">{$_('cancel')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("cancel")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 					{#if !delete_triggered} | 					{#if !delete_triggered} | ||||||
| 						<button | 						<button | ||||||
| @@ -143,7 +135,9 @@ | |||||||
| 								delete_triggered = true; | 								delete_triggered = true; | ||||||
| 							}} | 							}} | ||||||
| 							type="button" | 							type="button" | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('delete-group')}</button> | 							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 							>{$_("delete-group")}</button | ||||||
|  | 						> | ||||||
| 					{/if} | 					{/if} | ||||||
| 				{/if} | 				{/if} | ||||||
| 				{#if !delete_triggered} | 				{#if !delete_triggered} | ||||||
| @@ -152,64 +146,77 @@ | |||||||
| 						class:opacity-50={!save_enabled} | 						class:opacity-50={!save_enabled} | ||||||
| 						type="button" | 						type="button" | ||||||
| 						on:click={submit} | 						on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('save-changes')}</button> | 						class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 						>{$_("save-changes")}</button | ||||||
|  | 					> | ||||||
| 				{/if} | 				{/if} | ||||||
|       </span> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<!--  --> | 		<!--  --> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="title" class="font-semibold text-gray-700">{$_("name")}</label | ||||||
|         for="title" | 			> | ||||||
|         class="font-medium text-gray-700">{$_('name')}</label> |  | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('name')} | 				placeholder={$_("name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.name} | 				bind:value={editable.name} | ||||||
| 				class:border-red-500={!isGroupnameValid} | 				class:border-red-500={!isGroupnameValid} | ||||||
| 				class:focus:border-red-500={!isGroupnameValid} | 				class:focus:border-red-500={!isGroupnameValid} | ||||||
| 				class:focus:ring-red-500={!isGroupnameValid} | 				class:focus:ring-red-500={!isGroupnameValid} | ||||||
| 				name="title" | 				name="title" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 			{#if !isGroupnameValid} | 			{#if !isGroupnameValid} | ||||||
| 				<span | 				<span | ||||||
|           class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|           {$_('group-name-is-required')} | 				> | ||||||
|  | 					{$_("group-name-is-required")} | ||||||
| 				</span> | 				</span> | ||||||
| 			{/if} | 			{/if} | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="groupdescription" class="font-semibold text-gray-700" | ||||||
|         for="firstname" | 				>{$_("description")}</label | ||||||
|         class="font-medium text-gray-700">{$_('description')}</label> | 			> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('description')} | 				placeholder={$_("description")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.description} | 				bind:value={editable.description} | ||||||
|         name="firstname" | 				name="groupdescription" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full mt-8"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <p class="font-medium mb-4"> | 			<p class="font-semibold mb-4"> | ||||||
|         {$_('permissions')} | 				{$_("permissions")} | ||||||
|  | 			</p> | ||||||
|  | 			<div> | ||||||
| 				<a | 				<a | ||||||
| 					class="px-4 py-2 bg-gray-500 rounded-md text-white" | 					class="px-4 py-2 bg-gray-500 rounded-md text-white" | ||||||
|           href="/groups/{params.groupid}/permissions/">{$_('edit-permissions')}</a> | 					href="/groups/{params.groupid}/permissions/" | ||||||
|       </p> | 					>{$_("edit-permissions")}</a | ||||||
|  | 				> | ||||||
|  | 			</div> | ||||||
| 			<div class="w-full sm:my-px sm:px-px sm:w-1/2"> | 			<div class="w-full sm:my-px sm:px-px sm:w-1/2"> | ||||||
| 				<input | 				<input | ||||||
| 					autocomplete="off" | 					autocomplete="off" | ||||||
|           placeholder="{$_('search-for-permission')}" | 					placeholder={$_("search-for-permission")} | ||||||
| 					type="text" | 					type="text" | ||||||
| 					bind:value={search_permission} | 					bind:value={search_permission} | ||||||
|           class="mt-4 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 dark:bg-gray-900 dark:text-gray-100 rounded-md p-2" /> | 					class="mt-4 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 				/> | ||||||
| 			</div> | 			</div> | ||||||
| 			{#each original_data.permissions as p} | 			{#each original_data.permissions as p} | ||||||
| 				{#if p.toLowerCase().includes(search_permission.toLowerCase())} | 				{#if p.toLowerCase().includes(search_permission.toLowerCase())} | ||||||
| 					<span | 					<span | ||||||
|             style="background:{matched_colors[p.split(':')[0]][0]};color:{matched_colors[p.split(':')[0]][1]};" | 						style="background:{matched_colors[ | ||||||
|             class="mt-1 inline-flex items-center justify-center px-2 py-1 text-xs font-bold leading-none text-indigo-100 rounded">{p}</span> | 							p.split(':')[0] | ||||||
|  | 						][0]};color:{matched_colors[p.split(':')[0]][1]};" | ||||||
|  | 						class="mt-1 inline-flex items-center justify-center px-2 py-1 text-xs font-bold leading-none text-indigo-100 rounded" | ||||||
|  | 						>{p}</span | ||||||
|  | 					> | ||||||
| 					<!--  --> | 					<!--  --> | ||||||
| 				{/if} | 				{/if} | ||||||
| 			{/each} | 			{/each} | ||||||
|   | |||||||
| @@ -5,7 +5,8 @@ | |||||||
|     CreatePermission, |     CreatePermission, | ||||||
|     UserGroupService, |     UserGroupService, | ||||||
|   } from "@odit/lfk-client-js"; |   } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |   import toast from 'svelte-french-toast' | ||||||
|  |  | ||||||
|   import PromiseError from "../base/PromiseError.svelte"; |   import PromiseError from "../base/PromiseError.svelte"; | ||||||
|   export let params; |   export let params; | ||||||
|   let [ |   let [ | ||||||
| @@ -20,15 +21,14 @@ UserGroupService, | |||||||
|   $: save_enabled = |   $: save_enabled = | ||||||
|     JSON.stringify(grantedPermissions) === |     JSON.stringify(grantedPermissions) === | ||||||
|     JSON.stringify(grantedPermissions_initial); |     JSON.stringify(grantedPermissions_initial); | ||||||
|   const group_promise = UserGroupService.userGroupControllerGetOne(params.groupid); |   const group_promise = UserGroupService.userGroupControllerGetOne( | ||||||
|  |     params.groupid | ||||||
|  |   ); | ||||||
|   group_promise.then((data) => { |   group_promise.then((data) => { | ||||||
|     original_data = Object.assign(original_data, data); |     original_data = Object.assign(original_data, data); | ||||||
|   }); |   }); | ||||||
|   function submit() { |   function submit() { | ||||||
|     Toastify({ |     toast.loading($_("updating-permissions")); | ||||||
|       text: $_('updating-permissions'), |  | ||||||
|       duration: 2500, |  | ||||||
|     }).showToast(); |  | ||||||
|     to_delete.forEach((d) => { |     to_delete.forEach((d) => { | ||||||
|       promises = promises.concat([ |       promises = promises.concat([ | ||||||
|         PermissionService.permissionControllerRemove(d, true), |         PermissionService.permissionControllerRemove(d, true), | ||||||
| @@ -50,11 +50,7 @@ UserGroupService, | |||||||
|         ); |         ); | ||||||
|       }); |       }); | ||||||
|       grantedPermissions_initial = grantedPermissions; |       grantedPermissions_initial = grantedPermissions; | ||||||
|       Toastify({ |       toast.success($_("permissions-updated")); | ||||||
|         text: $_("permissions-updated"), |  | ||||||
|         duration: 2500, |  | ||||||
|         backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|       }).showToast(); |  | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|   Object.values(CreatePermission.target).forEach((t) => { |   Object.values(CreatePermission.target).forEach((t) => { | ||||||
| @@ -62,13 +58,15 @@ UserGroupService, | |||||||
|       allpermissions = allpermissions.concat([{ target: t, action: a }]); |       allpermissions = allpermissions.concat([{ target: t, action: a }]); | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
|   UserGroupService.userGroupControllerGetPermissions(params.groupid).then((val) => { |   UserGroupService.userGroupControllerGetPermissions(params.groupid).then( | ||||||
|  |     (val) => { | ||||||
|       val.directlyGranted.forEach((p) => { |       val.directlyGranted.forEach((p) => { | ||||||
|         delete p.responseType; |         delete p.responseType; | ||||||
|         grantedPermissions = grantedPermissions.concat([p]); |         grantedPermissions = grantedPermissions.concat([p]); | ||||||
|       }); |       }); | ||||||
|       grantedPermissions_initial = grantedPermissions; |       grantedPermissions_initial = grantedPermissions; | ||||||
|   }); |     } | ||||||
|  |   ); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#await group_promise} | {#await group_promise} | ||||||
| @@ -86,12 +84,15 @@ UserGroupService, | |||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24" |                 height="24" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 640 512"><path |                 viewBox="0 0 640 512" | ||||||
|  |                 ><path | ||||||
|                   fill="currentColor" |                   fill="currentColor" | ||||||
|                   d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" /></svg> |                   d="M610.5 341.3c2.6-14.1 2.6-28.5 0-42.6l25.8-14.9c3-1.7 4.3-5.2 3.3-8.5-6.7-21.6-18.2-41.2-33.2-57.4-2.3-2.5-6-3.1-9-1.4l-25.8 14.9c-10.9-9.3-23.4-16.5-36.9-21.3v-29.8c0-3.4-2.4-6.4-5.7-7.1-22.3-5-45-4.8-66.2 0-3.3.7-5.7 3.7-5.7 7.1v29.8c-13.5 4.8-26 12-36.9 21.3l-25.8-14.9c-2.9-1.7-6.7-1.1-9 1.4-15 16.2-26.5 35.8-33.2 57.4-1 3.3.4 6.8 3.3 8.5l25.8 14.9c-2.6 14.1-2.6 28.5 0 42.6l-25.8 14.9c-3 1.7-4.3 5.2-3.3 8.5 6.7 21.6 18.2 41.1 33.2 57.4 2.3 2.5 6 3.1 9 1.4l25.8-14.9c10.9 9.3 23.4 16.5 36.9 21.3v29.8c0 3.4 2.4 6.4 5.7 7.1 22.3 5 45 4.8 66.2 0 3.3-.7 5.7-3.7 5.7-7.1v-29.8c13.5-4.8 26-12 36.9-21.3l25.8 14.9c2.9 1.7 6.7 1.1 9-1.4 15-16.2 26.5-35.8 33.2-57.4 1-3.3-.4-6.8-3.3-8.5l-25.8-14.9zM496 368.5c-26.8 0-48.5-21.8-48.5-48.5s21.8-48.5 48.5-48.5 48.5 21.8 48.5 48.5-21.7 48.5-48.5 48.5zM96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm224 32c1.9 0 3.7-.5 5.6-.6 8.3-21.7 20.5-42.1 36.3-59.2 7.4-8 17.9-12.6 28.9-12.6 6.9 0 13.7 1.8 19.6 5.3l7.9 4.6c.8-.5 1.6-.9 2.4-1.4 7-14.6 11.2-30.8 11.2-48 0-61.9-50.1-112-112-112S208 82.1 208 144c0 61.9 50.1 112 112 112zm105.2 194.5c-2.3-1.2-4.6-2.6-6.8-3.9-8.2 4.8-15.3 9.8-27.5 9.8-10.9 0-21.4-4.6-28.9-12.6-18.3-19.8-32.3-43.9-40.2-69.6-10.7-34.5 24.9-49.7 25.8-50.3-.1-2.6-.1-5.2 0-7.8l-7.9-4.6c-3.8-2.2-7-5-9.8-8.1-3.3.2-6.5.6-9.8.6-24.6 0-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h255.4c-3.7-6-6.2-12.8-6.2-20.3v-9.2zM173.1 274.6C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <a class="mr-2" href="../../">{$_('user-groups')}</a><svg |               <a class="mr-2" href="../../">{$_("user-groups")}</a><svg | ||||||
|                 stroke="currentColor" |                 stroke="currentColor" | ||||||
|                 fill="none" |                 fill="none" | ||||||
|                 stroke-width="2" |                 stroke-width="2" | ||||||
| @@ -101,12 +102,10 @@ UserGroupService, | |||||||
|                 class="h-3 w-3 mr-2 stroke-current" |                 class="h-3 w-3 mr-2 stroke-current" | ||||||
|                 height="1em" |                 height="1em" | ||||||
|                 width="1em" |                 width="1em" | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                   x1="5" |                 ><line x1="5" y1="12" x2="19" y2="12" /> | ||||||
|                   y1="12" |                 <polyline points="12 5 19 12 12 19" /></svg | ||||||
|                   x2="19" |               > | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <span class="mr-2"><a href="../">{original_data.name}</a></span> |               <span class="mr-2"><a href="../">{original_data.name}</a></span> | ||||||
| @@ -122,45 +121,45 @@ UserGroupService, | |||||||
|                 class="h-3 w-3 mr-2 stroke-current" |                 class="h-3 w-3 mr-2 stroke-current" | ||||||
|                 height="1em" |                 height="1em" | ||||||
|                 width="1em" |                 width="1em" | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                   x1="5" |                 ><line x1="5" y1="12" x2="19" y2="12" /> | ||||||
|                   y1="12" |                 <polyline points="12 5 19 12 12 19" /></svg | ||||||
|                   x2="19" |               > | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |             </li> | ||||||
|             <li class="flex items-center"> |             <li class="flex items-center"> | ||||||
|               <span class="mr-2">{$_('permissions')}</span> |               <span class="mr-2">{$_("permissions")}</span> | ||||||
|             </li> |             </li> | ||||||
|           </ol> |           </ol> | ||||||
|         </nav> |         </nav> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <div class="mb-8 text-3xl font-extrabold"> |     <div class="mb-4 text-3xl font-extrabold"> | ||||||
|       {$_('permissions')}: |       <div> | ||||||
|       {original_data.name} |  | ||||||
|       <span> |  | ||||||
|         {#if promises.length === 0} |         {#if promises.length === 0} | ||||||
|           <button |           <button | ||||||
|             disabled={save_enabled} |             disabled={save_enabled} | ||||||
|             class:opacity-50={save_enabled} |             class:opacity-50={save_enabled} | ||||||
|             type="button" |             type="button" | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('save-changes')}</button> |             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  |             >{$_("save-changes")}</button | ||||||
|  |           > | ||||||
|         {:else} |         {:else} | ||||||
|           <button |           <button | ||||||
|             type="button" |             type="button" | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-yellow-600 text-base font-medium text-white hover:bg-yellow-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-yellow-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('applying-changes')}</button> |             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-yellow-600 text-base font-medium text-white hover:bg-yellow-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-yellow-500 sm:w-auto sm:text-sm" | ||||||
|  |             >{$_("applying-changes")}</button | ||||||
|  |           > | ||||||
|         {/if} |         {/if} | ||||||
|       </span> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <!--  --> |     <!--  --> | ||||||
|     <div class="flex flex-wrap -mx-1 overflow-hidden"> |     <div class="flex flex-wrap -mx-1 overflow-hidden"> | ||||||
|       <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> |       <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> | ||||||
|         {$_('verfuegbare')} |         {$_("available-permissions")} | ||||||
|       </div> |       </div> | ||||||
|       <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> |       <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> | ||||||
|         {$_('granted')} |         {$_("granted")} | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|     <!--  --> |     <!--  --> | ||||||
| @@ -168,12 +167,14 @@ UserGroupService, | |||||||
|       {#if allpermissions.length > 0} |       {#if allpermissions.length > 0} | ||||||
|         <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> |         <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> | ||||||
|           <div |           <div | ||||||
|             class="border-4 border-dashed rounded mb-4 p-5 text-lg text-center"> |             class="border-4 border-dashed rounded mb-4 p-5 text-lg text-center" | ||||||
|  |           > | ||||||
|             {#each allpermissions as p} |             {#each allpermissions as p} | ||||||
|               {#if !(grantedPermissions.filter((o) => p.target == o.target && p.action == o.action).length > 0)} |               {#if !(grantedPermissions.filter((o) => p.target == o.target && p.action == o.action).length > 0)} | ||||||
|                 <p |                 <p | ||||||
|                   class="block w-full mt-1 text-sm dark:border-gray-600 dark:bg-gray-700 bg-gray-200 p-2 focus:border-purple-400 focus:outline-none focus:shadow-outline-purple dark:text-gray-300 dark:focus:shadow-outline-gray form-input"> |                   class="block w-full mt-1 text-sm bg-gray-200 p-2 focus:border-purple-400 focus:outline-none focus:shadow-outline-purple form-input" | ||||||
|                   {p.target + ':' + p.action} |                 > | ||||||
|  |                   {p.target + ":" + p.action} | ||||||
|                   <button |                   <button | ||||||
|                     on:click={() => { |                     on:click={() => { | ||||||
|                       grantedPermissions = grantedPermissions.concat([p]); |                       grantedPermissions = grantedPermissions.concat([p]); | ||||||
| @@ -190,7 +191,9 @@ UserGroupService, | |||||||
|                       } |                       } | ||||||
|                     }} |                     }} | ||||||
|                     type="button" |                     type="button" | ||||||
|                     class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-green-200 text-base font-medium text-black hover:bg-green-700 hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-green-500 sm:ml-3 sm:w-auto sm:text-sm">+</button> |                     class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-green-200 text-base font-medium text-black hover:bg-green-700 hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-green-500 sm:w-auto sm:text-sm" | ||||||
|  |                     >+</button | ||||||
|  |                   > | ||||||
|                 </p> |                 </p> | ||||||
|               {/if} |               {/if} | ||||||
|             {/each} |             {/each} | ||||||
| @@ -198,22 +201,39 @@ UserGroupService, | |||||||
|         </div> |         </div> | ||||||
|         <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> |         <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2"> | ||||||
|           <div |           <div | ||||||
|             class="border-4 border-dashed rounded mb-4 p-5 text-lg text-center"> |             class="border-4 border-dashed rounded mb-4 p-5 text-lg text-center" | ||||||
|  |           > | ||||||
|             {#each grantedPermissions as p} |             {#each grantedPermissions as p} | ||||||
|               <p |               <p | ||||||
|                 class="block w-full mt-1 text-sm dark:border-gray-600 dark:bg-gray-700 bg-gray-200 p-2 focus:border-purple-400 focus:outline-none focus:shadow-outline-purple dark:text-gray-300 dark:focus:shadow-outline-gray form-input"> |                 class="block w-full mt-1 text-sm bg-gray-200 p-2 focus:border-purple-400 focus:outline-none focus:shadow-outline-purple form-input" | ||||||
|                 {p.target + ':' + p.action} |               > | ||||||
|  |                 {p.target + ":" + p.action} | ||||||
|                 <button |                 <button | ||||||
|                   on:click={() => { |                   on:click={() => { | ||||||
|                     grantedPermissions = grantedPermissions.filter((o) => o.target + ':' + o.action !== p.target + ':' + p.action); |                     grantedPermissions = grantedPermissions.filter( | ||||||
|                     if (to_add.some((o) => o.target + ':' + o.action === p.target + ':' + p.action)) { |                       (o) => | ||||||
|                       to_add = to_add.filter((o) => o.target + ':' + o.action !== p.target + ':' + p.action); |                         o.target + ":" + o.action !== p.target + ":" + p.action | ||||||
|  |                     ); | ||||||
|  |                     if ( | ||||||
|  |                       to_add.some( | ||||||
|  |                         (o) => | ||||||
|  |                           o.target + ":" + o.action === | ||||||
|  |                           p.target + ":" + p.action | ||||||
|  |                       ) | ||||||
|  |                     ) { | ||||||
|  |                       to_add = to_add.filter( | ||||||
|  |                         (o) => | ||||||
|  |                           o.target + ":" + o.action !== | ||||||
|  |                           p.target + ":" + p.action | ||||||
|  |                       ); | ||||||
|                     } else { |                     } else { | ||||||
|                       to_delete = to_delete.concat([p.id]); |                       to_delete = to_delete.concat([p.id]); | ||||||
|                     } |                     } | ||||||
|                   }} |                   }} | ||||||
|                   type="button" |                   type="button" | ||||||
|                   class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-300 text-base font-medium text-black hover:bg-red-700 hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm">-</button> |                   class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-300 text-base font-medium text-black hover:bg-red-700 hover:text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:w-auto sm:text-sm" | ||||||
|  |                   >-</button | ||||||
|  |                 > | ||||||
|               </p> |               </p> | ||||||
|             {/each} |             {/each} | ||||||
|           </div> |           </div> | ||||||
|   | |||||||
| @@ -8,22 +8,23 @@ | |||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> |   <h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('user-groups')} |     {$_("user-groups")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:CREATE')} |   </h4> | ||||||
|  |   {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:CREATE")} | ||||||
|     <button |     <button | ||||||
|       on:click={() => { |       on:click={() => { | ||||||
|         modal_open = true; |         modal_open = true; | ||||||
|       }} |       }} | ||||||
|       type="button" |       type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |       class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm" | ||||||
|         {$_('add-user-group')} |     > | ||||||
|  |       {$_("add-user-group")} | ||||||
|     </button> |     </button> | ||||||
|   {/if} |   {/if} | ||||||
|   </span> |  | ||||||
|   <UserGroupsOverview bind:current_groups /> |   <UserGroupsOverview bind:current_groups /> | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:CREATE")} | ||||||
|   <AddGroupModal bind:current_groups bind:modal_open /> |   <AddGroupModal bind:current_groups bind:modal_open /> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
| <div class="text-center items-center justify-center"> | <div class="text-center items-center justify-center"> | ||||||
|   <p class="mb-16 text-lg text-gray-500"> |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|     <img class="w-full h-44" src={groups_empty} alt="" /> |     <img class="w-full h-44" src={groups_empty} alt="" /> | ||||||
|     <span class="font-bold">{$_('there-are-no-groups-yet')}.</span><br /> |     <span class="font-bold">{$_("there-are-no-groups-yet")}.</span><br /> | ||||||
|     <span>{$_('add-your-first-group')}</span> |     <span>{$_("add-your-first-group")}</span> | ||||||
|   </p> |   </p> | ||||||
| </div> | </div> | ||||||
| @@ -13,13 +13,14 @@ | |||||||
|   ); |   ); | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:GET')} | {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:GET")} | ||||||
|   {#await groups_promise} |   {#await groups_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">{$_('groups-are-being-loaded')}</p> |     > | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |       <p class="font-bold">{$_("groups-are-being-loaded")}</p> | ||||||
|  |       <p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|     </div> |     </div> | ||||||
|   {:then} |   {:then} | ||||||
|     {#if current_groups.length === 0} |     {#if current_groups.length === 0} | ||||||
| @@ -28,26 +29,30 @@ | |||||||
|       <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="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border" | ||||||
|  |       /> | ||||||
|       <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 class="odd:bg-white even:bg-gray-100"> | ||||||
|               <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" | ||||||
|                 {$_('description')} |               > | ||||||
|  |                 {$_("description")} | ||||||
|               </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> | ||||||
| @@ -57,7 +62,10 @@ | |||||||
|                 .toString() |                 .toString() | ||||||
|                 .toLowerCase() |                 .toLowerCase() | ||||||
|                 .includes(searchvalue)} |                 .includes(searchvalue)} | ||||||
|                 <tr data-rowid="user_{group.id}"> |                 <tr | ||||||
|  |                   class="odd:bg-white even:bg-gray-100" | ||||||
|  |                   data-rowid="user_{group.id}" | ||||||
|  |                 > | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |                   <td class="px-6 py-4 whitespace-nowrap"> | ||||||
|                     <div class="flex items-center"> |                     <div class="flex items-center"> | ||||||
|                       <div class="ml-4"> |                       <div class="ml-4"> | ||||||
| @@ -72,39 +80,53 @@ | |||||||
|                   </td> |                   </td> | ||||||
|                   {#if active_deletes[group.id] === true} |                   {#if active_deletes[group.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[group.id] = false; |                           active_deletes[group.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={() => { | ||||||
|                           UserGroupService.userGroupControllerRemove(group.id, true) |                           UserGroupService.userGroupControllerRemove( | ||||||
|  |                             group.id, | ||||||
|  |                             true | ||||||
|  |                           ) | ||||||
|                             .then((resp) => { |                             .then((resp) => { | ||||||
|                               current_groups = current_groups.filter((obj) => obj.id !== group.id); |                               current_groups = current_groups.filter( | ||||||
|  |                                 (obj) => obj.id !== group.id | ||||||
|  |                               ); | ||||||
|                             }) |                             }) | ||||||
|                             .catch((err) => { |                             .catch((err) => { | ||||||
|                               // error deleting user |                               // error deleting user | ||||||
|                             }); |                             }); | ||||||
|                         }} |                         }} | ||||||
|                         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="./{group.id}" |                         href="./{group.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('USERGROUP:DELETE')} |                       > | ||||||
|  |                       {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:DELETE")} | ||||||
|                         <button |                         <button | ||||||
|                           on:click={() => { |                           on:click={() => { | ||||||
|                             active_deletes[group.id] = true; |                             active_deletes[group.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} | ||||||
| @@ -118,7 +140,7 @@ | |||||||
|   {: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> | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; |   import { _ } from "svelte-i18n"; | ||||||
|   import { clickOutside } from "../base/outsideclick"; |   import { clickOutside } from "../base/outsideclick"; | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |   import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|  |   import toast from "svelte-french-toast"; | ||||||
|   export let modal_open; |   export let modal_open; | ||||||
|   export let current_organizations; |   export let current_organizations; | ||||||
|   let name_input_dom; |   let name_input_dom; | ||||||
| @@ -48,10 +49,7 @@ | |||||||
|   function submit() { |   function submit() { | ||||||
|     if (processed_last_submit === true) { |     if (processed_last_submit === true) { | ||||||
|       processed_last_submit = false; |       processed_last_submit = false; | ||||||
|       const toast = Toastify({ |       toast.loading($_("organization-is-being-added")); | ||||||
|         text: $_("organization-is-being-added"), |  | ||||||
|         duration: -1, |  | ||||||
|       }).showToast(); |  | ||||||
|       let address = {}; |       let address = {}; | ||||||
|       if (address_checked === true) { |       if (address_checked === true) { | ||||||
|         address = { |         address = { | ||||||
| @@ -70,17 +68,13 @@ | |||||||
|         .then((result) => { |         .then((result) => { | ||||||
|           name = ""; |           name = ""; | ||||||
|           modal_open = false; |           modal_open = false; | ||||||
|           Toastify({ |           toast.dismiss(); | ||||||
|             text: $_("organization-added"), |           toast.success($_("organization-added")); | ||||||
|             duration: 500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
|           current_organizations = current_organizations.concat([result]); |           current_organizations = current_organizations.concat([result]); | ||||||
|         }) |         }) | ||||||
|         .catch((err) => {}) |         .catch((err) => {}) | ||||||
|         .finally(() => { |         .finally(() => { | ||||||
|           processed_last_submit = true; |           processed_last_submit = true; | ||||||
|           toast.hideToast(); |  | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -88,58 +82,70 @@ | |||||||
|  |  | ||||||
| {#if modal_open} | {#if modal_open} | ||||||
|   <div |   <div | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |     class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |     use:clickOutside | ||||||
|     on:click_outside={() => { |     on:click_outside={() => { | ||||||
|       modal_open = false; |       modal_open = false; | ||||||
|     }}> |     }} | ||||||
|  |   > | ||||||
|     <div |     <div | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |       class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  |     > | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|         <div |         <div | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |           class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|           data-id="modal_backdrop" /> |           data-id="modal_backdrop" | ||||||
|  |         /> | ||||||
|       </div> |       </div> | ||||||
|       <span |       <span | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |         class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|         aria-hidden="true">​</span> |         aria-hidden="true">​</span | ||||||
|  |       > | ||||||
|       <div |       <div | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |         class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|         role="dialog" |         role="dialog" | ||||||
|         aria-modal="true" |         aria-modal="true" | ||||||
|         aria-labelledby="modal-headline"> |         aria-labelledby="modal-headline" | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |       > | ||||||
|           <div class="sm:flex sm:items-start"> |         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4 rounded-t-xl"> | ||||||
|  |           <div class=""> | ||||||
|             <div |             <div | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10" | ||||||
|  |             > | ||||||
|               <svg |               <svg | ||||||
|                 class="h-6 w-6 text-blue-600" |                 class="h-6 w-6 text-blue-600" | ||||||
|                 fill="currentColor" |                 fill="currentColor" | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |                 xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |                 viewBox="0 0 24 24" | ||||||
|                 width="24" |                 width="24" | ||||||
|                 height="24"><path |                 height="24" | ||||||
|                   d="M17 19h2v-8h-6v8h2v-6h2v6zM3 19V4a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1v5h2v10h1v2H2v-2h1zm4-8v2h2v-2H7zm0 4v2h2v-2H7zm0-8v2h2V7H7z" /></svg> |                 ><path | ||||||
|  |                   d="M17 19h2v-8h-6v8h2v-6h2v6zM3 19V4a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1v5h2v10h1v2H2v-2h1zm4-8v2h2v-2H7zm0 4v2h2v-2H7zm0-8v2h2V7H7z" | ||||||
|  |                 /></svg | ||||||
|  |               > | ||||||
|             </div> |             </div> | ||||||
|             <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left"> |             <div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900"> |               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|                 {$_('create-a-new-organization')} |                 {$_("create-a-new-organization")} | ||||||
|               </h3> |               </h3> | ||||||
|               <div class="mt-2 mb-6"> |               <div class="mt-2 mb-6"> | ||||||
|                 <p class="text-sm text-gray-500"> |                 <p class="text-sm text-gray-500"> | ||||||
|                   {$_('please-provide-the-required-information-to-add-a-new-organization')} |                   {$_( | ||||||
|  |                     "please-provide-the-required-information-to-add-a-new-organization" | ||||||
|  |                   )} | ||||||
|                 </p> |                 </p> | ||||||
|               </div> |               </div> | ||||||
|               <div class="grid grid-cols-6 gap-6"> |               <div class="grid grid-cols-6 gap-6 text-left"> | ||||||
|                 <div class="col-span-6"> |                 <div class="col-span-6"> | ||||||
|                   <label |                   <label | ||||||
|                     for="firstname" |                     for="firstname" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('name')}</label> |                     class="block text-sm font-medium text-gray-700" | ||||||
|  |                     >{$_("name")}</label | ||||||
|  |                   > | ||||||
|                   <input |                   <input | ||||||
|                     use:focus |                     use:focus | ||||||
|                     autocomplete="off" |                     autocomplete="off" | ||||||
|                     placeholder={$_('name')} |                     placeholder={$_("name")} | ||||||
|                     class:border-red-500={!isOrgnameValid} |                     class:border-red-500={!isOrgnameValid} | ||||||
|                     class:focus:border-red-500={!isOrgnameValid} |                     class:focus:border-red-500={!isOrgnameValid} | ||||||
|                     class:focus:ring-red-500={!isOrgnameValid} |                     class:focus:ring-red-500={!isOrgnameValid} | ||||||
| @@ -147,11 +153,13 @@ | |||||||
|                     bind:this={name_input_dom} |                     bind:this={name_input_dom} | ||||||
|                     type="text" |                     type="text" | ||||||
|                     name="firstname" |                     name="firstname" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                   /> | ||||||
|                   {#if !isOrgnameValid} |                   {#if !isOrgnameValid} | ||||||
|                     <span |                     <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('organization-name-is-required')} |                     > | ||||||
|  |                       {$_("organization-name-is-required")} | ||||||
|                     </span> |                     </span> | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </div> | ||||||
| @@ -162,22 +170,25 @@ | |||||||
|                       id="comments" |                       id="comments" | ||||||
|                       name="comments" |                       name="comments" | ||||||
|                       type="checkbox" |                       type="checkbox" | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="ml-3 text-sm"> |                   <div class="ml-3 text-sm"> | ||||||
|                     <label |                     <label for="comments" class="font-semibold text-gray-700" | ||||||
|                       for="comments" |                       >{$_("address")}</label | ||||||
|                       class="font-medium text-gray-700">{$_('address')}</label> |                     > | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|                 {#if address_checked === true} |                 {#if address_checked === true} | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address1" |                       for="address1" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('address')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("address")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                     placeholder="{$_('address')}" |                       placeholder={$_("address")} | ||||||
|                       class:border-red-500={!isAddress1Valid} |                       class:border-red-500={!isAddress1Valid} | ||||||
|                       class:focus:border-red-500={!isAddress1Valid} |                       class:focus:border-red-500={!isAddress1Valid} | ||||||
|                       class:focus:ring-red-500={!isAddress1Valid} |                       class:focus:ring-red-500={!isAddress1Valid} | ||||||
| @@ -185,34 +196,41 @@ | |||||||
|                       bind:this={address_input1} |                       bind:this={address_input1} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address1" |                       name="address1" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !isAddress1Valid} |                     {#if !isAddress1Valid} | ||||||
|                       <span |                       <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('address-is-required')} |                       > | ||||||
|  |                         {$_("address-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="address2" |                       for="address2" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("apartment-suite-etc")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                     placeholder="{$_('apartment-suite-etc')}" |                       placeholder={$_("apartment-suite-etc")} | ||||||
|                       bind:value={address_input2_value} |                       bind:value={address_input2_value} | ||||||
|                       bind:this={address_input2} |                       bind:this={address_input2} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="address2" |                       name="address2" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="zipcode" |                       for="zipcode" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("zip-postal-code")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                     placeholder="{$_('zip-postal-code')}" |                       placeholder={$_("zip-postal-code")} | ||||||
|                       class:border-red-500={!iszipcodevalid} |                       class:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:border-red-500={!iszipcodevalid} |                       class:focus:border-red-500={!iszipcodevalid} | ||||||
|                       class:focus:ring-red-500={!iszipcodevalid} |                       class:focus:ring-red-500={!iszipcodevalid} | ||||||
| @@ -220,21 +238,25 @@ | |||||||
|                       bind:this={address_zipcode} |                       bind:this={address_zipcode} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="zipcode" |                       name="zipcode" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iszipcodevalid} |                     {#if !iszipcodevalid} | ||||||
|                       <span |                       <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('valid-zipcode-postal-code-is-required')} |                       > | ||||||
|  |                         {$_("valid-zipcode-postal-code-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="col-span-6"> |                   <div class="col-span-6"> | ||||||
|                     <label |                     <label | ||||||
|                       for="city" |                       for="city" | ||||||
|                     class="block text-sm font-medium text-gray-700">{$_('city')}</label> |                       class="block text-sm font-medium text-gray-700" | ||||||
|  |                       >{$_("city")}</label | ||||||
|  |                     > | ||||||
|                     <input |                     <input | ||||||
|                       autocomplete="off" |                       autocomplete="off" | ||||||
|                     placeholder="{$_('city')}" |                       placeholder={$_("city")} | ||||||
|                       class:border-red-500={!iscityvalid} |                       class:border-red-500={!iscityvalid} | ||||||
|                       class:focus:border-red-500={!iscityvalid} |                       class:focus:border-red-500={!iscityvalid} | ||||||
|                       class:focus:ring-red-500={!iscityvalid} |                       class:focus:ring-red-500={!iscityvalid} | ||||||
| @@ -242,11 +264,13 @@ | |||||||
|                       bind:this={address_city} |                       bind:this={address_city} | ||||||
|                       type="text" |                       type="text" | ||||||
|                       name="city" |                       name="city" | ||||||
|                     class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> |                       class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  |                     /> | ||||||
|                     {#if !iscityvalid} |                     {#if !iscityvalid} | ||||||
|                       <span |                       <span | ||||||
|                       class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> |                         class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|                       {$_('valid-city-is-required')} |                       > | ||||||
|  |                         {$_("valid-city-is-required")} | ||||||
|                       </span> |                       </span> | ||||||
|                     {/if} |                     {/if} | ||||||
|                   </div> |                   </div> | ||||||
| @@ -255,22 +279,24 @@ | |||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> |         <div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|           <button |           <button | ||||||
|             disabled={!createbtnenabled} |             disabled={!createbtnenabled} | ||||||
|             class:opacity-50={!createbtnenabled} |             class:opacity-50={!createbtnenabled} | ||||||
|             on:click={submit} |             on:click={submit} | ||||||
|             type="button" |             type="button" | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" | ||||||
|             {$_('create')} |           > | ||||||
|  |             {$_("create")} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|             on:click={() => { |             on:click={() => { | ||||||
|               modal_open = false; |               modal_open = false; | ||||||
|             }} |             }} | ||||||
|             type="button" |             type="button" | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |             class="w-full 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 hidden lg:block" | ||||||
|             {$_('cancel')} |           > | ||||||
|  |             {$_("cancel")} | ||||||
|           </button> |           </button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|   | |||||||
| @@ -1,102 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import { _ } from "svelte-i18n"; |  | ||||||
|   import { clickOutside } from "../base/outsideclick"; |  | ||||||
|   import { focusTrap } from "svelte-focus-trap"; |  | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |  | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import { createEventDispatcher } from "svelte"; |  | ||||||
|   export let modal_open; |  | ||||||
|   export let delete_org; |  | ||||||
|   const dispatch = createEventDispatcher(); |  | ||||||
|   function cancelDelete() { |  | ||||||
|     modal_open = false; |  | ||||||
|     dispatch("cancelDelete", { id: delete_org.id }); |  | ||||||
|   } |  | ||||||
|   function deleteOrg() { |  | ||||||
|     RunnerOrganizationService.runnerOrganizationControllerRemove( |  | ||||||
|       delete_org.id, |  | ||||||
|       true |  | ||||||
|     ) |  | ||||||
|       .then((resp) => { |  | ||||||
|         Toastify({ |  | ||||||
|           text: "Organization deleted", |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|         location.replace("./"); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => {}); |  | ||||||
|   } |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| {#if modal_open} |  | ||||||
|   <div |  | ||||||
|     class="fixed z-10 inset-0 overflow-y-auto" |  | ||||||
|     use:focusTrap |  | ||||||
|     use:clickOutside |  | ||||||
|     on:click_outside={cancelDelete}> |  | ||||||
|     <div |  | ||||||
|       class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0"> |  | ||||||
|       <div class="fixed inset-0 transition-opacity" aria-hidden="true"> |  | ||||||
|         <div |  | ||||||
|           class="absolute inset-0 bg-gray-500 opacity-75" |  | ||||||
|           data-id="modal_backdrop" /> |  | ||||||
|       </div> |  | ||||||
|       <span |  | ||||||
|         class="hidden sm:inline-block sm:align-middle sm:h-screen" |  | ||||||
|         aria-hidden="true">​</span> |  | ||||||
|       <div |  | ||||||
|         class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full" |  | ||||||
|         role="dialog" |  | ||||||
|         aria-modal="true" |  | ||||||
|         aria-labelledby="modal-headline"> |  | ||||||
|         <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4"> |  | ||||||
|           <div class="sm:flex sm:items-start"> |  | ||||||
|             <div |  | ||||||
|               class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"> |  | ||||||
|               <svg |  | ||||||
|                 class="h-6 w-6 text-blue-600" |  | ||||||
|                 fill="currentColor" |  | ||||||
|                 width="24" |  | ||||||
|                 height="24" |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                 viewBox="0 0 640 512"><path |  | ||||||
|                   fill="currentColor" |  | ||||||
|                   d="M96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm448 0c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm32 32h-64c-17.6 0-33.5 7.1-45.1 18.6 40.3 22.1 68.9 62 75.1 109.4h66c17.7 0 32-14.3 32-32v-32c0-35.3-28.7-64-64-64zm-256 0c61.9 0 112-50.1 112-112S381.9 32 320 32 208 82.1 208 144s50.1 112 112 112zm76.8 32h-8.3c-20.8 10-43.9 16-68.5 16s-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h288c26.5 0 48-21.5 48-48v-28.8c0-63.6-51.6-115.2-115.2-115.2zm-223.7-13.4C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.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"> |  | ||||||
|                 {$_('attention')} |  | ||||||
|               </h3> |  | ||||||
|               <div class="mt-2 mb-6"> |  | ||||||
|                 <p class="text-sm text-gray-500"> |  | ||||||
|                   {$_( |  | ||||||
|                     'do-you-want-to-delete-the-organization-delete_org-name', |  | ||||||
|                     { |  | ||||||
|                       values: { orgname: delete_org.name }, |  | ||||||
|                     } |  | ||||||
|                   )}<br /> |  | ||||||
|                   {$_('all-associated-teams-and-runners-will-be-deleted-too')} |  | ||||||
|                 </p> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|         </div> |  | ||||||
|         <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse"> |  | ||||||
|           <button |  | ||||||
|             on:click={deleteOrg} |  | ||||||
|             type="button" |  | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm"> |  | ||||||
|             {$_('confirm-delete-organization-and-associated-teams-runners')} |  | ||||||
|           </button> |  | ||||||
|           <button |  | ||||||
|             on:click={cancelDelete} |  | ||||||
|             type="button" |  | ||||||
|             class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm"> |  | ||||||
|             {$_('cancel-keep-organization')} |  | ||||||
|           </button> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| {/if} |  | ||||||
							
								
								
									
										104
									
								
								src/components/orgs/ConfirmOrgDeletionModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/components/orgs/ConfirmOrgDeletionModal.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | |||||||
|  | <script> | ||||||
|  | 	import { _ } from "svelte-i18n"; | ||||||
|  | 	import { clickOutside } from "../base/outsideclick"; | ||||||
|  |  | ||||||
|  | 	import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
|  |  | ||||||
|  | 	import { createEventDispatcher } from "svelte"; | ||||||
|  | 	export let modal_open; | ||||||
|  | 	export let delete_org; | ||||||
|  | 	const dispatch = createEventDispatcher(); | ||||||
|  | 	function cancelDelete() { | ||||||
|  | 		modal_open = false; | ||||||
|  | 		dispatch("cancelDelete", { id: delete_org.id }); | ||||||
|  | 	} | ||||||
|  | 	function deleteOrg() { | ||||||
|  | 		RunnerOrganizationService.runnerOrganizationControllerRemove( | ||||||
|  | 			delete_org.id, | ||||||
|  | 			true | ||||||
|  | 		) | ||||||
|  | 			.then((resp) => { | ||||||
|  | 				toast.success($_("organization-deleted")); | ||||||
|  | 				location.replace("./"); | ||||||
|  | 			}) | ||||||
|  | 			.catch((err) => {}); | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if modal_open} | ||||||
|  | 	<div | ||||||
|  | 		class="fixed z-10 inset-0 overflow-y-hidden" | ||||||
|  | 		use:clickOutside | ||||||
|  | 		on:click_outside={cancelDelete} | ||||||
|  | 	> | ||||||
|  | 		<div | ||||||
|  | 			class="flex items-end justify-center h-screen text-center sm:block p-0 lg:p-4" | ||||||
|  | 		> | ||||||
|  | 			<div class="fixed inset-0 transition-opacity" aria-hidden="true"> | ||||||
|  | 				<div | ||||||
|  | 					class="absolute inset-0 bg-gray-500 opacity-75" | ||||||
|  | 					data-id="modal_backdrop" | ||||||
|  | 				/> | ||||||
|  | 			</div> | ||||||
|  | 			<span | ||||||
|  | 				class="hidden sm:inline-block sm:align-middle sm:h-screen" | ||||||
|  | 				aria-hidden="true">​</span | ||||||
|  | 			> | ||||||
|  | 			<div | ||||||
|  | 				class="inline-block align-bottom text-left shadow-xl transform transition-all sm:align-middle w-full lg:w-auto min-w-auto lg:min-w-[35vw]" | ||||||
|  | 				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 rounded-t-xl"> | ||||||
|  | 					<div class=""> | ||||||
|  | 						<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" | ||||||
|  | 								width="24" | ||||||
|  | 								height="24" | ||||||
|  | 								xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 								viewBox="0 0 640 512" | ||||||
|  | 								><path | ||||||
|  | 									fill="currentColor" | ||||||
|  | 									d="M96 224c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm448 0c35.3 0 64-28.7 64-64s-28.7-64-64-64-64 28.7-64 64 28.7 64 64 64zm32 32h-64c-17.6 0-33.5 7.1-45.1 18.6 40.3 22.1 68.9 62 75.1 109.4h66c17.7 0 32-14.3 32-32v-32c0-35.3-28.7-64-64-64zm-256 0c61.9 0 112-50.1 112-112S381.9 32 320 32 208 82.1 208 144s50.1 112 112 112zm76.8 32h-8.3c-20.8 10-43.9 16-68.5 16s-47.6-6-68.5-16h-8.3C179.6 288 128 339.6 128 403.2V432c0 26.5 21.5 48 48 48h288c26.5 0 48-21.5 48-48v-28.8c0-63.6-51.6-115.2-115.2-115.2zm-223.7-13.4C161.5 263.1 145.6 256 128 256H64c-35.3 0-64 28.7-64 64v32c0 17.7 14.3 32 32 32h65.9c6.3-47.4 34.9-87.3 75.2-109.4z" | ||||||
|  | 								/></svg | ||||||
|  | 							> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="mt-3 text-center sm:text-left max-h-[75vh] overflow-y-auto"> | ||||||
|  | 							<h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||||
|  | 								{$_("do-you-want-to-delete-the-organization-delete_org-name", { | ||||||
|  | 									values: { orgname: delete_org.name }, | ||||||
|  | 								})} | ||||||
|  | 							</h3> | ||||||
|  | 							<div class="mt-2 mb-6"> | ||||||
|  | 								<p class="text-sm text-gray-500"> | ||||||
|  | 									{$_("all-associated-teams-and-runners-will-be-deleted-too")} | ||||||
|  | 								</p> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="bg-gray-50 px-4 py-3 sm:px-6 grid gap-2 lg:rounded-b-xl"> | ||||||
|  | 					<button | ||||||
|  | 						on:click={deleteOrg} | ||||||
|  | 						type="button" | ||||||
|  | 						class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500" | ||||||
|  | 					> | ||||||
|  | 						{$_("confirm-delete-organization-and-associated-teams-runners")} | ||||||
|  | 					</button> | ||||||
|  | 					<button | ||||||
|  | 						on:click={cancelDelete} | ||||||
|  | 						type="button" | ||||||
|  | 						class="w-full 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 hidden lg:block" | ||||||
|  | 					> | ||||||
|  | 						{$_("cancel-keep-organization")} | ||||||
|  | 					</button> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | {/if} | ||||||
| @@ -3,14 +3,17 @@ | |||||||
| 		GroupContactService, | 		GroupContactService, | ||||||
| 		RunnerOrganizationService, | 		RunnerOrganizationService, | ||||||
| 	} from "@odit/lfk-client-js"; | 	} from "@odit/lfk-client-js"; | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; | 	import toast from "svelte-french-toast"; | ||||||
|   import Toastify from "toastify-js"; | 	import { _ } from "svelte-i18n"; | ||||||
|   import store from "../../store"; | 	import { tick } from "svelte"; | ||||||
|   import ConfirmOrgDeletion from "./ConfirmOrgDeletion.svelte"; |  | ||||||
|   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; |  | ||||||
|   import PromiseError from "../base/PromiseError.svelte"; |  | ||||||
| 	import Select from "svelte-select"; | 	import Select from "svelte-select"; | ||||||
|   $: delete_triggered = false; | 	import store from "../../store"; | ||||||
|  | 	import PromiseError from "../base/PromiseError.svelte"; | ||||||
|  | 	import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|  | 	import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|  | 	import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  | 	import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; | ||||||
|  | 	import ConfirmOrgDeletionModal from "./ConfirmOrgDeletionModal.svelte"; | ||||||
| 	$: address_valid_or_none = | 	$: address_valid_or_none = | ||||||
| 		(isAddress1Valid && iszipcodevalid && iscityvalid) || | 		(isAddress1Valid && iszipcodevalid && iscityvalid) || | ||||||
| 		editable.address_checked === false; | 		editable.address_checked === false; | ||||||
| @@ -18,6 +21,8 @@ | |||||||
| 	let original = ""; | 	let original = ""; | ||||||
| 	let original_object = {}; | 	let original_object = {}; | ||||||
| 	let contacts = []; | 	let contacts = []; | ||||||
|  | 	let valueCopy = null; | ||||||
|  | 	let areaDom; | ||||||
| 	export let params; | 	export let params; | ||||||
| 	$: editable = {}; | 	$: editable = {}; | ||||||
| 	$: contact = {}; | 	$: contact = {}; | ||||||
| @@ -26,7 +31,11 @@ | |||||||
| 	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0; | 	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0; | ||||||
| 	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; | 	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0; | ||||||
| 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | 	$: iscityvalid = editable.address?.city?.trim().length !== 0; | ||||||
|   $: sponsoring_contracts_download_open = false; | 	$: sponsoring_contracts_show = true; | ||||||
|  | 	$: cards_show = true; | ||||||
|  | 	$: certificates_show = true; | ||||||
|  | 	$: generate_orgs = [original_object]; | ||||||
|  | 	$: registrationLink = `${config.baseurl_selfservice}/register/${editable.registrationKey}`; | ||||||
| 	const getContactLabel = (option) => | 	const getContactLabel = (option) => | ||||||
| 		option.firstname + " " + (option.middlename || "") + " " + option.lastname; | 		option.firstname + " " + (option.middlename || "") + " " + option.lastname; | ||||||
| 	const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne( | 	const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne( | ||||||
| @@ -60,38 +69,20 @@ | |||||||
| 	}); | 	}); | ||||||
| 	let modal_open = false; | 	let modal_open = false; | ||||||
| 	let delete_org = {}; | 	let delete_org = {}; | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|     if ( |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|     ) { |  | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| 	function deleteOrganization() { | 	function deleteOrganization() { | ||||||
| 		RunnerOrganizationService.runnerOrganizationControllerRemove( | 		RunnerOrganizationService.runnerOrganizationControllerRemove( | ||||||
| 			original_object.id, | 			original_object.id, | ||||||
| 			false | 			false | ||||||
| 		) | 		) | ||||||
| 			.then((resp) => { | 			.then((resp) => { | ||||||
|         Toastify({ | 				toast.success($_("organization-deleted")); | ||||||
|           text: $_("organization-deleted"), |  | ||||||
|           duration: 500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
| 				location.replace("./"); | 				location.replace("./"); | ||||||
| 			}) | 			}) | ||||||
|       .catch((err) => { | 			.catch((err) => {}); | ||||||
|         modal_open = true; |  | ||||||
|         delete_org = original_object; |  | ||||||
|       }); |  | ||||||
| 	} | 	} | ||||||
| 	function submit() { | 	function submit() { | ||||||
| 		if (data_loaded === true && save_enabled) { | 		if (data_loaded === true && save_enabled) { | ||||||
|       Toastify({ | 			toast($_("updating-organization")); | ||||||
|         text: $_("updating-organization"), |  | ||||||
|         duration: 2500, |  | ||||||
|       }).showToast(); |  | ||||||
| 			let postdata = Object.assign({}, editable); | 			let postdata = Object.assign({}, editable); | ||||||
| 			if (postdata.address_checked === false) { | 			if (postdata.address_checked === false) { | ||||||
| 				postdata.address = null; | 				postdata.address = null; | ||||||
| @@ -102,70 +93,40 @@ | |||||||
| 				postdata | 				postdata | ||||||
| 			) | 			) | ||||||
| 				.then((resp) => { | 				.then((resp) => { | ||||||
|  | 					editable.registrationKey = resp.registrationKey; | ||||||
| 					original_object = Object.assign({}, editable); | 					original_object = Object.assign({}, editable); | ||||||
| 					original = JSON.stringify(original_object); | 					original = JSON.stringify(original_object); | ||||||
|           Toastify({ | 					toast.success($_("updated-organization")); | ||||||
|             text: $_("updated-organization"), |  | ||||||
|             duration: 2500, |  | ||||||
|             backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|           }).showToast(); |  | ||||||
| 				}) | 				}) | ||||||
| 				.catch((err) => {}); | 				.catch((err) => {}); | ||||||
| 		} else { | 		} else { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	async function copy() { | ||||||
|  | 		if (!editable.registrationKey) { | ||||||
|  | 			toast.error($_("you-have-to-save-your-changes-to-generate-a-link")); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 		valueCopy = registrationLink; | ||||||
|  | 		await tick(); | ||||||
|  | 		areaDom.focus(); | ||||||
|  | 		areaDom.select(); | ||||||
|  | 		try { | ||||||
|  | 			const successful = document.execCommand("copy"); | ||||||
|  | 			if (!successful) { | ||||||
|  | 				throw new Error(); | ||||||
|  | 			} | ||||||
|  | 			toast($_("copied-link-to-clipboard")); | ||||||
|  | 		} catch (err) { | ||||||
|  | 			toast.error($_("error-whyile-copying-to-clipboard")); | ||||||
|  | 		} | ||||||
|  | 		// we can notifi by event or storage about copy status | ||||||
|  | 		valueCopy = null; | ||||||
|  | 	} | ||||||
| 	export let import_modal_open = false; | 	export let import_modal_open = false; | ||||||
|   async function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |  | ||||||
|       original_object.id |  | ||||||
|     ); |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdf"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     fetch( |  | ||||||
|       `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|       { |  | ||||||
|         method: "POST", |  | ||||||
|         headers: { |  | ||||||
|           "Content-Type": "application/json", |  | ||||||
|         }, |  | ||||||
|         body: JSON.stringify(runners), |  | ||||||
|       } |  | ||||||
|     ) |  | ||||||
|       .then((response) => { |  | ||||||
|         if (response.status != "200") { |  | ||||||
|           toast.hideToast(); |  | ||||||
|           Toastify({ |  | ||||||
|             text: $_("pdf-generation-failed"), |  | ||||||
|             duration: 3500, |  | ||||||
|             backgroundColor: |  | ||||||
|               "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|           }).showToast(); |  | ||||||
|         } else { |  | ||||||
|           return response.blob(); |  | ||||||
|         } |  | ||||||
|       }) |  | ||||||
|       .then((blob) => { |  | ||||||
|         const url = window.URL.createObjectURL(blob); |  | ||||||
|         let a = document.createElement("a"); |  | ||||||
|         a.href = url; |  | ||||||
|         a.download = "Sponsorings_" + original_object.name + ".pdf"; |  | ||||||
|         document.body.appendChild(a); |  | ||||||
|         a.click(); |  | ||||||
|         a.remove(); |  | ||||||
|         toast.hideToast(); |  | ||||||
|         Toastify({ |  | ||||||
|           text: $_("pdf-successfully-generated"), |  | ||||||
|           duration: 3500, |  | ||||||
|           backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|         }).showToast(); |  | ||||||
|       }) |  | ||||||
|       .catch((err) => {}); |  | ||||||
|   } |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | {#if valueCopy != null}<textarea bind:this={areaDom}>{valueCopy}</textarea>{/if} | ||||||
| <ImportRunnerModal | <ImportRunnerModal | ||||||
| 	on:cancelDelete={(event) => { | 	on:cancelDelete={(event) => { | ||||||
| 		import_modal_open = false; | 		import_modal_open = false; | ||||||
| @@ -175,234 +136,191 @@ | |||||||
| 	passed_orgs={[]} | 	passed_orgs={[]} | ||||||
| 	passed_org={editable} | 	passed_org={editable} | ||||||
| 	opened_from="OrgDetail" | 	opened_from="OrgDetail" | ||||||
|   bind:import_modal_open /> | 	bind:import_modal_open | ||||||
| <ConfirmOrgDeletion bind:modal_open bind:delete_org /> | /> | ||||||
|  | <ConfirmOrgDeletionModal bind:modal_open bind:delete_org /> | ||||||
| {#if data_loaded} | {#if data_loaded} | ||||||
| 	<section class="container p-5"> | 	<section class="container p-5"> | ||||||
|     <div class="mb-8 text-3xl font-extrabold leading-tight"> |  | ||||||
|       {original_object.name} |  | ||||||
|       <span data-id="org_actions_${editable.id}"> |  | ||||||
|         <div id="sponsoring:dropdown" class="relative inline-block"> |  | ||||||
|           <div> |  | ||||||
|             <button |  | ||||||
|               on:click={() => { |  | ||||||
|                 sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |  | ||||||
|               }} |  | ||||||
|               type="button" |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |  | ||||||
|               id="options-menu" |  | ||||||
|               aria-haspopup="true" |  | ||||||
|               aria-expanded="true"> |  | ||||||
|               {$_('generate-sponsoring-contracts')} |  | ||||||
|               <svg |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg" |  | ||||||
|                 width="24" |  | ||||||
|                 height="24" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 class="-mr-1 ml-2 h-5 w-5"><path |  | ||||||
|                   fill="none" |  | ||||||
|                   d="M0 0h24v24H0z" /> |  | ||||||
|                 <path |  | ||||||
|                   fill="currentColor" |  | ||||||
|                   d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg> |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|           {#if sponsoring_contracts_download_open} |  | ||||||
|             <div |  | ||||||
|               class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|               id="sponsoring:dropdown:menu"> |  | ||||||
|               <div |  | ||||||
|                 class="py-1" |  | ||||||
|                 role="menu" |  | ||||||
|                 aria-orientation="vertical" |  | ||||||
|                 aria-labelledby="options-menu"> |  | ||||||
|                 <span |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('de'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('german')} |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('en'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('english')} |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           {/if} |  | ||||||
|         </div> |  | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} |  | ||||||
|           <button |  | ||||||
|             on:click={() => { |  | ||||||
|               import_modal_open = true; |  | ||||||
|             }} |  | ||||||
|             type="button" |  | ||||||
|             class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> |  | ||||||
|             {$_('import-runners')} |  | ||||||
|           </button> |  | ||||||
|         {/if} |  | ||||||
|         {#if store.state.jwtinfo.userdetails.permissions.includes('USER:DELETE')} |  | ||||||
|           {#if delete_triggered} |  | ||||||
|             <button |  | ||||||
|               on:click={deleteOrganization} |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('confirm-delete')}</button> |  | ||||||
|             <button |  | ||||||
|               on:click={() => { |  | ||||||
|                 delete_triggered = !delete_triggered; |  | ||||||
|               }} |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm">{$_('cancel')}</button> |  | ||||||
|           {/if} |  | ||||||
|           {#if !delete_triggered} |  | ||||||
|             <button |  | ||||||
|               on:click={() => { |  | ||||||
|                 delete_triggered = true; |  | ||||||
|               }} |  | ||||||
|               type="button" |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('delete-organization')}</button> |  | ||||||
|           {/if} |  | ||||||
|         {/if} |  | ||||||
|         {#if !delete_triggered} |  | ||||||
|           <button |  | ||||||
|             on:click={submit} |  | ||||||
|             disabled={!save_enabled} |  | ||||||
|             class:opacity-50={!save_enabled} |  | ||||||
|             type="button" |  | ||||||
|             class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">{$_('save-changes')}</button> |  | ||||||
|         {/if} |  | ||||||
|       </span> |  | ||||||
|     </div> |  | ||||||
| 		<div class="flex flex-row mb-4"> | 		<div class="flex flex-row mb-4"> | ||||||
| 			<div class="w-full"> | 			<div class="w-full"> | ||||||
| 				<nav class="w-full flex"> | 				<nav class="w-full flex"> | ||||||
| 					<ol class="list-none flex flex-row items-center justify-start"> | 					<ol class="list-none flex flex-row items-center justify-start"> | ||||||
|             <li class="mr-2 flex items-center"> |  | ||||||
|               <svg |  | ||||||
|                 stroke="currentColor" |  | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 stroke-linecap="round" |  | ||||||
|                 stroke-linejoin="round" |  | ||||||
|                 class="h-3 w-3 stroke-current" |  | ||||||
|                 height="1em" |  | ||||||
|                 width="1em" |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><path |  | ||||||
|                   d="M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z" /> |  | ||||||
|                 <polyline points="9 22 9 12 15 12 15 22" /></svg> |  | ||||||
|             </li> |  | ||||||
| 						<li class="flex items-center"> | 						<li class="flex items-center"> | ||||||
|               <a class="mr-2" href="/">{$_('home')}</a><svg | 							<a class="mr-2" href="./" | ||||||
|                 stroke="currentColor" | 								><svg | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
|                 stroke-linecap="round" |  | ||||||
|                 stroke-linejoin="round" |  | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" |  | ||||||
|                 height="1em" |  | ||||||
|                 width="1em" |  | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line |  | ||||||
|                   x1="5" |  | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="mr-2 flex items-center"> |  | ||||||
|               <svg |  | ||||||
| 									xmlns="http://www.w3.org/2000/svg" | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|                 viewBox="0 0 24 24" |  | ||||||
| 									width="24" | 									width="24" | ||||||
|                 height="24"><path fill="none" d="M0 0h24v24H0z" /> | 									height="24" | ||||||
|                 <path |  | ||||||
|                   d="M21 20h2v2H1v-2h2V3a1 1 0 0 1 1-1h16a1 1 0 0 1 1 1v17zm-2 0V4H5v16h14zM8 11h3v2H8v-2zm0-4h3v2H8V7zm0 8h3v2H8v-2zm5 0h3v2h-3v-2zm0-4h3v2h-3v-2zm0-4h3v2h-3V7z" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <a class="mr-2" href="./">{$_('organizations')}</a><svg |  | ||||||
|                 stroke="currentColor" |  | ||||||
|                 fill="none" |  | ||||||
|                 stroke-width="2" |  | ||||||
| 									viewBox="0 0 24 24" | 									viewBox="0 0 24 24" | ||||||
|  | 									fill="none" | ||||||
|  | 									stroke="currentColor" | ||||||
|  | 									stroke-width="2" | ||||||
| 									stroke-linecap="round" | 									stroke-linecap="round" | ||||||
| 									stroke-linejoin="round" | 									stroke-linejoin="round" | ||||||
|                 class="h-3 w-3 mr-2 stroke-current" | 									class="inline-block" | ||||||
|                 height="1em" | 									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg | ||||||
|                 width="1em" | 								> | ||||||
|                 xmlns="http://www.w3.org/2000/svg"><line | 								{$_("organizations")}</a | ||||||
|                   x1="5" | 							> | ||||||
|                   y1="12" |  | ||||||
|                   x2="19" |  | ||||||
|                   y2="12" /> |  | ||||||
|                 <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|             </li> |  | ||||||
|             <li class="flex items-center"> |  | ||||||
|               <span class="mr-2">Org-Details #{params.orgid}</span> |  | ||||||
| 						</li> | 						</li> | ||||||
| 					</ol> | 					</ol> | ||||||
| 				</nav> | 				</nav> | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="mb-4 text-3xl font-extrabold leading-tight"> | ||||||
|       <label for="name" class="font-medium text-gray-700">{$_('name')}</label> | 			{original_object.name} [#{params.orgid}] | ||||||
|  | 			<div data-id="org_actions_${editable.id}"> | ||||||
|  | 				<GenerateSponsoringContracts | ||||||
|  | 					bind:sponsoring_contracts_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")} | ||||||
|  | 					<button | ||||||
|  | 						on:click={() => { | ||||||
|  | 							import_modal_open = true; | ||||||
|  | 						}} | ||||||
|  | 						type="button" | ||||||
|  | 						class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm" | ||||||
|  | 					> | ||||||
|  | 						{$_("import-runners")} | ||||||
|  | 					</button> | ||||||
|  | 				{/if} | ||||||
|  | 				{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:DELETE")} | ||||||
|  | 					<button | ||||||
|  | 						on:click={() => { | ||||||
|  | 							modal_open = true; | ||||||
|  | 							delete_org = original_object; | ||||||
|  | 						}} | ||||||
|  | 						type="button" | ||||||
|  | 						class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm" | ||||||
|  | 						>{$_("delete-organization")}</button | ||||||
|  | 					> | ||||||
|  | 				{/if} | ||||||
|  | 				<button | ||||||
|  | 					on:click={submit} | ||||||
|  | 					disabled={!save_enabled} | ||||||
|  | 					class:opacity-50={!save_enabled} | ||||||
|  | 					type="button" | ||||||
|  | 					class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 					>{$_("save-changes")}</button | ||||||
|  | 				> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="text-sm w-full mt-2"> | ||||||
|  | 			<label for="name" class="font-semibold text-gray-700">{$_("name")}</label> | ||||||
| 			<input | 			<input | ||||||
| 				autocomplete="off" | 				autocomplete="off" | ||||||
|         placeholder={$_('name')} | 				placeholder={$_("name")} | ||||||
| 				type="text" | 				type="text" | ||||||
| 				bind:value={editable.name} | 				bind:value={editable.name} | ||||||
| 				name="name" | 				name="name" | ||||||
|         class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 				class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|     <div class="text-sm w-full"> | 		<div class="text-sm w-full mt-2"> | ||||||
|       <label | 			<label for="contact" class="font-semibold text-gray-700" | ||||||
|         for="contact" | 				>{$_("contact")}</label | ||||||
|         class="font-medium text-gray-700">{$_('contact')}</label> | 			> | ||||||
| 			<Select | 			<Select | ||||||
|         containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" | 				containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|         itemFilter={(label, filterText, option) => label | 				itemFilter={(label, filterText, option) => | ||||||
|             .toLowerCase() | 					label.toLowerCase().includes(filterText.toLowerCase()) || | ||||||
|             .includes( | 					option.value.id.toString().startsWith(filterText.toLowerCase())} | ||||||
|               filterText.toLowerCase() |  | ||||||
|             ) || option.value.id |  | ||||||
|             .toString() |  | ||||||
|             .startsWith(filterText.toLowerCase())} |  | ||||||
| 				items={contacts} | 				items={contacts} | ||||||
| 				showChevron={true} | 				showChevron={true} | ||||||
|         placeholder={$_('no-contact-selected')} | 				placeholder={$_("no-contact-selected")} | ||||||
|         noOptionsMessage={$_('no-contact-found')} | 				noOptionsMessage={$_("no-contact-found")} | ||||||
| 				bind:selectedValue={contact} | 				bind:selectedValue={contact} | ||||||
|         on:select={(selectedValue) => (editable.contact = selectedValue.detail.value)} | 				on:select={(selectedValue) => | ||||||
|         on:clear={() => (editable.contact = null)} /> | 					(editable.contact = selectedValue.detail.value)} | ||||||
|  | 				on:clear={() => (editable.contact = null)} | ||||||
|  | 			/> | ||||||
| 		</div> | 		</div> | ||||||
|  | 		<div> | ||||||
|  | 			<div class="flex items-start mt-2"> | ||||||
|  | 				<div class="flex items-center h-5"> | ||||||
|  | 					<input | ||||||
|  | 						bind:checked={editable.registrationEnabled} | ||||||
|  | 						id="toggle_selfservice_feature" | ||||||
|  | 						name="toggle_selfservice_feature" | ||||||
|  | 						type="checkbox" | ||||||
|  | 						class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  | 					/> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="ml-3 text-sm"> | ||||||
|  | 					<label | ||||||
|  | 						for="toggle_selfservice_feature" | ||||||
|  | 						class="font-semibold text-gray-700" | ||||||
|  | 						>{$_("selfservice-registration")}</label | ||||||
|  | 					> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div> | ||||||
|  | 				{#if editable.registrationEnabled} | ||||||
|  | 					<div class="text-sm w-full mt-2"> | ||||||
|  | 						<button on:click={copy} class="inline-flex w-full"> | ||||||
|  | 							<p | ||||||
|  | 								name="token" | ||||||
|  | 								class="focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 p-2 break-all font-mono text-left" | ||||||
|  | 							> | ||||||
|  | 								{#if editable.registrationKey} | ||||||
|  | 									{registrationLink} | ||||||
|  | 								{:else} | ||||||
|  | 									{$_("you-have-to-save-your-changes-to-generate-a-link")} | ||||||
|  | 								{/if} | ||||||
|  | 							</p> | ||||||
|  | 							<div | ||||||
|  | 								class="bg-gray-200 border-gray-300 border-t border-b border-r text-black rounded-r-md sm:text-sm p-2 cursor-pointer flex items-center justify-center" | ||||||
|  | 							> | ||||||
|  | 								<svg | ||||||
|  | 									xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 									viewBox="0 0 24 24" | ||||||
|  | 									width="24" | ||||||
|  | 									height="24" | ||||||
|  | 									><path fill="none" d="M0 0h24v24H0z" /> | ||||||
|  | 									<path | ||||||
|  | 										fill="currentColor" | ||||||
|  | 										d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" | ||||||
|  | 									/></svg | ||||||
|  | 								> | ||||||
|  | 							</div> | ||||||
|  | 						</button> | ||||||
|  | 						{#if editable.registrationKey} | ||||||
|  | 							<p class="text-gray-500 text-xs"> | ||||||
|  | 								{$_("click-to-copy-the-link-into-your-clipboard")} | ||||||
|  | 							</p> | ||||||
|  | 						{/if} | ||||||
|  | 					</div> | ||||||
|  | 				{/if} | ||||||
| 				<!--  --> | 				<!--  --> | ||||||
|  | 				<div> | ||||||
| 					<div class="flex items-start mt-2"> | 					<div class="flex items-start mt-2"> | ||||||
| 						<div class="flex items-center h-5"> | 						<div class="flex items-center h-5"> | ||||||
| 							<input | 							<input | ||||||
| 								bind:checked={editable.address_checked} | 								bind:checked={editable.address_checked} | ||||||
|           id="comments" | 								id="toggle_address_checkbox" | ||||||
|           name="comments" | 								name="toggle_address_checkbox" | ||||||
| 								type="checkbox" | 								type="checkbox" | ||||||
|           class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> | 								class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  | 							/> | ||||||
| 						</div> | 						</div> | ||||||
| 						<div class="ml-3 text-sm"> | 						<div class="ml-3 text-sm"> | ||||||
| 							<label | 							<label | ||||||
|           for="comments" | 								for="toggle_address_checkbox" | ||||||
|           class="font-medium text-gray-700">{$_('address')}</label> | 								class="font-semibold text-gray-700">{$_("address")}</label | ||||||
|  | 							> | ||||||
|  | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
| 				</div> | 				</div> | ||||||
| 				{#if editable.address_checked === true} | 				{#if editable.address_checked === true} | ||||||
| 					<div class="col-span-6"> | 					<div class="col-span-6"> | ||||||
| 						<label | 						<label | ||||||
| 							for="address1" | 							for="address1" | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('address')}</label> | 							class="block text-sm font-medium text-gray-700" | ||||||
|  | 							>{$_("address")}</label | ||||||
|  | 						> | ||||||
| 						<input | 						<input | ||||||
| 							autocomplete="off" | 							autocomplete="off" | ||||||
| 							placeholder="Address" | 							placeholder="Address" | ||||||
| @@ -412,73 +330,91 @@ | |||||||
| 							bind:value={editable.address.address1} | 							bind:value={editable.address.address1} | ||||||
| 							type="text" | 							type="text" | ||||||
| 							name="address1" | 							name="address1" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 						/> | ||||||
| 						{#if !isAddress1Valid} | 						{#if !isAddress1Valid} | ||||||
| 							<span | 							<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('address-is-required')} | 							> | ||||||
|  | 								{$_("address-is-required")} | ||||||
| 							</span> | 							</span> | ||||||
| 						{/if} | 						{/if} | ||||||
| 					</div> | 					</div> | ||||||
| 					<div class="col-span-6"> | 					<div class="col-span-6"> | ||||||
| 						<label | 						<label | ||||||
| 							for="address2" | 							for="address2" | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label> | 							class="block text-sm font-medium text-gray-700" | ||||||
|  | 							>{$_("apartment-suite-etc")}</label | ||||||
|  | 						> | ||||||
| 						<input | 						<input | ||||||
| 							autocomplete="off" | 							autocomplete="off" | ||||||
|           placeholder={$_('apartment-suite-etc')} | 							placeholder={$_("apartment-suite-etc")} | ||||||
| 							bind:value={editable.address.address2} | 							bind:value={editable.address.address2} | ||||||
| 							type="text" | 							type="text" | ||||||
| 							name="address2" | 							name="address2" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 						/> | ||||||
| 					</div> | 					</div> | ||||||
| 					<div class="col-span-6"> | 					<div class="col-span-6"> | ||||||
|         <label | 						<label for="zipcode" class="block text-sm font-medium text-gray-700" | ||||||
|           for="zipcode" | 							>{$_("zip-postal-code")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label> | 						> | ||||||
| 						<input | 						<input | ||||||
| 							autocomplete="off" | 							autocomplete="off" | ||||||
|           placeholder={$_('zip-postal-code')} | 							placeholder={$_("zip-postal-code")} | ||||||
| 							class:border-red-500={!iszipcodevalid} | 							class:border-red-500={!iszipcodevalid} | ||||||
| 							class:focus:border-red-500={!iszipcodevalid} | 							class:focus:border-red-500={!iszipcodevalid} | ||||||
| 							class:focus:ring-red-500={!iszipcodevalid} | 							class:focus:ring-red-500={!iszipcodevalid} | ||||||
| 							bind:value={editable.address.postalcode} | 							bind:value={editable.address.postalcode} | ||||||
| 							type="text" | 							type="text" | ||||||
| 							name="zipcode" | 							name="zipcode" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 						/> | ||||||
| 						{#if !iszipcodevalid} | 						{#if !iszipcodevalid} | ||||||
| 							<span | 							<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-zipcode-postal-code-is-required')} | 							> | ||||||
|  | 								{$_("valid-zipcode-postal-code-is-required")} | ||||||
| 							</span> | 							</span> | ||||||
| 						{/if} | 						{/if} | ||||||
| 					</div> | 					</div> | ||||||
| 					<div class="col-span-6"> | 					<div class="col-span-6"> | ||||||
|         <label | 						<label for="city" class="block text-sm font-medium text-gray-700" | ||||||
|           for="city" | 							>{$_("city")}</label | ||||||
|           class="block text-sm font-medium text-gray-700">{$_('city')}</label> | 						> | ||||||
| 						<input | 						<input | ||||||
| 							autocomplete="off" | 							autocomplete="off" | ||||||
|           placeholder={$_('city')} | 							placeholder={$_("city")} | ||||||
| 							class:border-red-500={!iscityvalid} | 							class:border-red-500={!iscityvalid} | ||||||
| 							class:focus:border-red-500={!iscityvalid} | 							class:focus:border-red-500={!iscityvalid} | ||||||
| 							class:focus:ring-red-500={!iscityvalid} | 							class:focus:ring-red-500={!iscityvalid} | ||||||
| 							bind:value={editable.address.city} | 							bind:value={editable.address.city} | ||||||
| 							type="text" | 							type="text" | ||||||
| 							name="city" | 							name="city" | ||||||
|           class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" /> | 							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||||
|  | 						/> | ||||||
| 						{#if !iscityvalid} | 						{#if !iscityvalid} | ||||||
| 							<span | 							<span | ||||||
|             class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"> | 								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1" | ||||||
|             {$_('valid-city-is-required')} | 							> | ||||||
|  | 								{$_("valid-city-is-required")} | ||||||
| 							</span> | 							</span> | ||||||
| 						{/if} | 						{/if} | ||||||
| 					</div> | 					</div> | ||||||
| 				{/if} | 				{/if} | ||||||
|  | 				<div class="text-sm w-full mt-2"> | ||||||
|  | 					<span class="font-semibold text-gray-700">{$_("distance")}</span> | ||||||
|  | 					<br /> | ||||||
|  | 					<span class="text-gray-700" | ||||||
|  | 						>{(original_object.total_distance / 1000).toFixed(2)} km</span | ||||||
|  | 					> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
| 	</section> | 	</section> | ||||||
| {:else} | {:else} | ||||||
| 	{#await promise} | 	{#await promise} | ||||||
|     {$_('organization-detail-is-being-loaded')} | 		{$_("organization-detail-is-being-loaded")} | ||||||
| 	{:catch error} | 	{:catch error} | ||||||
| 		<PromiseError /> | 		<PromiseError /> | ||||||
| 	{/await} | 	{/await} | ||||||
|   | |||||||
| @@ -1,318 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import { getLocaleFromNavigator, _ } from "svelte-i18n"; |  | ||||||
|   let modal_open = false; |  | ||||||
|   let delete_org = {}; |  | ||||||
|   import { RunnerOrganizationService } from "@odit/lfk-client-js"; |  | ||||||
|   import store from "../../store"; |  | ||||||
|   import OrgsEmptyState from "./OrgsEmptyState.svelte"; |  | ||||||
|   import Toastify from "toastify-js"; |  | ||||||
|   import ConfirmOrgDeletion from "./ConfirmOrgDeletion.svelte"; |  | ||||||
|   $: searchvalue = ""; |  | ||||||
|   $: active_deletes = []; |  | ||||||
|   $: sponsoring_contracts_download_open = false; |  | ||||||
|   export let current_organizations = []; |  | ||||||
|  |  | ||||||
|   const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then( |  | ||||||
|     (val) => { |  | ||||||
|       current_organizations = val; |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   document.addEventListener("click", function (e) { |  | ||||||
|     if ( |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" && |  | ||||||
|       e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu" |  | ||||||
|     ) { |  | ||||||
|       sponsoring_contracts_download_open = false; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   async function generateSponsoringContract(locale) { |  | ||||||
|     sponsoring_contracts_download_open = false; |  | ||||||
|     const orgs = current_organizations.filter((r) => r.is_selected === true); |  | ||||||
|     const toast = Toastify({ |  | ||||||
|       text: $_("generating-pdfs"), |  | ||||||
|       duration: -1, |  | ||||||
|     }).showToast(); |  | ||||||
|     let count = 0; |  | ||||||
|     for await (const o of orgs) { |  | ||||||
|       const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners( |  | ||||||
|         o.id |  | ||||||
|       ); |  | ||||||
|       fetch( |  | ||||||
|         `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`, |  | ||||||
|         { |  | ||||||
|           method: "POST", |  | ||||||
|           headers: { |  | ||||||
|             "Content-Type": "application/json", |  | ||||||
|           }, |  | ||||||
|           body: JSON.stringify(runners), |  | ||||||
|         } |  | ||||||
|       ) |  | ||||||
|         .then((response) => { |  | ||||||
|           if (response.status != "200") { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdf-generation-failed"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: |  | ||||||
|                 "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } else { |  | ||||||
|             return response.blob(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .then((blob) => { |  | ||||||
|           count++; |  | ||||||
|           const url = window.URL.createObjectURL(blob); |  | ||||||
|           let a = document.createElement("a"); |  | ||||||
|           a.href = url; |  | ||||||
|           a.download = "Sponsorings_" + o.name + ".pdf"; |  | ||||||
|           document.body.appendChild(a); |  | ||||||
|           a.click(); |  | ||||||
|           a.remove(); |  | ||||||
|           if (count === orgs.length) { |  | ||||||
|             toast.hideToast(); |  | ||||||
|             Toastify({ |  | ||||||
|               text: $_("pdfs-successfully-generated"), |  | ||||||
|               duration: 3500, |  | ||||||
|               backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)", |  | ||||||
|             }).showToast(); |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|         .catch((err) => {}); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <ConfirmOrgDeletion |  | ||||||
|   on:cancelDelete={(event) => { |  | ||||||
|     modal_open = false; |  | ||||||
|     active_deletes[event.detail.id] = false; |  | ||||||
|   }} |  | ||||||
|   bind:modal_open |  | ||||||
|   bind:delete_org /> |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:GET')} |  | ||||||
|   {#await promise} |  | ||||||
|     <div |  | ||||||
|       class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" |  | ||||||
|       role="alert"> |  | ||||||
|       <p class="font-bold">{$_('organizations-are-being-loaded')}</p> |  | ||||||
|       <p class="text-sm">{$_('this-might-take-a-moment')}</p> |  | ||||||
|     </div> |  | ||||||
|   {:then} |  | ||||||
|     {#if current_organizations.length === 0} |  | ||||||
|       <OrgsEmptyState /> |  | ||||||
|     {:else} |  | ||||||
|       <input |  | ||||||
|         type="search" |  | ||||||
|         bind:value={searchvalue} |  | ||||||
|         placeholder={$_('datatable.search')} |  | ||||||
|         aria-label={$_('datatable.search')} |  | ||||||
|         class="gridjs-input gridjs-search-input mb-4" /> |  | ||||||
|       <div class="h-12"> |  | ||||||
|         {#if current_organizations.some((r) => r.is_selected === true)} |  | ||||||
|         <div id="sponsoring:dropdown" class="relative inline-block"> |  | ||||||
|           <div> |  | ||||||
|             <button |  | ||||||
|               on:click={() => { |  | ||||||
|                 sponsoring_contracts_download_open = !sponsoring_contracts_download_open; |  | ||||||
|               }} |  | ||||||
|               type="button" |  | ||||||
|               class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex" |  | ||||||
|               id="options-menu" |  | ||||||
|               aria-haspopup="true" |  | ||||||
|               aria-expanded="true"> |  | ||||||
|               {$_('generate-sponsoring-contracts')} |  | ||||||
|               <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" class="-mr-1 ml-2 h-5 w-5"><path fill="none" d="M0 0h24v24H0z"/><path fill="currentColor" d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z"/></svg> |  | ||||||
|             </button> |  | ||||||
|           </div> |  | ||||||
|           {#if sponsoring_contracts_download_open} |  | ||||||
|             <div |  | ||||||
|               class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5" |  | ||||||
|               id="sponsoring:dropdown:menu"> |  | ||||||
|               <div |  | ||||||
|                 class="py-1" |  | ||||||
|                 role="menu" |  | ||||||
|                 aria-orientation="vertical" |  | ||||||
|                 aria-labelledby="options-menu"> |  | ||||||
|                 <span |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('de'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('german')} |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                   on:click={() => { |  | ||||||
|                     generateSponsoringContract('en'); |  | ||||||
|                   }} |  | ||||||
|                   type="submit" |  | ||||||
|                   class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900 inline-flex" |  | ||||||
|                   role="menuitem"> |  | ||||||
|                   {$_('english')} |  | ||||||
|                 </button> |  | ||||||
|               </div> |  | ||||||
|             </div> |  | ||||||
|           {/if} |  | ||||||
|         </div> |  | ||||||
|         {/if} |  | ||||||
|       </div> |  | ||||||
|       <div |  | ||||||
|         class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"> |  | ||||||
|         <table class="divide-y divide-gray-200 w-full"> |  | ||||||
|           <thead class="bg-gray-50"> |  | ||||||
|             <tr> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 <span |  | ||||||
|                   on:click={() => { |  | ||||||
|                     const newstate = !current_organizations.some((r) => r.is_selected === true); |  | ||||||
|                     current_organizations = current_organizations.map((r) => { |  | ||||||
|                       r.is_selected = newstate; |  | ||||||
|                       return r; |  | ||||||
|                     }); |  | ||||||
|                   }} |  | ||||||
|                   class="underline cursor-pointer select-none">{#if current_organizations.some((r) => r.is_selected === true)} |  | ||||||
|                     {$_('deselect-all')} |  | ||||||
|                   {:else}{$_('select-all')}{/if} |  | ||||||
|                 </span> |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('name')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('address')} |  | ||||||
|               </th> |  | ||||||
|               <th |  | ||||||
|                 scope="col" |  | ||||||
|                 class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> |  | ||||||
|                 {$_('contact')} |  | ||||||
|               </th> |  | ||||||
|               <th scope="col" class="relative px-6 py-3"> |  | ||||||
|                 <span class="sr-only">{$_('action')}</span> |  | ||||||
|               </th> |  | ||||||
|             </tr> |  | ||||||
|           </thead> |  | ||||||
|           <tbody class="divide-y divide-gray-200"> |  | ||||||
|             {#each current_organizations as o} |  | ||||||
|               {#if Object.values(o) |  | ||||||
|                 .toString() |  | ||||||
|                 .toLowerCase() |  | ||||||
|                 .includes(searchvalue)} |  | ||||||
|                 <tr data-rowid="org_{o.id}"> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <input |  | ||||||
|                       bind:checked={o.is_selected} |  | ||||||
|                       type="checkbox" |  | ||||||
|                       class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" /> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       <div class="ml-4"> |  | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                           {o.name} |  | ||||||
|                         </div> |  | ||||||
|                       </div> |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       <div class="ml-4"> |  | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                           {#if o.address.address1 !== null} |  | ||||||
|                             {o.address.address1}<br /> |  | ||||||
|                             {o.address.address2 || ''}<br /> |  | ||||||
|                             {o.address.postalcode} |  | ||||||
|                             {o.address.city} |  | ||||||
|                             {o.address.country} |  | ||||||
|                           {/if} |  | ||||||
|                         </div> |  | ||||||
|                       </div> |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   <td class="px-6 py-4 whitespace-nowrap"> |  | ||||||
|                     <div class="flex items-center"> |  | ||||||
|                       <div class="ml-4"> |  | ||||||
|                         <div class="text-sm font-medium text-gray-900"> |  | ||||||
|                           {#if o.contact} |  | ||||||
|                             <a |  | ||||||
|                               href="../contacts/{o.contact.id}" |  | ||||||
|                               class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{o.contact.firstname} |  | ||||||
|                               {o.contact.middlename || ''} |  | ||||||
|                               {o.contact.lastname}</a> |  | ||||||
|                           {:else}{$_('no-contact-specified')}{/if} |  | ||||||
|                         </div> |  | ||||||
|                       </div> |  | ||||||
|                     </div> |  | ||||||
|                   </td> |  | ||||||
|                   {#if active_deletes[o.id] === true} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           active_deletes[o.id] = false; |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button> |  | ||||||
|                       <button |  | ||||||
|                         on:click={() => { |  | ||||||
|                           RunnerOrganizationService.runnerOrganizationControllerRemove(o.id, false) |  | ||||||
|                             .then((resp) => { |  | ||||||
|                               current_organizations = current_organizations.filter((obj) => obj.id !== o.id); |  | ||||||
|                               Toastify({ |  | ||||||
|                                 text: 'Organization deleted', |  | ||||||
|                                 duration: 500, |  | ||||||
|                                 backgroundColor: |  | ||||||
|                                   'linear-gradient(to right, #00b09b, #96c93d)', |  | ||||||
|                               }).showToast(); |  | ||||||
|                             }) |  | ||||||
|                             .catch((err) => { |  | ||||||
|                               modal_open = true; |  | ||||||
|                               delete_org = o; |  | ||||||
|                             }); |  | ||||||
|                         }} |  | ||||||
|                         tabindex="0" |  | ||||||
|                         class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button> |  | ||||||
|                     </td> |  | ||||||
|                   {:else} |  | ||||||
|                     <td |  | ||||||
|                       class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> |  | ||||||
|                       <a |  | ||||||
|                         href="./{o.id}" |  | ||||||
|                         class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a> |  | ||||||
|                       {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:DELETE')} |  | ||||||
|                         <button |  | ||||||
|                           on:click={() => { |  | ||||||
|                             active_deletes[o.id] = true; |  | ||||||
|                           }} |  | ||||||
|                           tabindex="0" |  | ||||||
|                           class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button> |  | ||||||
|                       {/if} |  | ||||||
|                     </td> |  | ||||||
|                   {/if} |  | ||||||
|                 </tr> |  | ||||||
|               {/if} |  | ||||||
|             {/each} |  | ||||||
|           </tbody> |  | ||||||
|         </table> |  | ||||||
|       </div> |  | ||||||
|     {/if} |  | ||||||
|   {:catch error} |  | ||||||
|     <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500"> |  | ||||||
|       <span class="inline-block align-middle mr-8"> |  | ||||||
|         <b class="capitalize">{$_('general_promise_error')}</b> |  | ||||||
|         {error} |  | ||||||
|       </span> |  | ||||||
|     </div> |  | ||||||
|   {/await} |  | ||||||
| {/if} |  | ||||||
| @@ -1,42 +1,244 @@ | |||||||
| <script> | <script> | ||||||
|   import { _ } from "svelte-i18n"; | 	import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte"; | ||||||
|  | 	let delete_org = {}; | ||||||
|  | 	import { RunnerOrganizationService } from "@odit/lfk-client-js"; | ||||||
| 	import store from "../../store"; | 	import store from "../../store"; | ||||||
|   import AddOrgModal from "./AddOrgModal.svelte"; | 	import OrgsEmptyState from "./OrgsEmptyState.svelte"; | ||||||
|   export let modal_open = false; | 	import ConfirmOrgDeletionModal from "./ConfirmOrgDeletionModal.svelte"; | ||||||
|   import OrgOverview from "./OrgOverview.svelte"; | 	import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte"; | ||||||
|   import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; | 	import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
|  | 	$: searchvalue = ""; | ||||||
|  | 	$: active_deletes = []; | ||||||
|  | 	$: sponsoring_contracts_show = current_organizations.some( | ||||||
|  | 		(r) => r.is_selected === true | ||||||
|  | 	); | ||||||
|  | 	$: cards_show = current_organizations.some((r) => r.is_selected === true); | ||||||
|  | 	$: generate_orgs = current_organizations.filter( | ||||||
|  | 		(r) => r.is_selected === true | ||||||
|  | 	); | ||||||
|  | 	$: certificates_show = current_organizations.some( | ||||||
|  | 		(r) => r.is_selected === true | ||||||
|  | 	); | ||||||
| 	let current_organizations = []; | 	let current_organizations = []; | ||||||
|   export let import_modal_open = false; |  | ||||||
|  | 	const promise = | ||||||
|  | 		RunnerOrganizationService.runnerOrganizationControllerGetAll().then( | ||||||
|  | 			(val) => { | ||||||
|  | 				current_organizations = val; | ||||||
|  | 			} | ||||||
|  | 		); | ||||||
|  | 	import { _ } from "svelte-i18n"; | ||||||
|  | 	import AddOrgModal from "./AddOrgModal.svelte"; | ||||||
|  | 	let delete_modal_open = false; | ||||||
|  | 	let modal_open = false; | ||||||
|  | 	import ImportRunnerModal from "../runners/ImportRunnerModal.svelte"; | ||||||
|  | 	let import_modal_open = false; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <section class="container p-5"> | <section class="container p-5"> | ||||||
|   <span class="mb-1 text-3xl font-extrabold leading-tight"> | 	<h4 class="mb-1 text-3xl font-extrabold leading-tight"> | ||||||
|     {$_('organizations')} | 		{$_("organizations")} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:CREATE')} | 	</h4> | ||||||
|  | 	{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:CREATE")} | ||||||
| 		<button | 		<button | ||||||
| 			on:click={() => { | 			on:click={() => { | ||||||
| 				modal_open = true; | 				modal_open = true; | ||||||
| 			}} | 			}} | ||||||
| 			type="button" | 			type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | 			class="w-full 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:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|         {$_('create-organization')} | 		> | ||||||
|  | 			{$_("create-organization")} | ||||||
| 		</button> | 		</button> | ||||||
| 	{/if} | 	{/if} | ||||||
|     {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')} | 	{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:IMPORT")} | ||||||
| 		<button | 		<button | ||||||
| 			on:click={() => { | 			on:click={() => { | ||||||
| 				import_modal_open = true; | 				import_modal_open = true; | ||||||
| 			}} | 			}} | ||||||
| 			type="button" | 			type="button" | ||||||
|         class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm"> | 			class="w-full 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:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|         {$_('import-runners')} | 		> | ||||||
|  | 			{$_("import-runners")} | ||||||
| 		</button> | 		</button> | ||||||
| 	{/if} | 	{/if} | ||||||
|  | 	<ConfirmOrgDeletionModal | ||||||
|  | 		on:cancelDelete={(event) => { | ||||||
|  | 			delete_modal_open = false; | ||||||
|  | 			active_deletes[event.detail.id] = false; | ||||||
|  | 		}} | ||||||
|  | 		bind:modal_open={delete_modal_open} | ||||||
|  | 		bind:delete_org | ||||||
|  | 	/> | ||||||
|  | 	{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:GET")} | ||||||
|  | 		{#await promise} | ||||||
|  | 			<div | ||||||
|  | 				class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2" | ||||||
|  | 				role="alert" | ||||||
|  | 			> | ||||||
|  | 				<p class="font-bold">{$_("organizations-are-being-loaded")}</p> | ||||||
|  | 				<p class="text-sm">{$_("this-might-take-a-moment")}</p> | ||||||
|  | 			</div> | ||||||
|  | 		{:then} | ||||||
|  | 			{#if current_organizations.length === 0} | ||||||
|  | 				<OrgsEmptyState /> | ||||||
|  | 			{:else} | ||||||
|  | 				<input | ||||||
|  | 					type="search" | ||||||
|  | 					bind:value={searchvalue} | ||||||
|  | 					placeholder={$_("datatable.search")} | ||||||
|  | 					aria-label={$_("datatable.search")} | ||||||
|  | 					class="w-full sm:w-auto sm:mt-0 p-2 rounded-md border mb-1 lg:mb-0" | ||||||
|  | 				/> | ||||||
|  | 				<GenerateSponsoringContracts | ||||||
|  | 					bind:sponsoring_contracts_show | ||||||
|  | 					bind:generate_orgs | ||||||
|  | 				/> | ||||||
|  | 				<GenerateRunnerCards bind:cards_show bind:generate_orgs /> | ||||||
|  | 				<GenerateRunnerCertificates bind:certificates_show bind:generate_orgs /> | ||||||
|  | 				<div | ||||||
|  | 					class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll" | ||||||
|  | 				> | ||||||
|  | 					<table class="divide-y divide-gray-200 w-full"> | ||||||
|  | 						<thead class="bg-gray-50"> | ||||||
|  | 							<tr class="odd:bg-white even:bg-gray-100"> | ||||||
|  | 								<th | ||||||
|  | 									scope="col" | ||||||
|  | 									class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider" | ||||||
|  | 								> | ||||||
|  | 									<button | ||||||
|  | 										on:click={() => { | ||||||
|  | 											const newstate = !current_organizations.some( | ||||||
|  | 												(r) => r.is_selected === true | ||||||
|  | 											); | ||||||
|  | 											current_organizations = current_organizations.map((r) => { | ||||||
|  | 												r.is_selected = newstate; | ||||||
|  | 												return r; | ||||||
|  | 											}); | ||||||
|  | 										}} | ||||||
|  | 										class="underline cursor-pointer select-none" | ||||||
|  | 										>{#if current_organizations.some((r) => r.is_selected === true)} | ||||||
|  | 											{$_("deselect-all")} | ||||||
|  | 										{:else}{$_("select-all")}{/if} | ||||||
|  | 									</button> | ||||||
|  | 								</th> | ||||||
|  | 								<th | ||||||
|  | 									scope="col" | ||||||
|  | 									class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider" | ||||||
|  | 								> | ||||||
|  | 									{$_("name")} | ||||||
|  | 								</th> | ||||||
|  | 								<th | ||||||
|  | 									scope="col" | ||||||
|  | 									class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider" | ||||||
|  | 								> | ||||||
|  | 									{$_("address")} | ||||||
|  | 								</th> | ||||||
|  | 								<th | ||||||
|  | 									scope="col" | ||||||
|  | 									class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider" | ||||||
|  | 								> | ||||||
|  | 									{$_("contact")} | ||||||
|  | 								</th> | ||||||
|  | 								<th scope="col" class="relative px-6 py-3"> | ||||||
|  | 									<span class="sr-only">{$_("action")}</span> | ||||||
|  | 								</th> | ||||||
|  | 							</tr> | ||||||
|  | 						</thead> | ||||||
|  | 						<tbody class="divide-y divide-gray-200"> | ||||||
|  | 							{#each current_organizations as o} | ||||||
|  | 								{#if Object.values(o) | ||||||
|  | 									.toString() | ||||||
|  | 									.toLowerCase() | ||||||
|  | 									.includes(searchvalue)} | ||||||
|  | 									<tr | ||||||
|  | 										class="odd:bg-white even:bg-gray-100" | ||||||
|  | 										data-rowid="org_{o.id}" | ||||||
|  | 									> | ||||||
|  | 										<td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  | 											<input | ||||||
|  | 												bind:checked={o.is_selected} | ||||||
|  | 												type="checkbox" | ||||||
|  | 												class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" | ||||||
|  | 											/> | ||||||
|  | 										</td> | ||||||
|  | 										<td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  | 											<div class="flex items-center"> | ||||||
|  | 												<div class="text-sm font-medium text-gray-900"> | ||||||
|  | 													{o.name} | ||||||
|  | 												</div> | ||||||
|  | 											</div> | ||||||
|  | 										</td> | ||||||
|  | 										<td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  | 											<div class="flex items-center"> | ||||||
|  | 												<div class="text-sm font-medium text-gray-900"> | ||||||
|  | 													{#if o.address.address1 !== null} | ||||||
|  | 														{o.address.address1}<br /> | ||||||
|  | 														<!-- {o.address.address2 || ''}<br /> --> | ||||||
|  | 														{o.address.postalcode} | ||||||
|  | 														{o.address.city} | ||||||
|  | 														{o.address.country} | ||||||
|  | 													{/if} | ||||||
|  | 												</div> | ||||||
|  | 											</div> | ||||||
|  | 										</td> | ||||||
|  | 										<td class="px-6 py-4 whitespace-nowrap"> | ||||||
|  | 											<div class="flex items-center"> | ||||||
|  | 												<div class="text-sm font-medium text-gray-900"> | ||||||
|  | 													{#if o.contact} | ||||||
|  | 														<a | ||||||
|  | 															href="../contacts/{o.contact.id}" | ||||||
|  | 															class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800 border border-current" | ||||||
|  | 															>{o.contact.firstname} | ||||||
|  | 															{o.contact.middlename || ""} | ||||||
|  | 															{o.contact.lastname}</a | ||||||
|  | 														> | ||||||
|  | 													{:else}{$_("no-contact-specified")}{/if} | ||||||
|  | 												</div> | ||||||
|  | 											</div> | ||||||
|  | 										</td> | ||||||
|  | 										<td | ||||||
|  | 											class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium" | ||||||
|  | 										> | ||||||
|  | 											<a | ||||||
|  | 												href="./{o.id}" | ||||||
|  | 												class="text-indigo-600 hover:text-indigo-900" | ||||||
|  | 												>{$_("details")}</a | ||||||
|  | 											> | ||||||
|  | 											{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:DELETE")} | ||||||
|  | 												<button | ||||||
|  | 													on:click={() => { | ||||||
|  | 														active_deletes[o.id] = true; | ||||||
|  | 														delete_modal_open = true; | ||||||
|  | 														delete_org = o; | ||||||
|  | 													}} | ||||||
|  | 													tabindex="0" | ||||||
|  | 													class="ml-4 text-red-600 hover:text-red-900 cursor-pointer" | ||||||
|  | 													>{$_("delete")}</button | ||||||
|  | 												> | ||||||
|  | 											{/if} | ||||||
|  | 										</td> | ||||||
|  | 									</tr> | ||||||
|  | 								{/if} | ||||||
|  | 							{/each} | ||||||
|  | 						</tbody> | ||||||
|  | 					</table> | ||||||
|  | 				</div> | ||||||
|  | 			{/if} | ||||||
|  | 		{:catch error} | ||||||
|  | 			<div | ||||||
|  | 				class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500" | ||||||
|  | 			> | ||||||
|  | 				<span class="inline-block align-middle mr-8"> | ||||||
|  | 					<b class="capitalize">{$_("general_promise_error")}</b> | ||||||
|  | 					{error} | ||||||
| 				</span> | 				</span> | ||||||
|   <OrgOverview bind:current_organizations /> | 			</div> | ||||||
|  | 		{/await} | ||||||
|  | 	{/if} | ||||||
| </section> | </section> | ||||||
|  |  | ||||||
| {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:CREATE')} | {#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:CREATE")} | ||||||
| 	<AddOrgModal bind:current_organizations bind:modal_open /> | 	<AddOrgModal bind:current_organizations bind:modal_open /> | ||||||
| 	<ImportRunnerModal | 	<ImportRunnerModal | ||||||
| 		on:cancelDelete={(event) => { | 		on:cancelDelete={(event) => { | ||||||
| @@ -46,6 +248,6 @@ | |||||||
| 		passed_org={{}} | 		passed_org={{}} | ||||||
| 		passed_orgs={current_organizations} | 		passed_orgs={current_organizations} | ||||||
| 		opened_from="OrgOverview" | 		opened_from="OrgOverview" | ||||||
|     current_runners={[]} | 		bind:import_modal_open | ||||||
|     bind:import_modal_open /> | 	/> | ||||||
| {/if} | {/if} | ||||||
|   | |||||||
| @@ -9,9 +9,9 @@ | |||||||
| <div class="text-center items-center justify-center"> | <div class="text-center items-center justify-center"> | ||||||
|   <p class="mb-16 text-lg text-gray-500"> |   <p class="mb-16 text-lg text-gray-500"> | ||||||
|     <img class="w-full h-44" src={org_empty} alt="" /> |     <img class="w-full h-44" src={org_empty} alt="" /> | ||||||
|     <span |     <span class="font-bold">{$_("there-are-no-organizations-added-yet")}</span | ||||||
|       class="font-bold">{$_('there-are-no-organizations-added-yet')}</span><br /> |     ><br /> | ||||||
|     <span>{$_('add-your-first-organization')}</span> |     <span>{$_("add-your-first-organization")}</span> | ||||||
|   </p> |   </p> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										150
									
								
								src/components/pdf_generation/DocumentServer.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								src/components/pdf_generation/DocumentServer.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | |||||||
|  | class DocumentServer { | ||||||
|  |   baseUrl: string; | ||||||
|  |   apiKey: string; | ||||||
|  |  | ||||||
|  |   constructor(baseUrl: string, apiKey: string) { | ||||||
|  |     this.baseUrl = baseUrl; | ||||||
|  |     this.apiKey = apiKey; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async generateCards(cards: any[], locale: string) { | ||||||
|  |     const generateCards = new Array<any>(); | ||||||
|  |  | ||||||
|  |     for (let i = 0; i < cards.length; i++) { | ||||||
|  |       const card = { | ||||||
|  |         id: cards[i].id, | ||||||
|  |         enabled: cards[i].enabled, | ||||||
|  |         code: cards[i].code, | ||||||
|  |         runner: { | ||||||
|  |           id: cards[i]?.runner?.id, | ||||||
|  |           first_name: cards[i]?.runner?.firstname, | ||||||
|  |           middle_name: cards[i]?.runner?.middlename, | ||||||
|  |           last_name: cards[i]?.runner?.lastname, | ||||||
|  |           group: { | ||||||
|  |             id: cards[i]?.runner?.group.id, | ||||||
|  |             name: cards[i]?.runner?.group.name, | ||||||
|  |             parent_group: { | ||||||
|  |               id: cards[i]?.runner?.group?.parentGroup?.id, | ||||||
|  |               name: cards[i]?.runner?.group?.parentGroup?.name, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |       generateCards.push(card); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const response = await fetch( | ||||||
|  |       `${this.baseUrl}/v1/pdfs/cards?key=${this.apiKey}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify({ | ||||||
|  |           locale, | ||||||
|  |           cards: generateCards, | ||||||
|  |         }), | ||||||
|  |       }, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     const blob = await response.blob(); | ||||||
|  |     return blob; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async generateContracts(runners: any[], locale: string) { | ||||||
|  |     const generateRunners = new Array<any>(); | ||||||
|  |  | ||||||
|  |     for (let i = 0; i < runners.length; i++) { | ||||||
|  |       console.log(runners[i]); | ||||||
|  |       const card = { | ||||||
|  |         id: runners[i].id, | ||||||
|  |         first_name: runners[i].firstname, | ||||||
|  |         middle_name: runners[i].middlename, | ||||||
|  |         last_name: runners[i].lastname, | ||||||
|  |         group: { | ||||||
|  |           id: runners[i].group.id, | ||||||
|  |           name: runners[i].group.name, | ||||||
|  |           parent_group: { | ||||||
|  |             id: runners[i]?.group?.parentGroup?.id, | ||||||
|  |             name: runners[i]?.group?.parentGroup?.name, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |       generateRunners.push(card); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const response = await fetch( | ||||||
|  |       `${this.baseUrl}/v1/pdfs/contracts?key=${this.apiKey}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify({ | ||||||
|  |           locale, | ||||||
|  |           runners: generateRunners, | ||||||
|  |         }), | ||||||
|  |       }, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     const blob = await response.blob(); | ||||||
|  |     return blob; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async generateCertificates(runners: any[], locale: string) { | ||||||
|  |     const generateRunners = new Array<any>(); | ||||||
|  |  | ||||||
|  |     for (let i = 0; i < runners.length; i++) { | ||||||
|  |       const certificate = { | ||||||
|  |         id: runners[i].id, | ||||||
|  |         first_name: runners[i].firstname, | ||||||
|  |         middle_name: runners[i].middlename, | ||||||
|  |         last_name: runners[i].lastname, | ||||||
|  |         group: { | ||||||
|  |           id: runners[i].group.id, | ||||||
|  |           name: runners[i].group.name, | ||||||
|  |           parent_group: { | ||||||
|  |             id: runners[i]?.group?.parentGroup?.id, | ||||||
|  |             name: runners[i]?.group?.parentGroup?.name, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |         distance: runners[i].distance, | ||||||
|  |         distance_donations: runners[i].distanceDonations.map( | ||||||
|  |           (distanceDonation: any) => { | ||||||
|  |             return { | ||||||
|  |               id: distanceDonation.id, | ||||||
|  |               amount: distanceDonation.amount, | ||||||
|  |               amount_per_distance: distanceDonation.amountPerDistance, | ||||||
|  |               donor: { | ||||||
|  |                 id: distanceDonation.donor.id, | ||||||
|  |                 first_name: distanceDonation.donor.firstname, | ||||||
|  |                 middle_name: distanceDonation.donor.middlename, | ||||||
|  |                 last_name: distanceDonation.donor.lastname, | ||||||
|  |               }, | ||||||
|  |             }; | ||||||
|  |           }, | ||||||
|  |         ), | ||||||
|  |       }; | ||||||
|  |       generateRunners.push(certificate); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const response = await fetch( | ||||||
|  |       `${this.baseUrl}/v1/pdfs/certificates?key=${this.apiKey}`, | ||||||
|  |       { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify({ | ||||||
|  |           locale, | ||||||
|  |           runners: generateRunners, | ||||||
|  |         }), | ||||||
|  |       }, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     const blob = await response.blob(); | ||||||
|  |     return blob; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export default DocumentServer; | ||||||
							
								
								
									
										197
									
								
								src/components/pdf_generation/GenerateRunnerCards.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								src/components/pdf_generation/GenerateRunnerCards.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,197 @@ | |||||||
|  | <script> | ||||||
|  | 	import { _ } from "svelte-i18n"; | ||||||
|  | 	import { | ||||||
|  | 		RunnerCardService, | ||||||
|  | 		RunnerOrganizationService, | ||||||
|  | 		RunnerTeamService, | ||||||
|  | 	} from "@odit/lfk-client-js"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
|  | 	import DocumentServer from "./DocumentServer.ts"; | ||||||
|  |  | ||||||
|  | 	import { init } from "@paralleldrive/cuid2"; | ||||||
|  | 	const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  | 	const documentServer = new DocumentServer( | ||||||
|  | 		config.baseurl_documentserver, | ||||||
|  | 		config.documentserver_key | ||||||
|  | 	); | ||||||
|  |  | ||||||
|  | 	export let cards_show = false; | ||||||
|  | 	export let generate_cards = []; | ||||||
|  | 	export let generate_runners = []; | ||||||
|  | 	export let generate_orgs = []; | ||||||
|  | 	export let generate_teams = []; | ||||||
|  |  | ||||||
|  | 	function download(blob, fileName) { | ||||||
|  | 		const url = window.URL.createObjectURL(blob); | ||||||
|  | 		let a = document.createElement("a"); | ||||||
|  | 		a.href = url; | ||||||
|  | 		a.download = fileName; | ||||||
|  | 		document.body.appendChild(a); | ||||||
|  | 		a.click(); | ||||||
|  | 		a.remove(); | ||||||
|  | 		toast.dismiss(); | ||||||
|  | 		toast.success($_("pdf-successfully-generated")); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function generateRunnerCards(locale) { | ||||||
|  | 		if (generate_orgs.length > 0) { | ||||||
|  | 			generateOrgCards(locale); | ||||||
|  | 		} else if (generate_teams.length > 0) { | ||||||
|  | 			generateTeamCards(locale); | ||||||
|  | 		} else if (generate_runners.length > 0) { | ||||||
|  | 			generateRunnersCards(locale); | ||||||
|  | 		} else { | ||||||
|  | 			generateCards(locale); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function generateCards(locale) { | ||||||
|  | 		toast.loading($_("generating-pdf")); | ||||||
|  | 		documentServer | ||||||
|  | 			.generateCards(generate_cards, locale) | ||||||
|  | 			.then((blob) => { | ||||||
|  | 				download(blob, `${$_("runnercards")}-${locale}-${createId()}.pdf`); | ||||||
|  | 			}) | ||||||
|  | 			.catch((err) => { | ||||||
|  | 				console.error(err); | ||||||
|  | 			}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateRunnersCards(locale) { | ||||||
|  | 		toast.loading($_("generating-pdf")); | ||||||
|  | 		const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  | 		let cards = []; | ||||||
|  | 		for (let runner of generate_runners) { | ||||||
|  | 			let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  | 			if (!card) { | ||||||
|  | 				card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  | 					runner: runner.id, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
|  | 			cards.push(card); | ||||||
|  | 		} | ||||||
|  | 		documentServer | ||||||
|  | 			.generateCards(cards, locale) | ||||||
|  | 			.then((blob) => { | ||||||
|  | 				let fileName = `${$_("runnercards")}-${locale}-${createId()}.pdf`; | ||||||
|  | 				if (generate_runners.length == 1) { | ||||||
|  | 					fileName = `${$_("runnercards")}_${generate_runners[0].firstname}_${ | ||||||
|  | 						generate_runners[0].lastname | ||||||
|  | 					}-${locale}-${createId()}.pdf`; | ||||||
|  | 				} | ||||||
|  | 				download(blob, fileName); | ||||||
|  | 			}) | ||||||
|  | 			.catch((err) => {}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateTeamCards(locale) { | ||||||
|  | 		toast.loading($_("generating-pdfs")); | ||||||
|  | 		let count = 0; | ||||||
|  | 		const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  | 		for (const t of generate_teams) { | ||||||
|  | 			const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  | 				t.id | ||||||
|  | 			); | ||||||
|  | 			let cards = []; | ||||||
|  | 			for (let runner of runners) { | ||||||
|  | 				let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  | 				if (!card) { | ||||||
|  | 					card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  | 						runner: runner.id, | ||||||
|  | 					}); | ||||||
|  | 				} | ||||||
|  | 				cards.push(card); | ||||||
|  | 			} | ||||||
|  | 			documentServer | ||||||
|  | 				.generateCards(cards, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("runnercards")}_${t.name}-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.catch((err) => {}); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateOrgCards(locale) { | ||||||
|  | 		toast.loading($_("generating-pdfs")); | ||||||
|  | 		const current_cards = await RunnerCardService.runnerCardControllerGetAll(); | ||||||
|  | 		let count = 0; | ||||||
|  | 		let count_orgs = 0; | ||||||
|  | 		for (const o of generate_orgs) { | ||||||
|  | 			count_orgs++; | ||||||
|  | 			let count = 0; | ||||||
|  | 			let runners = | ||||||
|  | 				await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  | 					o.id, | ||||||
|  | 					true | ||||||
|  | 				); | ||||||
|  | 			let cards = []; | ||||||
|  | 			for (let runner of runners) { | ||||||
|  | 				let card = current_cards.find((c) => c.runner?.id == runner.id); | ||||||
|  | 				if (!card) { | ||||||
|  | 					card = await RunnerCardService.runnerCardControllerPost({ | ||||||
|  | 						runner: runner.id, | ||||||
|  | 					}); | ||||||
|  | 				} | ||||||
|  | 				cards.push(card); | ||||||
|  | 			} | ||||||
|  | 			await documentServer | ||||||
|  | 				.generateCards(cards, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("runnercards")}_${o.name}_direct-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.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 documentServer | ||||||
|  | 					.generateCards(cards, locale) | ||||||
|  | 					.then((blob) => { | ||||||
|  | 						download( | ||||||
|  | 							blob, | ||||||
|  | 							`${$_("runnercards")}_${o.name}_${ | ||||||
|  | 								t.name | ||||||
|  | 							}-${locale}-${createId()}.pdf` | ||||||
|  | 						); | ||||||
|  | 					}) | ||||||
|  | 					.catch((err) => {}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if cards_show} | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateRunnerCards("de"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-runnercards")}: DE | ||||||
|  | 	</button> | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateRunnerCards("en"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-runnercards")}: EN | ||||||
|  | 	</button> | ||||||
|  | {/if} | ||||||
							
								
								
									
										175
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,175 @@ | |||||||
|  | <script> | ||||||
|  | 	import { _ } from "svelte-i18n"; | ||||||
|  | 	import { | ||||||
|  | 		DonationService, | ||||||
|  | 		RunnerTeamService, | ||||||
|  | 		RunnerOrganizationService, | ||||||
|  | 	} from "@odit/lfk-client-js"; | ||||||
|  | 	import { init } from "@paralleldrive/cuid2"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
|  | 	import DocumentServer from "./DocumentServer"; | ||||||
|  | 	const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  | 	const documentServer = new DocumentServer( | ||||||
|  | 		config.baseurl_documentserver, | ||||||
|  | 		config.documentserver_key | ||||||
|  | 	); | ||||||
|  |  | ||||||
|  | 	export let certificates_show = false; | ||||||
|  | 	export let generate_runners = []; | ||||||
|  | 	export let generate_orgs = []; | ||||||
|  | 	export let generate_teams = []; | ||||||
|  |  | ||||||
|  | 	function generateCertificates(locale) { | ||||||
|  | 		if (generate_orgs.length > 0) { | ||||||
|  | 			generateOrgCertificates(locale); | ||||||
|  | 		} else if (generate_teams.length > 0) { | ||||||
|  | 			generateTeamCertificates(locale); | ||||||
|  | 		} else { | ||||||
|  | 			generateRunnerCertificates(locale); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function download(blob, fileName) { | ||||||
|  | 		const url = window.URL.createObjectURL(blob); | ||||||
|  | 		let a = document.createElement("a"); | ||||||
|  | 		a.href = url; | ||||||
|  | 		a.download = fileName; | ||||||
|  | 		document.body.appendChild(a); | ||||||
|  | 		a.click(); | ||||||
|  | 		a.remove(); | ||||||
|  | 		toast.dismiss(); | ||||||
|  | 		toast.success($_("pdf-successfully-generated")); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateRunnerCertificates(locale) { | ||||||
|  | 		toast.loading($_("generating-pdf")); | ||||||
|  | 		const current_donations = | ||||||
|  | 			(await DonationService.donationControllerGetAll()) || []; | ||||||
|  | 		let certificateRunners = []; | ||||||
|  | 		for (let runner of generate_runners) { | ||||||
|  | 			runner.distanceDonations = | ||||||
|  | 				current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  | 			certificateRunners.push(runner); | ||||||
|  | 		} | ||||||
|  | 		documentServer | ||||||
|  | 			.generateCertificates(certificateRunners, locale) | ||||||
|  | 			.then((blob) => { | ||||||
|  | 				let fileName = `${$_("certificates")}-${locale}.pdf`; | ||||||
|  | 				if (generate_runners.length == 1) { | ||||||
|  | 					fileName = `${$_("certificates")}_${ | ||||||
|  | 						generate_runners[0].firstname | ||||||
|  | 					}_${generate_runners[0].lastname}-${locale}-${createId()}.pdf`; | ||||||
|  | 				} | ||||||
|  | 				download(blob, fileName); | ||||||
|  | 			}) | ||||||
|  | 			.catch((err) => {}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateTeamCertificates(locale) { | ||||||
|  | 		toast.loading($_("generating-pdfs")); | ||||||
|  | 		let count = 0; | ||||||
|  | 		const current_donations = | ||||||
|  | 			(await DonationService.donationControllerGetAll()) || []; | ||||||
|  | 		for (const t of generate_teams) { | ||||||
|  | 			const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  | 				t.id | ||||||
|  | 			); | ||||||
|  | 			let certificateRunners = []; | ||||||
|  | 			for (let runner of runners) { | ||||||
|  | 				runner.distanceDonations = | ||||||
|  | 					current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  | 				certificateRunners.push(runner); | ||||||
|  | 			} | ||||||
|  | 			documentServer | ||||||
|  | 				.generateCertificates(certificateRunners, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					count++; | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("certificates")}_${t.name}-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.catch((err) => {}); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateOrgCertificates(locale) { | ||||||
|  | 		toast.loading($_("generating-pdfs")); | ||||||
|  | 		const current_donations = | ||||||
|  | 			(await DonationService.donationControllerGetAll()) || []; | ||||||
|  | 		let count = 0; | ||||||
|  | 		let count_orgs = 0; | ||||||
|  | 		for (const o of generate_orgs) { | ||||||
|  | 			count_orgs++; | ||||||
|  | 			let count = 0; | ||||||
|  | 			let runners = | ||||||
|  | 				await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  | 					o.id, | ||||||
|  | 					true | ||||||
|  | 				); | ||||||
|  | 			let certificateRunners = []; | ||||||
|  | 			for (let runner of runners) { | ||||||
|  | 				runner.distanceDonations = | ||||||
|  | 					current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  | 				certificateRunners.push(runner); | ||||||
|  | 			} | ||||||
|  | 			await documentServer | ||||||
|  | 				.generateCertificates(certificateRunners, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("certificates")}_${o.name}-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.catch((err) => {}); | ||||||
|  | 			for (const t of o.teams) { | ||||||
|  | 				count++; | ||||||
|  | 				let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  | 					t.id | ||||||
|  | 				); | ||||||
|  | 				let certificateRunners = []; | ||||||
|  | 				for (let runner of runners) { | ||||||
|  | 					runner.distanceDonations = | ||||||
|  | 						current_donations.filter((d) => d.runner?.id == runner.id) || []; | ||||||
|  | 					certificateRunners.push(runner); | ||||||
|  | 				} | ||||||
|  | 				await documentServer | ||||||
|  | 					.generateCertificates(certificateRunners, locale) | ||||||
|  | 					.then((blob) => { | ||||||
|  | 						download( | ||||||
|  | 							blob, | ||||||
|  | 							`${$_("certificates")}_${o.name}_${ | ||||||
|  | 								t.name | ||||||
|  | 							}-${locale}-${createId()}.pdf` | ||||||
|  | 						); | ||||||
|  | 						if ( | ||||||
|  | 							count === o.teams.length && | ||||||
|  | 							count_orgs === generate_orgs.length | ||||||
|  | 						) { | ||||||
|  | 							toast.dismiss(); | ||||||
|  | 							toast.success($_("pdfs-successfully-generated")); | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  | 					.catch((err) => {}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if certificates_show} | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateCertificates("de"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-runner-certificates")}: DE | ||||||
|  | 	</button> | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateCertificates("en"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-runner-certificates")}: EN | ||||||
|  | 	</button> | ||||||
|  | {/if} | ||||||
							
								
								
									
										138
									
								
								src/components/pdf_generation/GenerateSponsoringContracts.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								src/components/pdf_generation/GenerateSponsoringContracts.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | |||||||
|  | <script> | ||||||
|  | 	import { _ } from "svelte-i18n"; | ||||||
|  | 	import { | ||||||
|  | 		RunnerOrganizationService, | ||||||
|  | 		RunnerTeamService, | ||||||
|  | 	} from "@odit/lfk-client-js"; | ||||||
|  | 	import DocumentServer from "./DocumentServer"; | ||||||
|  | 	import { init } from "@paralleldrive/cuid2"; | ||||||
|  | 	import toast from "svelte-french-toast"; | ||||||
|  | 	const createId = init({ length: 10, fingerprint: "lfk-frontend" }); | ||||||
|  | 	const documentServer = new DocumentServer( | ||||||
|  | 		config.baseurl_documentserver, | ||||||
|  | 		config.documentserver_key | ||||||
|  | 	); | ||||||
|  |  | ||||||
|  | 	export let sponsoring_contracts_show = false; | ||||||
|  | 	export let generate_runners = []; | ||||||
|  | 	export let generate_orgs = []; | ||||||
|  | 	export let generate_teams = []; | ||||||
|  |  | ||||||
|  | 	function generateSponsoringContract(locale) { | ||||||
|  | 		if (generate_orgs.length > 0) { | ||||||
|  | 			generateOrgContracts(locale); | ||||||
|  | 		} else if (generate_teams.length > 0) { | ||||||
|  | 			generateTeamContracts(locale); | ||||||
|  | 		} else { | ||||||
|  | 			generateRunnerContracts(locale); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function download(blob, fileName) { | ||||||
|  | 		const url = window.URL.createObjectURL(blob); | ||||||
|  | 		let a = document.createElement("a"); | ||||||
|  | 		a.href = url; | ||||||
|  | 		a.download = fileName; | ||||||
|  | 		document.body.appendChild(a); | ||||||
|  | 		a.click(); | ||||||
|  | 		a.remove(); | ||||||
|  | 		toast.dismiss(); | ||||||
|  | 		toast.success($_("pdf-successfully-generated")); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateTeamContracts(locale) { | ||||||
|  | 		toast.loading($_("generating-pdfs")); | ||||||
|  | 		let count = 0; | ||||||
|  | 		for (const t of generate_teams) { | ||||||
|  | 			count++; | ||||||
|  | 			const runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  | 				t.id | ||||||
|  | 			); | ||||||
|  | 			documentServer | ||||||
|  | 				.generateContracts(runners, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("sponsorings")}_${t.name}-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.catch((err) => {}); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async function generateOrgContracts(locale) { | ||||||
|  | 		toast.loading($_("generating-pdf")); | ||||||
|  | 		let count_orgs = 0; | ||||||
|  | 		for (const o of generate_orgs) { | ||||||
|  | 			count_orgs++; | ||||||
|  | 			let count = 0; | ||||||
|  | 			let runners = | ||||||
|  | 				await RunnerOrganizationService.runnerOrganizationControllerGetRunners( | ||||||
|  | 					o.id, | ||||||
|  | 					true | ||||||
|  | 				); | ||||||
|  | 			await documentServer | ||||||
|  | 				.generateContracts(runners, locale) | ||||||
|  | 				.then((blob) => { | ||||||
|  | 					download( | ||||||
|  | 						blob, | ||||||
|  | 						`${$_("sponsorings")}_${o.name}_direct-${locale}-${createId()}.pdf` | ||||||
|  | 					); | ||||||
|  | 				}) | ||||||
|  | 				.catch((err) => {}); | ||||||
|  | 			for (const t of o.teams) { | ||||||
|  | 				count++; | ||||||
|  | 				let runners = await RunnerTeamService.runnerTeamControllerGetRunners( | ||||||
|  | 					t.id | ||||||
|  | 				); | ||||||
|  | 				await documentServer | ||||||
|  | 					.generateContracts(runners, locale) | ||||||
|  | 					.then((blob) => { | ||||||
|  | 						download( | ||||||
|  | 							blob, | ||||||
|  | 							`${$_("sponsorings")}_${o.name}_${ | ||||||
|  | 								t.name | ||||||
|  | 							}-${locale}-${createId()}.pdf` | ||||||
|  | 						); | ||||||
|  | 					}) | ||||||
|  | 					.catch((err) => {}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function generateRunnerContracts(locale) { | ||||||
|  | 		toast.loading($_("generating-pdf")); | ||||||
|  | 		documentServer | ||||||
|  | 			.generateContracts(generate_runners, locale) | ||||||
|  | 			.then((blob) => { | ||||||
|  | 				let fileName = `${$_("sponsorings")}-${locale}-${createId()}.pdf`; | ||||||
|  | 				if (generate_runners.length == 1) { | ||||||
|  | 					fileName = `${$_("sponsorings")}_${generate_runners[0].firstname}_${ | ||||||
|  | 						generate_runners[0].lastname | ||||||
|  | 					}-${locale}-${createId()}.pdf`; | ||||||
|  | 				} | ||||||
|  | 				download(blob, fileName); | ||||||
|  | 			}) | ||||||
|  | 			.catch((err) => { | ||||||
|  | 				console.error(err); | ||||||
|  | 			}); | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | {#if sponsoring_contracts_show} | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateSponsoringContract("de"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-sponsoring-contracts")}: DE | ||||||
|  | 	</button> | ||||||
|  | 	<button | ||||||
|  | 		on:click={() => { | ||||||
|  | 			generateSponsoringContract("en"); | ||||||
|  | 		}} | ||||||
|  | 		class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:w-auto sm:text-sm mb-1 lg:mb-0" | ||||||
|  | 	> | ||||||
|  | 		{$_("generate-sponsoring-contracts")}: EN | ||||||
|  | 	</button> | ||||||
|  | {/if} | ||||||
| @@ -1,80 +0,0 @@ | |||||||
| <h3 class="text-lg">Standard Avatars</h3> |  | ||||||
| <div class="relative rounded-full w-4 h-4"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-8 h-8"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-12 h-12"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-16 h-16"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-20 h-20"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-24 h-24"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
| </div> |  | ||||||
| <h3 class="text-lg">Status Avatars</h3> |  | ||||||
| <div class="relative rounded-full w-4 h-4"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-1 h-1 bg-gray-200" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-8 h-8"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-2 h-2 bg-green-400" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-12 h-12"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-4 h-4 bg-red-600" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-16 h-16"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-5 h-5 bg-gray-200" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-20 h-20"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-6 h-6 bg-green-400" /> |  | ||||||
| </div> |  | ||||||
| <div class="relative rounded-full w-24 h-24"> |  | ||||||
|   <img |  | ||||||
|     alt="" |  | ||||||
|     src="https://gustui.s3.amazonaws.com/avatar.png" |  | ||||||
|     class="absolute left-0 top-0 w-full h-full rounded-full object-cover" /> |  | ||||||
|   <div class="absolute rounded-full right-0 bottom-0 w-6 h-6 bg-red-600" /> |  | ||||||
| </div> |  | ||||||
| @@ -1,53 +0,0 @@ | |||||||
| <h3 class="text-lg">badges</h3> |  | ||||||
| <span |  | ||||||
|   class="text-sm font-medium bg-green-100 py-1 px-2 rounded text-green-500 align-middle">Paid</span> |  | ||||||
| <span |  | ||||||
|   class="text-sm font-medium bg-red-100 py-1 px-2 rounded text-red-500 align-middle">Overdue</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-blue-600">Primary</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-gray-600">Secondary</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-green-600">Success</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-red-600">Danger</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-black bg-yellow-400">Warning</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-indigo-300">Info</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-black bg-gray-200">Light</span> |  | ||||||
| <span |  | ||||||
|   class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-gray-900">Dark</span> |  | ||||||
| <h3 class="text-lg">closable badges</h3> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-blue-600"> |  | ||||||
|   Primary |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-gray-600"> |  | ||||||
|   Secondary |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-green-600"> |  | ||||||
|   Success |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-red-600"> |  | ||||||
|   Danger |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-black bg-yellow-400"> |  | ||||||
|   Warning |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-indigo-300"> |  | ||||||
|   Info |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-black bg-gray-200"> |  | ||||||
|   Light<span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| <span class="rounded-sm py-1 px-2 text-xs font-medium text-white bg-gray-900"> |  | ||||||
|   Dark |  | ||||||
|   <span class="ml-2 text-base cursor-pointer">×</span> |  | ||||||
| </span> |  | ||||||
| @@ -1,62 +0,0 @@ | |||||||
| <div class="flex flex-row mb-4"> |  | ||||||
|   <div class="w-full"> |  | ||||||
|     <nav class="w-full flex"> |  | ||||||
|       <ol class="list-none flex flex-row items-center justify-start"> |  | ||||||
|         <li class="mr-2 flex items-center"> |  | ||||||
|           <svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="none" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             class="h-3 w-3 stroke-current" |  | ||||||
|             height="1em" |  | ||||||
|             width="1em" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><path |  | ||||||
|               d="M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z" /> |  | ||||||
|             <polyline points="9 22 9 12 15 12 15 22" /></svg> |  | ||||||
|         </li> |  | ||||||
|         <li class="flex items-center"> |  | ||||||
|           <a class="mr-2" href="/">Home</a><svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="none" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             class="h-3 w-3 mr-2 stroke-current" |  | ||||||
|             height="1em" |  | ||||||
|             width="1em" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><line |  | ||||||
|               x1="5" |  | ||||||
|               y1="12" |  | ||||||
|               x2="19" |  | ||||||
|               y2="12" /> |  | ||||||
|             <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|         </li> |  | ||||||
|         <li class="flex items-center"> |  | ||||||
|           <a class="mr-2" href="/">Second level</a><svg |  | ||||||
|             stroke="currentColor" |  | ||||||
|             fill="none" |  | ||||||
|             stroke-width="2" |  | ||||||
|             viewBox="0 0 24 24" |  | ||||||
|             stroke-linecap="round" |  | ||||||
|             stroke-linejoin="round" |  | ||||||
|             class="h-3 w-3 mr-2 stroke-current" |  | ||||||
|             height="1em" |  | ||||||
|             width="1em" |  | ||||||
|             xmlns="http://www.w3.org/2000/svg"><line |  | ||||||
|               x1="5" |  | ||||||
|               y1="12" |  | ||||||
|               x2="19" |  | ||||||
|               y2="12" /> |  | ||||||
|             <polyline points="12 5 19 12 12 19" /></svg> |  | ||||||
|         </li> |  | ||||||
|         <li class="flex items-center"> |  | ||||||
|           <a class="mr-2" href="/">Third level</a> |  | ||||||
|         </li> |  | ||||||
|       </ol> |  | ||||||
|     </nav> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| @@ -1,75 +0,0 @@ | |||||||
| <script> |  | ||||||
|   import Avatars from "./Avatars.svelte"; |  | ||||||
|   import Badges from "./Badges.svelte"; |  | ||||||
|   import BreadcrumbNav from "./BreadcrumbNav.svelte"; |  | ||||||
|   import FileUpload from "./FileUpload.svelte"; |  | ||||||
|   import Pagination from "./Pagination.svelte"; |  | ||||||
|   import Table from "./Table.svelte"; |  | ||||||
|   import Tabs from "./Tabs.svelte"; |  | ||||||
|   import Tags from "./Tags.svelte"; |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <div class="border-4 border-dashed rounded h-96 mb-4" /> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <FileUpload /> |  | ||||||
| </div> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <Tabs /> |  | ||||||
| </div> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <Tags /> |  | ||||||
| </div> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <Badges /> |  | ||||||
| </div> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <Avatars /> |  | ||||||
| </div> |  | ||||||
| <Pagination /> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <Table /> |  | ||||||
| </div> |  | ||||||
| <div |  | ||||||
|   class="widget w-full p-4 mb-4 rounded-lg bg-white border border-grey-100"> |  | ||||||
|   <div class="flex flex-row items-center justify-between mb-6"> |  | ||||||
|     <div class="flex flex-col"> |  | ||||||
|       <div class="text-sm font-light text-grey-500">Regular</div> |  | ||||||
|       <div class="text-sm font-bold"><span>Text inputs</span></div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
|   <div class="flex flex-col lg:flex-row lg:flex-wrap w-full lg:space-x-4"> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div class="form-element "> |  | ||||||
|         <div class="form-label">Label</div><input |  | ||||||
|           name="name" |  | ||||||
|           type="text" |  | ||||||
|           class="form-input" |  | ||||||
|           placeholder="Enter something..." /> |  | ||||||
|         <div class="form-hint">This is a hint</div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div class="form-element "> |  | ||||||
|         <div class="form-label">First name</div><input |  | ||||||
|           name="name" |  | ||||||
|           type="text" |  | ||||||
|           class="form-input form-input-invalid" |  | ||||||
|           placeholder="john@example.com" /> |  | ||||||
|         <div class="form-error">First name is required</div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="w-full lg:w-1/4"> |  | ||||||
|       <div class="form-element "> |  | ||||||
|         <div class="form-label">First name</div><input |  | ||||||
|           name="name" |  | ||||||
|           type="text" |  | ||||||
|           class="form-input form-input-valid" |  | ||||||
|           placeholder="john@example.com" /> |  | ||||||
|         <div class="form-success">First name is valid</div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="mb-8"> |  | ||||||
|   <BreadcrumbNav /> |  | ||||||
| </div> |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user