Compare commits
	
		
			614 Commits
		
	
	
		
			18ec100c33
			...
			1.10.6
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						
						
							
						
						3fab344779
	
				 | 
					
					
						|||
| 
						
						
							
						
						d841727439
	
				 | 
					
					
						|||
| 
						
						
							
						
						87df34bb56
	
				 | 
					
					
						|||
| 
						
						
							
						
						9c3b742a98
	
				 | 
					
					
						|||
| 
						
						
							
						
						68bf3717f9
	
				 | 
					
					
						|||
| 
						
						
							
						
						f88c6e0dba
	
				 | 
					
					
						|||
| 
						
						
							
						
						12050cdda9
	
				 | 
					
					
						|||
| 
						
						
							
						
						01e77a97f3
	
				 | 
					
					
						|||
| 
						
						
							
						
						10824b5d9b
	
				 | 
					
					
						|||
| 
						
						
							
						
						d9870e03bc
	
				 | 
					
					
						|||
| 
						
						
							
						
						785b9e0b60
	
				 | 
					
					
						|||
| 
						
						
							
						
						fce2bc645e
	
				 | 
					
					
						|||
| 
						
						
							
						
						991716a7f5
	
				 | 
					
					
						|||
| 
						
						
							
						
						aa720f2460
	
				 | 
					
					
						|||
| 
						
						
							
						
						ac4ef8fb6d
	
				 | 
					
					
						|||
| 
						
						
							
						
						eae0afda23
	
				 | 
					
					
						|||
| 
						
						
							
						
						5291f8a4d1
	
				 | 
					
					
						|||
| 
						
						
							
						
						e2d7de1e9e
	
				 | 
					
					
						|||
| 
						
						
							
						
						d7c9c27ec7
	
				 | 
					
					
						|||
| 
						
						
							
						
						153b1b3c2b
	
				 | 
					
					
						|||
| 
						
						
							
						
						ec63c7c1c5
	
				 | 
					
					
						|||
| 
						
						
							
						
						05c2535698
	
				 | 
					
					
						|||
| 
						
						
							
						
						e261d5e345
	
				 | 
					
					
						|||
| 
						
						
							
						
						c00497d776
	
				 | 
					
					
						|||
| 
						
						
							
						
						766eeab49f
	
				 | 
					
					
						|||
| 
						
						
							
						
						3c9b404234
	
				 | 
					
					
						|||
| 
						
						
							
						
						9c56b3883e
	
				 | 
					
					
						|||
| 
						
						
							
						
						3d506db975
	
				 | 
					
					
						|||
| 
						
						
							
						
						d7e84a79a8
	
				 | 
					
					
						|||
| 
						
						
							
						
						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 | 
@@ -1,6 +0,0 @@
 | 
			
		||||
FROM mcr.microsoft.com/vscode/devcontainers/base:alpine-3.12
 | 
			
		||||
RUN apk update
 | 
			
		||||
RUN apk add --upgrade nodejs-current npm
 | 
			
		||||
RUN npm i -g yarn rimraf
 | 
			
		||||
RUN rimraf node_modules
 | 
			
		||||
RUN yarn set version berry
 | 
			
		||||
@@ -1,20 +0,0 @@
 | 
			
		||||
{
 | 
			
		||||
	"name": "Node.js",
 | 
			
		||||
	"build": {
 | 
			
		||||
		"dockerfile": "Dockerfile"
 | 
			
		||||
	},
 | 
			
		||||
	"settings": {
 | 
			
		||||
		"terminal.integrated.shell.linux": "/bin/sh"
 | 
			
		||||
	},
 | 
			
		||||
	"extensions": [
 | 
			
		||||
		"dbaeumer.vscode-eslint",
 | 
			
		||||
		"2gua.rainbow-brackets",
 | 
			
		||||
		"christian-kohler.npm-intellisense",
 | 
			
		||||
		"remimarsal.prettier-now",
 | 
			
		||||
		"svelte.svelte-vscode",
 | 
			
		||||
		"lokalise.i18n-ally",
 | 
			
		||||
		"fivethree.vscode-svelte-snippets",
 | 
			
		||||
		"voorjaar.windicss-intellisense"
 | 
			
		||||
	],
 | 
			
		||||
	"postCreateCommand": "yarn && yarn dev --open"
 | 
			
		||||
}
 | 
			
		||||
@@ -1 +1,4 @@
 | 
			
		||||
public/env.sample.js
 | 
			
		||||
.pnpm-store
 | 
			
		||||
.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
 | 
			
		||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,10 +1,8 @@
 | 
			
		||||
node_modules
 | 
			
		||||
package-lock.json
 | 
			
		||||
yarn.lock
 | 
			
		||||
*.map
 | 
			
		||||
public/env.js
 | 
			
		||||
public/index.html
 | 
			
		||||
/dist
 | 
			
		||||
.pnpm-store
 | 
			
		||||
.yarn
 | 
			
		||||
.pnp.js
 | 
			
		||||
.yarnrc.yml
 | 
			
		||||
.pnp.*
 | 
			
		||||
							
								
								
									
										22
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							@@ -1,14 +1,12 @@
 | 
			
		||||
{
 | 
			
		||||
    "recommendations": [
 | 
			
		||||
        "2gua.rainbow-brackets",
 | 
			
		||||
        "christian-kohler.npm-intellisense",
 | 
			
		||||
        "remimarsal.prettier-now",
 | 
			
		||||
        "svelte.svelte-vscode",
 | 
			
		||||
        "lokalise.i18n-ally",
 | 
			
		||||
        "fivethree.vscode-svelte-snippets",
 | 
			
		||||
        "voorjaar.windicss-intellisense"
 | 
			
		||||
    ],
 | 
			
		||||
    "unwantedRecommendations": [
 | 
			
		||||
        "antfu.i18n-ally"
 | 
			
		||||
    ]
 | 
			
		||||
  "recommendations": [
 | 
			
		||||
    "2gua.rainbow-brackets",
 | 
			
		||||
    "christian-kohler.npm-intellisense",
 | 
			
		||||
    "remimarsal.prettier-now",
 | 
			
		||||
    "svelte.svelte-vscode",
 | 
			
		||||
    "lokalise.i18n-ally",
 | 
			
		||||
    "fivethree.vscode-svelte-snippets",
 | 
			
		||||
    "voorjaar.windicss-intellisense"
 | 
			
		||||
  ],
 | 
			
		||||
  "unwantedRecommendations": ["antfu.i18n-ally"]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										8
									
								
								.vscode/i18n-ally-custom-framework.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.vscode/i18n-ally-custom-framework.yml
									
									
									
									
										vendored
									
									
								
							@@ -1,7 +1,7 @@
 | 
			
		||||
languageIds:
 | 
			
		||||
    - javascript
 | 
			
		||||
    - svelte
 | 
			
		||||
    - html
 | 
			
		||||
  - javascript
 | 
			
		||||
  - svelte
 | 
			
		||||
  - html
 | 
			
		||||
monopoly: false
 | 
			
		||||
refactorTemplates:
 | 
			
		||||
    - "{$_('$1')}"
 | 
			
		||||
  - "{$_('$1')}"
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										6
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							@@ -1,5 +1,5 @@
 | 
			
		||||
{
 | 
			
		||||
    "i18n-ally.localesPaths": "src/locales",
 | 
			
		||||
    "i18n-ally.keystyle": "nested",
 | 
			
		||||
    "windicss.enableCodeFolding": false,
 | 
			
		||||
  "i18n-ally.localesPaths": "src/locales",
 | 
			
		||||
  "i18n-ally.keystyle": "nested",
 | 
			
		||||
  "windicss.enableCodeFolding": false
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1031
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										1031
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										20
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								Dockerfile
									
									
									
									
									
								
							@@ -1,14 +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
 | 
			
		||||
COPY package.json ./
 | 
			
		||||
RUN yarn
 | 
			
		||||
COPY package.json *.config.js index.html ./
 | 
			
		||||
 | 
			
		||||
COPY package.json pnpm-lock.yaml vite.config.js tailwind.config.cjs postcss.config.cjs index.html ./
 | 
			
		||||
RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@10.7
 | 
			
		||||
RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i
 | 
			
		||||
 | 
			
		||||
COPY src ./src
 | 
			
		||||
COPY public ./public
 | 
			
		||||
RUN yarn build
 | 
			
		||||
RUN pnpm build
 | 
			
		||||
 | 
			
		||||
# final image
 | 
			
		||||
FROM alpine
 | 
			
		||||
COPY --from=0 /app/dist /app
 | 
			
		||||
FROM fholzer/nginx-brotli:v1.19.1
 | 
			
		||||
COPY --from=1 /app /usr/share/nginx/html
 | 
			
		||||
FROM registry.odit.services/library/nginx-brotli:3.15 AS final
 | 
			
		||||
COPY --from=build /app/dist /usr/share/nginx/html
 | 
			
		||||
COPY ./nginx.conf /etc/nginx/nginx.conf
 | 
			
		||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							@@ -1,22 +1,27 @@
 | 
			
		||||
# @odit/lfk-frontend
 | 
			
		||||
 | 
			
		||||
## ✒️ Overview
 | 
			
		||||
 | 
			
		||||
This is an API client for [https://git.odit.services/lfk/backend](@lfk/backend)
 | 
			
		||||
- WebApp built with [Svelte](https://svelte.dev), [WindiCSS](https://windicss.org/) (to compile [TailwindCSS](https://tailwindcss.com/)) and [Vite](https://vitejs.dev).
 | 
			
		||||
 | 
			
		||||
This application is intended for use by admin users/ members only.
 | 
			
		||||
 | 
			
		||||
## 🚀 Getting Started
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
yarn
 | 
			
		||||
pnpm i
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Development
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
yarn dev
 | 
			
		||||
pnpm dev
 | 
			
		||||
/
 | 
			
		||||
yarn dev --open
 | 
			
		||||
pnpm dev --open
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Build
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
yarn build
 | 
			
		||||
pnpm build
 | 
			
		||||
```
 | 
			
		||||
@@ -1,8 +1,8 @@
 | 
			
		||||
version: "3"
 | 
			
		||||
services:
 | 
			
		||||
    httpd:
 | 
			
		||||
        build: .
 | 
			
		||||
        volumes:
 | 
			
		||||
            - ./public/env.sample.js:/usr/share/nginx/html/env.js
 | 
			
		||||
        ports:
 | 
			
		||||
            - 4050:80
 | 
			
		||||
  httpd:
 | 
			
		||||
    build: .
 | 
			
		||||
    volumes:
 | 
			
		||||
      - ./public/env.sample.js:/usr/share/nginx/html/env.js
 | 
			
		||||
    ports:
 | 
			
		||||
      - 4050:80
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										38
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										38
									
								
								index.html
									
									
									
									
									
								
							@@ -1,22 +1,22 @@
 | 
			
		||||
<!DOCTYPE html>
 | 
			
		||||
<!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>
 | 
			
		||||
 | 
			
		||||
<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-0.8.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>
 | 
			
		||||
 | 
			
		||||
  <body>
 | 
			
		||||
    <span style="display: none; visibility: hidden" id="buildinfo"
 | 
			
		||||
      >RELEASE_INFO-1.10.6-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>
 | 
			
		||||
@@ -6,6 +6,11 @@ http {
 | 
			
		||||
    server {
 | 
			
		||||
        error_page 404 /index.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';
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										28
									
								
								order.js
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								order.js
									
									
									
									
									
								
							@@ -1,16 +1,18 @@
 | 
			
		||||
const fs = require('fs');
 | 
			
		||||
import fs from "fs";
 | 
			
		||||
// get all language files
 | 
			
		||||
const files = fs.readdirSync('./src/locales/');
 | 
			
		||||
const files = fs.readdirSync("./src/locales/");
 | 
			
		||||
files.forEach((f) => {
 | 
			
		||||
	// read file as object
 | 
			
		||||
	const unordered = JSON.parse(fs.readFileSync(`src/locales/${f}`));
 | 
			
		||||
	// order object by keys alpabetically A-Z
 | 
			
		||||
	const ordered = Object.keys(unordered).sort().reduce((obj, key) => {
 | 
			
		||||
		obj[key] = unordered[key];
 | 
			
		||||
		return obj;
 | 
			
		||||
	}, {});
 | 
			
		||||
	// format output as json for commit diff compatibility
 | 
			
		||||
	const out = JSON.stringify(ordered, 0, 4);
 | 
			
		||||
	// write output file
 | 
			
		||||
	fs.writeFileSync(`src/locales/${f}`, out);
 | 
			
		||||
  // read file as object
 | 
			
		||||
  const unordered = JSON.parse(fs.readFileSync(`src/locales/${f}`));
 | 
			
		||||
  // order object by keys alpabetically A-Z
 | 
			
		||||
  const ordered = Object.keys(unordered)
 | 
			
		||||
    .sort()
 | 
			
		||||
    .reduce((obj, key) => {
 | 
			
		||||
      obj[key] = unordered[key];
 | 
			
		||||
      return obj;
 | 
			
		||||
    }, {});
 | 
			
		||||
  // format output as json for commit diff compatibility
 | 
			
		||||
  const out = JSON.stringify(ordered, 0, 4);
 | 
			
		||||
  // write output file
 | 
			
		||||
  fs.writeFileSync(`src/locales/${f}`, out);
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										117
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										117
									
								
								package.json
									
									
									
									
									
								
							@@ -1,56 +1,65 @@
 | 
			
		||||
{
 | 
			
		||||
	"name": "@odit/lfk-frontend",
 | 
			
		||||
	"version": "0.8.4",
 | 
			
		||||
	"scripts": {
 | 
			
		||||
		"i18n-order": "node order.js",
 | 
			
		||||
		"dev": "vite",
 | 
			
		||||
		"build": "vite build",
 | 
			
		||||
		"release": "release-it",
 | 
			
		||||
		"licenses:export": "license-exporter --json -o public"
 | 
			
		||||
	},
 | 
			
		||||
	"license": "CC-BY-NC-SA-4.0",
 | 
			
		||||
	"devDependencies": {
 | 
			
		||||
		"check-password-strength": "2.0.2",
 | 
			
		||||
		"@odit/lfk-client-js": "0.10.1",
 | 
			
		||||
		"@odit/license-exporter": "0.0.11",
 | 
			
		||||
		"@sveltejs/vite-plugin-svelte": "1.0.0-next.5",
 | 
			
		||||
		"@types/html-minifier": "4.0.0",
 | 
			
		||||
		"auto-changelog": "2.2.1",
 | 
			
		||||
		"autoprefixer": "10.2.5",
 | 
			
		||||
		"csvtojson": "2.0.10",
 | 
			
		||||
		"gridjs": "3.4.0",
 | 
			
		||||
		"html-minifier": "4.0.0",
 | 
			
		||||
		"localforage": "1.9.0",
 | 
			
		||||
		"marked": "2.0.1",
 | 
			
		||||
		"release-it": "14.5.1",
 | 
			
		||||
		"svelte": "3.37.0",
 | 
			
		||||
		"svelte-focus-trap": "1.2.0",
 | 
			
		||||
		"svelte-i18n": "3.3.9",
 | 
			
		||||
		"svelte-preprocess": "4.7.0",
 | 
			
		||||
		"svelte-select": "3.17.0",
 | 
			
		||||
		"tailwindcss": "2.0.4",
 | 
			
		||||
		"tinro": "0.6.1",
 | 
			
		||||
		"toastify-js": "1.10.0",
 | 
			
		||||
		"validator": "13.5.2",
 | 
			
		||||
		"vite": "2.1.5",
 | 
			
		||||
		"vite-plugin-windicss": "0.12.2",
 | 
			
		||||
		"xlsx": "0.16.9"
 | 
			
		||||
	},
 | 
			
		||||
	"release-it": {
 | 
			
		||||
		"git": {
 | 
			
		||||
			"commit": true,
 | 
			
		||||
			"requireCleanWorkingDir": false,
 | 
			
		||||
			"commitMessage": "🚀RELEASE v${version}",
 | 
			
		||||
			"push": false,
 | 
			
		||||
			"tag": true,
 | 
			
		||||
			"tagName": null,
 | 
			
		||||
			"tagAnnotation": "v${version}"
 | 
			
		||||
		},
 | 
			
		||||
		"npm": {
 | 
			
		||||
			"publish": false
 | 
			
		||||
		},
 | 
			
		||||
		"hooks": {
 | 
			
		||||
			"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.html && node order.js  && git add src/locales"
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
  "name": "@odit/lfk-frontend",
 | 
			
		||||
  "version": "1.10.6",
 | 
			
		||||
  "type": "module",
 | 
			
		||||
  "scripts": {
 | 
			
		||||
    "i18n-order": "node order.js",
 | 
			
		||||
    "dev": "vite",
 | 
			
		||||
    "format": "prettier --write --plugin-search-dir=. .",
 | 
			
		||||
    "build": "vite build",
 | 
			
		||||
    "release": "release-it",
 | 
			
		||||
    "licenses:export": "license-exporter --json -o public"
 | 
			
		||||
  },
 | 
			
		||||
  "license": "CC-BY-NC-SA-4.0",
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@odit/license-exporter": "0.2.0",
 | 
			
		||||
    "@sveltejs/vite-plugin-svelte": "2.1.1",
 | 
			
		||||
    "auto-changelog": "2.5.0",
 | 
			
		||||
    "autoprefixer": "10.4.21",
 | 
			
		||||
    "postcss": "8.5.3",
 | 
			
		||||
    "prettier": "3.5.3",
 | 
			
		||||
    "prettier-plugin-svelte": "3.3.3",
 | 
			
		||||
    "release-it": "17.10.0",
 | 
			
		||||
    "svelte-select": "3.17.0",
 | 
			
		||||
    "tailwindcss": "3.4.15",
 | 
			
		||||
    "vite": "4.3.3"
 | 
			
		||||
  },
 | 
			
		||||
  "release-it": {
 | 
			
		||||
    "git": {
 | 
			
		||||
      "commit": true,
 | 
			
		||||
      "requireCleanWorkingDir": false,
 | 
			
		||||
      "commitMessage": "chore(release): ${version}",
 | 
			
		||||
      "push": true,
 | 
			
		||||
      "tag": true,
 | 
			
		||||
      "tagName": "${version}",
 | 
			
		||||
      "tagAnnotation": "${version}"
 | 
			
		||||
    },
 | 
			
		||||
    "npm": {
 | 
			
		||||
      "publish": false
 | 
			
		||||
    },
 | 
			
		||||
    "hooks": {
 | 
			
		||||
      "after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node versionbuilder.js  && git add index.html && node order.js  && git add src/locales"
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "@fontsource/athiti": "^5.2.5",
 | 
			
		||||
    "@odit/lfk-client-js": "1.2.4",
 | 
			
		||||
    "@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",
 | 
			
		||||
    "html5-qrcode": "^2.3.8",
 | 
			
		||||
    "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"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										3761
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										3761
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										3
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
onlyBuiltDependencies:
 | 
			
		||||
  - es5-ext
 | 
			
		||||
  - esbuild
 | 
			
		||||
							
								
								
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								postcss.config.cjs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,6 @@
 | 
			
		||||
module.exports = {
 | 
			
		||||
  plugins: {
 | 
			
		||||
    tailwindcss: {},
 | 
			
		||||
    autoprefixer: {},
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								public/beep.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/beep.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							@@ -1,8 +1,12 @@
 | 
			
		||||
const config = {
 | 
			
		||||
	baseurl: 'http://localhost:4010',
 | 
			
		||||
	documentserver_key: 'NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe',
 | 
			
		||||
	// optional
 | 
			
		||||
	default_username: 'demo',
 | 
			
		||||
	default_password: 'demo',
 | 
			
		||||
	prefersHashRouting: true
 | 
			
		||||
  baseurl: "http://localhost:4010",
 | 
			
		||||
  baseurl_selfservice: "http://localhost:5174",
 | 
			
		||||
  baseurl_documentserver: "http://localhost:4010/documents",
 | 
			
		||||
  documentserver_key:
 | 
			
		||||
    "NqZSYTy5AFQ7MppbLW5moqpTk7u7YrNUHKYhKYuThnnya2WpCOIU694hIZT1FzYe",
 | 
			
		||||
  // optional
 | 
			
		||||
  default_username: "demo",
 | 
			
		||||
  default_password: "demo",
 | 
			
		||||
  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  | 
@@ -1,34 +1,39 @@
 | 
			
		||||
{
 | 
			
		||||
	"name": "Lauf für Kaya! - Admin",
 | 
			
		||||
	"short_name": "LfK!Admin",
 | 
			
		||||
	"start_url": "/?utm_source=pwa",
 | 
			
		||||
	"orientation": "portrait-primary",
 | 
			
		||||
	"display": "standalone",
 | 
			
		||||
	"background_color": "#fff",
 | 
			
		||||
	"theme_color": "#fff",
 | 
			
		||||
	"description": "Lauf für Kaya! - Admin",
 | 
			
		||||
	"shortcuts": [
 | 
			
		||||
		{
 | 
			
		||||
			"name": "Users",
 | 
			
		||||
			"url": "/users/?utm_source=pwa"
 | 
			
		||||
		}
 | 
			
		||||
	],
 | 
			
		||||
	"icons": [
 | 
			
		||||
		{
 | 
			
		||||
			"src": "/favicon.png",
 | 
			
		||||
			"sizes": "48x48",
 | 
			
		||||
			"type": "image/png"
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			"src": "/favicon.png",
 | 
			
		||||
			"sizes": "144x144",
 | 
			
		||||
			"type": "image/png"
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			"src": "/lfk-logo.png",
 | 
			
		||||
			"sizes": "1540x144",
 | 
			
		||||
			"type": "image/png"
 | 
			
		||||
		},
 | 
			
		||||
		{ "src": "/maskable_icon_x1.png", "sizes": "750x750", "type": "image/png", "purpose": "any maskable" }
 | 
			
		||||
	]
 | 
			
		||||
  "name": "Lauf für Kaya! - Admin",
 | 
			
		||||
  "short_name": "LfK!Admin",
 | 
			
		||||
  "start_url": "/?utm_source=pwa",
 | 
			
		||||
  "orientation": "portrait-primary",
 | 
			
		||||
  "display": "standalone",
 | 
			
		||||
  "background_color": "#fff",
 | 
			
		||||
  "theme_color": "#fff",
 | 
			
		||||
  "description": "Lauf für Kaya! - Admin",
 | 
			
		||||
  "shortcuts": [
 | 
			
		||||
    {
 | 
			
		||||
      "name": "Users",
 | 
			
		||||
      "url": "/users/?utm_source=pwa"
 | 
			
		||||
    }
 | 
			
		||||
  ],
 | 
			
		||||
  "icons": [
 | 
			
		||||
    {
 | 
			
		||||
      "src": "/favicon.png",
 | 
			
		||||
      "sizes": "48x48",
 | 
			
		||||
      "type": "image/png"
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      "src": "/favicon.png",
 | 
			
		||||
      "sizes": "144x144",
 | 
			
		||||
      "type": "image/png"
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      "src": "/lfk-logo.png",
 | 
			
		||||
      "sizes": "1540x144",
 | 
			
		||||
      "type": "image/png"
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      "src": "/maskable_icon_x1.png",
 | 
			
		||||
      "sizes": "750x750",
 | 
			
		||||
      "type": "image/png",
 | 
			
		||||
      "purpose": "any maskable"
 | 
			
		||||
    }
 | 
			
		||||
  ]
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,4 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import "toastify-js/src/toastify.css";
 | 
			
		||||
  import "gridjs/dist/theme/mermaid.css";
 | 
			
		||||
  import { Route, router } from "tinro";
 | 
			
		||||
  router.subscribe((routeInfo) => {
 | 
			
		||||
    window.scrollTo(0, 0);
 | 
			
		||||
@@ -24,11 +22,10 @@
 | 
			
		||||
    name: "lfk_admin",
 | 
			
		||||
    version: 1.0,
 | 
			
		||||
    storeName: "lfk_admin",
 | 
			
		||||
    description: "LfK! admin dashbaord",
 | 
			
		||||
    description: "LfK! admin dashboard",
 | 
			
		||||
  });
 | 
			
		||||
  window.onunhandledrejection = (event) => {
 | 
			
		||||
    if (event.reason.toString() == "Error: Unauthorized") {
 | 
			
		||||
      console.log("Found 1");
 | 
			
		||||
      localForage.clear();
 | 
			
		||||
      location.replace("/");
 | 
			
		||||
    }
 | 
			
		||||
@@ -44,6 +41,7 @@
 | 
			
		||||
  import Settings from "./components/settings/Settings.svelte";
 | 
			
		||||
  import Transition from "./components/base/Transition.svelte";
 | 
			
		||||
  import Orgs from "./components/orgs/Orgs.svelte";
 | 
			
		||||
  import CardAssignment from "./components/general/CardAssignment.svelte";
 | 
			
		||||
  import Runners from "./components/runners/Runners.svelte";
 | 
			
		||||
  import Footer from "./components/general/Footer.svelte";
 | 
			
		||||
  import TracksOverview from "./components/tracks/TracksOverview.svelte";
 | 
			
		||||
@@ -72,27 +70,29 @@
 | 
			
		||||
  import Scans from "./components/scans/Scans.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();
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<Route>
 | 
			
		||||
  {#if $router.path === '/forgot_password'}
 | 
			
		||||
  {#if $router.path === "/forgot_password"}
 | 
			
		||||
    <Route path="/forgot_password">
 | 
			
		||||
      <ForgotPassword />
 | 
			
		||||
    </Route>
 | 
			
		||||
  {:else if $router.path.includes('/reset')}
 | 
			
		||||
  {:else if $router.path.includes("/reset")}
 | 
			
		||||
    <Route path="/reset/:resetkey" let:params>
 | 
			
		||||
      <ResetPassword {params} />
 | 
			
		||||
    </Route>
 | 
			
		||||
  {:else if $router.path === '/about'}
 | 
			
		||||
  {:else if $router.path === "/about"}
 | 
			
		||||
    <Route path="/about">
 | 
			
		||||
      <About />
 | 
			
		||||
    </Route>
 | 
			
		||||
  {:else if $router.path === '/imprint'}
 | 
			
		||||
  {:else if $router.path === "/imprint"}
 | 
			
		||||
    <Route path="/imprint">
 | 
			
		||||
      <Imprint />
 | 
			
		||||
    </Route>
 | 
			
		||||
  {:else if $router.path === '/privacy'}
 | 
			
		||||
  {:else if $router.path === "/privacy"}
 | 
			
		||||
    <Route path="/privacy">
 | 
			
		||||
      <Privacy />
 | 
			
		||||
    </Route>
 | 
			
		||||
@@ -136,12 +136,17 @@
 | 
			
		||||
        </Route>
 | 
			
		||||
        <Route path="/runners/*">
 | 
			
		||||
          <Route path="/">
 | 
			
		||||
            <Runners />
 | 
			
		||||
            <Runners created_via="all" />
 | 
			
		||||
          </Route>
 | 
			
		||||
          <Route path="/:runnerid" let:params>
 | 
			
		||||
            <RunnerDetail {params} />
 | 
			
		||||
          </Route>
 | 
			
		||||
        </Route>
 | 
			
		||||
        <Route path="/cardassignment/*">
 | 
			
		||||
          <Route path="/">
 | 
			
		||||
            <CardAssignment />
 | 
			
		||||
          </Route>
 | 
			
		||||
        </Route>
 | 
			
		||||
        <Route path="/teams/*">
 | 
			
		||||
          <Route path="/">
 | 
			
		||||
            <Teams />
 | 
			
		||||
@@ -206,6 +211,14 @@
 | 
			
		||||
            <ScanStationDetail {params} />
 | 
			
		||||
          </Route>
 | 
			
		||||
        </Route>
 | 
			
		||||
        <Route path="/statsclients/*">
 | 
			
		||||
          <Route path="/">
 | 
			
		||||
            <StatsClients />
 | 
			
		||||
          </Route>
 | 
			
		||||
          <Route path="/:clientid" let:params>
 | 
			
		||||
            <StatsClientDetail {params} />
 | 
			
		||||
          </Route>
 | 
			
		||||
        </Route>
 | 
			
		||||
        <Route path="/about">
 | 
			
		||||
          <About />
 | 
			
		||||
        </Route>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,28 +1,22 @@
 | 
			
		||||
<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 Toastify from "toastify-js";
 | 
			
		||||
  import "toastify-js/src/toastify.css";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
 | 
			
		||||
  let reset_mail_sent = false;
 | 
			
		||||
  let usersEmail = "";
 | 
			
		||||
  function reset() {
 | 
			
		||||
    if (isEmail(usersEmail)) {
 | 
			
		||||
      toast.loading($_("mail-validation-in-progress"));
 | 
			
		||||
      AuthService.authControllerGetResetToken("de", { email: usersEmail })
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("mail-validation-in-progress"),
 | 
			
		||||
            duration: 3500,
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          reset_mail_sent = true;
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("invalid-mail-reset"),
 | 
			
		||||
        duration: 3500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast($_("invalid-mail-reset"));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
</script>
 | 
			
		||||
@@ -32,17 +26,18 @@
 | 
			
		||||
    <div class="max-w-md w-full py-12 px-6">
 | 
			
		||||
      <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">
 | 
			
		||||
        {$_('application_name')}
 | 
			
		||||
        {$_("application_name")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <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>
 | 
			
		||||
      <div class="mt-6">
 | 
			
		||||
        <div class="mt-6">
 | 
			
		||||
          <a
 | 
			
		||||
            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')}
 | 
			
		||||
            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>
 | 
			
		||||
@@ -53,25 +48,26 @@
 | 
			
		||||
    <div class="max-w-md w-full py-12 px-6">
 | 
			
		||||
      <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">
 | 
			
		||||
        {$_('application_name')}
 | 
			
		||||
        {$_("application_name")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <p class="mt-6 text-sm text-center text-gray-900">
 | 
			
		||||
        {$_('forgot_password')}
 | 
			
		||||
        {$_("forgot_password")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <p class="mt-2 mb-2 text-sm text-center text-gray-900">
 | 
			
		||||
        {$_('dont-panic-were-resetting-it')}
 | 
			
		||||
        {$_("dont-panic-were-resetting-it")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <div>
 | 
			
		||||
        <div class="rounded-md shadow-sm">
 | 
			
		||||
          <div>
 | 
			
		||||
            <input
 | 
			
		||||
              aria-label={$_('e-mail-adress')}
 | 
			
		||||
              aria-label={$_("e-mail-adress")}
 | 
			
		||||
              name="email"
 | 
			
		||||
              type="email"
 | 
			
		||||
              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"
 | 
			
		||||
              placeholder={$_('e-mail-adress')}
 | 
			
		||||
              bind:value={usersEmail} />
 | 
			
		||||
              placeholder={$_("e-mail-adress")}
 | 
			
		||||
              bind:value={usersEmail}
 | 
			
		||||
            />
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@@ -79,19 +75,22 @@
 | 
			
		||||
          <button
 | 
			
		||||
            on:click={reset}
 | 
			
		||||
            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">
 | 
			
		||||
              <svg
 | 
			
		||||
                class="h-5 w-5 text-gray-500"
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                viewBox="0 0 20 20">
 | 
			
		||||
                viewBox="0 0 20 20"
 | 
			
		||||
              >
 | 
			
		||||
                <path
 | 
			
		||||
                  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"
 | 
			
		||||
                  clip-rule="evenodd" />
 | 
			
		||||
                  clip-rule="evenodd"
 | 
			
		||||
                />
 | 
			
		||||
              </svg>
 | 
			
		||||
            </span>
 | 
			
		||||
            {$_('reset-my-password')}
 | 
			
		||||
            {$_("reset-my-password")}
 | 
			
		||||
          </button>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="mt-6">
 | 
			
		||||
@@ -100,24 +99,30 @@
 | 
			
		||||
              <div class="w-full border-t border-gray-300" />
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="relative flex justify-center text-sm">
 | 
			
		||||
              <span
 | 
			
		||||
                class="px-2 bg-gray-100 text-gray-500">{$_('dont-have-your-email-connected')}</span>
 | 
			
		||||
              <span class="px-2 bg-gray-100 text-gray-500"
 | 
			
		||||
                >{$_("dont-have-your-email-connected")}</span
 | 
			
		||||
              >
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
          <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">
 | 
			
		||||
            <a
 | 
			
		||||
              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">
 | 
			
		||||
              {$_('send-a-mail-to-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"
 | 
			
		||||
            >
 | 
			
		||||
              {$_("send-a-mail-to-lfk-odit-services")}
 | 
			
		||||
            </a>
 | 
			
		||||
          </div>
 | 
			
		||||
          <div class="mt-6">
 | 
			
		||||
            <a
 | 
			
		||||
              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>
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@
 | 
			
		||||
  import { OpenAPI, AuthService } from "@odit/lfk-client-js";
 | 
			
		||||
  import Footer from "../general/Footer.svelte";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  // ------
 | 
			
		||||
  let username = config.default_username || "";
 | 
			
		||||
  let password = config.default_password || "";
 | 
			
		||||
@@ -20,11 +20,6 @@
 | 
			
		||||
        OpenAPI.TOKEN = value.access_token;
 | 
			
		||||
        const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1]));
 | 
			
		||||
        store.login(value, jwtinfo);
 | 
			
		||||
        Toastify({
 | 
			
		||||
          text: $_("welcome_wavinghand"),
 | 
			
		||||
          duration: 500,
 | 
			
		||||
          backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
        }).showToast();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
@@ -33,10 +28,7 @@
 | 
			
		||||
    // prevent login button spamming
 | 
			
		||||
    if (last_loginclick_processed && is_blocked_by_autologin === false) {
 | 
			
		||||
      last_loginclick_processed = false;
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("login_is_checked"),
 | 
			
		||||
        duration: 500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("login_is_checked"));
 | 
			
		||||
      let postdata = {};
 | 
			
		||||
      if (isEmail(username)) {
 | 
			
		||||
        postdata = {
 | 
			
		||||
@@ -56,31 +48,18 @@
 | 
			
		||||
          const jwtinfo = JSON.parse(atob(OpenAPI.TOKEN.split(".")[1]));
 | 
			
		||||
          store.login(result.access_token, jwtinfo);
 | 
			
		||||
          location.replace("/");
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("welcome_wavinghand"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("error_on_login"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor:
 | 
			
		||||
              "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.error($_("error_on_login"));
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          last_loginclick_processed = true;
 | 
			
		||||
        });
 | 
			
		||||
      // last login was not processed yet
 | 
			
		||||
    } else {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: "chill...",
 | 
			
		||||
        duration: 1500,
 | 
			
		||||
        backgroundColor:
 | 
			
		||||
          "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast($_("please-wait-a-moment-your-login-is-still-being-processed"));
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
  function handleKeydown(e) {
 | 
			
		||||
@@ -91,34 +70,37 @@
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<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">
 | 
			
		||||
    <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-sm text-center">{$_('log_in_to_your_account')}</p>
 | 
			
		||||
    <p class="mt-6 text-xl text-center font-bold">{$_("application_name")}</p>
 | 
			
		||||
    <p class="mt-2 mb-6 text-sm text-center">{$_("log_in_to_your_account")}</p>
 | 
			
		||||
    <div>
 | 
			
		||||
      <div class="rounded-md shadow-sm">
 | 
			
		||||
        <div>
 | 
			
		||||
          <!-- svelte-ignore a11y-autofocus -->
 | 
			
		||||
          <input
 | 
			
		||||
            autofocus
 | 
			
		||||
            aria-label={$_('email_address_or_username')}
 | 
			
		||||
            aria-label={$_("email_address_or_username")}
 | 
			
		||||
            type="text"
 | 
			
		||||
            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"
 | 
			
		||||
            on:keydown={handleKeydown}
 | 
			
		||||
            placeholder={$_('email_address_or_username')}
 | 
			
		||||
            bind:value={username} />
 | 
			
		||||
            placeholder={$_("email_address_or_username")}
 | 
			
		||||
            bind:value={username}
 | 
			
		||||
          />
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="-mt-px relative">
 | 
			
		||||
          <input
 | 
			
		||||
            aria-label={$_('password')}
 | 
			
		||||
            aria-label={$_("password")}
 | 
			
		||||
            type="password"
 | 
			
		||||
            required=""
 | 
			
		||||
            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"
 | 
			
		||||
            on:keydown={handleKeydown}
 | 
			
		||||
            placeholder={$_('password')} />
 | 
			
		||||
            placeholder={$_("password")}
 | 
			
		||||
          />
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
@@ -126,29 +108,33 @@
 | 
			
		||||
        <button
 | 
			
		||||
          on:click={login}
 | 
			
		||||
          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">
 | 
			
		||||
            <svg
 | 
			
		||||
              class="h-5 w-5 text-gray-500"
 | 
			
		||||
              fill="currentColor"
 | 
			
		||||
              viewBox="0 0 20 20">
 | 
			
		||||
              viewBox="0 0 20 20"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                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"
 | 
			
		||||
                clip-rule="evenodd" />
 | 
			
		||||
                clip-rule="evenodd"
 | 
			
		||||
              />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </span>
 | 
			
		||||
          {$_('log_in')}
 | 
			
		||||
          {$_("log_in")}
 | 
			
		||||
        </button>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mt-2">
 | 
			
		||||
    <!-- <div class="mt-2">
 | 
			
		||||
      <a
 | 
			
		||||
        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">
 | 
			
		||||
        {$_('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"
 | 
			
		||||
      >
 | 
			
		||||
        {$_("forgot_password")}
 | 
			
		||||
      </a>
 | 
			
		||||
    </div>
 | 
			
		||||
    </div> -->
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
<Footer />
 | 
			
		||||
 
 | 
			
		||||
@@ -1,52 +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
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
  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;
 | 
			
		||||
  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;
 | 
			
		||||
  $: 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>
 | 
			
		||||
  <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,8 +1,7 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { AuthService } from "@odit/lfk-client-js";
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import "toastify-js/src/toastify.css";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  import PasswordStrength, {
 | 
			
		||||
    password_strong_enough,
 | 
			
		||||
  } from "../auth/PasswordStrength.svelte";
 | 
			
		||||
@@ -11,101 +10,97 @@
 | 
			
		||||
  export let params;
 | 
			
		||||
  function set_new_password() {
 | 
			
		||||
    if (password.trim() !== "") {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("password-reset-in-progress"),
 | 
			
		||||
        duration: 3500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("password-reset-in-progress"));
 | 
			
		||||
      AuthService.authControllerResetPassword(atob(params.resetkey), {
 | 
			
		||||
        password,
 | 
			
		||||
      })
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("password-reset-successful"),
 | 
			
		||||
            duration: 3500,
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast($_("password-reset-successful"));
 | 
			
		||||
          state = "reset_success";
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          state = "reset_error";
 | 
			
		||||
        });
 | 
			
		||||
    } else {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("please-provide-a-password"),
 | 
			
		||||
        duration: 3500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.dismiss();
 | 
			
		||||
      toast.error($_("please-provide-a-password"));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if state === 'reset_success'}
 | 
			
		||||
{#if state === "reset_success"}
 | 
			
		||||
  <div class="min-h-screen flex items-center justify-center bg-gray-100">
 | 
			
		||||
    <div class="max-w-md w-full py-12 px-6">
 | 
			
		||||
      <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">
 | 
			
		||||
        {$_('application_name')}
 | 
			
		||||
        {$_("application_name")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold">
 | 
			
		||||
        {$_('successful-password-reset')}
 | 
			
		||||
        {$_("successful-password-reset")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <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>
 | 
			
		||||
      <div class="mt-6">
 | 
			
		||||
        <div class="mt-6">
 | 
			
		||||
          <a
 | 
			
		||||
            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">
 | 
			
		||||
            {$_('go-to-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"
 | 
			
		||||
          >
 | 
			
		||||
            {$_("go-to-login")}
 | 
			
		||||
          </a>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
{:else if state === 'reset_error'}
 | 
			
		||||
{:else if state === "reset_error"}
 | 
			
		||||
  <div class="min-h-screen flex items-center justify-center bg-gray-100">
 | 
			
		||||
    <div class="max-w-md w-full py-12 px-6">
 | 
			
		||||
      <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">
 | 
			
		||||
        {$_('application_name')}
 | 
			
		||||
        {$_("application_name")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <p class="mt-2 mb-2 text-sm text-center text-gray-900 font-bold">
 | 
			
		||||
        {$_('password-reset-failed')}
 | 
			
		||||
        {$_("password-reset-failed")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <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>
 | 
			
		||||
      <div class="mt-6">
 | 
			
		||||
        <div class="mt-6">
 | 
			
		||||
          <a
 | 
			
		||||
            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">
 | 
			
		||||
            {$_('request-a-new-reset-mail')}
 | 
			
		||||
            class="text-center relative block w-full py-2 px-3 border border-transparent rounded-md text-white font-semibold bg-gray-800 hover:bg-gray-700 focus:bg-gray-900 focus:outline-none focus:shadow-outline sm:text-sm"
 | 
			
		||||
          >
 | 
			
		||||
            {$_("request-a-new-reset-mail")}
 | 
			
		||||
          </a>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
{:else if state === 'reset_in_progress'}
 | 
			
		||||
{:else if state === "reset_in_progress"}
 | 
			
		||||
  <div class="min-h-screen flex items-center justify-center bg-gray-100">
 | 
			
		||||
    <div class="max-w-md w-full py-12 px-6">
 | 
			
		||||
      <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">
 | 
			
		||||
        {$_('application_name')}
 | 
			
		||||
        {$_("application_name")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <p class="mt-2 mb-4 text-md text-center text-gray-900">
 | 
			
		||||
        {$_('reset-password')}
 | 
			
		||||
        {$_("reset-password")}
 | 
			
		||||
      </p>
 | 
			
		||||
      <div>
 | 
			
		||||
        <div class="rounded-md shadow-sm">
 | 
			
		||||
          <div>
 | 
			
		||||
            <input
 | 
			
		||||
              aria-label={$_('new-password')}
 | 
			
		||||
              aria-label={$_("new-password")}
 | 
			
		||||
              name="password"
 | 
			
		||||
              type="password"
 | 
			
		||||
              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"
 | 
			
		||||
              placeholder={$_('new-password')}
 | 
			
		||||
              bind:value={password} />
 | 
			
		||||
              placeholder={$_("new-password")}
 | 
			
		||||
              bind:value={password}
 | 
			
		||||
            />
 | 
			
		||||
          </div>
 | 
			
		||||
          <PasswordStrength bind:password_change={password} />
 | 
			
		||||
        </div>
 | 
			
		||||
@@ -116,19 +111,22 @@
 | 
			
		||||
            disabled={!password_strong_enough(password)}
 | 
			
		||||
            class:opacity-50={!password_strong_enough(password)}
 | 
			
		||||
            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">
 | 
			
		||||
              <svg
 | 
			
		||||
                class="h-5 w-5 text-gray-500"
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                viewBox="0 0 20 20">
 | 
			
		||||
                viewBox="0 0 20 20"
 | 
			
		||||
              >
 | 
			
		||||
                <path
 | 
			
		||||
                  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"
 | 
			
		||||
                  clip-rule="evenodd" />
 | 
			
		||||
                  clip-rule="evenodd"
 | 
			
		||||
                />
 | 
			
		||||
              </svg>
 | 
			
		||||
            </span>
 | 
			
		||||
            {$_('reset-my-password')}
 | 
			
		||||
            {$_("reset-my-password")}
 | 
			
		||||
          </button>
 | 
			
		||||
        </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>
 | 
			
		||||
@@ -4,19 +4,22 @@
 | 
			
		||||
 | 
			
		||||
<body class="antialiased font-sans">
 | 
			
		||||
  <div class="flex min-h-screen">
 | 
			
		||||
    <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="text-black text-5xl md:text-15xl font-black">
 | 
			
		||||
          {$_('internal-error')}
 | 
			
		||||
          {$_("internal-error")}
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="w-16 h-1 bg-purple-light my-3 md:my-6" />
 | 
			
		||||
        <p
 | 
			
		||||
          class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal">
 | 
			
		||||
          {$_('generic-ui-logic-error')}
 | 
			
		||||
          class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal"
 | 
			
		||||
        >
 | 
			
		||||
          {$_("generic-ui-logic-error")}
 | 
			
		||||
        </p>
 | 
			
		||||
        <a
 | 
			
		||||
          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>
 | 
			
		||||
 
 | 
			
		||||
@@ -5,7 +5,7 @@
 | 
			
		||||
 | 
			
		||||
<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>
 | 
			
		||||
    <b class="capitalize">{$_("general_promise_error")}</b>
 | 
			
		||||
    {error}
 | 
			
		||||
  </span>
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,24 +1,25 @@
 | 
			
		||||
export function getlang(langkeys) {
 | 
			
		||||
	return {
 | 
			
		||||
		search: {
 | 
			
		||||
			placeholder: langkeys.search
 | 
			
		||||
		},
 | 
			
		||||
		sort: {
 | 
			
		||||
			sortAsc: langkeys.sort_column_ascending,
 | 
			
		||||
			sortDesc: langkeys.sort_column_descending
 | 
			
		||||
		},
 | 
			
		||||
		pagination: {
 | 
			
		||||
			previous: langkeys.previous,
 | 
			
		||||
			next: langkeys.next,
 | 
			
		||||
			navigate: (page, pages) => `${langkeys.page} ${page} ${langkeys.of} ${pages}`,
 | 
			
		||||
			page: (page) => `${langkeys.page} ${page}`,
 | 
			
		||||
			showing: langkeys.showing,
 | 
			
		||||
			of: langkeys.of,
 | 
			
		||||
			to: langkeys.to,
 | 
			
		||||
			results: langkeys.records
 | 
			
		||||
		},
 | 
			
		||||
		loading: langkeys.loading,
 | 
			
		||||
		noRecordsFound: langkeys.no_matching_records_found,
 | 
			
		||||
		error: langkeys.an_error_happened_while_fetching_the_data
 | 
			
		||||
	};
 | 
			
		||||
  return {
 | 
			
		||||
    search: {
 | 
			
		||||
      placeholder: langkeys.search,
 | 
			
		||||
    },
 | 
			
		||||
    sort: {
 | 
			
		||||
      sortAsc: langkeys.sort_column_ascending,
 | 
			
		||||
      sortDesc: langkeys.sort_column_descending,
 | 
			
		||||
    },
 | 
			
		||||
    pagination: {
 | 
			
		||||
      previous: langkeys.previous,
 | 
			
		||||
      next: langkeys.next,
 | 
			
		||||
      navigate: (page, pages) =>
 | 
			
		||||
        `${langkeys.page} ${page} ${langkeys.of} ${pages}`,
 | 
			
		||||
      page: (page) => `${langkeys.page} ${page}`,
 | 
			
		||||
      showing: langkeys.showing,
 | 
			
		||||
      of: langkeys.of,
 | 
			
		||||
      to: langkeys.to,
 | 
			
		||||
      results: langkeys.records,
 | 
			
		||||
    },
 | 
			
		||||
    loading: langkeys.loading,
 | 
			
		||||
    noRecordsFound: langkeys.no_matching_records_found,
 | 
			
		||||
    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 */
 | 
			
		||||
export function clickOutside(node) {
 | 
			
		||||
	const handleClick = (event) => {
 | 
			
		||||
		if (event.target.getAttribute('data-id') === 'modal_backdrop') {
 | 
			
		||||
			node.dispatchEvent(new CustomEvent('click_outside', node));
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
	document.removeEventListener('click', handleClick, true);
 | 
			
		||||
	document.addEventListener('click', handleClick, true);
 | 
			
		||||
  const handleClick = (event) => {
 | 
			
		||||
    if (event.target.getAttribute("data-id") === "modal_backdrop") {
 | 
			
		||||
      node.dispatchEvent(new CustomEvent("click_outside", node));
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
  document.removeEventListener("click", handleClick, true);
 | 
			
		||||
  document.addEventListener("click", handleClick, true);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@@ -1,14 +1,16 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
 | 
			
		||||
  import { RunnerCardService } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { createEventDispatcher } from "svelte";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  import DocumentServer from "../pdf_generation/DocumentServer";
 | 
			
		||||
  export let bulk_modal_open;
 | 
			
		||||
  export let current_cards;
 | 
			
		||||
  function focus(el) {
 | 
			
		||||
    el.focus();
 | 
			
		||||
  }
 | 
			
		||||
  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;
 | 
			
		||||
@@ -30,27 +32,20 @@
 | 
			
		||||
  function submit_without_print() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_("creating-blanco-cards"),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, false)
 | 
			
		||||
      toast.loading($_("creating-blanco-cards"));
 | 
			
		||||
      RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true)
 | 
			
		||||
        .then((result) => {
 | 
			
		||||
          bulk_modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("created-blanco-cards"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("created-blanco-cards"));
 | 
			
		||||
          dispatch("created", { cards: result });
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          //
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -58,47 +53,17 @@
 | 
			
		||||
  function submit_with_print() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_("creating-blanco-cards"),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.dismiss();
 | 
			
		||||
      toast.loading($_("creating-blanco-cards"));
 | 
			
		||||
      RunnerCardService.runnerCardControllerPostBlancoBulk(card_count, true)
 | 
			
		||||
        .then((result) => {
 | 
			
		||||
          bulk_modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("created-blanco-cards"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          current_cards = current_cards.concat(result);
 | 
			
		||||
          const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          fetch(
 | 
			
		||||
            `${config.baseurl}/documents/cards?&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
            {
 | 
			
		||||
              method: "POST",
 | 
			
		||||
              headers: {
 | 
			
		||||
                "Content-Type": "application/json",
 | 
			
		||||
              },
 | 
			
		||||
              body: JSON.stringify(result),
 | 
			
		||||
            }
 | 
			
		||||
          )
 | 
			
		||||
            .then((response) => {
 | 
			
		||||
              if (response.status != "200") {
 | 
			
		||||
                toast.hideToast();
 | 
			
		||||
                Toastify({
 | 
			
		||||
                  text: $_("pdf-generation-failed"),
 | 
			
		||||
                  duration: 3500,
 | 
			
		||||
                  backgroundColor:
 | 
			
		||||
                    "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                }).showToast();
 | 
			
		||||
              } else {
 | 
			
		||||
                return response.blob();
 | 
			
		||||
              }
 | 
			
		||||
            })
 | 
			
		||||
          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");
 | 
			
		||||
@@ -107,12 +72,8 @@
 | 
			
		||||
              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();
 | 
			
		||||
              toast.dismiss();
 | 
			
		||||
              toast.success($_("pdf-successfully-generated"));
 | 
			
		||||
            })
 | 
			
		||||
            .catch((err) => {
 | 
			
		||||
              console.error(err);
 | 
			
		||||
@@ -123,8 +84,6 @@
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -132,56 +91,68 @@
 | 
			
		||||
 | 
			
		||||
{#if bulk_modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    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 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="absolute inset-0 bg-gray-500 opacity-75"
 | 
			
		||||
          data-id="modal_backdrop" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        aria-hidden="true">​</span
 | 
			
		||||
      >
 | 
			
		||||
      <div
 | 
			
		||||
        class="inline-block align-bottom bg-white rounded-lg text-left shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-2xl sm:w-full"
 | 
			
		||||
        role="dialog"
 | 
			
		||||
        aria-modal="true"
 | 
			
		||||
        aria-labelledby="modal-headline">
 | 
			
		||||
        <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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" />
 | 
			
		||||
                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>
 | 
			
		||||
                  d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z"
 | 
			
		||||
                /></svg
 | 
			
		||||
              >
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:mt-0">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-bulk-blanco-cards')}
 | 
			
		||||
                {$_("create-bulk-blanco-cards")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <p class="text-sm text-gray-500">
 | 
			
		||||
                  {$_('just-enter-how-many-you-want-and-the-system-will-create-them')}
 | 
			
		||||
                  {$_(
 | 
			
		||||
                    "just-enter-how-many-you-want-and-the-system-will-create-them"
 | 
			
		||||
                  )}
 | 
			
		||||
                </p>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="amount"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('amount')}</label>
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                    >{$_("amount")}</label
 | 
			
		||||
                  >
 | 
			
		||||
                  <div class="mt-1 flex rounded-md shadow-sm">
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
@@ -192,15 +163,19 @@
 | 
			
		||||
                      type="number"
 | 
			
		||||
                      step="1"
 | 
			
		||||
                      name="amount"
 | 
			
		||||
                      class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 p-2"
 | 
			
		||||
                      placeholder="400" />
 | 
			
		||||
                      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>
 | 
			
		||||
                      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')}
 | 
			
		||||
                      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>
 | 
			
		||||
@@ -208,30 +183,33 @@
 | 
			
		||||
            </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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!createbtnenabled}
 | 
			
		||||
            class:opacity-50={!createbtnenabled}
 | 
			
		||||
            on:click={submit_with_print}
 | 
			
		||||
            type="button"
 | 
			
		||||
            class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
            {$_('create-and-generate-pdf')}
 | 
			
		||||
            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 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
            {$_('create-without-pdf')}
 | 
			
		||||
            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 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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,32 +1,44 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
  import {
 | 
			
		||||
    RunnerCardService,
 | 
			
		||||
    RunnerService,
 | 
			
		||||
    ScanService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
 | 
			
		||||
  import { RunnerCardService, RunnerService } from "@odit/lfk-client-js";
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { createEventDispatcher } from "svelte";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  export let current_cards;
 | 
			
		||||
  const getRunnerLabel = (option) =>
 | 
			
		||||
    option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
  const filterRunners = (label, filterText, option) =>
 | 
			
		||||
    label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
    option.value.toString().startsWith(filterText.toLowerCase());
 | 
			
		||||
 | 
			
		||||
  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;
 | 
			
		||||
  $: runners = [];
 | 
			
		||||
  $: 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;
 | 
			
		||||
  (() => {
 | 
			
		||||
@@ -46,10 +58,7 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_("adding-card"),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("adding-card"));
 | 
			
		||||
      let postdata = {
 | 
			
		||||
        runner,
 | 
			
		||||
        enabled,
 | 
			
		||||
@@ -59,21 +68,15 @@
 | 
			
		||||
          runner = 0;
 | 
			
		||||
          modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("card-added"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          current_cards.push(result);
 | 
			
		||||
          current_cards = current_cards;
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("card-added"));
 | 
			
		||||
          dispatch("created", { cards: [result] });
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          //
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -81,87 +84,105 @@
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-hidden"
 | 
			
		||||
    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">
 | 
			
		||||
      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" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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" />
 | 
			
		||||
                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>
 | 
			
		||||
                  d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z"
 | 
			
		||||
                /></svg
 | 
			
		||||
              >
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:mt-0">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-a-new-card')}
 | 
			
		||||
                {$_("create-a-new-card")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="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')}
 | 
			
		||||
                  {$_("you-can-provide-a-runner-but-you-dont-have-to")}
 | 
			
		||||
                  {$_(
 | 
			
		||||
                    "if-you-want-to-create-multiple-blanco-cards-try-the-add-bulk-button"
 | 
			
		||||
                  )}
 | 
			
		||||
                </p>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="donor"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('runner')}</label>
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                    >{$_("runner")}</label
 | 
			
		||||
                  >
 | 
			
		||||
                  <Select
 | 
			
		||||
                    containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2"
 | 
			
		||||
                    itemFilter={(label, filterText, option) => filterRunners(label, filterText, option)}
 | 
			
		||||
                    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')}
 | 
			
		||||
                    on:select={(selectedValue) => (runner = selectedValue.detail.value.id)}
 | 
			
		||||
                    on:clear={() => (runner = null)} />
 | 
			
		||||
                    bind:loading
 | 
			
		||||
                    showChevron={!loading}
 | 
			
		||||
                    placeholder={$_("search-for-runner-by-name-or-id")}
 | 
			
		||||
                    noOptionsMessage={$_("no-runners-found")}
 | 
			
		||||
                    on:select={(selectedValue) =>
 | 
			
		||||
                      (runner = selectedValue.detail.value.id)}
 | 
			
		||||
                    on:clear={() => (runner = null)}
 | 
			
		||||
                  />
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
        <div class="bg-gray-50 px-4 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,26 +1,34 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
 | 
			
		||||
  import { RunnerCardService, RunnerService } from "@odit/lfk-client-js";
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { createEventDispatcher } from "svelte";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  export let edit_modal_open;
 | 
			
		||||
  export let current_cards;
 | 
			
		||||
  export let runner = {};
 | 
			
		||||
  export let editable = {};
 | 
			
		||||
  export let original_data = {};
 | 
			
		||||
  const getRunnerLabel = (option) =>
 | 
			
		||||
    option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
  const filterRunners = (label, filterText, option) =>
 | 
			
		||||
    label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
    option.value.toString().startsWith(filterText.toLowerCase());
 | 
			
		||||
  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 };
 | 
			
		||||
@@ -46,33 +54,23 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_("updating-card"),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("updating-card"));
 | 
			
		||||
      RunnerCardService.runnerCardControllerPut(original_data.id, editable)
 | 
			
		||||
        .then((result) => {
 | 
			
		||||
          let id = original_data.id;
 | 
			
		||||
          runner = {};
 | 
			
		||||
          editable = {};
 | 
			
		||||
          original_data = {};
 | 
			
		||||
          edit_modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("card-updated"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          current_cards[current_cards.findIndex((c) => c.id === id)] = result;
 | 
			
		||||
          current_cards = current_cards;
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("card-updated"));
 | 
			
		||||
          dispatch("dataUpdated", { card: result });
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          //
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -80,66 +78,79 @@
 | 
			
		||||
 | 
			
		||||
{#if edit_modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    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 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="absolute inset-0 bg-gray-500 opacity-75"
 | 
			
		||||
          data-id="modal_backdrop" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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" />
 | 
			
		||||
                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>
 | 
			
		||||
                  d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z"
 | 
			
		||||
                /></svg
 | 
			
		||||
              >
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:text-left max-h-[75vh] overflow-y-auto">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('edit-a-card')}
 | 
			
		||||
                {$_("edit-a-card")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <p class="text-sm text-gray-500">
 | 
			
		||||
                  {$_('you-can-provide-a-runner-but-you-dont-have-to')}
 | 
			
		||||
                  {$_("you-can-provide-a-runner-but-you-dont-have-to")}
 | 
			
		||||
                </p>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="runner"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('runner')}</label>
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                    >{$_("runner")}</label
 | 
			
		||||
                  >
 | 
			
		||||
                  <Select
 | 
			
		||||
                    containerClasses="rounded-l-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2"
 | 
			
		||||
                    itemFilter={(label, filterText, option) => filterRunners(label, filterText, option)}
 | 
			
		||||
                    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')}
 | 
			
		||||
                    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)} />
 | 
			
		||||
                    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">
 | 
			
		||||
@@ -151,33 +162,36 @@
 | 
			
		||||
                      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')}
 | 
			
		||||
                      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}
 | 
			
		||||
                      {$_("enabled")}
 | 
			
		||||
                    {:else}{$_("disabled")}{/if}
 | 
			
		||||
                  </p>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
        <div class="bg-gray-50 px-4 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!createbtnenabled}
 | 
			
		||||
            class:opacity-50={!createbtnenabled}
 | 
			
		||||
            on:click={submit}
 | 
			
		||||
            type="button"
 | 
			
		||||
            class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
            {$_('save-changes')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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}
 | 
			
		||||
@@ -7,34 +7,47 @@
 | 
			
		||||
  $: current_cards = [];
 | 
			
		||||
  export let modal_open = false;
 | 
			
		||||
  export let bulk_modal_open = false;
 | 
			
		||||
  let addCards;
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('cards')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('add-card')}
 | 
			
		||||
      </button>
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          bulk_modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('create-bulk-cards')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </span>
 | 
			
		||||
  <CardsOverview bind:current_cards />
 | 
			
		||||
  <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:current_cards bind:modal_open />
 | 
			
		||||
  <AddCardBulkModal bind:current_cards bind:bulk_modal_open />
 | 
			
		||||
{#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}
 | 
			
		||||
 
 | 
			
		||||
@@ -5,8 +5,8 @@
 | 
			
		||||
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="m-auto" style="height:15rem" src={cards_empty} alt="" />
 | 
			
		||||
    <span class="font-bold">{$_('there-are-no-cards-yet')}</span><br />
 | 
			
		||||
    <span>{$_('add-your-first-card')}</span>
 | 
			
		||||
    <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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,237 +1,316 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, json, _ } from "svelte-i18n";
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { RunnerCardService } from "@odit/lfk-client-js";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  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 = [];
 | 
			
		||||
  $: searchvalue = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  $: cards_show = current_cards.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: generate_cards = current_cards.filter((r) => r.is_selected === true);
 | 
			
		||||
  const cards_promise = RunnerCardService.runnerCardControllerGetAll().then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      current_cards = val;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  function should_display_based_on_id(id) {
 | 
			
		||||
    if (searchvalue.toString().slice(-1) === "*") {
 | 
			
		||||
      return id.toString().startsWith(searchvalue.replace("*", ""));
 | 
			
		||||
    }
 | 
			
		||||
    return id.toString() === searchvalue;
 | 
			
		||||
  }
 | 
			
		||||
  const getRunnerLabel = (option) =>
 | 
			
		||||
    option?.firstname + " " + (option?.middlename || "") + " " + (option?.lastname || "{$_('non-blanko')}");
 | 
			
		||||
  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) {
 | 
			
		||||
    if(card.runner?.id){
 | 
			
		||||
    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
 | 
			
		||||
    } 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')}
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes("CARD:UPDATE")}
 | 
			
		||||
  <CardDetailModal
 | 
			
		||||
    bind:current_cards
 | 
			
		||||
    bind:edit_modal_open
 | 
			
		||||
    bind:runner
 | 
			
		||||
    bind:editable
 | 
			
		||||
    bind:original_data />
 | 
			
		||||
    bind:original_data
 | 
			
		||||
    on:dataUpdated={(event) => {
 | 
			
		||||
      current_cards[
 | 
			
		||||
        current_cards.findIndex((c) => c.id === event.detail.card.id)
 | 
			
		||||
      ] = event.detail.card;
 | 
			
		||||
      current_cards = current_cards;
 | 
			
		||||
      options.update((options) => ({
 | 
			
		||||
        ...options,
 | 
			
		||||
        data: current_cards,
 | 
			
		||||
      }));
 | 
			
		||||
    }}
 | 
			
		||||
  />
 | 
			
		||||
{/if}
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('CARD:GET')}
 | 
			
		||||
  {#await cards_promise}
 | 
			
		||||
{#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>
 | 
			
		||||
      role="alert"
 | 
			
		||||
    >
 | 
			
		||||
      <p class="font-bold">{$_("loading-cards")}</p>
 | 
			
		||||
      <p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_cards.length === 0}
 | 
			
		||||
      <CardsEmptyState />
 | 
			
		||||
    {:else}
 | 
			
		||||
      <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">
 | 
			
		||||
          <GenerateRunnerCards
 | 
			
		||||
            bind:cards_show
 | 
			
		||||
            bind:generate_cards />
 | 
			
		||||
        </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_cards.some((r) => r.is_selected === true);
 | 
			
		||||
                    current_cards = current_cards.map((r) => {
 | 
			
		||||
                      r.is_selected = newstate;
 | 
			
		||||
                      return r;
 | 
			
		||||
                    });
 | 
			
		||||
                  }}
 | 
			
		||||
                  class="underline cursor-pointer select-none">{#if current_cards.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">
 | 
			
		||||
                {$_('code')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('runner')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('status')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th scope="col" class="relative px-6 py-3">
 | 
			
		||||
                <span class="sr-only">{$_('action')}</span>
 | 
			
		||||
  {: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>
 | 
			
		||||
          </thead>
 | 
			
		||||
          <tbody class="divide-y divide-gray-200">
 | 
			
		||||
            {#each current_cards as card}
 | 
			
		||||
              {#if card.code
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .includes(
 | 
			
		||||
                  searchvalue.toLowerCase()
 | 
			
		||||
                ) || card.runner?.firstname
 | 
			
		||||
                  .toLowerCase()
 | 
			
		||||
                  .includes(
 | 
			
		||||
                    searchvalue.toLowerCase()
 | 
			
		||||
                  ) || card.runner?.middlename
 | 
			
		||||
                  .toLowerCase()
 | 
			
		||||
                  .includes(
 | 
			
		||||
                    searchvalue.toLowerCase()
 | 
			
		||||
                  ) || card.runner?.lastname
 | 
			
		||||
                  .toLowerCase()
 | 
			
		||||
                  .includes(
 | 
			
		||||
                    searchvalue.toLowerCase()
 | 
			
		||||
                  ) || should_display_based_on_id(card.id)}
 | 
			
		||||
                <tr data-rowid="card_{card.id}">
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    <input
 | 
			
		||||
                      bind:checked={card.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">{card.code}</div>
 | 
			
		||||
                  </td>
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    <div class="flex items-center">
 | 
			
		||||
                      {#if card.runner}
 | 
			
		||||
                        <a
 | 
			
		||||
                          href="../runners/{card.runner.id}"
 | 
			
		||||
                          class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{card.runner.firstname}
 | 
			
		||||
                          {card.runner.middlename || ''}
 | 
			
		||||
                          {card.runner.lastname}</a>
 | 
			
		||||
                      {:else}{$_('non-blanko')}{/if}
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </td>
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    <div class="flex items-center">
 | 
			
		||||
                      {#if card.enabled}
 | 
			
		||||
                        <span
 | 
			
		||||
                          class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">{$_('enabled')}</span>
 | 
			
		||||
                      {:else}
 | 
			
		||||
                        <span
 | 
			
		||||
                          class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">{$_('disabled')}</span>
 | 
			
		||||
                      {/if}
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </td>
 | 
			
		||||
 | 
			
		||||
                  {#if active_deletes[card.id] === true}
 | 
			
		||||
                    <td
 | 
			
		||||
                      class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          active_deletes[card.id] = false;
 | 
			
		||||
                        }}
 | 
			
		||||
                        tabindex="0"
 | 
			
		||||
                        class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button>
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          RunnerCardService.runnerCardControllerRemove(card.id, false).then(
 | 
			
		||||
                            (resp) => {
 | 
			
		||||
                              current_cards = current_cards.filter(
 | 
			
		||||
                                (obj) => obj.id !== card.id
 | 
			
		||||
                              );
 | 
			
		||||
                              Toastify({
 | 
			
		||||
                                text: $_('card-deleted'),
 | 
			
		||||
                                duration: 500,
 | 
			
		||||
                                backgroundColor:
 | 
			
		||||
                                  'linear-gradient(to right, #00b09b, #96c93d)',
 | 
			
		||||
                              }).showToast();
 | 
			
		||||
                            }
 | 
			
		||||
                          );
 | 
			
		||||
                        }}
 | 
			
		||||
                        tabindex="0"
 | 
			
		||||
                        class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('confirm-delete')}</button>
 | 
			
		||||
                    </td>
 | 
			
		||||
                  {:else}
 | 
			
		||||
                    <td
 | 
			
		||||
                      class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          open_edit_modal(card);
 | 
			
		||||
                        }}
 | 
			
		||||
                        class="text-indigo-600 hover:text-indigo-900">{$_('details')}</button>
 | 
			
		||||
                      {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:DELETE')}
 | 
			
		||||
                        <button
 | 
			
		||||
                          on:click={() => {
 | 
			
		||||
                            active_deletes[card.id] = true;
 | 
			
		||||
                          }}
 | 
			
		||||
                          tabindex="0"
 | 
			
		||||
                          class="ml-4 text-red-600 hover:text-red-900 cursor-pointer">{$_('delete')}</button>
 | 
			
		||||
                      {/if}
 | 
			
		||||
                    </td>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </tr>
 | 
			
		||||
              {/if}
 | 
			
		||||
            {/each}
 | 
			
		||||
          </tbody>
 | 
			
		||||
        </table>
 | 
			
		||||
      </div>
 | 
			
		||||
    {/if}
 | 
			
		||||
  {:catch error}
 | 
			
		||||
    <div class="text-white px-6 py-4 border-0 rounded relative mb-4 bg-red-500">
 | 
			
		||||
      <span class="inline-block align-middle mr-8">
 | 
			
		||||
        <b class="capitalize">{$_('general_promise_error')}</b>
 | 
			
		||||
        {error}
 | 
			
		||||
      </span>
 | 
			
		||||
          {/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>
 | 
			
		||||
  {/await}
 | 
			
		||||
    <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="flex-shrink-0 flex items-center justify-center size-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 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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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,7 +1,7 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
 | 
			
		||||
  import {
 | 
			
		||||
    GroupContactService,
 | 
			
		||||
    RunnerTeamService,
 | 
			
		||||
@@ -9,7 +9,7 @@
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  import isMobilePhone from "validator/es/lib/isMobilePhone";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  export let current_contacts;
 | 
			
		||||
  $: selected_team = [];
 | 
			
		||||
@@ -43,7 +43,7 @@
 | 
			
		||||
  $: address_zipcode_value = "";
 | 
			
		||||
  $: address_city_value = "";
 | 
			
		||||
  $: processed_last_submit = true;
 | 
			
		||||
  $: address_checked = true;
 | 
			
		||||
  $: address_checked = false;
 | 
			
		||||
  $: isPhoneValidOrEmpty =
 | 
			
		||||
    (phone_input_value.includes("+") &&
 | 
			
		||||
      isMobilePhone(
 | 
			
		||||
@@ -85,10 +85,7 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: "Contact is being added...",
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("contact-is-being-added"));
 | 
			
		||||
      let address = {};
 | 
			
		||||
      if (address_checked === true) {
 | 
			
		||||
        address = {
 | 
			
		||||
@@ -122,11 +119,8 @@
 | 
			
		||||
          email_input_value = "";
 | 
			
		||||
          modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: "Contact added",
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("contact-added"));
 | 
			
		||||
          current_contacts.push(result);
 | 
			
		||||
          current_contacts = current_contacts;
 | 
			
		||||
        })
 | 
			
		||||
@@ -135,8 +129,6 @@
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -144,59 +136,71 @@
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-hidden"
 | 
			
		||||
    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">
 | 
			
		||||
      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" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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" />
 | 
			
		||||
                height="24"
 | 
			
		||||
                ><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                <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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:text-left max-h-[75vh] overflow-y-auto">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-a-new-contact')}
 | 
			
		||||
                {$_("create-a-new-contact")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <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>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    use:focus
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('first-name')}
 | 
			
		||||
                    placeholder={$_("first-name")}
 | 
			
		||||
                    class:border-red-500={!isFirstnameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
@@ -204,34 +208,41 @@
 | 
			
		||||
                    bind:this={firstname_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('first-name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("first-name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('middle-name')}
 | 
			
		||||
                    placeholder={$_("middle-name")}
 | 
			
		||||
                    bind:value={middlename_input_value}
 | 
			
		||||
                    bind:this={middlename_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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 class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('last-name')}"
 | 
			
		||||
                    placeholder={$_("last-name")}
 | 
			
		||||
                    class:border-red-500={!isLastnameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
@@ -239,23 +250,28 @@
 | 
			
		||||
                    bind:this={lastname_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('last-name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("last-name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    name="team"
 | 
			
		||||
                    multiple
 | 
			
		||||
                    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}
 | 
			
		||||
                      <option value={team.id}>
 | 
			
		||||
                        {team.parentGroup.name}
 | 
			
		||||
@@ -271,10 +287,12 @@
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('phone')}
 | 
			
		||||
                    placeholder={$_("phone")}
 | 
			
		||||
                    class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
                    class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
                    class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
@@ -282,21 +300,27 @@
 | 
			
		||||
                    bind:this={phone_input}
 | 
			
		||||
                    type="tel"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      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')}
 | 
			
		||||
                      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"
 | 
			
		||||
                      )}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('e-mail-adress')}
 | 
			
		||||
                    placeholder={$_("e-mail-adress")}
 | 
			
		||||
                    class:border-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
                    class:focus:border-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
                    class:focus:ring-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
@@ -304,11 +328,13 @@
 | 
			
		||||
                    bind:this={email_input}
 | 
			
		||||
                    type="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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('valid-email-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("valid-email-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
@@ -319,22 +345,25 @@
 | 
			
		||||
                      id="comments"
 | 
			
		||||
                      name="comments"
 | 
			
		||||
                      type="checkbox"
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" />
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="ml-3 text-sm">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="comments"
 | 
			
		||||
                      class="font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
                    <label for="comments" class="font-semibold text-gray-700"
 | 
			
		||||
                      >{$_("address")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                {#if address_checked === true}
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      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
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder="{$_('address')}"
 | 
			
		||||
                      placeholder={$_("address")}
 | 
			
		||||
                      class:border-red-500={!isAddress1Valid}
 | 
			
		||||
                      class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
                      class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
@@ -342,34 +371,41 @@
 | 
			
		||||
                      bind:this={address_input1}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address1"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('address-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("address-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="address2"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("apartment-suite-etc")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_('apartment-suite-etc')}
 | 
			
		||||
                      placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
                      bind:value={address_input2_value}
 | 
			
		||||
                      bind:this={address_input2}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address2"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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 class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      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
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_('zip-postal-code')}
 | 
			
		||||
                      placeholder={$_("zip-postal-code")}
 | 
			
		||||
                      class:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
@@ -377,21 +413,25 @@
 | 
			
		||||
                      bind:this={address_zipcode}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="zipcode"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="city"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700">{$_('city')}</label>
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("city")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder="{$_('city')}"
 | 
			
		||||
                      placeholder={$_("city")}
 | 
			
		||||
                      class:border-red-500={!iscityvalid}
 | 
			
		||||
                      class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
                      class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
@@ -399,11 +439,13 @@
 | 
			
		||||
                      bind:this={address_city}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="city"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('valid-city-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-city-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
@@ -412,22 +454,24 @@
 | 
			
		||||
            </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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,394 +1,399 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import {
 | 
			
		||||
    GroupContactService,
 | 
			
		||||
    RunnerTeamService,
 | 
			
		||||
    RunnerOrganizationService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  let data_loaded = false;
 | 
			
		||||
  let orgs = [];
 | 
			
		||||
  let teams = [];
 | 
			
		||||
  export let params;
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: original_data = {};
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: changes_performed = !(
 | 
			
		||||
    JSON.stringify(original_data) === JSON.stringify(editable)
 | 
			
		||||
  );
 | 
			
		||||
  $: isEmailValid =
 | 
			
		||||
    (editable.email || "") === "" ||
 | 
			
		||||
    (editable.email && isEmail(editable.email || ""));
 | 
			
		||||
  $: isFirstnameValid = editable.firstname !== "";
 | 
			
		||||
  $: isLastnameValid = editable.lastname !== "";
 | 
			
		||||
  $: save_enabled =
 | 
			
		||||
    changes_performed &&
 | 
			
		||||
    isFirstnameValid &&
 | 
			
		||||
    isLastnameValid &&
 | 
			
		||||
    isEmailValid &&
 | 
			
		||||
    isPhoneValidOrEmpty &&
 | 
			
		||||
    ((isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
    editable.address_checked === false);
 | 
			
		||||
  const promise = GroupContactService.groupContactControllerGetOne(
 | 
			
		||||
    params.contact
 | 
			
		||||
  ).then((data) => {
 | 
			
		||||
    data_loaded = true;
 | 
			
		||||
    original_data = Object.assign(original_data, data);
 | 
			
		||||
    editable = Object.assign(editable, original_data);
 | 
			
		||||
    editable.groups = editable.groups.map((g) => g.id);
 | 
			
		||||
    original_data.groups = original_data.groups.map((g) => g.id);
 | 
			
		||||
    editable.address_checked = editable.address.address1 !== null;
 | 
			
		||||
    original_data.address_checked = editable.address.address1 !== null;
 | 
			
		||||
    if(editable.address_checked===false){
 | 
			
		||||
      editable.address = {
 | 
			
		||||
        address1: "",
 | 
			
		||||
        address2: "",
 | 
			
		||||
        city: "",
 | 
			
		||||
        postalcode: "",
 | 
			
		||||
        country: ""
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => {
 | 
			
		||||
    orgs = val;
 | 
			
		||||
  });
 | 
			
		||||
  RunnerTeamService.runnerTeamControllerGetAll().then((val) => {
 | 
			
		||||
    teams = val;
 | 
			
		||||
  });
 | 
			
		||||
  $: isPhoneValidOrEmpty =
 | 
			
		||||
    editable.phone?.includes("+") ||
 | 
			
		||||
    editable.phone === "" ||
 | 
			
		||||
    editable.phone === null;
 | 
			
		||||
  $: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
  $: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
  $: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("contact-is-being-updated"),
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      editable.address.country = "DE";
 | 
			
		||||
      if (editable.address_checked === false) {
 | 
			
		||||
        editable.address = null;
 | 
			
		||||
      }
 | 
			
		||||
      if (editable.email) editable.email = editable.email;
 | 
			
		||||
      if (editable.phone) editable.phone = editable.phone;
 | 
			
		||||
      if (editable.middlename) editable.middlename = editable.middlename;
 | 
			
		||||
      GroupContactService.groupContactControllerPut(original_data.id, editable)
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Object.assign(original_data, editable);
 | 
			
		||||
          original_data=original_data;
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("updated-contact"),
 | 
			
		||||
            duration: 2500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  function deleteContact() {
 | 
			
		||||
    GroupContactService.groupContactControllerRemove(original_data.id, true)
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {});
 | 
			
		||||
  }
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import {
 | 
			
		||||
		GroupContactService,
 | 
			
		||||
		RunnerTeamService,
 | 
			
		||||
		RunnerOrganizationService,
 | 
			
		||||
	} from "@odit/lfk-client-js";
 | 
			
		||||
	import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
	import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	let data_loaded = false;
 | 
			
		||||
	let orgs = [];
 | 
			
		||||
	let teams = [];
 | 
			
		||||
	export let params;
 | 
			
		||||
	$: delete_triggered = false;
 | 
			
		||||
	$: original_data = {};
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: changes_performed = !(
 | 
			
		||||
		JSON.stringify(original_data) === JSON.stringify(editable)
 | 
			
		||||
	);
 | 
			
		||||
	$: isEmailValid =
 | 
			
		||||
		(editable.email || "") === "" ||
 | 
			
		||||
		(editable.email && isEmail(editable.email || ""));
 | 
			
		||||
	$: isFirstnameValid = editable.firstname !== "";
 | 
			
		||||
	$: isLastnameValid = editable.lastname !== "";
 | 
			
		||||
	$: save_enabled =
 | 
			
		||||
		changes_performed &&
 | 
			
		||||
		isFirstnameValid &&
 | 
			
		||||
		isLastnameValid &&
 | 
			
		||||
		isEmailValid &&
 | 
			
		||||
		isPhoneValidOrEmpty &&
 | 
			
		||||
		((isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
			editable.address_checked === false);
 | 
			
		||||
	const promise = GroupContactService.groupContactControllerGetOne(
 | 
			
		||||
		params.contact
 | 
			
		||||
	).then((data) => {
 | 
			
		||||
		data_loaded = true;
 | 
			
		||||
		original_data = Object.assign(original_data, data);
 | 
			
		||||
		editable = Object.assign(editable, original_data);
 | 
			
		||||
		editable.groups = editable.groups.map((g) => g.id);
 | 
			
		||||
		original_data.groups = original_data.groups.map((g) => g.id);
 | 
			
		||||
		editable.address_checked = editable.address.address1 !== null;
 | 
			
		||||
		original_data.address_checked = editable.address.address1 !== null;
 | 
			
		||||
		if (editable.address_checked === false) {
 | 
			
		||||
			editable.address = {
 | 
			
		||||
				address1: "",
 | 
			
		||||
				address2: "",
 | 
			
		||||
				city: "",
 | 
			
		||||
				postalcode: "",
 | 
			
		||||
				country: "",
 | 
			
		||||
			};
 | 
			
		||||
		}
 | 
			
		||||
	});
 | 
			
		||||
	RunnerOrganizationService.runnerOrganizationControllerGetAll().then((val) => {
 | 
			
		||||
		orgs = val;
 | 
			
		||||
	});
 | 
			
		||||
	RunnerTeamService.runnerTeamControllerGetAll().then((val) => {
 | 
			
		||||
		teams = val;
 | 
			
		||||
	});
 | 
			
		||||
	$: isPhoneValidOrEmpty =
 | 
			
		||||
		editable.phone?.includes("+") ||
 | 
			
		||||
		editable.phone === "" ||
 | 
			
		||||
		editable.phone === null;
 | 
			
		||||
	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
	$: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast.loading($_("contact-is-being-updated"));
 | 
			
		||||
			editable.address.country = "DE";
 | 
			
		||||
			if (editable.address_checked === false) {
 | 
			
		||||
				editable.address = null;
 | 
			
		||||
			}
 | 
			
		||||
			if (editable.email) editable.email = editable.email;
 | 
			
		||||
			if (editable.phone) editable.phone = editable.phone;
 | 
			
		||||
			if (editable.middlename) editable.middlename = editable.middlename;
 | 
			
		||||
			GroupContactService.groupContactControllerPut(original_data.id, editable)
 | 
			
		||||
				.then((resp) => {
 | 
			
		||||
					Object.assign(original_data, editable);
 | 
			
		||||
					original_data = original_data;
 | 
			
		||||
					toast.dismiss();
 | 
			
		||||
					toast.success($_("updated-contact"));
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		} else {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	function deleteContact() {
 | 
			
		||||
		GroupContactService.groupContactControllerRemove(original_data.id, true)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#await promise}
 | 
			
		||||
  {$_('loading-contact-details')}
 | 
			
		||||
	{$_("loading-contact-details")}
 | 
			
		||||
{:then}
 | 
			
		||||
  <section class="container p-5 select-none">
 | 
			
		||||
    <div class="flex flex-row mb-4">
 | 
			
		||||
      <div class="w-full">
 | 
			
		||||
        <nav class="w-full flex">
 | 
			
		||||
          <ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <svg
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                width="24"
 | 
			
		||||
                height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                <path
 | 
			
		||||
                  d="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"
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                class="h-3 w-3 mr-2 stroke-current"
 | 
			
		||||
                height="1em"
 | 
			
		||||
                width="1em"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"><line
 | 
			
		||||
                  x1="5"
 | 
			
		||||
                  y1="12"
 | 
			
		||||
                  x2="19"
 | 
			
		||||
                  y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">{original_data.firstname}
 | 
			
		||||
                {original_data.middlename || ''}
 | 
			
		||||
                {original_data.lastname}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mb-8 text-3xl font-extrabold leading-tight">
 | 
			
		||||
      {original_data.firstname}
 | 
			
		||||
      {original_data.middlename || ''}
 | 
			
		||||
      {original_data.lastname}
 | 
			
		||||
      <span data-id="contact_actions_${editable.id}">
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:DELETE')}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              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>
 | 
			
		||||
            <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-contact')}</button>
 | 
			
		||||
          {/if}
 | 
			
		||||
        {/if}
 | 
			
		||||
        {#if !delete_triggered}
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!save_enabled}
 | 
			
		||||
            class:opacity-50={!save_enabled}
 | 
			
		||||
            type="button"
 | 
			
		||||
            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>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="firstname"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('first-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('first-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        class:border-red-500={!isFirstnameValid}
 | 
			
		||||
        class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
        class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
      {#if !isFirstnameValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('first-name-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="middlename"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('middle-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('middle-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="lastname"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('last-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('last-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.lastname}
 | 
			
		||||
        class:border-red-500={!isLastnameValid}
 | 
			
		||||
        class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
        class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
        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" />
 | 
			
		||||
      {#if !isLastnameValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('last-name-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="email"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('e-mail-adress')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('e-mail-adress')}
 | 
			
		||||
        type="email"
 | 
			
		||||
        bind:value={editable.email}
 | 
			
		||||
        class:border-red-500={!isEmailValid}
 | 
			
		||||
        class:focus:border-red-500={!isEmailValid}
 | 
			
		||||
        class:focus:ring-red-500={!isEmailValid}
 | 
			
		||||
        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" />
 | 
			
		||||
      {#if !isEmailValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('valid-email-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label for="phone" class="font-medium text-gray-700">{$_('phone')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('phone')}
 | 
			
		||||
        type="tel"
 | 
			
		||||
        class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
      {#if !isPhoneValidOrEmpty}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('valid-international-phone-number-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <span class="font-medium text-gray-700">{$_('groups')}</span>
 | 
			
		||||
      <select
 | 
			
		||||
        bind:value={editable.groups}
 | 
			
		||||
        name="team"
 | 
			
		||||
        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">
 | 
			
		||||
        {#each teams as team}
 | 
			
		||||
          <option value={team.id}>
 | 
			
		||||
            {team.parentGroup.name}
 | 
			
		||||
            >
 | 
			
		||||
            {team.name}
 | 
			
		||||
          </option>
 | 
			
		||||
        {/each}
 | 
			
		||||
        {#each orgs as org}
 | 
			
		||||
          <option value={org.id}>{org.name}</option>
 | 
			
		||||
        {/each}
 | 
			
		||||
      </select>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div class="flex items-start mt-2">
 | 
			
		||||
      <div class="flex items-center h-5">
 | 
			
		||||
        <input
 | 
			
		||||
          bind:checked={editable.address_checked}
 | 
			
		||||
          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">{$_('address')}</label>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    {#if editable.address_checked === true}
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="address1"
 | 
			
		||||
          class="block text-sm font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder="Address"
 | 
			
		||||
          class:border-red-500={!isAddress1Valid}
 | 
			
		||||
          class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
          class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
          bind:value={editable.address.address1}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="address1"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !isAddress1Valid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('address-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="address2"
 | 
			
		||||
          class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('apartment-suite-etc')}
 | 
			
		||||
          bind:value={editable.address.address2}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="address2"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="zipcode"
 | 
			
		||||
          class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('zip-postal-code')}
 | 
			
		||||
          class:border-red-500={!iszipcodevalid}
 | 
			
		||||
          class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
          class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
          bind:value={editable.address.postalcode}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="zipcode"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !iszipcodevalid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="city"
 | 
			
		||||
          class="block text-sm font-medium text-gray-700">{$_('city')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('city')}
 | 
			
		||||
          class:border-red-500={!iscityvalid}
 | 
			
		||||
          class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
          class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
          bind:value={editable.address.city}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="city"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !iscityvalid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('valid-city-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </section>
 | 
			
		||||
	<section class="container p-5 select-none">
 | 
			
		||||
		<div class="flex flex-row mb-4">
 | 
			
		||||
			<div class="w-full">
 | 
			
		||||
				<nav class="w-full flex">
 | 
			
		||||
					<ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="./"
 | 
			
		||||
								><svg
 | 
			
		||||
									xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
									width="24"
 | 
			
		||||
									height="24"
 | 
			
		||||
									viewBox="0 0 24 24"
 | 
			
		||||
									fill="none"
 | 
			
		||||
									stroke="currentColor"
 | 
			
		||||
									stroke-width="2"
 | 
			
		||||
									stroke-linecap="round"
 | 
			
		||||
									stroke-linejoin="round"
 | 
			
		||||
									class="inline-block"
 | 
			
		||||
									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg
 | 
			
		||||
								>
 | 
			
		||||
								{$_("contacts")}</a
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
					</ol>
 | 
			
		||||
				</nav>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mb-4 text-3xl font-extrabold leading-tight">
 | 
			
		||||
			{original_data.firstname}
 | 
			
		||||
			{original_data.middlename || ""}
 | 
			
		||||
			{original_data.lastname}
 | 
			
		||||
			<div data-id="contact_actions_${editable.id}">
 | 
			
		||||
				{#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:DELETE")}
 | 
			
		||||
					{#if delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							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:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("confirm-deletion")}</button
 | 
			
		||||
						>
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = !delete_triggered;
 | 
			
		||||
							}}
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("cancel")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
					{#if !delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = true;
 | 
			
		||||
							}}
 | 
			
		||||
							type="button"
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("delete-contact")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
				{/if}
 | 
			
		||||
				{#if !delete_triggered}
 | 
			
		||||
					<button
 | 
			
		||||
						disabled={!save_enabled}
 | 
			
		||||
						class:opacity-50={!save_enabled}
 | 
			
		||||
						type="button"
 | 
			
		||||
						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:w-auto sm:text-sm mb-1 lg:mb-0"
 | 
			
		||||
						>{$_("save-changes")}</button
 | 
			
		||||
					>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="firstname" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("first-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("first-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				class:border-red-500={!isFirstnameValid}
 | 
			
		||||
				class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
				class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
				bind:value={editable.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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
			{#if !isFirstnameValid}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("first-name-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="middlename" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("middle-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("middle-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="lastname" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("last-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("last-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.lastname}
 | 
			
		||||
				class:border-red-500={!isLastnameValid}
 | 
			
		||||
				class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
				class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
				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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
			{#if !isLastnameValid}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("last-name-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="email" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("e-mail-adress")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("e-mail-adress")}
 | 
			
		||||
				type="email"
 | 
			
		||||
				bind:value={editable.email}
 | 
			
		||||
				class:border-red-500={!isEmailValid}
 | 
			
		||||
				class:focus:border-red-500={!isEmailValid}
 | 
			
		||||
				class:focus:ring-red-500={!isEmailValid}
 | 
			
		||||
				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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
			{#if !isEmailValid}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("valid-email-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="phone" class="font-semibold text-gray-700">{$_("phone")}</label>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("phone")}
 | 
			
		||||
				type="tel"
 | 
			
		||||
				class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				bind:value={editable.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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
			{#if !isPhoneValidOrEmpty}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("valid-international-phone-number-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<span class="font-semibold text-gray-700">{$_("groups")}</span>
 | 
			
		||||
			<select
 | 
			
		||||
				bind:value={editable.groups}
 | 
			
		||||
				name="team"
 | 
			
		||||
				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-neutral-800 rounded-md p-2"
 | 
			
		||||
			>
 | 
			
		||||
				{#each teams as team}
 | 
			
		||||
					<option value={team.id}>
 | 
			
		||||
						{team.parentGroup.name}
 | 
			
		||||
						>
 | 
			
		||||
						{team.name}
 | 
			
		||||
					</option>
 | 
			
		||||
				{/each}
 | 
			
		||||
				{#each orgs as org}
 | 
			
		||||
					<option value={org.id}>{org.name}</option>
 | 
			
		||||
				{/each}
 | 
			
		||||
			</select>
 | 
			
		||||
		</div>
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		<div class="flex items-start mt-2">
 | 
			
		||||
			<div class="flex items-center h-5">
 | 
			
		||||
				<input
 | 
			
		||||
					bind:checked={editable.address_checked}
 | 
			
		||||
					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-semibold text-gray-700"
 | 
			
		||||
					>{$_("address")}</label
 | 
			
		||||
				>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		{#if editable.address_checked === true}
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="address1" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
					>{$_("address")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder="Address"
 | 
			
		||||
					class:border-red-500={!isAddress1Valid}
 | 
			
		||||
					class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
					class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
					bind:value={editable.address.address1}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="address1"
 | 
			
		||||
					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
				/>
 | 
			
		||||
				{#if !isAddress1Valid}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("address-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="address2" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
					>{$_("apartment-suite-etc")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
					bind:value={editable.address.address2}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="address2"
 | 
			
		||||
					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="zipcode" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
					>{$_("zip-postal-code")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("zip-postal-code")}
 | 
			
		||||
					class:border-red-500={!iszipcodevalid}
 | 
			
		||||
					class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
					class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
					bind:value={editable.address.postalcode}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="zipcode"
 | 
			
		||||
					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
				/>
 | 
			
		||||
				{#if !iszipcodevalid}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="city" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
					>{$_("city")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("city")}
 | 
			
		||||
					class:border-red-500={!iscityvalid}
 | 
			
		||||
					class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
					class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
					bind:value={editable.address.city}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="city"
 | 
			
		||||
					class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
				/>
 | 
			
		||||
				{#if !iscityvalid}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("valid-city-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		{/if}
 | 
			
		||||
	</section>
 | 
			
		||||
{:catch error}
 | 
			
		||||
  <PromiseError {error} />
 | 
			
		||||
	<PromiseError {error} />
 | 
			
		||||
{/await}
 | 
			
		||||
 
 | 
			
		||||
@@ -8,22 +8,23 @@
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('contacts')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('create-a-new-contact')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </span>
 | 
			
		||||
  <h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_("contacts")}
 | 
			
		||||
  </h4>
 | 
			
		||||
  {#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT: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"
 | 
			
		||||
    >
 | 
			
		||||
      {$_("create-a-new-contact")}
 | 
			
		||||
    </button>
 | 
			
		||||
  {/if}
 | 
			
		||||
  <ContactsOverview bind:current_contacts />
 | 
			
		||||
</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 />
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="w-full h-44" src={team_empty} alt="" />
 | 
			
		||||
    <span class="font-bold">{$_('there-are-no-contacts-added-yet')}</span><br />
 | 
			
		||||
    <span>{$_('add-your-first-contact')}</span>
 | 
			
		||||
    <span class="font-bold">{$_("there-are-no-contacts-added-yet")}</span><br />
 | 
			
		||||
    <span>{$_("add-your-first-contact")}</span>
 | 
			
		||||
  </p>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,177 +1,198 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { GroupContactService } from "@odit/lfk-client-js";
 | 
			
		||||
  const promise = GroupContactService.groupContactControllerGetAll().then(
 | 
			
		||||
    (result) => {
 | 
			
		||||
      current_contacts = result;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import ContactsEmptyState from "./ContactsEmptyState.svelte";
 | 
			
		||||
  $: searchvalue = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  export let current_contacts = [];
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { GroupContactService } from "@odit/lfk-client-js";
 | 
			
		||||
	const promise = GroupContactService.groupContactControllerGetAll().then(
 | 
			
		||||
		(result) => {
 | 
			
		||||
			current_contacts = result;
 | 
			
		||||
		}
 | 
			
		||||
	);
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import ContactsEmptyState from "./ContactsEmptyState.svelte";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	$: searchvalue = "";
 | 
			
		||||
	$: active_deletes = [];
 | 
			
		||||
	export let current_contacts = [];
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('TEAM: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">{$_('contacts-are-being-loaded')}</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_contacts.length === 0}
 | 
			
		||||
      <ContactsEmptyState />
 | 
			
		||||
    {:else}
 | 
			
		||||
      <input
 | 
			
		||||
        type="search"
 | 
			
		||||
        bind:value={searchvalue}
 | 
			
		||||
        placeholder={$_('datatable.search')}
 | 
			
		||||
        aria-label={$_('datatable.search')}
 | 
			
		||||
        class="gridjs-input gridjs-search-input mb-4" />
 | 
			
		||||
      <div
 | 
			
		||||
        class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll">
 | 
			
		||||
        <table class="divide-y divide-gray-200 w-full">
 | 
			
		||||
          <thead class="bg-gray-50">
 | 
			
		||||
            <tr>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('name')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('groups')}
 | 
			
		||||
              </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="relative px-6 py-3">
 | 
			
		||||
                <span class="sr-only">{$_('action')}</span>
 | 
			
		||||
              </th>
 | 
			
		||||
            </tr>
 | 
			
		||||
          </thead>
 | 
			
		||||
          <tbody class="divide-y divide-gray-200">
 | 
			
		||||
            {#each current_contacts as t}
 | 
			
		||||
              {#if Object.values(t)
 | 
			
		||||
                .toString()
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .includes(searchvalue)}
 | 
			
		||||
                <tr data-rowid="team_{t.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">
 | 
			
		||||
                          {t.firstname}
 | 
			
		||||
                          {t.middlename || ''}
 | 
			
		||||
                          {t.lastname}
 | 
			
		||||
                        </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 t.groups.length > 0}
 | 
			
		||||
                            {#each t.groups as g}
 | 
			
		||||
                              {#if g.responseType === 'RUNNERORGANIZATION'}
 | 
			
		||||
                                <a
 | 
			
		||||
                                  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>
 | 
			
		||||
                              {:else}
 | 
			
		||||
                                <a
 | 
			
		||||
                                  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}
 | 
			
		||||
                                  >
 | 
			
		||||
                                  {g.name}</a>
 | 
			
		||||
                              {/if}
 | 
			
		||||
                            {/each}
 | 
			
		||||
                          {:else}
 | 
			
		||||
                            {$_('contact-is-not-a-member-in-any-group')}
 | 
			
		||||
                          {/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 t.address.address1 !== null}
 | 
			
		||||
                            {t.address.address1}<br />
 | 
			
		||||
                            {t.address.address2 || ''}<br />
 | 
			
		||||
                            {t.address.postalcode}
 | 
			
		||||
                            {t.address.city}
 | 
			
		||||
                            {t.address.country}
 | 
			
		||||
                          {/if}
 | 
			
		||||
                        </div>
 | 
			
		||||
                      </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </td>
 | 
			
		||||
                  {#if active_deletes[t.id] === true}
 | 
			
		||||
                    <td
 | 
			
		||||
                      class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          active_deletes[t.id] = false;
 | 
			
		||||
                        }}
 | 
			
		||||
                        tabindex="0"
 | 
			
		||||
                        class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer">{$_('cancel-delete')}</button>
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          GroupContactService.groupContactControllerRemove(t.id, false).then(
 | 
			
		||||
                            (resp) => {
 | 
			
		||||
                              current_contacts = current_contacts.filter(
 | 
			
		||||
                                (obj) => obj.id !== t.id
 | 
			
		||||
                              );
 | 
			
		||||
                              Toastify({
 | 
			
		||||
                                text: $_('contact-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="./{t.id}"
 | 
			
		||||
                        class="text-indigo-600 hover:text-indigo-900">{$_('details')}</a>
 | 
			
		||||
                      {#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:DELETE')}
 | 
			
		||||
                        <button
 | 
			
		||||
                          on:click={() => {
 | 
			
		||||
                            active_deletes[t.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 store.state.jwtinfo.userdetails.permissions.includes("TEAM: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">{$_("contacts-are-being-loaded")}</p>
 | 
			
		||||
			<p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
		</div>
 | 
			
		||||
	{:then}
 | 
			
		||||
		{#if current_contacts.length === 0}
 | 
			
		||||
			<ContactsEmptyState />
 | 
			
		||||
		{:else}
 | 
			
		||||
			<input
 | 
			
		||||
				type="search"
 | 
			
		||||
				bind:value={searchvalue}
 | 
			
		||||
				placeholder={$_("datatable.search")}
 | 
			
		||||
				aria-label={$_("datatable.search")}
 | 
			
		||||
				class="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border"
 | 
			
		||||
			/>
 | 
			
		||||
			<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"
 | 
			
		||||
							>
 | 
			
		||||
								{$_("name")}
 | 
			
		||||
							</th>
 | 
			
		||||
							<th
 | 
			
		||||
								scope="col"
 | 
			
		||||
								class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"
 | 
			
		||||
							>
 | 
			
		||||
								{$_("groups")}
 | 
			
		||||
							</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="relative px-6 py-3">
 | 
			
		||||
								<span class="sr-only">{$_("action")}</span>
 | 
			
		||||
							</th>
 | 
			
		||||
						</tr>
 | 
			
		||||
					</thead>
 | 
			
		||||
					<tbody class="divide-y divide-gray-200">
 | 
			
		||||
						{#each current_contacts as t}
 | 
			
		||||
							{#if Object.values(t)
 | 
			
		||||
								.toString()
 | 
			
		||||
								.toLowerCase()
 | 
			
		||||
								.includes(searchvalue)}
 | 
			
		||||
								<tr
 | 
			
		||||
									class="odd:bg-white even:bg-gray-100"
 | 
			
		||||
									data-rowid="team_{t.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">
 | 
			
		||||
													{t.firstname}
 | 
			
		||||
													{t.middlename || ""}
 | 
			
		||||
													{t.lastname}
 | 
			
		||||
												</div>
 | 
			
		||||
											</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 gap-0.5 flex flex-wrap"
 | 
			
		||||
											>
 | 
			
		||||
												{#if t.groups.length > 0}
 | 
			
		||||
													{#each t.groups as g}
 | 
			
		||||
														{#if g.responseType === "RUNNERORGANIZATION"}
 | 
			
		||||
															<a
 | 
			
		||||
																href="../orgs/{g.id}"
 | 
			
		||||
																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}
 | 
			
		||||
															<a
 | 
			
		||||
																href="../teams/{g.id}"
 | 
			
		||||
																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
 | 
			
		||||
															>
 | 
			
		||||
														{/if}
 | 
			
		||||
													{/each}
 | 
			
		||||
												{:else}
 | 
			
		||||
													{$_("contact-is-not-a-member-in-any-group")}
 | 
			
		||||
												{/if}
 | 
			
		||||
											</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 t.address.address1 !== null}
 | 
			
		||||
														{t.address.address1}<br />
 | 
			
		||||
														{t.address.address2 || ""}<br />
 | 
			
		||||
														{t.address.postalcode}
 | 
			
		||||
														{t.address.city}
 | 
			
		||||
														{t.address.country}
 | 
			
		||||
													{/if}
 | 
			
		||||
												</div>
 | 
			
		||||
											</div>
 | 
			
		||||
										</div>
 | 
			
		||||
									</td>
 | 
			
		||||
									{#if active_deletes[t.id] === true}
 | 
			
		||||
										<td
 | 
			
		||||
											class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"
 | 
			
		||||
										>
 | 
			
		||||
											<button
 | 
			
		||||
												on:click={() => {
 | 
			
		||||
													active_deletes[t.id] = false;
 | 
			
		||||
												}}
 | 
			
		||||
												tabindex="0"
 | 
			
		||||
												class="ml-4 text-indigo-600 hover:text-indigo-900 cursor-pointer"
 | 
			
		||||
												>{$_("cancel-delete")}</button
 | 
			
		||||
											>
 | 
			
		||||
											<button
 | 
			
		||||
												on:click={() => {
 | 
			
		||||
													toast.loading($_("deleting-contact"));
 | 
			
		||||
													GroupContactService.groupContactControllerRemove(
 | 
			
		||||
														t.id,
 | 
			
		||||
														false
 | 
			
		||||
													).then((resp) => {
 | 
			
		||||
														current_contacts = current_contacts.filter(
 | 
			
		||||
															(obj) => obj.id !== t.id
 | 
			
		||||
														);
 | 
			
		||||
														toast.dismiss();
 | 
			
		||||
														toast.success($_("contact-deleted"));
 | 
			
		||||
													});
 | 
			
		||||
												}}
 | 
			
		||||
												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="./{t.id}"
 | 
			
		||||
												class="text-indigo-600 hover:text-indigo-900"
 | 
			
		||||
												>{$_("details")}</a
 | 
			
		||||
											>
 | 
			
		||||
											{#if store.state.jwtinfo.userdetails.permissions.includes("TEAM:DELETE")}
 | 
			
		||||
												<button
 | 
			
		||||
													on:click={() => {
 | 
			
		||||
														active_deletes[t.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,327 +1,459 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import localForage from "localforage";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import { router } from "tinro";
 | 
			
		||||
  import NoComponentLoaded from "../base/NoComponentLoaded.svelte";
 | 
			
		||||
  import { AuthService } from "@odit/lfk-client-js";
 | 
			
		||||
  $: navOpen = false;
 | 
			
		||||
  function logout() {
 | 
			
		||||
    localForage.clear();
 | 
			
		||||
    location.replace("/");
 | 
			
		||||
  }
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import localForage from "localforage";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import { router } from "tinro";
 | 
			
		||||
	import NoComponentLoaded from "../base/NoComponentLoaded.svelte";
 | 
			
		||||
	import { AuthService } from "@odit/lfk-client-js";
 | 
			
		||||
	import { Toaster } from "svelte-french-toast";
 | 
			
		||||
	$: navOpen = false;
 | 
			
		||||
	function logout() {
 | 
			
		||||
		localForage.clear();
 | 
			
		||||
		location.replace("/");
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="min-h-screen bg-gray-50">
 | 
			
		||||
  <nav
 | 
			
		||||
    class:-translate-x-full={!navOpen}
 | 
			
		||||
    class:translate-x-0={navOpen}
 | 
			
		||||
    class="select-none fixed top-0 left-0 z-20 h-full pb-10 overflow-x-hidden overflow-y-auto transition origin-left transform border-r w-60 md:translate-x-0 bg-gray-50">
 | 
			
		||||
    <a href="/" class="flex items-center px-4 py-5">
 | 
			
		||||
      <img src="/lfk-logo.png" alt="Logo" class="h-10" />
 | 
			
		||||
      <h3 class="text-lg">Lauf für Kaya! Admin</h3>
 | 
			
		||||
    </a>
 | 
			
		||||
    <nav class="text-sm font-medium text-gray-600" aria-label="Main Navigation">
 | 
			
		||||
      <a
 | 
			
		||||
        class:bg-gray-100={$router.path === '/'}
 | 
			
		||||
        class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
        href="/">
 | 
			
		||||
        <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 20 20"
 | 
			
		||||
          fill="currentColor">
 | 
			
		||||
          <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" />
 | 
			
		||||
        </svg>
 | 
			
		||||
        <span>{$_('dashboard-title')}</span>
 | 
			
		||||
      </a>
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path.includes('/orgs/')}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/orgs/">
 | 
			
		||||
          <svg
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-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="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"
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 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:bg-gray-100={$router.path === '/groups/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/groups/">
 | 
			
		||||
          <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="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"
 | 
			
		||||
            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>
 | 
			
		||||
          <span>{$_('donors')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path.includes('/donations/')}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/donations/">
 | 
			
		||||
          <svg
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-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>
 | 
			
		||||
          <span>{$_('donations')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('TRACK:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path === '/tracks/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/tracks/">
 | 
			
		||||
          <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="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>
 | 
			
		||||
          <span>{$_('tracks')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('CARD:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path === '/cards/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/cards/">
 | 
			
		||||
          <svg
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600"
 | 
			
		||||
            fill="currentColor"
 | 
			
		||||
            width="24"
 | 
			
		||||
            height="24"
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
            viewBox="0 0 24 24">
 | 
			
		||||
            <path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
            <path
 | 
			
		||||
              fill="currentColor"
 | 
			
		||||
              d="M22 10v10a1 1 0 01-1 1H3a1 1 0 01-1-1V10h20zm0-2H2V4a1 1 0 011-1h18a1 1 0 011 1v4zm-7 8v2h4v-2h-4z" /></svg>
 | 
			
		||||
          <span>{$_('cards')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('SCAN:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path === '/scans/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/scans/">
 | 
			
		||||
          <svg
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600"
 | 
			
		||||
            fill="currentColor"
 | 
			
		||||
            width="24"
 | 
			
		||||
            height="24"
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
            viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
            <path
 | 
			
		||||
              fill="currentColor"
 | 
			
		||||
              d="M2 4h2v16H2V4zm4 0h1v16H6V4zm2 0h2v16H8V4zm3 0h2v16h-2V4zm3 0h2v16h-2V4zm3 0h1v16h-1V4zm2 0h3v16h-3V4z" /></svg>
 | 
			
		||||
          <span>Scans</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('CONTACT:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path === '/contacts/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/contacts/">
 | 
			
		||||
          <svg
 | 
			
		||||
            fill="currentColor"
 | 
			
		||||
            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"
 | 
			
		||||
            width="24"
 | 
			
		||||
            height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
            <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>
 | 
			
		||||
          <span>{$_('contacts')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      {#if store.state.jwtinfo.userdetails.permissions.includes('STATION:GET')}
 | 
			
		||||
        <a
 | 
			
		||||
          class:bg-gray-100={$router.path === '/scanstations/'}
 | 
			
		||||
          class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
          href="/scanstations/">
 | 
			
		||||
          <svg
 | 
			
		||||
            class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600"
 | 
			
		||||
            fill="currentColor"
 | 
			
		||||
            width="24"
 | 
			
		||||
            height="24"
 | 
			
		||||
            viewBox="0 0 24 24"
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"><path
 | 
			
		||||
              fill="none"
 | 
			
		||||
              d="M0 0h24v24H0z" />
 | 
			
		||||
            <path
 | 
			
		||||
              fill="currentColor"
 | 
			
		||||
              d="M4 5v11h16V5H4zM2 4a1 1 0 011-1h18a1 1 0 011 1v14H2V4zM1 19h22v2H1v-2z" /></svg>
 | 
			
		||||
          <span>{$_('scanstations')}</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      {/if}
 | 
			
		||||
      <a
 | 
			
		||||
        class:bg-gray-100={$router.path === '/settings/'}
 | 
			
		||||
        class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
        href="/settings/">
 | 
			
		||||
        <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 20 20"
 | 
			
		||||
          fill="currentColor">
 | 
			
		||||
          <path
 | 
			
		||||
            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"
 | 
			
		||||
            clip-rule="evenodd" />
 | 
			
		||||
        </svg>
 | 
			
		||||
        <span>{$_('settings')}</span>
 | 
			
		||||
      </a>
 | 
			
		||||
      <a
 | 
			
		||||
        class:bg-gray-100={$router.path === '/about/'}
 | 
			
		||||
        class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
        href="/about/">
 | 
			
		||||
        <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"
 | 
			
		||||
          fill="none"
 | 
			
		||||
          stroke="currentColor"
 | 
			
		||||
          stroke-width="2"
 | 
			
		||||
          stroke-linecap="round"
 | 
			
		||||
          stroke-linejoin="round"
 | 
			
		||||
          viewBox="0 0 24 24"><circle cx="12" cy="12" r="10" />
 | 
			
		||||
          <path d="M12 16v-4M12 8h.01" /></svg>
 | 
			
		||||
        <span>{$_('about')}</span>
 | 
			
		||||
      </a>
 | 
			
		||||
      <span
 | 
			
		||||
        tabindex="0"
 | 
			
		||||
        class="flex items-center px-4 py-3 transition cursor-pointer group hover:bg-gray-100 hover:text-gray-900"
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          AuthService.authControllerLogout();
 | 
			
		||||
          logout();
 | 
			
		||||
        }}>
 | 
			
		||||
        <svg
 | 
			
		||||
          class="flex-shrink-0 w-5 h-5 mr-2 text-gray-400 transition group-hover:text-gray-600"
 | 
			
		||||
          fill="currentColor"
 | 
			
		||||
          width="24"
 | 
			
		||||
          height="24"
 | 
			
		||||
          xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
          viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
          <path
 | 
			
		||||
            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>
 | 
			
		||||
        <span>{$_('logout')}</span>
 | 
			
		||||
      </span>
 | 
			
		||||
    </nav>
 | 
			
		||||
  </nav>
 | 
			
		||||
  <div class="ml-0 transition md:ml-60">
 | 
			
		||||
    <header
 | 
			
		||||
      on:click={() => {
 | 
			
		||||
        navOpen = true;
 | 
			
		||||
      }}
 | 
			
		||||
      class="flex items-center justify-between w-full px-4 bg-white border-b h-14 md:hidden">
 | 
			
		||||
      <button class="block btn btn-light md:hidden">
 | 
			
		||||
        <span class="sr-only">Menu</span><svg
 | 
			
		||||
          class="w-4 h-4"
 | 
			
		||||
          xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
          viewBox="0 0 20 20"
 | 
			
		||||
          fill="currentcolor"><path
 | 
			
		||||
            fill-rule="evenodd"
 | 
			
		||||
            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"
 | 
			
		||||
            clip-rule="evenodd" /></svg></button>
 | 
			
		||||
    </header>
 | 
			
		||||
    <slot>
 | 
			
		||||
      <NoComponentLoaded />
 | 
			
		||||
    </slot>
 | 
			
		||||
  </div>
 | 
			
		||||
  <div
 | 
			
		||||
    on:click={() => {
 | 
			
		||||
      navOpen = false;
 | 
			
		||||
    }}
 | 
			
		||||
    class:hidden={!navOpen}
 | 
			
		||||
    class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden" />
 | 
			
		||||
	<div
 | 
			
		||||
		class:collapsed_navigation={!navOpen}
 | 
			
		||||
		style="z-index:11;"
 | 
			
		||||
		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">
 | 
			
		||||
			<img src="/lfk-logo.png" alt="Logo" class="h-10" />
 | 
			
		||||
			<h3 class="text-lg font-bold">LfK!Admin</h3>
 | 
			
		||||
		</a>
 | 
			
		||||
		<nav class="text-sm font-medium text-gray-600" aria-label="Main Navigation">
 | 
			
		||||
			<a
 | 
			
		||||
				class:activenav={$router.path === "/"}
 | 
			
		||||
				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="/"
 | 
			
		||||
			>
 | 
			
		||||
				<svg
 | 
			
		||||
					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600"
 | 
			
		||||
					xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
					viewBox="0 0 20 20"
 | 
			
		||||
					fill="currentColor"
 | 
			
		||||
				>
 | 
			
		||||
					<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"
 | 
			
		||||
					/>
 | 
			
		||||
				</svg>
 | 
			
		||||
				<span>{$_("dashboard-title")}</span>
 | 
			
		||||
			</a>
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:GET") && store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/cardassignment/")}
 | 
			
		||||
					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="/cardassignment/"
 | 
			
		||||
				>
 | 
			
		||||
					<svg
 | 
			
		||||
						xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
						viewBox="0 0 24 24"
 | 
			
		||||
						fill="currentColor"
 | 
			
		||||
						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600"
 | 
			
		||||
					>
 | 
			
		||||
						<path
 | 
			
		||||
							fill-rule="evenodd"
 | 
			
		||||
							d="M14.615 1.595a.75.75 0 0 1 .359.852L12.982 9.75h7.268a.75.75 0 0 1 .548 1.262l-10.5 11.25a.75.75 0 0 1-1.272-.71l1.992-7.302H3.75a.75.75 0 0 1-.548-1.262l10.5-11.25a.75.75 0 0 1 .913-.143Z"
 | 
			
		||||
							clip-rule="evenodd"
 | 
			
		||||
						/>
 | 
			
		||||
					</svg>
 | 
			
		||||
 | 
			
		||||
					<span>{$_('card_assignment_menu')}</span>
 | 
			
		||||
				</a>
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/runners/")}
 | 
			
		||||
					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="/runners/"
 | 
			
		||||
				>
 | 
			
		||||
					<svg
 | 
			
		||||
						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"
 | 
			
		||||
						xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
						viewBox="0 0 24 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("DONOR:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/donors/")}
 | 
			
		||||
					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="/donors/"
 | 
			
		||||
				>
 | 
			
		||||
					<svg
 | 
			
		||||
						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-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="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
 | 
			
		||||
					>
 | 
			
		||||
					<span>{$_("donors")}</span>
 | 
			
		||||
				</a>
 | 
			
		||||
			{/if}
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/donations/")}
 | 
			
		||||
					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/"
 | 
			
		||||
				>
 | 
			
		||||
					<svg
 | 
			
		||||
						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-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
 | 
			
		||||
					>
 | 
			
		||||
					<span>{$_("donations")}</span>
 | 
			
		||||
				</a>
 | 
			
		||||
			{/if}
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("TRACK:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path === "/tracks/"}
 | 
			
		||||
					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/"
 | 
			
		||||
				>
 | 
			
		||||
					<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="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
 | 
			
		||||
					>
 | 
			
		||||
					<span>{$_("tracks")}</span>
 | 
			
		||||
				</a>
 | 
			
		||||
			{/if}
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("CARD:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path === "/cards/"}
 | 
			
		||||
					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="/cards/"
 | 
			
		||||
				>
 | 
			
		||||
					<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="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>
 | 
			
		||||
				</a>
 | 
			
		||||
			{/if}
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("CONTACT:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/contacts/")}
 | 
			
		||||
					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/"
 | 
			
		||||
				>
 | 
			
		||||
					<svg
 | 
			
		||||
						fill="currentColor"
 | 
			
		||||
						class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600"
 | 
			
		||||
						xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
						viewBox="0 0 24 24"
 | 
			
		||||
						width="24"
 | 
			
		||||
						height="24"
 | 
			
		||||
						><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
						<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
 | 
			
		||||
					>
 | 
			
		||||
					<span>{$_("contacts")}</span>
 | 
			
		||||
				</a>
 | 
			
		||||
			{/if}
 | 
			
		||||
			{#if store.state.jwtinfo.userdetails.permissions.includes("STATION:GET")}
 | 
			
		||||
				<a
 | 
			
		||||
					class:activenav={$router.path.includes("/scanstations/")}
 | 
			
		||||
					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/"
 | 
			
		||||
				>
 | 
			
		||||
					<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>{$_("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>
 | 
			
		||||
			{/if}
 | 
			
		||||
			<a
 | 
			
		||||
				class:activenav={$router.path === "/settings/"}
 | 
			
		||||
				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/"
 | 
			
		||||
			>
 | 
			
		||||
				<svg
 | 
			
		||||
					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600"
 | 
			
		||||
					xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
					viewBox="0 0 20 20"
 | 
			
		||||
					fill="currentColor"
 | 
			
		||||
				>
 | 
			
		||||
					<path
 | 
			
		||||
						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"
 | 
			
		||||
						clip-rule="evenodd"
 | 
			
		||||
					/>
 | 
			
		||||
				</svg>
 | 
			
		||||
				<span>{$_("settings")}</span>
 | 
			
		||||
			</a>
 | 
			
		||||
			<a
 | 
			
		||||
				class:activenav={$router.path === "/about/"}
 | 
			
		||||
				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/"
 | 
			
		||||
			>
 | 
			
		||||
				<svg
 | 
			
		||||
					class="flex-shrink-0 w-5 h-5 mr-2 transition group-hover:text-gray-600"
 | 
			
		||||
					xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
					fill="none"
 | 
			
		||||
					stroke="currentColor"
 | 
			
		||||
					stroke-width="2"
 | 
			
		||||
					stroke-linecap="round"
 | 
			
		||||
					stroke-linejoin="round"
 | 
			
		||||
					viewBox="0 0 24 24"
 | 
			
		||||
					><circle cx="12" cy="12" r="10" />
 | 
			
		||||
					<path d="M12 16v-4M12 8h.01" /></svg
 | 
			
		||||
				>
 | 
			
		||||
				<span>{$_("about")}</span>
 | 
			
		||||
			</a>
 | 
			
		||||
			<button
 | 
			
		||||
				tabindex="0"
 | 
			
		||||
				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={() => {
 | 
			
		||||
					AuthService.authControllerLogout();
 | 
			
		||||
					logout();
 | 
			
		||||
				}}
 | 
			
		||||
			>
 | 
			
		||||
				<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
 | 
			
		||||
						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
 | 
			
		||||
				>
 | 
			
		||||
				<span>{$_("logout")}</span>
 | 
			
		||||
			</button>
 | 
			
		||||
		</nav>
 | 
			
		||||
	</div>
 | 
			
		||||
	<div class="ml-0 transition md:ml-60">
 | 
			
		||||
		<header
 | 
			
		||||
			class="flex items-center w-full px-4 bg-white border-b h-14 md:hidden"
 | 
			
		||||
		>
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					navOpen = true;
 | 
			
		||||
				}}
 | 
			
		||||
				class="block btn btn-light md:hidden"
 | 
			
		||||
			>
 | 
			
		||||
				<span class="sr-only">Menu</span><svg
 | 
			
		||||
					xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
					fill="none"
 | 
			
		||||
					viewBox="0 0 24 24"
 | 
			
		||||
					stroke-width="1.5"
 | 
			
		||||
					stroke="currentColor"
 | 
			
		||||
					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>
 | 
			
		||||
		<Toaster position="top-right" />
 | 
			
		||||
		<slot>
 | 
			
		||||
			<NoComponentLoaded />
 | 
			
		||||
		</slot>
 | 
			
		||||
	</div>
 | 
			
		||||
	{#if navOpen === true}
 | 
			
		||||
		<button
 | 
			
		||||
			on:click={() => {
 | 
			
		||||
				navOpen = false;
 | 
			
		||||
			}}
 | 
			
		||||
			class:hidden={!navOpen}
 | 
			
		||||
			class="fixed inset-0 z-10 w-screen h-screen bg-black bg-opacity-25 md:hidden"
 | 
			
		||||
		/>
 | 
			
		||||
	{/if}
 | 
			
		||||
</section>
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
	.collapsed_navigation {
 | 
			
		||||
		transform: translateX(-100%);
 | 
			
		||||
	}
 | 
			
		||||
	@media (min-width: 768px) {
 | 
			
		||||
		.collapsed_navigation {
 | 
			
		||||
			transform: translateX(0px);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,22 +1,263 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import StatCards from "./StatCards.svelte";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  let navOpen = false;
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { StatsService } from "@odit/lfk-client-js";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import StatCard from "./StatCard.svelte";
 | 
			
		||||
	const stats_promise = StatsService.statsControllerGet();
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<div
 | 
			
		||||
  class="p-5 overflow-x-hidden"
 | 
			
		||||
  on:click={() => {
 | 
			
		||||
    navOpen = false;
 | 
			
		||||
  }}>
 | 
			
		||||
  <h1 class="text-3xl leading-tight">
 | 
			
		||||
    <span class="font-extrabold">{$_('dashboard-title')}</span>
 | 
			
		||||
    <span>
 | 
			
		||||
      -
 | 
			
		||||
      {$_('dashboard-greeting')},
 | 
			
		||||
      <span
 | 
			
		||||
        class="text-blue-500">{store.state.jwtinfo.userdetails.firstname} {store.state.jwtinfo.userdetails.lastname}</span></span>
 | 
			
		||||
  </h1>
 | 
			
		||||
  <StatCards />
 | 
			
		||||
<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
 | 
			
		||||
			class="bg-teal-lightest border-t-4 border-teal rounded-b text-teal-darkest px-4 py-3 shadow-md my-2"
 | 
			
		||||
			role="alert"
 | 
			
		||||
		>
 | 
			
		||||
			<p class="font-bold">{$_("stats-are-being-loaded")}</p>
 | 
			
		||||
			<p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
		</div>
 | 
			
		||||
	{:then stats}
 | 
			
		||||
		<div
 | 
			
		||||
			class="grid gap-1 grid-cols-2 lg:grid-cols-3 xl:grid-cols-5 2xl:grid-cols-6 sm:gap-4"
 | 
			
		||||
		>
 | 
			
		||||
			<StatCard
 | 
			
		||||
				title={$_("runners")}
 | 
			
		||||
				value={stats.total_runners}
 | 
			
		||||
				href="/runners/"
 | 
			
		||||
			>
 | 
			
		||||
				<svg
 | 
			
		||||
					height="24"
 | 
			
		||||
					width="24"
 | 
			
		||||
					fill="currentColor"
 | 
			
		||||
					xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
					viewBox="0 0 24 24"
 | 
			
		||||
					><path d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
					<path
 | 
			
		||||
						d="M13.49 5.48c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm-3.6 13.9l1-4.4 2.1 2v6h2v-7.5l-2.1-2 .6-3c1.3 1.5 3.3 2.5 5.5 2.5v-2c-1.9 0-3.5-1-4.3-2.4l-1-1.6c-.4-.6-1-1-1.7-1-.3 0-.5.1-.8.1l-5.2 2.2v4.7h2v-3.4l1.8-.7-1.6 8.1-4.9-1-.4 2 7 1.4z"
 | 
			
		||||
					/></svg
 | 
			
		||||
				>
 | 
			
		||||
			</StatCard>
 | 
			
		||||
			<StatCard
 | 
			
		||||
				title={$_("total-scans")}
 | 
			
		||||
				value={stats.total_scans}
 | 
			
		||||
				href="/scans/"
 | 
			
		||||
			>
 | 
			
		||||
				<svg
 | 
			
		||||
					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>
 | 
			
		||||
			<StatCard
 | 
			
		||||
				title={$_('runners_via_kiosk')}
 | 
			
		||||
				value={stats.runnersViaKiosk}
 | 
			
		||||
				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>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
 | 
			
		||||
  import {
 | 
			
		||||
    DonationService,
 | 
			
		||||
    DonorService,
 | 
			
		||||
    RunnerService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  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 current_donations;
 | 
			
		||||
  const dispatch = createEventDispatcher();
 | 
			
		||||
  const getDonorLabel = (option) =>
 | 
			
		||||
    option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
  const filterDonors = (label, filterText, option) =>
 | 
			
		||||
    label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
    option.value.id.toString().startsWith(filterText.toLowerCase());
 | 
			
		||||
  function focus(el) {
 | 
			
		||||
    el.focus();
 | 
			
		||||
  }
 | 
			
		||||
  $: donor = 0;
 | 
			
		||||
  $: runner = 0;
 | 
			
		||||
  $: donors = [];
 | 
			
		||||
  $: runners = [];
 | 
			
		||||
  $: is_fixed = false;
 | 
			
		||||
  DonorService.donorControllerGetAll().then((val) => {
 | 
			
		||||
    donors = val.map((r) => {
 | 
			
		||||
      return { label: getDonorLabel(r), value: r };
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
  RunnerService.runnerControllerGetAll().then((val) => {
 | 
			
		||||
    runners = val.map((r) => {
 | 
			
		||||
      return { label: getDonorLabel(r), value: r };
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
  $: is_paid = false;
 | 
			
		||||
  $: amount_input = 0;
 | 
			
		||||
  $: processed_last_submit = true;
 | 
			
		||||
  $: is_amount_valid = amount_input > 0;
 | 
			
		||||
@@ -56,15 +45,16 @@
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      let amount_cent = Math.floor(amount_input * 100);
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: "adding donation",
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("adding-donation"));
 | 
			
		||||
      if (is_fixed) {
 | 
			
		||||
        let postdata = {
 | 
			
		||||
          donor,
 | 
			
		||||
          amount: amount_cent,
 | 
			
		||||
          paidAmount: 0,
 | 
			
		||||
        };
 | 
			
		||||
        if (is_paid) {
 | 
			
		||||
          postdata.paidAmount = amount_cent;
 | 
			
		||||
        }
 | 
			
		||||
        DonationService.donationControllerPostFixed(postdata)
 | 
			
		||||
          .then((result) => {
 | 
			
		||||
            donor = donors[0].id || 0;
 | 
			
		||||
@@ -72,21 +62,15 @@
 | 
			
		||||
            amount_input = 0;
 | 
			
		||||
            modal_open = false;
 | 
			
		||||
            //
 | 
			
		||||
            Toastify({
 | 
			
		||||
              text: "donation_added",
 | 
			
		||||
              duration: 500,
 | 
			
		||||
              backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
            }).showToast();
 | 
			
		||||
            current_donations.push(result);
 | 
			
		||||
            current_donations = current_donations;
 | 
			
		||||
            toast.dismiss();
 | 
			
		||||
            toast.success($_("donation_added"));
 | 
			
		||||
            dispatch("created", { donations: [result] });
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err) => {
 | 
			
		||||
            //
 | 
			
		||||
          })
 | 
			
		||||
          .finally(() => {
 | 
			
		||||
            processed_last_submit = true;
 | 
			
		||||
            //
 | 
			
		||||
            toast.hideToast();
 | 
			
		||||
          });
 | 
			
		||||
      } else {
 | 
			
		||||
        let postdata = {
 | 
			
		||||
@@ -101,29 +85,237 @@
 | 
			
		||||
            amount_input = 0;
 | 
			
		||||
            modal_open = false;
 | 
			
		||||
            //
 | 
			
		||||
            Toastify({
 | 
			
		||||
              text: "donation_added",
 | 
			
		||||
              duration: 500,
 | 
			
		||||
              backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
            }).showToast();
 | 
			
		||||
            current_donations.push(result);
 | 
			
		||||
            current_donations = current_donations;
 | 
			
		||||
            toast.dismiss();
 | 
			
		||||
            toast.success($_("donation_added"));
 | 
			
		||||
            dispatch("created", { donations: [result] });
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err) => {
 | 
			
		||||
            //
 | 
			
		||||
          })
 | 
			
		||||
          .finally(() => {
 | 
			
		||||
            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>
 | 
			
		||||
 | 
			
		||||
{#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="flex-shrink-0 flex items-center justify-center size-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">
 | 
			
		||||
              <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="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-2 lg: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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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>
 | 
			
		||||
  input:before {
 | 
			
		||||
  .toggle:before {
 | 
			
		||||
    content: "";
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    width: 1.25rem;
 | 
			
		||||
@@ -137,159 +329,12 @@
 | 
			
		||||
    transition: 0.2s ease-in-out;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  input:checked {
 | 
			
		||||
  .toggle:checked {
 | 
			
		||||
    /* @apply: bg-indigo-400; */
 | 
			
		||||
    background-color: #7f9cf5;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  input:checked:before {
 | 
			
		||||
  .toggle:checked:before {
 | 
			
		||||
    left: 1.25rem;
 | 
			
		||||
  }
 | 
			
		||||
</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 size-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="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-2 lg: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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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="flex-shrink-0 flex items-center justify-center size-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 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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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,286 +1,352 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import {
 | 
			
		||||
    DonationService,
 | 
			
		||||
    DonorService,
 | 
			
		||||
    RunnerService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  let data_loaded = false;
 | 
			
		||||
  export let params;
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: original_data = {};
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: donor = {};
 | 
			
		||||
  $: runner = {};
 | 
			
		||||
  $: current_donors = [];
 | 
			
		||||
  $: current_runners = [];
 | 
			
		||||
  $: amount_input = 0;
 | 
			
		||||
  $: is_amount_valid = amount_input > 0;
 | 
			
		||||
  $: is_everything_set =
 | 
			
		||||
    editable.donor != null &&
 | 
			
		||||
    ((original_data.responseType == "DISTANCEDONATION" &&
 | 
			
		||||
      editable?.runner != null) ||
 | 
			
		||||
      original_data.responseType !== "DISTANCEDONATION");
 | 
			
		||||
  $: changes_performed =
 | 
			
		||||
    !(JSON.stringify(original_data) === JSON.stringify(editable)) ||
 | 
			
		||||
    (original_data.responseType == "DISTANCEDONATION" &&
 | 
			
		||||
      !(Math.floor(amount_input * 100) === original_data.amountPerDistance)) ||
 | 
			
		||||
    (original_data.responseType !== "DISTANCEDONATION" &&
 | 
			
		||||
      !(Math.floor(amount_input * 100) === original_data.amount));
 | 
			
		||||
  $: save_enabled = changes_performed && is_amount_valid && is_everything_set;
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import {
 | 
			
		||||
		DonationService,
 | 
			
		||||
		DonorService,
 | 
			
		||||
		RunnerService,
 | 
			
		||||
	} from "@odit/lfk-client-js";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
 | 
			
		||||
  const promise = DonationService.donationControllerGetOne(
 | 
			
		||||
    params.donationid
 | 
			
		||||
  ).then((data) => {
 | 
			
		||||
    data_loaded = true;
 | 
			
		||||
    original_data = Object.assign(original_data, data);
 | 
			
		||||
    editable = Object.assign(editable, original_data);
 | 
			
		||||
    if (data.responseType == "DISTANCEDONATION") {
 | 
			
		||||
      amount_input = data.amountPerDistance / 100;
 | 
			
		||||
      RunnerService.runnerControllerGetAll().then((val) => {
 | 
			
		||||
        current_runners = val.map((r) => {
 | 
			
		||||
          return { label: getDonorLabel(r), value: r };
 | 
			
		||||
        });
 | 
			
		||||
        runner = current_runners.find((g) => g.value.id == editable.runner.id);
 | 
			
		||||
      });
 | 
			
		||||
    } else {
 | 
			
		||||
      amount_input = data.amount / 100;
 | 
			
		||||
    }
 | 
			
		||||
    DonorService.donorControllerGetAll().then((val) => {
 | 
			
		||||
      current_donors = val.map((r) => {
 | 
			
		||||
        return { label: getDonorLabel(r), value: r };
 | 
			
		||||
      });
 | 
			
		||||
      donor = current_donors.find((g) => g.value.id == editable.donor.id);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
  const getDonorLabel = (option) =>
 | 
			
		||||
    option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
  const filterDonors = (label, filterText, option) =>
 | 
			
		||||
    label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
    option.value.id.toString().startsWith(filterText.toLowerCase());
 | 
			
		||||
	import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
	import Select from "svelte-select";
 | 
			
		||||
	let data_loaded = false;
 | 
			
		||||
	export let params;
 | 
			
		||||
	$: delete_triggered = false;
 | 
			
		||||
	$: original_data = {};
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: donor = {};
 | 
			
		||||
	$: runner = {};
 | 
			
		||||
	$: current_donors = [];
 | 
			
		||||
	$: current_runners = [];
 | 
			
		||||
	$: amount_input = 0;
 | 
			
		||||
	$: is_amount_valid = amount_input > 0;
 | 
			
		||||
	$: paid_amount_input = 0;
 | 
			
		||||
	$: is_paid_amount_valid = paid_amount_input > 0;
 | 
			
		||||
	$: is_everything_set =
 | 
			
		||||
		editable.donor != null &&
 | 
			
		||||
		((original_data.responseType == "DISTANCEDONATION" &&
 | 
			
		||||
			editable?.runner != null) ||
 | 
			
		||||
			original_data.responseType !== "DISTANCEDONATION");
 | 
			
		||||
	$: changes_performed =
 | 
			
		||||
		!(JSON.stringify(original_data) === JSON.stringify(editable)) ||
 | 
			
		||||
		(original_data.responseType == "DISTANCEDONATION" &&
 | 
			
		||||
			!(Math.floor(amount_input * 100) === original_data.amountPerDistance)) ||
 | 
			
		||||
		(original_data.responseType !== "DISTANCEDONATION" &&
 | 
			
		||||
			!(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;
 | 
			
		||||
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: "Donation is being updated",
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      let postdata = {};
 | 
			
		||||
      if (original_data.responseType === "DISTANCEDONATION") {
 | 
			
		||||
        editable.amountPerDistance = Math.floor(amount_input * 100);
 | 
			
		||||
        postdata = Object.assign(postdata, editable);
 | 
			
		||||
        postdata.runner = postdata.runner.id;
 | 
			
		||||
        postdata.donor = postdata.donor.id;
 | 
			
		||||
        DonationService.donationControllerPutDistance(
 | 
			
		||||
          original_data.id,
 | 
			
		||||
          postdata
 | 
			
		||||
        )
 | 
			
		||||
          .then((resp) => {
 | 
			
		||||
            Object.assign(original_data, editable);
 | 
			
		||||
            original_data = original_data;
 | 
			
		||||
            Toastify({
 | 
			
		||||
              text: "updated donation",
 | 
			
		||||
              duration: 2500,
 | 
			
		||||
              backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
            }).showToast();
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err) => {});
 | 
			
		||||
      } else {
 | 
			
		||||
        editable.amount = Math.floor(amount_input * 100);
 | 
			
		||||
        postdata = Object.assign(postdata, editable);
 | 
			
		||||
        postdata.donor = postdata.donor.id;
 | 
			
		||||
        DonationService.donationControllerPutFixed(original_data.id, postdata)
 | 
			
		||||
          .then((resp) => {
 | 
			
		||||
            Object.assign(original_data, editable);
 | 
			
		||||
            original_data = original_data;
 | 
			
		||||
            Toastify({
 | 
			
		||||
              text: "updated donation",
 | 
			
		||||
              duration: 2500,
 | 
			
		||||
              backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
            }).showToast();
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err) => {});
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  function deleteDonation() {
 | 
			
		||||
    DonationService.donationControllerRemove(original_data.id, false)
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        Toastify({
 | 
			
		||||
          text: "Donation delete",
 | 
			
		||||
          duration: 500,
 | 
			
		||||
          backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {
 | 
			
		||||
        modal_open = true;
 | 
			
		||||
        delete_donor = original_data;
 | 
			
		||||
      });
 | 
			
		||||
  }
 | 
			
		||||
	const promise = DonationService.donationControllerGetOne(
 | 
			
		||||
		params.donationid
 | 
			
		||||
	).then((data) => {
 | 
			
		||||
		data_loaded = true;
 | 
			
		||||
		original_data = Object.assign({}, data);
 | 
			
		||||
		editable = Object.assign({}, original_data);
 | 
			
		||||
		paid_amount_input = data.paidAmount / 100;
 | 
			
		||||
		if (data.responseType == "DISTANCEDONATION") {
 | 
			
		||||
			amount_input = data.amountPerDistance / 100;
 | 
			
		||||
			RunnerService.runnerControllerGetAll().then((val) => {
 | 
			
		||||
				current_runners = val.map((r) => {
 | 
			
		||||
					return { label: getDonorLabel(r), value: r };
 | 
			
		||||
				});
 | 
			
		||||
				runner = current_runners.find((g) => g.value.id == editable.runner.id);
 | 
			
		||||
			});
 | 
			
		||||
		} else {
 | 
			
		||||
			amount_input = data.amount / 100;
 | 
			
		||||
		}
 | 
			
		||||
		DonorService.donorControllerGetAll().then((val) => {
 | 
			
		||||
			current_donors = val.map((r) => {
 | 
			
		||||
				return { label: getDonorLabel(r), value: r };
 | 
			
		||||
			});
 | 
			
		||||
			donor = current_donors.find((g) => g.value.id == editable.donor.id);
 | 
			
		||||
		});
 | 
			
		||||
	});
 | 
			
		||||
	const getDonorLabel = (option) =>
 | 
			
		||||
		option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
	const filterDonors = (label, filterText, option) =>
 | 
			
		||||
		label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
		option.value.id.toString().startsWith(filterText.toLowerCase());
 | 
			
		||||
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast($_("updating-donation"));
 | 
			
		||||
			let postdata = {};
 | 
			
		||||
			editable.paidAmount = paid_amount_input * 100;
 | 
			
		||||
			if (original_data.responseType === "DISTANCEDONATION") {
 | 
			
		||||
				editable.amountPerDistance = Math.floor(amount_input * 100);
 | 
			
		||||
				postdata = Object.assign(postdata, editable);
 | 
			
		||||
				postdata.runner = postdata.runner.id;
 | 
			
		||||
				postdata.donor = postdata.donor.id;
 | 
			
		||||
				DonationService.donationControllerPutDistance(
 | 
			
		||||
					original_data.id,
 | 
			
		||||
					postdata
 | 
			
		||||
				)
 | 
			
		||||
					.then((resp) => {
 | 
			
		||||
						Object.assign(original_data, editable);
 | 
			
		||||
						original_data = original_data;
 | 
			
		||||
						toast.success($_("donation-updated"));
 | 
			
		||||
					})
 | 
			
		||||
					.catch((err) => {});
 | 
			
		||||
			} else {
 | 
			
		||||
				editable.amount = Math.floor(amount_input * 100);
 | 
			
		||||
				postdata = Object.assign(postdata, editable);
 | 
			
		||||
				postdata.donor = postdata.donor.id;
 | 
			
		||||
				DonationService.donationControllerPutFixed(original_data.id, postdata)
 | 
			
		||||
					.then((resp) => {
 | 
			
		||||
						Object.assign(original_data, editable);
 | 
			
		||||
						original_data = original_data;
 | 
			
		||||
						toast.success($_("donation-updated"));
 | 
			
		||||
					})
 | 
			
		||||
					.catch((err) => {});
 | 
			
		||||
			}
 | 
			
		||||
		} else {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	function deleteDonation() {
 | 
			
		||||
		DonationService.donationControllerRemove(original_data.id, false)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				toast.success($_("donation-deleted"));
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {
 | 
			
		||||
				modal_open = true;
 | 
			
		||||
				delete_donor = original_data;
 | 
			
		||||
			});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#await promise}
 | 
			
		||||
  {$_('loading-donation-details')}
 | 
			
		||||
	{$_("loading-donation-details")}
 | 
			
		||||
{:then}
 | 
			
		||||
  <section class="container p-5 select-none">
 | 
			
		||||
    <div class="flex flex-row mb-4">
 | 
			
		||||
      <div class="w-full">
 | 
			
		||||
        <nav class="w-full flex">
 | 
			
		||||
          <ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <svg
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                width="24"
 | 
			
		||||
                height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                <path
 | 
			
		||||
                  d="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"
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                class="h-3 w-3 mr-2 stroke-current"
 | 
			
		||||
                height="1em"
 | 
			
		||||
                width="1em"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"><line
 | 
			
		||||
                  x1="5"
 | 
			
		||||
                  y1="12"
 | 
			
		||||
                  x2="19"
 | 
			
		||||
                  y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">{original_data.id}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mb-8 text-3xl font-extrabold leading-tight">
 | 
			
		||||
      {original_data.donor.firstname}
 | 
			
		||||
      {original_data.donor.middlename || ''}
 | 
			
		||||
      {original_data.donor.lastname}
 | 
			
		||||
      >
 | 
			
		||||
      {#if original_data.responseType == 'DISTANCEDONATION'}
 | 
			
		||||
        {original_data.runner.firstname}
 | 
			
		||||
        {original_data.runner.middlename || ''}
 | 
			
		||||
        {original_data.runner.lastname}
 | 
			
		||||
      {:else}
 | 
			
		||||
        {$_('fixed-donation')}:
 | 
			
		||||
        {amount_input.toFixed(2).toLocaleString('de-DE', { valute: 'EUR' })}€
 | 
			
		||||
      {/if}
 | 
			
		||||
      <span data-id="donation_actions_${original_data.id}">
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:DELETE')}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              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>
 | 
			
		||||
            <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:">{$_('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:">{$_('delete-donation')}</button>
 | 
			
		||||
          {/if}
 | 
			
		||||
        {/if}
 | 
			
		||||
        {#if !delete_triggered}
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!save_enabled}
 | 
			
		||||
            class:opacity-50={!save_enabled}
 | 
			
		||||
            type="button"
 | 
			
		||||
            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>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="font-medium text-gray-700">{$_('total-donation-amount')}:</span>
 | 
			
		||||
      <span>{(editable.amount / 100)
 | 
			
		||||
          .toFixed(2)
 | 
			
		||||
          .toLocaleString('de-DE', { valute: 'EUR' })}€</span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="donor"
 | 
			
		||||
        class="block  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={current_donors}
 | 
			
		||||
        showChevron={true}
 | 
			
		||||
        placeholder={$_('search-for-donor-name-or-id')}
 | 
			
		||||
        noOptionsMessage={$_('no-donors-found')}
 | 
			
		||||
        bind:selectedValue={donor}
 | 
			
		||||
        on:select={(selectedValue) => (editable.donor = selectedValue.detail.value)}
 | 
			
		||||
        on:clear={() => (editable.donor = null)} />
 | 
			
		||||
    </div>
 | 
			
		||||
    {#if original_data.responseType == 'DISTANCEDONATION'}
 | 
			
		||||
      <div class=" w-full">
 | 
			
		||||
        <label
 | 
			
		||||
          for="donor"
 | 
			
		||||
          class="block  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={current_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)}
 | 
			
		||||
          on:clear={() => (editable.runner = null)} />
 | 
			
		||||
      </div>
 | 
			
		||||
    {/if}
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label for="lastname" class="font-medium text-gray-700">
 | 
			
		||||
        {#if original_data.responseType == 'DISTANCEDONATION'}
 | 
			
		||||
          {$_('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: 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 ">€</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>
 | 
			
		||||
  </section>
 | 
			
		||||
	<section class="container p-5 select-none">
 | 
			
		||||
		<div class="flex flex-row mb-4">
 | 
			
		||||
			<div class="mt-2 w-full">
 | 
			
		||||
				<nav class="w-full flex">
 | 
			
		||||
					<ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="./"
 | 
			
		||||
								><svg
 | 
			
		||||
									xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
									width="24"
 | 
			
		||||
									height="24"
 | 
			
		||||
									viewBox="0 0 24 24"
 | 
			
		||||
									fill="none"
 | 
			
		||||
									stroke="currentColor"
 | 
			
		||||
									stroke-width="2"
 | 
			
		||||
									stroke-linecap="round"
 | 
			
		||||
									stroke-linejoin="round"
 | 
			
		||||
									class="inline-block"
 | 
			
		||||
									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg
 | 
			
		||||
								>
 | 
			
		||||
								{$_("donations")}</a
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
					</ol>
 | 
			
		||||
				</nav>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mb-4 text-3xl font-extrabold leading-tight">
 | 
			
		||||
			{original_data.donor.firstname}
 | 
			
		||||
			{original_data.donor.middlename || ""}
 | 
			
		||||
			{original_data.donor.lastname}
 | 
			
		||||
			>
 | 
			
		||||
			{#if original_data.responseType == "DISTANCEDONATION"}
 | 
			
		||||
				{original_data.runner.firstname}
 | 
			
		||||
				{original_data.runner.middlename || ""}
 | 
			
		||||
				{original_data.runner.lastname}
 | 
			
		||||
			{:else}
 | 
			
		||||
				{$_("fixed-donation")}:
 | 
			
		||||
				{amount_input.toFixed(2).toLocaleString("de-DE", { valute: "EUR" })}€
 | 
			
		||||
			{/if}
 | 
			
		||||
			[#{original_data.id}]
 | 
			
		||||
			<div data-id="donation_actions_${original_data.id}">
 | 
			
		||||
				{#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:DELETE")}
 | 
			
		||||
					{#if delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							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:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("confirm-deletion")}</button
 | 
			
		||||
						>
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = !delete_triggered;
 | 
			
		||||
							}}
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("cancel")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
					{#if !delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = true;
 | 
			
		||||
							}}
 | 
			
		||||
							type="button"
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("delete-donation")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
				{/if}
 | 
			
		||||
				{#if !delete_triggered}
 | 
			
		||||
					<button
 | 
			
		||||
						disabled={!save_enabled}
 | 
			
		||||
						class:opacity-50={!save_enabled}
 | 
			
		||||
						type="button"
 | 
			
		||||
						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:w-auto sm:text-sm mb-1 lg:mb-0"
 | 
			
		||||
						>{$_("save-changes")}</button
 | 
			
		||||
					>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		<div>
 | 
			
		||||
			<span class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("total-donation-amount")}:</span
 | 
			
		||||
			>
 | 
			
		||||
			<span
 | 
			
		||||
				>{(editable.amount / 100)
 | 
			
		||||
					.toFixed(2)
 | 
			
		||||
					.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>
 | 
			
		||||
		<br />
 | 
			
		||||
		<div class=" mt-2 w-full">
 | 
			
		||||
			<label for="donor" class="block font-semibold 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={current_donors}
 | 
			
		||||
				showChevron={true}
 | 
			
		||||
				placeholder={$_("search-for-donor-name-or-id")}
 | 
			
		||||
				noOptionsMessage={$_("no-donors-found")}
 | 
			
		||||
				bind:selectedValue={donor}
 | 
			
		||||
				on:select={(selectedValue) => {
 | 
			
		||||
					editable.donor = selectedValue.detail.value;
 | 
			
		||||
					editable.donor.donationAmount = original_data.donor.donationAmount;
 | 
			
		||||
					editable.donor.paidDonationAmount =
 | 
			
		||||
						original_data.donor.paidDonationAmount;
 | 
			
		||||
				}}
 | 
			
		||||
				on:clear={() => (editable.donor = null)}
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		{#if original_data.responseType == "DISTANCEDONATION"}
 | 
			
		||||
			<div class=" mt-2 w-full">
 | 
			
		||||
				<label for="donor" class="block font-semibold 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={current_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)}
 | 
			
		||||
					on:clear={() => (editable.runner = null)}
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
		{/if}
 | 
			
		||||
		<div class=" mt-2 w-full">
 | 
			
		||||
			<label for="lastname" class="font-semibold text-gray-700">
 | 
			
		||||
				{#if original_data.responseType == "DISTANCEDONATION"}
 | 
			
		||||
					{$_("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: 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"
 | 
			
		||||
					>€</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 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>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
	</section>
 | 
			
		||||
{:catch error}
 | 
			
		||||
  <PromiseError {error} />
 | 
			
		||||
	<PromiseError {error} />
 | 
			
		||||
{/await}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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";
 | 
			
		||||
  $: current_donations = [];
 | 
			
		||||
  export let modal_open = false;
 | 
			
		||||
  let addDonations;
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('donations')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('add-donation')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </span>
 | 
			
		||||
  <DonationsOverview bind:current_donations />
 | 
			
		||||
  <h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_("donations")}
 | 
			
		||||
  </h4>
 | 
			
		||||
  {#if store.state.jwtinfo.userdetails.permissions.includes("DONATION: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"
 | 
			
		||||
    >
 | 
			
		||||
      {$_("add-donation")}
 | 
			
		||||
    </button>
 | 
			
		||||
  {/if}
 | 
			
		||||
  <DonationsOverview bind:current_donations bind:addDonations />
 | 
			
		||||
</section>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:CREATE')}
 | 
			
		||||
  <AddDonationModal bind:current_donations bind:modal_open />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes("DONATION:CREATE")}
 | 
			
		||||
  <AddDonationModal
 | 
			
		||||
    on:created={(event) => {
 | 
			
		||||
      addDonations(event.detail.donations);
 | 
			
		||||
    }}
 | 
			
		||||
    bind:modal_open
 | 
			
		||||
  />
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -5,8 +5,8 @@
 | 
			
		||||
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="m-auto" style="height:15rem" src={donations_empty} alt="" />
 | 
			
		||||
    <span class="font-bold">{$_('there-are-no-donations-yet')}</span><br />
 | 
			
		||||
    <span>{$_('add-your-fist-donation')}</span>
 | 
			
		||||
    <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>{$_("add-your-fist-donation")}</span>
 | 
			
		||||
  </p>
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,194 +1,291 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
  import { DonationService, DonorService } from "@odit/lfk-client-js";
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { DonationService } from "@odit/lfk-client-js";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  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 = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  $: active_edits = [];
 | 
			
		||||
  $: selected =
 | 
			
		||||
    $table?.getSelectedRowModel().rows.map((row) => row.index) || [];
 | 
			
		||||
  $: dataLoaded = false;
 | 
			
		||||
 | 
			
		||||
  export let current_donations = [];
 | 
			
		||||
  const donations_promise = DonationService.donationControllerGetAll().then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      current_donations = val;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  function should_display_based_on_id(id) {
 | 
			
		||||
    if (searchvalue.toString().slice(-1) === "*") {
 | 
			
		||||
      return id.toString().startsWith(searchvalue.replace("*", ""));
 | 
			
		||||
    }
 | 
			
		||||
    return id.toString() === searchvalue;
 | 
			
		||||
  export const addDonations = (donations) => {
 | 
			
		||||
    current_donations = current_donations.concat(...donations);
 | 
			
		||||
    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
 | 
			
		||||
            );
 | 
			
		||||
          },
 | 
			
		||||
          paymentAction: () => {
 | 
			
		||||
            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"));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  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>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('DONATION:GET')}
 | 
			
		||||
  {#await donations_promise}
 | 
			
		||||
<AddDonationPaymentModal
 | 
			
		||||
  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
 | 
			
		||||
      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">donations are being loaded</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
      role="alert"
 | 
			
		||||
    >
 | 
			
		||||
      <p class="font-bold">{$_("donations-are-being-loaded")}</p>
 | 
			
		||||
      <p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_donations.length === 0}
 | 
			
		||||
      <DonationsEmptyState />
 | 
			
		||||
    {:else}
 | 
			
		||||
      <input
 | 
			
		||||
        type="search"
 | 
			
		||||
        bind:value={searchvalue}
 | 
			
		||||
        placeholder={$_('datatable.search')}
 | 
			
		||||
        aria-label={$_('datatable.search')}
 | 
			
		||||
        class="gridjs-input gridjs-search-input mb-4" />
 | 
			
		||||
      <div
 | 
			
		||||
        class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll">
 | 
			
		||||
        <table class="divide-y divide-gray-200 w-full">
 | 
			
		||||
          <thead class="bg-gray-50">
 | 
			
		||||
            <tr>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('donor')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('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>
 | 
			
		||||
  {:else if current_donations.length === 0}
 | 
			
		||||
    <DonationsEmptyState />
 | 
			
		||||
  {:else}
 | 
			
		||||
    <input
 | 
			
		||||
      type="search"
 | 
			
		||||
      bind:value={searchvalue}
 | 
			
		||||
      placeholder={$_("datatable.search")}
 | 
			
		||||
      aria-label={$_("datatable.search")}
 | 
			
		||||
      class="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border"
 | 
			
		||||
    />
 | 
			
		||||
    <div
 | 
			
		||||
      class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"
 | 
			
		||||
    >
 | 
			
		||||
      <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>
 | 
			
		||||
          </thead>
 | 
			
		||||
          <tbody class="divide-y divide-gray-200">
 | 
			
		||||
            {#each current_donations as donation}
 | 
			
		||||
              {#if donation.donor.firstname
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .includes(
 | 
			
		||||
                  searchvalue.toLowerCase()
 | 
			
		||||
                ) ||  donation.donor.lastname
 | 
			
		||||
                  .toLowerCase()
 | 
			
		||||
                  .includes(
 | 
			
		||||
                    searchvalue.toLowerCase()
 | 
			
		||||
                  ) || donation.runner?.firstname
 | 
			
		||||
                  .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 class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    {#if donation.runner}
 | 
			
		||||
                      <div class="text-sm font-medium text-gray-900">
 | 
			
		||||
                        <a
 | 
			
		||||
                          href="../runners/{donation.runner.id}"
 | 
			
		||||
                          class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{donation.runner.firstname}
 | 
			
		||||
                          {donation.runner.middlename || ''}
 | 
			
		||||
                          {donation.runner.lastname}</a>
 | 
			
		||||
                      </div>
 | 
			
		||||
                    {:else}
 | 
			
		||||
                      <div class="text-sm font-medium text-gray-900">
 | 
			
		||||
                        {$_('fixed-donation')}
 | 
			
		||||
                      </div>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </td>
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    {#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>
 | 
			
		||||
              {/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>
 | 
			
		||||
          {/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>
 | 
			
		||||
  {/await}
 | 
			
		||||
    <div class="h-2" />
 | 
			
		||||
    <TableBottom {table} {selected} />
 | 
			
		||||
  {/if}
 | 
			
		||||
{/if}
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
  table tbody tr td:nth-child(2) {
 | 
			
		||||
    font-family: monospace;
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,14 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
  import {
 | 
			
		||||
    DonorService
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
 | 
			
		||||
  import { DonorService } from "@odit/lfk-client-js";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  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 current_donors;
 | 
			
		||||
  let firstname_input;
 | 
			
		||||
  let lastname_input;
 | 
			
		||||
  let middlename_input;
 | 
			
		||||
@@ -19,6 +18,7 @@
 | 
			
		||||
  let address_input2;
 | 
			
		||||
  let address_zipcode;
 | 
			
		||||
  let address_city;
 | 
			
		||||
  const dispatch = createEventDispatcher();
 | 
			
		||||
  function focus(el) {
 | 
			
		||||
    el.focus();
 | 
			
		||||
  }
 | 
			
		||||
@@ -74,10 +74,7 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_('donor-is-being-added'),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("donor-is-being-added"));
 | 
			
		||||
      let address = {};
 | 
			
		||||
      if (address_checked === true) {
 | 
			
		||||
        address = {
 | 
			
		||||
@@ -92,7 +89,7 @@
 | 
			
		||||
        firstname: firstname_input_value,
 | 
			
		||||
        lastname: lastname_input_value,
 | 
			
		||||
        address,
 | 
			
		||||
        receiptNeeded: address_checked
 | 
			
		||||
        receiptNeeded: address_checked,
 | 
			
		||||
      };
 | 
			
		||||
      if (middlename_input_value) {
 | 
			
		||||
        postdata.middlename = middlename_input_value;
 | 
			
		||||
@@ -111,21 +108,15 @@
 | 
			
		||||
          email_input_value = "";
 | 
			
		||||
          modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_('donor-added'),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          current_donors.push(result);
 | 
			
		||||
          current_donors = current_donors;
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("donor-added"));
 | 
			
		||||
          dispatch("created", { donors: [result] });
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {
 | 
			
		||||
          //
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -133,59 +124,71 @@
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-hidden"
 | 
			
		||||
    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">
 | 
			
		||||
      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" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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" />
 | 
			
		||||
                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>
 | 
			
		||||
                  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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-a-new-donor')}
 | 
			
		||||
                {$_("create-a-new-donor")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <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>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    use:focus
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('first-name')}
 | 
			
		||||
                    placeholder={$_("first-name")}
 | 
			
		||||
                    class:border-red-500={!isFirstnameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
@@ -193,34 +196,41 @@
 | 
			
		||||
                    bind:this={firstname_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('first-name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("first-name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('middle-name')}
 | 
			
		||||
                    placeholder={$_("middle-name")}
 | 
			
		||||
                    bind:value={middlename_input_value}
 | 
			
		||||
                    bind:this={middlename_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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 class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('last-name')}"
 | 
			
		||||
                    placeholder={$_("last-name")}
 | 
			
		||||
                    class:border-red-500={!isLastnameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
@@ -228,21 +238,25 @@
 | 
			
		||||
                    bind:this={lastname_input}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('last-name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("last-name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('phone')}
 | 
			
		||||
                    placeholder={$_("phone")}
 | 
			
		||||
                    class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
                    class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
                    class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
@@ -250,21 +264,27 @@
 | 
			
		||||
                    bind:this={phone_input}
 | 
			
		||||
                    type="tel"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      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')}
 | 
			
		||||
                      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"
 | 
			
		||||
                      )}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('e-mail-adress')}
 | 
			
		||||
                    placeholder={$_("e-mail-adress")}
 | 
			
		||||
                    class:border-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
                    class:focus:border-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
                    class:focus:ring-red-500={!isEmailValidOrEmpty}
 | 
			
		||||
@@ -272,11 +292,13 @@
 | 
			
		||||
                    bind:this={email_input}
 | 
			
		||||
                    type="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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('valid-email-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("valid-email-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
@@ -287,19 +309,22 @@
 | 
			
		||||
                      id="comments"
 | 
			
		||||
                      name="comments"
 | 
			
		||||
                      type="checkbox"
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" />
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="ml-3 text-sm">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="comments"
 | 
			
		||||
                      class="font-medium text-gray-700">{$_('receipt-needed')}</label>
 | 
			
		||||
                    <label for="comments" class="font-semibold text-gray-700"
 | 
			
		||||
                      >{$_("receipt-needed")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                {#if address_checked === true}
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      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
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder="Address"
 | 
			
		||||
@@ -310,34 +335,41 @@
 | 
			
		||||
                      bind:this={address_input1}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address1"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('address-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("address-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="address2"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("apartment-suite-etc")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_('apartment-suite-etc')}
 | 
			
		||||
                      placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
                      bind:value={address_input2_value}
 | 
			
		||||
                      bind:this={address_input2}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address2"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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 class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      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
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_('zip-postal-code')}
 | 
			
		||||
                      placeholder={$_("zip-postal-code")}
 | 
			
		||||
                      class:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
@@ -345,18 +377,22 @@
 | 
			
		||||
                      bind:this={address_zipcode}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="zipcode"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="city"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700">City</label>
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >City</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder="City"
 | 
			
		||||
@@ -367,11 +403,13 @@
 | 
			
		||||
                      bind:this={address_city}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="city"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                      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}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                        {$_('valid-city-is-required')}
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-city-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
@@ -380,22 +418,24 @@
 | 
			
		||||
            </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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,92 +1,90 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  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";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  export let delete_donor;
 | 
			
		||||
  const dispatch = createEventDispatcher();
 | 
			
		||||
  function cancelDelete() {
 | 
			
		||||
    modal_open = false;
 | 
			
		||||
    dispatch("cancelDelete", { id: delete_donor.id });
 | 
			
		||||
  }
 | 
			
		||||
  function deleteDonor() {
 | 
			
		||||
    DonorService.donorControllerRemove(
 | 
			
		||||
      delete_donor.id,
 | 
			
		||||
      true
 | 
			
		||||
    )
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        Toastify({
 | 
			
		||||
          text: "Donor deleted",
 | 
			
		||||
          duration: 500,
 | 
			
		||||
          backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {});
 | 
			
		||||
  }
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
	import { createEventDispatcher } from "svelte";
 | 
			
		||||
	export let modal_open;
 | 
			
		||||
	export let delete_donor;
 | 
			
		||||
	const dispatch = createEventDispatcher();
 | 
			
		||||
	function cancelDelete() {
 | 
			
		||||
		modal_open = false;
 | 
			
		||||
		dispatch("cancelDelete", { id: delete_donor.id });
 | 
			
		||||
	}
 | 
			
		||||
	function deleteDonor() {
 | 
			
		||||
		dispatch("delete", { id: delete_donor.id });
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    use:clickOutside
 | 
			
		||||
    on:click_outside={cancelDelete}>
 | 
			
		||||
    <div
 | 
			
		||||
      class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
      <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
        <div
 | 
			
		||||
          class="absolute inset-0 bg-gray-500 opacity-75"
 | 
			
		||||
          data-id="modal_backdrop" />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
      <div
 | 
			
		||||
        class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full"
 | 
			
		||||
        role="dialog"
 | 
			
		||||
        aria-modal="true"
 | 
			
		||||
        aria-labelledby="modal-headline">
 | 
			
		||||
        <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
            <div
 | 
			
		||||
              class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10">
 | 
			
		||||
              <svg class="h-6 w-6 text-blue-600" fill="currentColor" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('attention')}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
                <p class="text-sm text-gray-500">
 | 
			
		||||
                  {$_(
 | 
			
		||||
                    'do-you-want-to-delete-this-donor-with-all-related-donations'
 | 
			
		||||
                  )}
 | 
			
		||||
                  <br /> 
 | 
			
		||||
                  {$_('all-associated-donations-will-get-deleted-as-well')}
 | 
			
		||||
                </p>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
          <button
 | 
			
		||||
            on:click={deleteDonor}
 | 
			
		||||
            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-donor-with-all-donations')}
 | 
			
		||||
          </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-donor')}
 | 
			
		||||
          </button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
	<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="flex-shrink-0 flex items-center justify-center size-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
 | 
			
		||||
							>
 | 
			
		||||
						</div>
 | 
			
		||||
						<div class="mt-3 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-this-donor-with-all-related-donations"
 | 
			
		||||
								)}
 | 
			
		||||
							</h3>
 | 
			
		||||
							<div class="mb-6">
 | 
			
		||||
								<p class="text-sm text-gray-500">
 | 
			
		||||
									{$_("all-associated-donations-will-get-deleted-as-well")}
 | 
			
		||||
								</p>
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
				<div class="bg-gray-50 px-4 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
					<button
 | 
			
		||||
						on:click={deleteDonor}
 | 
			
		||||
						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-donor-with-all-donations")}
 | 
			
		||||
					</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-donor")}
 | 
			
		||||
					</button>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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,406 +1,413 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import { DonorService, DonationService } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  import ConfirmDonorDeletion from "./ConfirmDonorDeletion.svelte";
 | 
			
		||||
  let data_loaded = false;
 | 
			
		||||
  export let params;
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: original_data = {};
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: current_donations = [];
 | 
			
		||||
  $: changes_performed = !(
 | 
			
		||||
    JSON.stringify(original_data) === JSON.stringify(editable)
 | 
			
		||||
  );
 | 
			
		||||
  $: isEmailValid =
 | 
			
		||||
    (editable.email || "") === "" ||
 | 
			
		||||
    (editable.email && isEmail(editable.email || ""));
 | 
			
		||||
  $: isFirstnameValid = editable.firstname !== "";
 | 
			
		||||
  $: isLastnameValid = editable.lastname !== "";
 | 
			
		||||
  $: save_enabled =
 | 
			
		||||
    changes_performed &&
 | 
			
		||||
    isFirstnameValid &&
 | 
			
		||||
    isLastnameValid &&
 | 
			
		||||
    isEmailValid &&
 | 
			
		||||
    isPhoneValidOrEmpty &&
 | 
			
		||||
    ((isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
      editable.address_checked === false);
 | 
			
		||||
  const donation_promise = DonationService.donationControllerGetAll().then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      current_donations = val;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  const promise = DonorService.donorControllerGetOne(params.donorid).then(
 | 
			
		||||
    (data) => {
 | 
			
		||||
      data_loaded = true;
 | 
			
		||||
      original_data = Object.assign(original_data, data);
 | 
			
		||||
      editable = Object.assign(editable, original_data);
 | 
			
		||||
      editable.address_checked = editable.address.address1 !== null;
 | 
			
		||||
      original_data.address_checked = editable.address.address1 !== null;
 | 
			
		||||
      if (editable.address_checked === false) {
 | 
			
		||||
        editable.address = {
 | 
			
		||||
          address1: "",
 | 
			
		||||
          address2: "",
 | 
			
		||||
          city: "",
 | 
			
		||||
          postalcode: "",
 | 
			
		||||
          country: "",
 | 
			
		||||
        };
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  $: isPhoneValidOrEmpty =
 | 
			
		||||
    editable.phone?.includes("+") ||
 | 
			
		||||
    editable.phone === "" ||
 | 
			
		||||
    editable.phone === null;
 | 
			
		||||
  $: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
  $: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
  $: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
  let modal_open = false;
 | 
			
		||||
  let delete_donor = {};
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("donor-is-being-updated"),
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      editable.address.country = "DE";
 | 
			
		||||
      if (editable.address_checked === false) {
 | 
			
		||||
        editable.address = null;
 | 
			
		||||
      }
 | 
			
		||||
      if (editable.email) editable.email = editable.email;
 | 
			
		||||
      if (editable.phone) editable.phone = editable.phone;
 | 
			
		||||
      if (editable.middlename) editable.middlename = editable.middlename;
 | 
			
		||||
      editable.receiptNeeded = editable.address_checked;
 | 
			
		||||
      DonorService.donorControllerPut(original_data.id, editable)
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Object.assign(original_data, editable);
 | 
			
		||||
          original_data = original_data;
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("updated-donor"),
 | 
			
		||||
            duration: 2500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  function deleteDonor() {
 | 
			
		||||
    DonorService.donorControllerRemove(original_data.id, false)
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        Toastify({
 | 
			
		||||
          text: $_("donor-deleted"),
 | 
			
		||||
          duration: 500,
 | 
			
		||||
          backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {
 | 
			
		||||
        modal_open = true;
 | 
			
		||||
        delete_donor = original_data;
 | 
			
		||||
      });
 | 
			
		||||
  }
 | 
			
		||||
	import { DonorService } from "@odit/lfk-client-js";
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
	import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
	let data_loaded = false;
 | 
			
		||||
	export let params;
 | 
			
		||||
	$: delete_triggered = false;
 | 
			
		||||
	$: original_data = {};
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: changes_performed = !(
 | 
			
		||||
		JSON.stringify(original_data) === JSON.stringify(editable)
 | 
			
		||||
	);
 | 
			
		||||
	$: isEmailValid =
 | 
			
		||||
		(editable.email || "") === "" ||
 | 
			
		||||
		(editable.email && isEmail(editable.email || ""));
 | 
			
		||||
	$: isFirstnameValid = editable.firstname !== "";
 | 
			
		||||
	$: isLastnameValid = editable.lastname !== "";
 | 
			
		||||
	$: save_enabled =
 | 
			
		||||
		changes_performed &&
 | 
			
		||||
		isFirstnameValid &&
 | 
			
		||||
		isLastnameValid &&
 | 
			
		||||
		isEmailValid &&
 | 
			
		||||
		isPhoneValidOrEmpty &&
 | 
			
		||||
		((isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
			editable.address_checked === false);
 | 
			
		||||
	const promise = DonorService.donorControllerGetOne(params.donorid).then(
 | 
			
		||||
		(data) => {
 | 
			
		||||
			data_loaded = true;
 | 
			
		||||
			original_data = Object.assign(original_data, data);
 | 
			
		||||
			editable = Object.assign(editable, original_data);
 | 
			
		||||
			editable.address_checked = editable.address.address1 !== null;
 | 
			
		||||
			original_data.address_checked = editable.address.address1 !== null;
 | 
			
		||||
			if (editable.address_checked === false) {
 | 
			
		||||
				editable.address = {
 | 
			
		||||
					address1: "",
 | 
			
		||||
					address2: "",
 | 
			
		||||
					city: "",
 | 
			
		||||
					postalcode: "",
 | 
			
		||||
					country: "",
 | 
			
		||||
				};
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	);
 | 
			
		||||
	$: isPhoneValidOrEmpty =
 | 
			
		||||
		editable.phone?.includes("+") ||
 | 
			
		||||
		editable.phone === "" ||
 | 
			
		||||
		editable.phone === null;
 | 
			
		||||
	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
	$: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast($_("donor-is-being-updated"));
 | 
			
		||||
			editable.address.country = "DE";
 | 
			
		||||
			if (editable.address_checked === false) {
 | 
			
		||||
				editable.address = null;
 | 
			
		||||
			}
 | 
			
		||||
			if (editable.email) editable.email = editable.email;
 | 
			
		||||
			else editable.email = null;
 | 
			
		||||
			if (editable.phone) editable.phone = editable.phone;
 | 
			
		||||
			else editable.phone = null;
 | 
			
		||||
			if (editable.middlename) editable.middlename = editable.middlename;
 | 
			
		||||
			editable.receiptNeeded = editable.address_checked;
 | 
			
		||||
			DonorService.donorControllerPut(original_data.id, editable)
 | 
			
		||||
				.then((resp) => {
 | 
			
		||||
					Object.assign(original_data, editable);
 | 
			
		||||
					original_data = original_data;
 | 
			
		||||
					toast.success($_("updated-donor"));
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		} else {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	function deleteDonor() {
 | 
			
		||||
		DonorService.donorControllerRemove(original_data.id, true)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				toast.success($_("donor-deleted"));
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {
 | 
			
		||||
				console.log(err);
 | 
			
		||||
			});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<ConfirmDonorDeletion bind:modal_open bind:delete_donor />
 | 
			
		||||
{#await promise && donation_promise}
 | 
			
		||||
  {$_('loading-donor-details')}
 | 
			
		||||
{#await promise}
 | 
			
		||||
	{$_("loading-donor-details")}
 | 
			
		||||
{:then}
 | 
			
		||||
  <section class="container p-5 select-none">
 | 
			
		||||
    <div class="flex flex-row mb-4">
 | 
			
		||||
      <div class="w-full">
 | 
			
		||||
        <nav class="w-full flex">
 | 
			
		||||
          <ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <svg
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                width="24"
 | 
			
		||||
                height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                <path
 | 
			
		||||
                  d="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"
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                class="h-3 w-3 mr-2 stroke-current"
 | 
			
		||||
                height="1em"
 | 
			
		||||
                width="1em"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"><line
 | 
			
		||||
                  x1="5"
 | 
			
		||||
                  y1="12"
 | 
			
		||||
                  x2="19"
 | 
			
		||||
                  y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">{original_data.firstname}
 | 
			
		||||
                {original_data.middlename || ''}
 | 
			
		||||
                {original_data.lastname}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mb-8 text-3xl font-extrabold leading-tight">
 | 
			
		||||
      {original_data.firstname}
 | 
			
		||||
      {original_data.middlename || ''}
 | 
			
		||||
      {original_data.lastname}
 | 
			
		||||
      <span data-id="donor_actions_${editable.id}">
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:DELETE')}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              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>
 | 
			
		||||
            <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:">{$_('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:">{$_('delete-donor')}</button>
 | 
			
		||||
          {/if}
 | 
			
		||||
        {/if}
 | 
			
		||||
        {#if !delete_triggered}
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!save_enabled}
 | 
			
		||||
            class:opacity-50={!save_enabled}
 | 
			
		||||
            type="button"
 | 
			
		||||
            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>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="font-medium text-gray-700">{$_('total-donation-amount')}:</span>
 | 
			
		||||
      <span>{(editable.donationAmount / 100)
 | 
			
		||||
          .toFixed(2)
 | 
			
		||||
          .toLocaleString('de-DE', { valute: 'EUR' })}€</span>
 | 
			
		||||
      <br />
 | 
			
		||||
      <span class="font-medium text-gray-700">{$_('donations')}:</span>
 | 
			
		||||
      {#if current_donations.filter((d) => d.donor.id == editable.id).length > 0}
 | 
			
		||||
        {#each current_donations.filter((o) => o.donor.id == editable.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}
 | 
			
		||||
      {:else}{$_('donor-has-no-associated-donations')}{/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="firstname"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('first-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('first-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        class:border-red-500={!isFirstnameValid}
 | 
			
		||||
        class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
        class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
      {#if !isFirstnameValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('first-name-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="middlename"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('middle-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('middle-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="lastname"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('last-name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('last-name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.lastname}
 | 
			
		||||
        class:border-red-500={!isLastnameValid}
 | 
			
		||||
        class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
        class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
        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" />
 | 
			
		||||
      {#if !isLastnameValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('last-name-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="email"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('e-mail-adress')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('e-mail-adress')}
 | 
			
		||||
        type="email"
 | 
			
		||||
        bind:value={editable.email}
 | 
			
		||||
        class:border-red-500={!isEmailValid}
 | 
			
		||||
        class:focus:border-red-500={!isEmailValid}
 | 
			
		||||
        class:focus:ring-red-500={!isEmailValid}
 | 
			
		||||
        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" />
 | 
			
		||||
      {#if !isEmailValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('valid-email-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class=" w-full">
 | 
			
		||||
      <label for="phone" class="font-medium text-gray-700">{$_('phone')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('phone')}
 | 
			
		||||
        type="tel"
 | 
			
		||||
        class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
      {#if !isPhoneValidOrEmpty}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('valid-international-phone-number-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="flex items-start mt-2">
 | 
			
		||||
      <div class="flex items-center h-5">
 | 
			
		||||
        <input
 | 
			
		||||
          bind:checked={editable.address_checked}
 | 
			
		||||
          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 ">
 | 
			
		||||
        <label
 | 
			
		||||
          for="comments"
 | 
			
		||||
          class="font-medium text-gray-700">{$_('receipt-needed')}</label>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    {#if editable.address_checked === true}
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="address1"
 | 
			
		||||
          class="block  font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder="Address"
 | 
			
		||||
          class:border-red-500={!isAddress1Valid}
 | 
			
		||||
          class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
          class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
          bind:value={editable.address.address1}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="address1"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !isAddress1Valid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('address-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="address2"
 | 
			
		||||
          class="block  font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('apartment-suite-etc')}
 | 
			
		||||
          bind:value={editable.address.address2}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="address2"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="zipcode"
 | 
			
		||||
          class="block  font-medium text-gray-700">{$_('zip-postal-code')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('zip-postal-code')}
 | 
			
		||||
          class:border-red-500={!iszipcodevalid}
 | 
			
		||||
          class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
          class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
          bind:value={editable.address.postalcode}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="zipcode"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !iszipcodevalid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="col-span-6">
 | 
			
		||||
        <label
 | 
			
		||||
          for="city"
 | 
			
		||||
          class="block  font-medium text-gray-700">{$_('city')}</label>
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder={$_('city')}
 | 
			
		||||
          class:border-red-500={!iscityvalid}
 | 
			
		||||
          class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
          class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
          bind:value={editable.address.city}
 | 
			
		||||
          type="text"
 | 
			
		||||
          name="city"
 | 
			
		||||
          class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm: border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
        {#if !iscityvalid}
 | 
			
		||||
          <span
 | 
			
		||||
            class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
            {$_('valid-city-is-required')}
 | 
			
		||||
          </span>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </section>
 | 
			
		||||
	<section class="container p-5 select-none">
 | 
			
		||||
		<div class="flex flex-row mb-4">
 | 
			
		||||
			<div class="w-full">
 | 
			
		||||
				<nav class="w-full flex">
 | 
			
		||||
					<ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="./"
 | 
			
		||||
								><svg
 | 
			
		||||
									xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
									width="24"
 | 
			
		||||
									height="24"
 | 
			
		||||
									viewBox="0 0 24 24"
 | 
			
		||||
									fill="none"
 | 
			
		||||
									stroke="currentColor"
 | 
			
		||||
									stroke-width="2"
 | 
			
		||||
									stroke-linecap="round"
 | 
			
		||||
									stroke-linejoin="round"
 | 
			
		||||
									class="inline-block"
 | 
			
		||||
									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg
 | 
			
		||||
								>
 | 
			
		||||
								{$_("donors")}</a
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
					</ol>
 | 
			
		||||
				</nav>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mb-4 text-3xl font-extrabold leading-tight">
 | 
			
		||||
			{original_data.firstname}
 | 
			
		||||
			{original_data.middlename || ""}
 | 
			
		||||
			{original_data.lastname}
 | 
			
		||||
			<div data-id="donor_actions_${editable.id}">
 | 
			
		||||
				{#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:DELETE")}
 | 
			
		||||
					{#if delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							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:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("confirm-deletion")}</button
 | 
			
		||||
						>
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = !delete_triggered;
 | 
			
		||||
							}}
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("cancel")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
					{#if !delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = true;
 | 
			
		||||
							}}
 | 
			
		||||
							type="button"
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("delete-donor")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
				{/if}
 | 
			
		||||
				{#if !delete_triggered}
 | 
			
		||||
					<button
 | 
			
		||||
						disabled={!save_enabled}
 | 
			
		||||
						class:opacity-50={!save_enabled}
 | 
			
		||||
						type="button"
 | 
			
		||||
						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:w-auto sm:text-sm mb-1 lg:mb-0"
 | 
			
		||||
						>{$_("save-changes")}</button
 | 
			
		||||
					>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		<div>
 | 
			
		||||
			<span class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("total-donation-amount")}:</span
 | 
			
		||||
			>
 | 
			
		||||
			<span
 | 
			
		||||
				>{(editable.donationAmount / 100)
 | 
			
		||||
					.toFixed(2)
 | 
			
		||||
					.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 />
 | 
			
		||||
			<span class="font-semibold text-gray-700">{$_("donations")}:</span>
 | 
			
		||||
			{#if original_data.donations.length > 0}
 | 
			
		||||
				{#each original_data.donations 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 border border-current bg-green-700 text-white mr-1"
 | 
			
		||||
							>{$_("fixed-donation")}:
 | 
			
		||||
							{(d.amount / 100)
 | 
			
		||||
								.toFixed(2)
 | 
			
		||||
								.toLocaleString("de-DE", { valute: "EUR" })}€</a
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
				{/each}
 | 
			
		||||
			{:else}{$_("donor-has-no-associated-donations")}{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mt-2 w-full">
 | 
			
		||||
			<label for="firstname" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("first-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("first-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				class:border-red-500={!isFirstnameValid}
 | 
			
		||||
				class:focus:border-red-500={!isFirstnameValid}
 | 
			
		||||
				class:focus:ring-red-500={!isFirstnameValid}
 | 
			
		||||
				bind:value={editable.firstname}
 | 
			
		||||
				name="firstname"
 | 
			
		||||
				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}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("first-name-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mt-2 w-full">
 | 
			
		||||
			<label for="middlename" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("middle-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("middle-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.middlename}
 | 
			
		||||
				name="middlename"
 | 
			
		||||
				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 class="mt-2 w-full">
 | 
			
		||||
			<label for="lastname" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("last-name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("last-name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.lastname}
 | 
			
		||||
				class:border-red-500={!isLastnameValid}
 | 
			
		||||
				class:focus:border-red-500={!isLastnameValid}
 | 
			
		||||
				class:focus:ring-red-500={!isLastnameValid}
 | 
			
		||||
				name="lastname"
 | 
			
		||||
				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}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("last-name-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mt-2 w-full">
 | 
			
		||||
			<label for="email" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("e-mail-adress")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("e-mail-adress")}
 | 
			
		||||
				type="email"
 | 
			
		||||
				bind:value={editable.email}
 | 
			
		||||
				class:border-red-500={!isEmailValid}
 | 
			
		||||
				class:focus:border-red-500={!isEmailValid}
 | 
			
		||||
				class:focus:ring-red-500={!isEmailValid}
 | 
			
		||||
				name="email"
 | 
			
		||||
				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}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("valid-email-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mt-2 w-full">
 | 
			
		||||
			<label for="phone" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("phone")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("phone")}
 | 
			
		||||
				type="tel"
 | 
			
		||||
				class:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				class:focus:border-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				class:focus:ring-red-500={!isPhoneValidOrEmpty}
 | 
			
		||||
				bind:value={editable.phone}
 | 
			
		||||
				name="phone"
 | 
			
		||||
				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}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("valid-international-phone-number-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="flex items-start mt-2">
 | 
			
		||||
			<div class="flex items-center h-5">
 | 
			
		||||
				<input
 | 
			
		||||
					bind:checked={editable.address_checked}
 | 
			
		||||
					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">
 | 
			
		||||
				<label for="comments" class="font-semibold text-gray-700"
 | 
			
		||||
					>{$_("receipt-needed")}</label
 | 
			
		||||
				>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		{#if editable.address_checked === true}
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="address1" class="block font-medium text-gray-700"
 | 
			
		||||
					>{$_("address")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder="Address"
 | 
			
		||||
					class:border-red-500={!isAddress1Valid}
 | 
			
		||||
					class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
					class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
					bind:value={editable.address.address1}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="address1"
 | 
			
		||||
					class="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}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("address-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="address2" class="block font-medium text-gray-700"
 | 
			
		||||
					>{$_("apartment-suite-etc")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
					bind:value={editable.address.address2}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="address2"
 | 
			
		||||
					class="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 class="col-span-6">
 | 
			
		||||
				<label for="zipcode" class="block font-medium text-gray-700"
 | 
			
		||||
					>{$_("zip-postal-code")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("zip-postal-code")}
 | 
			
		||||
					class:border-red-500={!iszipcodevalid}
 | 
			
		||||
					class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
					class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
					bind:value={editable.address.postalcode}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="zipcode"
 | 
			
		||||
					class="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}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="col-span-6">
 | 
			
		||||
				<label for="city" class="block font-medium text-gray-700"
 | 
			
		||||
					>{$_("city")}</label
 | 
			
		||||
				>
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("city")}
 | 
			
		||||
					class:border-red-500={!iscityvalid}
 | 
			
		||||
					class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
					class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
					bind:value={editable.address.city}
 | 
			
		||||
					type="text"
 | 
			
		||||
					name="city"
 | 
			
		||||
					class="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}
 | 
			
		||||
					<span
 | 
			
		||||
						class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("valid-city-is-required")}
 | 
			
		||||
					</span>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		{/if}
 | 
			
		||||
	</section>
 | 
			
		||||
{:catch error}
 | 
			
		||||
  <PromiseError {error} />
 | 
			
		||||
	<PromiseError {error} />
 | 
			
		||||
{/await}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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";
 | 
			
		||||
  $: current_donors = [];
 | 
			
		||||
  export let modal_open = false;
 | 
			
		||||
  let addDonors;
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('donors')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('add-donor')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </span>
 | 
			
		||||
  <DonorsOverview bind:current_donors />
 | 
			
		||||
  <h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_("donors")}
 | 
			
		||||
  </h4>
 | 
			
		||||
  {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR: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-donor")}
 | 
			
		||||
    </button>
 | 
			
		||||
  {/if}
 | 
			
		||||
  {#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:GET")}
 | 
			
		||||
    <button
 | 
			
		||||
      on:click={() => {
 | 
			
		||||
        const data = current_donors
 | 
			
		||||
          .filter((d) => d.receiptNeeded === true)
 | 
			
		||||
          .map(function (d) {
 | 
			
		||||
            d.address.address2 =
 | 
			
		||||
              d.address.address2 === "" ? "" : " " + d.address.address2;
 | 
			
		||||
            const address = `${d.address.address1}${d.address.address2}, ${d.address.postalcode} ${d.address.city}, ${d.address.country}`;
 | 
			
		||||
            return [
 | 
			
		||||
              d.firstname,
 | 
			
		||||
              d.middlename,
 | 
			
		||||
              d.lastname,
 | 
			
		||||
              (d.paidDonationAmount/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>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:CREATE')}
 | 
			
		||||
  <AddDonorModal bind:current_donors bind:modal_open />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes("DONOR:CREATE")}
 | 
			
		||||
  <AddDonorModal
 | 
			
		||||
    on:created={(event) => {
 | 
			
		||||
      addDonors(event.detail.donors);
 | 
			
		||||
    }}
 | 
			
		||||
    bind:modal_open
 | 
			
		||||
  />
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="w-full" style="height:15rem" src={donors_empty} alt="" />
 | 
			
		||||
    <span class="font-bold">{$_('there-are-no-donors-yet')}</span><br />
 | 
			
		||||
    <span>{$_('add-your-first-donor')}</span>
 | 
			
		||||
    <span class="font-bold">{$_("there-are-no-donors-yet")}</span><br />
 | 
			
		||||
    <span>{$_("add-your-first-donor")}</span>
 | 
			
		||||
  </p>
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,208 +1,262 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
  import { DonationService, DonorService } from "@odit/lfk-client-js";
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { DonorService } from "@odit/lfk-client-js";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import DonorsEmptyState from "./DonorsEmptyState.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 = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  $: current_donations = [];
 | 
			
		||||
  let modal_open = false;
 | 
			
		||||
  let delete_donor = {};
 | 
			
		||||
  $: selectedDonors =
 | 
			
		||||
    $table?.getSelectedRowModel().rows.map((row) => row.original) || [];
 | 
			
		||||
  $: selected =
 | 
			
		||||
    $table?.getSelectedRowModel().rows.map((row) => row.index) || [];
 | 
			
		||||
 | 
			
		||||
  $: dataLoaded = false;
 | 
			
		||||
 | 
			
		||||
  export let current_donors = [];
 | 
			
		||||
  const donors_promise = DonorService.donorControllerGetAll().then((val) => {
 | 
			
		||||
    current_donors = val;
 | 
			
		||||
  export const addDonors = (donors) => {
 | 
			
		||||
    current_donors = current_donors.concat(...donors);
 | 
			
		||||
    options.update((options) => ({
 | 
			
		||||
      ...options,
 | 
			
		||||
      data: current_donors,
 | 
			
		||||
    }));
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  //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 donation_promise = DonationService.donationControllerGetAll().then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      current_donations = val;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
  const table = createSvelteTable(options);
 | 
			
		||||
 | 
			
		||||
  function should_display_based_on_id(id) {
 | 
			
		||||
    if (searchvalue.toString().slice(-1) === "*") {
 | 
			
		||||
      return id.toString().startsWith(searchvalue.replace("*", ""));
 | 
			
		||||
    }
 | 
			
		||||
    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>
 | 
			
		||||
 | 
			
		||||
<ConfirmDonorDeletion
 | 
			
		||||
  on:cancelDelete={(event) => {
 | 
			
		||||
    modal_open = false;
 | 
			
		||||
    active_deletes[event.detail.id] = false;
 | 
			
		||||
    active_deletes = active_deletes.filter((a) => a.id !== event.detail.id);
 | 
			
		||||
  }}
 | 
			
		||||
  bind:modal_open
 | 
			
		||||
  bind:delete_donor />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('DONOR:GET')}
 | 
			
		||||
  {#await donors_promise && donation_promise}
 | 
			
		||||
  on:delete={async (event) => {
 | 
			
		||||
    toast.loading($_("deleting-donor"));
 | 
			
		||||
    await DonorService.donorControllerRemove(event.detail.id, true);
 | 
			
		||||
    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
 | 
			
		||||
      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">{$_('donors-are-being-loaded')}</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
      role="alert"
 | 
			
		||||
    >
 | 
			
		||||
      <p class="font-bold">{$_("donors-are-being-loaded")}</p>
 | 
			
		||||
      <p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_donors.length === 0}
 | 
			
		||||
      <DonorsEmptyState />
 | 
			
		||||
    {:else}
 | 
			
		||||
      <input
 | 
			
		||||
        type="search"
 | 
			
		||||
        bind:value={searchvalue}
 | 
			
		||||
        placeholder={$_('datatable.search')}
 | 
			
		||||
        aria-label={$_('datatable.search')}
 | 
			
		||||
        class="gridjs-input gridjs-search-input mb-4" />
 | 
			
		||||
      <div
 | 
			
		||||
        class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll">
 | 
			
		||||
        <table class="divide-y divide-gray-200 w-full">
 | 
			
		||||
          <thead class="bg-gray-50">
 | 
			
		||||
            <tr>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('name')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('contact-information')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('donations')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th
 | 
			
		||||
                scope="col"
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
 | 
			
		||||
                {$_('total-donation-amount')}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th scope="col" class="relative px-6 py-3">
 | 
			
		||||
                <span class="sr-only">{$_('action')}</span>
 | 
			
		||||
  {:else if current_donors.length === 0}
 | 
			
		||||
    <DonorsEmptyState />
 | 
			
		||||
  {:else}
 | 
			
		||||
    <input
 | 
			
		||||
      type="search"
 | 
			
		||||
      bind:value={searchvalue}
 | 
			
		||||
      placeholder={$_("datatable.search")}
 | 
			
		||||
      aria-label={$_("datatable.search")}
 | 
			
		||||
      class="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border"
 | 
			
		||||
    />
 | 
			
		||||
    <div
 | 
			
		||||
      class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll"
 | 
			
		||||
    >
 | 
			
		||||
      <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>
 | 
			
		||||
          </thead>
 | 
			
		||||
          <tbody class="divide-y divide-gray-200">
 | 
			
		||||
            {#each current_donors as donor}
 | 
			
		||||
              {#if donor.firstname
 | 
			
		||||
                .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}
 | 
			
		||||
                    {: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>
 | 
			
		||||
              {/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>
 | 
			
		||||
          {/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>
 | 
			
		||||
  {/await}
 | 
			
		||||
    <div class="h-2" />
 | 
			
		||||
    <TableBottom {table} {selected} />
 | 
			
		||||
  {/if}
 | 
			
		||||
{/if}
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
  table tbody tr td:nth-child(2) {
 | 
			
		||||
    font-family: monospace;
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,193 +1,196 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  (function () {
 | 
			
		||||
    document.onkeydown = function (e) {
 | 
			
		||||
      e = e || window.event;
 | 
			
		||||
      if (e.key === "Escape") {
 | 
			
		||||
        modal_open = false;
 | 
			
		||||
      }
 | 
			
		||||
    };
 | 
			
		||||
  })();
 | 
			
		||||
  const license_promise = fetch("/licenses.json");
 | 
			
		||||
  let licenses = [];
 | 
			
		||||
  $: currentlicense = "";
 | 
			
		||||
  $: licensetext = "";
 | 
			
		||||
  license_promise
 | 
			
		||||
    .then((response) => response.json())
 | 
			
		||||
    .then((json) => {
 | 
			
		||||
      licenses = json;
 | 
			
		||||
    });
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
 | 
			
		||||
	let modal_open = false;
 | 
			
		||||
	(function () {
 | 
			
		||||
		document.onkeydown = function (e) {
 | 
			
		||||
			e = e || window.event;
 | 
			
		||||
			if (e.key === "Escape") {
 | 
			
		||||
				modal_open = false;
 | 
			
		||||
			}
 | 
			
		||||
		};
 | 
			
		||||
	})();
 | 
			
		||||
	const license_promise = fetch("/licenses.json");
 | 
			
		||||
	let licenses = [];
 | 
			
		||||
	$: currentlicense = "";
 | 
			
		||||
	$: licensetext = "";
 | 
			
		||||
	license_promise
 | 
			
		||||
		.then((response) => response.json())
 | 
			
		||||
		.then((json) => {
 | 
			
		||||
			licenses = json;
 | 
			
		||||
		});
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    use:clickOutside
 | 
			
		||||
    on:click_outside={() => {
 | 
			
		||||
      modal_open = false;
 | 
			
		||||
    }}>
 | 
			
		||||
    <div
 | 
			
		||||
      class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
      <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
        <div
 | 
			
		||||
          class="absolute inset-0 bg-gray-500 opacity-75"
 | 
			
		||||
          data-id="modal_backdrop" />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
      <div
 | 
			
		||||
        class="inline-block align-bottom bg-white rounded-lg text-left 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
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                class="h-6 w-6 text-blue-600"
 | 
			
		||||
                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 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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium">
 | 
			
		||||
                {$_('read-license')}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
                <p class="text-sm text-gray-500">{currentlicense}</p>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
                <p class="text-sm text-gray-500">{licensetext}</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={() => {
 | 
			
		||||
              modal_open = false;
 | 
			
		||||
            }}
 | 
			
		||||
            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">
 | 
			
		||||
            {$_('close')}
 | 
			
		||||
          </button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
	<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 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="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 size-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"
 | 
			
		||||
						>
 | 
			
		||||
							<svg
 | 
			
		||||
								fill="currentColor"
 | 
			
		||||
								class="h-6 w-6 text-blue-600"
 | 
			
		||||
								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 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 class="mt-3 sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
							<h3 class="text-lg leading-6 font-medium">
 | 
			
		||||
								{$_("read-license")}
 | 
			
		||||
							</h3>
 | 
			
		||||
							<div class="mb-6">
 | 
			
		||||
								<p class="text-sm text-gray-500">{currentlicense}</p>
 | 
			
		||||
							</div>
 | 
			
		||||
							<div class="mb-6">
 | 
			
		||||
								<p class="text-sm text-gray-500">{licensetext}</p>
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
				<div class="bg-gray-50 px-4 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
					<button
 | 
			
		||||
						on:click={() => {
 | 
			
		||||
							modal_open = false;
 | 
			
		||||
						}}
 | 
			
		||||
						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"
 | 
			
		||||
					>
 | 
			
		||||
						{$_("close")}
 | 
			
		||||
					</button>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
{/if}
 | 
			
		||||
<!-- /// -->
 | 
			
		||||
<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">
 | 
			
		||||
    <h1
 | 
			
		||||
      class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl">
 | 
			
		||||
      {$_('about')}
 | 
			
		||||
      🧾
 | 
			
		||||
    </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!
 | 
			
		||||
      <strong class="text-white font-medium">
 | 
			
		||||
        {$_('by')}
 | 
			
		||||
        <a href="https://odit.services" class="underline">ODIT.Services</a>
 | 
			
		||||
      </strong>
 | 
			
		||||
      <br />
 | 
			
		||||
      <span class="text-lg">{$_('lfk-is-os')}</span>
 | 
			
		||||
    </p>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<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">
 | 
			
		||||
    <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}
 | 
			
		||||
        <p class="text-center w-full">{$_('licenses-are-being-loaded')}</p>
 | 
			
		||||
      {:then}
 | 
			
		||||
        <table>
 | 
			
		||||
          <thead>
 | 
			
		||||
            <tr>
 | 
			
		||||
              <th>{$_('dependency_name')}</th>
 | 
			
		||||
              <th>{$_('license')}</th>
 | 
			
		||||
              <th>{$_('repo_link')}</th>
 | 
			
		||||
              <th>{$_('installed-version')}</th>
 | 
			
		||||
              <th>{$_('author')}</th>
 | 
			
		||||
            </tr>
 | 
			
		||||
          </thead>
 | 
			
		||||
          <tbody>
 | 
			
		||||
            {#each licenses as l}
 | 
			
		||||
              <tr>
 | 
			
		||||
                <td>{l.name}</td>
 | 
			
		||||
                <td>
 | 
			
		||||
                  {l.license || '?'}<br /><span
 | 
			
		||||
                    class="underline cursor-pointer"
 | 
			
		||||
                    on:click={() => {
 | 
			
		||||
                      modal_open = true;
 | 
			
		||||
                      currentlicense = l.name + '@' + l.version;
 | 
			
		||||
                      licensetext = l.licensetext || $_('no-license-text-could-be-found');
 | 
			
		||||
                    }}>{$_('read-license')}</span>
 | 
			
		||||
                </td>
 | 
			
		||||
                <td>
 | 
			
		||||
                  {(l.repo?.url || l.repo)
 | 
			
		||||
                    .replace('git+', '')
 | 
			
		||||
                    .replace('git://', '')}
 | 
			
		||||
                </td>
 | 
			
		||||
                <td>{l.version || '?'}</td>
 | 
			
		||||
                <td>{l.author?.name || l.author || '?'}</td>
 | 
			
		||||
              </tr>
 | 
			
		||||
            {/each}
 | 
			
		||||
          </tbody>
 | 
			
		||||
        </table>
 | 
			
		||||
      {: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 class="w-full leading-8 mt-8">
 | 
			
		||||
      <p class="text-xl font-medium">{$_('icon-image-credits')}</p>
 | 
			
		||||
      <ul class="list-disc">
 | 
			
		||||
        <li>
 | 
			
		||||
          <a
 | 
			
		||||
            class="underline"
 | 
			
		||||
            target="_blank"
 | 
			
		||||
            rel="noopener noreferrer"
 | 
			
		||||
            href="https://storyset.com">https://storyset.com</a>
 | 
			
		||||
        </li>
 | 
			
		||||
        <li>
 | 
			
		||||
          <a
 | 
			
		||||
            class="underline"
 | 
			
		||||
            target="_blank"
 | 
			
		||||
            rel="noopener noreferrer"
 | 
			
		||||
            href="https://undraw.co">https://undraw.co</a>
 | 
			
		||||
        </li>
 | 
			
		||||
        <li>
 | 
			
		||||
          <a
 | 
			
		||||
            class="underline"
 | 
			
		||||
            target="_blank"
 | 
			
		||||
            rel="noopener noreferrer"
 | 
			
		||||
            href="https://remixicon.com">https://remixicon.com</a>
 | 
			
		||||
        </li>
 | 
			
		||||
      </ul>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
	<h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
		{$_("about")}
 | 
			
		||||
	</h4>
 | 
			
		||||
	<p class="mt-2 mb-2">
 | 
			
		||||
		Lauf für Kaya!
 | 
			
		||||
		<strong class="font-medium">
 | 
			
		||||
			{$_("by")}
 | 
			
		||||
			<a href="https://odit.services" class="underline">ODIT.Services</a>
 | 
			
		||||
		</strong>
 | 
			
		||||
		<br />
 | 
			
		||||
		<span>{$_("lfk-is-os")}</span>
 | 
			
		||||
	</p>
 | 
			
		||||
	<h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
		{$_("credits")}
 | 
			
		||||
	</h4>
 | 
			
		||||
	<p class="text-left">{$_("oss_credit_description")}</p>
 | 
			
		||||
	<div class="mt-5 overflow-x-auto">
 | 
			
		||||
		{#await license_promise}
 | 
			
		||||
			<p>{$_("licenses-are-being-loaded")}</p>
 | 
			
		||||
		{:then}
 | 
			
		||||
			<table class="font-mono">
 | 
			
		||||
				<thead class="border-b border-gray-400">
 | 
			
		||||
					<tr class="odd:bg-white even:bg-gray-100">
 | 
			
		||||
						<th>{$_("dependency_name")}</th>
 | 
			
		||||
						<th>{$_("license")}</th>
 | 
			
		||||
						<th>{$_("repo_link")}</th>
 | 
			
		||||
						<th>{$_("installed-version")}</th>
 | 
			
		||||
						<th>{$_("author")}</th>
 | 
			
		||||
					</tr>
 | 
			
		||||
				</thead>
 | 
			
		||||
				<tbody>
 | 
			
		||||
					{#each licenses as l}
 | 
			
		||||
						<tr class="odd:bg-white even:bg-gray-100 *:p-2">
 | 
			
		||||
							<td>{l.name}</td>
 | 
			
		||||
							<td>
 | 
			
		||||
								<button
 | 
			
		||||
									class="underline cursor-pointer"
 | 
			
		||||
									on:click={() => {
 | 
			
		||||
										modal_open = true;
 | 
			
		||||
										currentlicense = l.name + "@" + l.version;
 | 
			
		||||
										licensetext =
 | 
			
		||||
											l.licensetext || $_("no-license-text-could-be-found");
 | 
			
		||||
									}}>{l.license || "?"}</button
 | 
			
		||||
								>
 | 
			
		||||
							</td>
 | 
			
		||||
							<td>
 | 
			
		||||
								{(l.repo?.url || l.repo)
 | 
			
		||||
									.replace("git+", "")
 | 
			
		||||
									.replace("git://", "")}
 | 
			
		||||
							</td>
 | 
			
		||||
							<td>{l.version || "?"}</td>
 | 
			
		||||
							<td>{l.author?.name || l.author || "?"}</td>
 | 
			
		||||
						</tr>
 | 
			
		||||
					{/each}
 | 
			
		||||
				</tbody>
 | 
			
		||||
			</table>
 | 
			
		||||
		{: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 class="w-full mt-8">
 | 
			
		||||
		<p class="font-medium">{$_("icon-image-credits")}</p>
 | 
			
		||||
		<ul class="list-disc ml-6">
 | 
			
		||||
			<li>
 | 
			
		||||
				<a
 | 
			
		||||
					class="underline"
 | 
			
		||||
					target="_blank"
 | 
			
		||||
					rel="noopener noreferrer"
 | 
			
		||||
					href="https://storyset.com">https://storyset.com</a
 | 
			
		||||
				>
 | 
			
		||||
			</li>
 | 
			
		||||
			<li>
 | 
			
		||||
				<a
 | 
			
		||||
					class="underline"
 | 
			
		||||
					target="_blank"
 | 
			
		||||
					rel="noopener noreferrer"
 | 
			
		||||
					href="https://undraw.co">https://undraw.co</a
 | 
			
		||||
				>
 | 
			
		||||
			</li>
 | 
			
		||||
			<li>
 | 
			
		||||
				<a
 | 
			
		||||
					class="underline"
 | 
			
		||||
					target="_blank"
 | 
			
		||||
					rel="noopener noreferrer"
 | 
			
		||||
					href="https://remixicon.com">https://remixicon.com</a
 | 
			
		||||
				>
 | 
			
		||||
			</li>
 | 
			
		||||
		</ul>
 | 
			
		||||
	</div>
 | 
			
		||||
</section>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										496
									
								
								src/components/general/CardAssignment.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										496
									
								
								src/components/general/CardAssignment.svelte
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,496 @@
 | 
			
		||||
<script>
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { RunnerCardService, RunnerService } from "@odit/lfk-client-js";
 | 
			
		||||
	import QrCodeScanner from "./QrCodeScanner.svelte";
 | 
			
		||||
	let state = "scan_runner";
 | 
			
		||||
	let runnerinfo = { id: 0, firstname: "", lastname: "" };
 | 
			
		||||
	let cardCode = "";
 | 
			
		||||
	let scannerActive = true;
 | 
			
		||||
	function resetAll() {
 | 
			
		||||
		state = "scan_runner";
 | 
			
		||||
		runnerinfo = { id: 0, firstname: "", lastname: "" };
 | 
			
		||||
		cardCode = "";
 | 
			
		||||
		scannerActive = true;
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<div class="p-4">
 | 
			
		||||
	<h3 class="text-3xl font-bold">{$_("card_assignment_for_mobile")}</h3>
 | 
			
		||||
	{#if state === "done"}
 | 
			
		||||
		<div class="text-center mx-auto">
 | 
			
		||||
			<svg
 | 
			
		||||
				class="h-64 mx-auto"
 | 
			
		||||
				xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
				viewBox="0 0 500 500"
 | 
			
		||||
				><path
 | 
			
		||||
					d="m339.91 38.77-2 1.14c-16.77 9.58-27.19 27.41-26.49 46.53a50.12 50.12 0 0 0 .81 7.33 54.34 54.34 0 0 0 32.62 39.6c9.79 3.94 21.14 5.27 28.8 12.46 10 9.36 9.88 24.8 9.1 38.36s-.83 29 9.18 38.32c6.27 5.84 15.36 7.88 24 7.65 22.8-.61 43.57-15.93 54.22-35.83s12.18-43.63 8.31-65.82c-5.46-31.09-21.78-60.95-47.75-79.33s-61.88-23.8-90.8-10.41ZM152.06 393l2.36-.64c19.89-5.4 35.44-21 39.85-41a55.18 55.18 0 0 0 1.13-7.79 58 58 0 0 0-23.09-49.67c-9.05-6.7-20.43-11.12-26.42-20.6-7.79-12.35-3.56-28.28.89-42.08s8.63-29.75.79-42.07c-4.91-7.71-13.75-12.25-22.71-14.32-23.71-5.49-49.29 4.77-65.63 22.48S34.94 239.14 33 263.11c-2.74 33.57 6.14 68.81 28.05 94.77s57.54 41.19 91.01 35.12Z"
 | 
			
		||||
					style="fill:#f5f5f5"
 | 
			
		||||
				/><circle
 | 
			
		||||
					cx="409.16"
 | 
			
		||||
					cy="320.55"
 | 
			
		||||
					r="36.72"
 | 
			
		||||
					style="fill:#f5f5f5"
 | 
			
		||||
					transform="rotate(-45 409.165 320.548)"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M470.34 473.11c0 .15-98.66.26-220.33.26s-220.35-.11-220.35-.26 98.64-.26 220.35-.26 220.33.15 220.33.26Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M453.56 473.11a9.53 9.53 0 0 1-1-2c-.54-1.27-1.24-3-2-5s-1.4-3.53-2-5a14.87 14.87 0 0 1-.8-2.09 7.37 7.37 0 0 1 1.14 1.93c.61 1.25 1.36 3 2.13 5s1.42 3.75 1.87 5a10.55 10.55 0 0 1 .66 2.16ZM458.16 473.31c-.15 0-.66-3.33-1.59-7.41s-1.9-7.32-1.76-7.37a8.61 8.61 0 0 1 .84 2.07c.44 1.31 1 3.14 1.43 5.19s.77 3.93.94 5.29a8.54 8.54 0 0 1 .14 2.23ZM466.32 459.42c.14.08-1.36 3.08-2.88 6.91s-2.51 7-2.66 7a9.06 9.06 0 0 1 .46-2.15 53.35 53.35 0 0 1 3.93-9.86 8.66 8.66 0 0 1 1.15-1.9ZM225.88 473.11a74 74 0 0 1-6.08-10.68 71.59 71.59 0 0 1 6.08 10.68ZM230.2 473a79 79 0 0 1-2.66-12.67A82 82 0 0 1 230.2 473ZM239.31 461.69a39.45 39.45 0 0 1-3.08 5.76 41.06 41.06 0 0 1-3.53 5.49 39.66 39.66 0 0 1 3.08-5.75 42.46 42.46 0 0 1 3.53-5.5ZM55.68 473.11a29 29 0 0 1-2.94-4.68 29 29 0 0 1-2.49-4.93 29.84 29.84 0 0 1 2.94 4.68 29.92 29.92 0 0 1 2.49 4.93ZM58.09 457.6a60.92 60.92 0 0 1 1.17 7.73 56.51 56.51 0 0 1 .64 7.78 118.57 118.57 0 0 1-1.81-15.51ZM67.49 462.53a29.15 29.15 0 0 1-1.38 5.37 29.37 29.37 0 0 1-1.88 5.21 29.37 29.37 0 0 1 1.38-5.36 31.72 31.72 0 0 1 1.88-5.22Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M141.72 440a6.16 6.16 0 0 1 3.2 3.62 11 11 0 0 1 .43 4.91c-.37 3.19-2.18 6.4-4.12 8.95-2.69-2.34-3.3-6.43-3.49-8.52-.29-3.31 1.2-9.46 4-9M144.34 462.69a5.24 5.24 0 0 1 1.71-5 7 7 0 0 1 5.15-1.61 2.94 2.94 0 0 1 2.27 1 2.26 2.26 0 0 1 0 2.29 4.49 4.49 0 0 1-1.69 1.66c-2.32 1.46-4.84 2.44-7.47 1.65"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M140.9 472.46a4.38 4.38 0 0 1 .1-.65c.08-.46.19-1.06.32-1.77a18.5 18.5 0 0 1 1.83-5.65 11.28 11.28 0 0 1 4.06-4.31 7.33 7.33 0 0 1 1.65-.73 3 3 0 0 1 .48-.12h.17a11.36 11.36 0 0 0-2.22 1 11.75 11.75 0 0 0-3.94 4.28 20.19 20.19 0 0 0-1.87 5.57c-.16.74-.29 1.34-.39 1.76a2.49 2.49 0 0 1-.19.62Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M141 444.74a1.36 1.36 0 0 1 0 .29l.05.82c0 .71.08 1.74.12 3 .08 2.55.14 6.08.13 10s-.11 7.42-.22 10c-.05 1.28-.1 2.31-.15 3 0 .33 0 .6-.05.82a1.33 1.33 0 0 1 0 .28 1.29 1.29 0 0 1 0-.29v-.82c0-.74.05-1.77.08-3 .06-2.55.13-6.08.15-10s0-7.42-.07-10V445c-.04-.16-.04-.26-.04-.26Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M139.7 464.63a12.32 12.32 0 0 0-6.72-9c-.9-.43-2.07-.71-2.81 0s-.57 1.86-.21 2.79a10.54 10.54 0 0 0 9.68 6.54"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M133.41 458.63a3 3 0 0 1 .57.25 5.17 5.17 0 0 1 .64.34 9.69 9.69 0 0 1 .82.52 9.89 9.89 0 0 1 .94.7 11.62 11.62 0 0 1 1 .91 13.72 13.72 0 0 1 1.89 2.43 14 14 0 0 1 1.25 2.82 10.26 10.26 0 0 1 .31 1.31 9.15 9.15 0 0 1 .16 1.16 8.75 8.75 0 0 1 0 1v.73a4 4 0 0 1-.07.62s0-.89-.12-2.3a10.83 10.83 0 0 0-.19-1.14 11.51 11.51 0 0 0-.33-1.28 13.92 13.92 0 0 0-1.24-2.77 14.19 14.19 0 0 0-1.85-2.4 10.11 10.11 0 0 0-1-.91 10.49 10.49 0 0 0-.9-.72c-1.1-.9-1.9-1.22-1.88-1.27Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M411.83 430.38a8.16 8.16 0 0 1 4.22 4.79 14.35 14.35 0 0 1 .57 6.48c-.48 4.21-2.87 8.46-5.45 11.82-3.54-3.09-4.35-8.49-4.6-11.25-.38-4.36 1.58-12.49 5.26-11.84M415.29 460.41a7 7 0 0 1 2.25-6.61 9.26 9.26 0 0 1 6.81-2.13 3.89 3.89 0 0 1 3 1.35 3 3 0 0 1 .05 3 6 6 0 0 1-2.23 2.2c-3.07 1.93-6.4 3.23-9.87 2.18"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M410.75 473.31a5.31 5.31 0 0 1 .13-.86l.42-2.34a24.66 24.66 0 0 1 2.42-7.46 14.9 14.9 0 0 1 5.35-5.69 10.1 10.1 0 0 1 2.19-1 6.47 6.47 0 0 1 .63-.16.61.61 0 0 1 .22 0 13.81 13.81 0 0 0-8.13 7 26.37 26.37 0 0 0-2.46 7.36c-.22 1-.39 1.77-.53 2.32a4.05 4.05 0 0 1-.24.83Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M410.88 436.7a1.65 1.65 0 0 1 .05.38q0 .42.06 1.08c.05.94.11 2.31.16 4 .11 3.38.19 8 .17 13.18s-.15 9.81-.28 13.18c-.07 1.69-.14 3.05-.2 4 0 .44 0 .8-.07 1.08a1.65 1.65 0 0 1 0 .38 2.21 2.21 0 0 1 0-.38v-1.08c0-1 .06-2.34.1-4 .08-3.37.17-8 .2-13.17s0-9.8-.09-13.17c0-1.66 0-3-.06-4v-1.09a2.79 2.79 0 0 1-.04-.39Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M409.16 463a16.22 16.22 0 0 0-8.88-11.92c-1.19-.57-2.73-.94-3.71-.06s-.75 2.46-.28 3.69a13.9 13.9 0 0 0 12.79 8.63"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M400.85 455.05a3.47 3.47 0 0 1 .75.32 9.58 9.58 0 0 1 .85.46 10.94 10.94 0 0 1 1.08.68 12.91 12.91 0 0 1 1.24.93 15.58 15.58 0 0 1 1.32 1.19 16.82 16.82 0 0 1 4.15 6.94 16.17 16.17 0 0 1 .42 1.74 13.28 13.28 0 0 1 .21 1.53c.06.47 0 .9.06 1.28a9.24 9.24 0 0 1 0 1 4.38 4.38 0 0 1-.09.81c-.07 0 .05-1.17-.16-3-.06-.46-.12-1-.24-1.51s-.27-1.1-.44-1.69a18.46 18.46 0 0 0-1.64-3.65 19.56 19.56 0 0 0-2.44-3.18 17.44 17.44 0 0 0-1.28-1.2c-.41-.37-.83-.67-1.2-.95-1.55-1.16-2.62-1.64-2.59-1.7ZM155.45 292c0 .14-14 .26-31.2.26S93 292.15 93 292s14-.26 31.21-.26 31.24.12 31.24.26ZM136.75 284.1c0 .14-14 .26-31.2.26s-31.21-.12-31.21-.26 14-.26 31.21-.26 31.2.16 31.2.26ZM294.43 412.9c0 .14-8.07.26-18 .26s-18-.12-18-.26 8.06-.26 18-.26 18 .12 18 .26ZM283.64 405c0 .15-8.07.26-18 .26s-18-.11-18-.26 8.06-.26 18-.26 18 .11 18 .26Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><circle cx="88.02" cy="72.37" r="14.66" style="fill:#5e9cff" /><path
 | 
			
		||||
					d="M88 87a14.68 14.68 0 0 1-13.52-9 14.78 14.78 0 0 1 0-11.43A14.84 14.84 0 0 1 77.64 62a14.69 14.69 0 0 1 20.76 0 14.84 14.84 0 0 1 3.14 4.66A14.67 14.67 0 0 1 88 87ZM85.93 105.14c-.08 0-.15-2.7-.15-6s.07-6 .15-6a56.83 56.83 0 0 1 .16 6 56.56 56.56 0 0 1-.16 6Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M85.93 105.16s-.08 0-.12-1.79v-8.55c0-1.78.09-1.78.12-1.78s.08 0 .13 1.78v8.55c-.06 1.79-.06 1.79-.13 1.79Zm0-12.08c-.07.22-.14 2.52-.14 6s.07 5.79.14 6c.07-.23.14-2.52.14-6s-.07-5.78-.14-6ZM77.48 89.64a63 63 0 0 1-3.07 5.64c-1.78 3.08-3.28 5.53-3.35 5.49a63.32 63.32 0 0 1 3.08-5.64c1.77-3.07 3.27-5.53 3.34-5.49Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M71.06 100.79s-.07 0 .84-1.7c.58-1.07 1.38-2.47 2.23-4s1.68-2.87 2.32-3.91c.89-1.46 1-1.58 1-1.58s.07 0-.84 1.7c-.58 1.07-1.37 2.48-2.23 4-1.52 2.64-3.22 5.46-3.35 5.5Zm6.41-11.12c-.18.17-1.46 2.25-3.32 5.47s-3 5.36-3.08 5.61c.18-.18 1.46-2.25 3.32-5.47s3.02-5.37 3.08-5.61ZM68.81 84.24a31 31 0 0 1-3.42 2.92 32.1 32.1 0 0 1-3.61 2.68 32.32 32.32 0 0 1 3.42-2.92 35.42 35.42 0 0 1 3.61-2.68Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M61.78 89.86s0-.06 1-.93c.65-.55 1.52-1.26 2.46-2a35.08 35.08 0 0 1 3.62-2.68 34.93 34.93 0 0 1-3.43 2.92c-.94.75-1.83 1.44-2.52 2a5 5 0 0 1-1.13.69Zm7-5.6a38.16 38.16 0 0 0-3.59 2.67 40.2 40.2 0 0 0-3.42 2.9 37.22 37.22 0 0 0 3.59-2.68 39.77 39.77 0 0 0 3.44-2.89Zm0 0ZM100.56 99.19a7.77 7.77 0 0 1-.39-1.49c-.21-.92-.51-2.19-.89-3.59s-.75-2.64-1-3.55a8.5 8.5 0 0 1-.41-1.48 7.14 7.14 0 0 1 .62 1.41c.33.89.75 2.14 1.12 3.54a36.32 36.32 0 0 1 .8 3.63 6.9 6.9 0 0 1 .15 1.53Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M100.55 99.2s-.11-.22-.4-1.5c-.19-.86-.5-2.16-.88-3.58s-.78-2.72-1-3.55c-.37-1.2-.45-1.46-.4-1.48s0 0 .22.37.27.64.42 1c.23.6.68 1.87 1.13 3.54s.69 3 .8 3.62a7.38 7.38 0 0 1 .17 1.54Zm-2.7-10.09a13.79 13.79 0 0 0 .41 1.43c.26.83.65 2.1 1 3.55s.69 2.73.88 3.59.32 1.37.37 1.46a10.18 10.18 0 0 0-.18-1.5c-.11-.64-.35-2-.8-3.62s-.9-2.94-1.13-3.54a9.12 9.12 0 0 0-.55-1.37ZM115.42 88.64a54.77 54.77 0 0 1-4.33-3.58 51.8 51.8 0 0 1-4.09-3.82 50.82 50.82 0 0 1 4.33 3.58 49.69 49.69 0 0 1 4.09 3.82Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M115.42 88.65a54.61 54.61 0 0 1-4.34-3.58c-1.13-1-2.18-1.93-3-2.65-1.22-1.13-1.19-1.17-1.17-1.19.15 0 2.34 1.85 4.33 3.59 1.13 1 2.19 1.93 3 2.65 1.22 1.13 1.18 1.17 1.16 1.19ZM107 81.26c.11.18 1.67 1.64 4.12 3.78s4.11 3.5 4.3 3.58c-.1-.19-1.67-1.64-4.12-3.79s-4.12-3.49-4.3-3.57ZM121.12 74.14a11.91 11.91 0 0 1-1.77-.26c-1.09-.18-2.59-.41-4.25-.6s-3.19-.3-4.29-.37a10.88 10.88 0 0 1-1.78-.16 8.74 8.74 0 0 1 1.79-.06 43.17 43.17 0 0 1 4.31.28c1.68.2 3.18.46 4.26.7a9.11 9.11 0 0 1 1.73.47Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M121.07 74.16c-.13 0-.49-.06-1.37-.2l-.35-.06c-1.49-.25-2.92-.45-4.26-.61s-2.77-.27-4.28-.37h-.35c-1.2-.08-1.44-.1-1.44-.15a9.68 9.68 0 0 1 1.79-.07 42.81 42.81 0 0 1 4.31.29 41.81 41.81 0 0 1 4.26.7 9.36 9.36 0 0 1 1.74.47s-.01 0-.05 0Zm-12-1.41a13.35 13.35 0 0 0 1.41.12h.35c1.51.1 3 .22 4.28.37s2.77.36 4.25.61l.35.06c.71.12 1.24.2 1.37.2a12.87 12.87 0 0 0-1.7-.44 41.81 41.81 0 0 0-4.26-.7 43 43 0 0 0-4.31-.29 13.47 13.47 0 0 0-1.75.07ZM121.14 59.88c0 .08-3.16.93-7.09 1.89s-7.14 1.66-7.16 1.58 3.15-.92 7.09-1.88 7.14-1.67 7.16-1.59Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M106.92 63.37s0-.08 2.07-.64c1.34-.35 3.13-.8 5-1.27s3.7-.88 5-1.18c2.11-.47 2.13-.42 2.13-.39s0 .07-2.07.63c-1.34.36-3.12.81-5 1.27s-3.69.88-5 1.18a16 16 0 0 1-2.13.4Zm14.18-3.48c-.29 0-3 .59-7.12 1.59s-6.8 1.73-7.06 1.86c.29 0 3-.58 7.13-1.59s6.8-1.75 7.05-1.86ZM104.57 43.27a63.61 63.61 0 0 1-3.39 5.41c-1.94 2.94-3.58 5.29-3.65 5.24s1.45-2.47 3.39-5.41 3.58-5.29 3.65-5.24Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M97.53 53.93c0-.14 1.72-2.9 3.39-5.42.94-1.42 1.84-2.75 2.53-3.74 1.06-1.52 1.12-1.52 1.14-1.5s.07 0-.93 1.64c-.65 1-1.52 2.37-2.46 3.79s-1.83 2.75-2.52 3.74c-.98 1.37-1.11 1.49-1.15 1.49Zm7-10.64c-.19.17-1.59 2.15-3.62 5.23s-3.32 5.14-3.39 5.37c.19-.16 1.58-2.14 3.62-5.22s3.34-5.14 3.42-5.38ZM88.13 39.8a12.88 12.88 0 0 1 0 2v4.85c0 1.85 0 3.61.06 4.85a14.66 14.66 0 0 1 0 2 11.36 11.36 0 0 1-.2-2c-.08-1.24-.14-3-.14-4.86s0-3.61.13-4.86a12.13 12.13 0 0 1 .15-1.98Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M88.15 53.54s-.05 0-.1-.55-.08-.88-.12-1.47c-.05-.84-.14-2.58-.14-4.86s.08-4 .13-4.86c0-.59.07-1.09.11-1.47s.08-.55.1-.55c.05 0 .07.19 0 2v4.85c0 1.91 0 3.54.05 4.77v2.08Zm0-13.71a18 18 0 0 0-.18 2c0 .83-.13 2.58-.13 4.85s.09 4 .14 4.86a16.64 16.64 0 0 0 .19 2v-2.08c0-1.23-.05-2.92-.05-4.77v-4.85c.03-.76.05-1.84.01-2.01ZM75.68 55.3c-.07.05-1.61-2.35-3.44-5.34s-3.24-5.47-3.19-5.51 1.62 2.35 3.45 5.35a63 63 0 0 1 3.18 5.5Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M75.68 55.32s-.16-.12-1.07-1.54c-.65-1-1.5-2.37-2.39-3.82-1.57-2.57-3.22-5.37-3.22-5.51s.07 0 1.09 1.54c.65 1 1.5 2.37 2.38 3.82s1.71 2.82 2.31 3.86c.94 1.63.9 1.65.87 1.67Zm-6.61-10.85c.06.24 1.26 2.34 3.18 5.48s3.24 5.16 3.42 5.33c-.07-.24-1.26-2.34-3.18-5.48s-3.24-5.16-3.42-5.33ZM68.25 61.62a7 7 0 0 1-1.56-.24c-1-.2-2.26-.52-3.69-.94s-2.7-.87-3.6-1.21a7.56 7.56 0 0 1-1.4-.64 8.28 8.28 0 0 1 1.52.43c.92.3 2.2.7 3.62 1.12s2.71.77 3.65 1a9.86 9.86 0 0 1 1.46.48Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M68.2 61.65h-.38c-.29 0-.68-.12-1.13-.22-.65-.13-2-.44-3.69-.94s-3-1-3.61-1.21l-1.07-.44c-.4-.17-.39-.2-.38-.22s.22 0 1.53.42c.74.24 2.12.68 3.62 1.13s2.6.74 3.65 1c1.34.36 1.53.43 1.52.48s0 0-.06 0ZM58 58.6a9.9 9.9 0 0 0 1.42.61c.62.24 1.9.71 3.61 1.21s3 .81 3.68.94a10.47 10.47 0 0 0 1.51.26 15.06 15.06 0 0 0-1.48-.44c-1-.28-2.29-.62-3.65-1s-2.89-.91-3.63-1.18c-.58-.15-1.36-.4-1.46-.4ZM65 72.89a38 38 0 0 1-4.68.6 37.31 37.31 0 0 1-4.71.3 35.07 35.07 0 0 1 4.68-.61 34.11 34.11 0 0 1 4.71-.29Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M55.88 73.83a1 1 0 0 1-.27 0 39.32 39.32 0 0 1 4.69-.6 39.75 39.75 0 0 1 4.72-.3 40.29 40.29 0 0 1-4.69.61c-1.81.14-3.74.29-4.45.29Zm-.24 0a43.5 43.5 0 0 0 4.68-.31 46 46 0 0 0 4.68-.63 43.61 43.61 0 0 0-4.69.31 39.33 39.33 0 0 0-4.67.59Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m321.8 444.95 7.41 19.13 16.31-3.83 3.59-8.75-5.52-18.14-21.79 11.59z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m349.94 449.25.58 1.45s29 0 31.09 4.61l-49.89 19.33-5.66-16.58c8.73.5 16.64-2.13 23.88-8.81Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><g style="opacity:.6000000000000001"
 | 
			
		||||
					><path
 | 
			
		||||
						d="M333.13 461.32a2.15 2.15 0 0 0-.52 2.83 2.11 2.11 0 0 0 2.84.5 2.25 2.25 0 0 0 .53-3 2.19 2.19 0 0 0-3-.2"
 | 
			
		||||
						style="fill:#fff"
 | 
			
		||||
					/></g
 | 
			
		||||
				><g style="opacity:.6000000000000001"
 | 
			
		||||
					><path
 | 
			
		||||
						d="m331.72 474.64-1.36-4 48.32-17.07s2.59.09 2.93 1.73Z"
 | 
			
		||||
						style="fill:#fff"
 | 
			
		||||
					/></g
 | 
			
		||||
				><path
 | 
			
		||||
					d="M350.75 450.77c.08.24-1.1.82-2 2.06s-1.15 2.48-1.41 2.48-.44-1.54.68-3 2.69-1.78 2.73-1.54ZM356.81 450.87c.14.21-.76 1-1.19 2.36s-.26 2.5-.5 2.58-.93-1.21-.36-2.86 1.95-2.31 2.05-2.08ZM362.17 455.87c-.2.14-1-.93-.89-2.48s1.05-2.52 1.23-2.36-.25 1.14-.32 2.41.21 2.32-.02 2.43ZM259.73 325.23c.43 1.14 27.19 31 33.15 39.7 6.74 9.84 8.16 18.16 8.49 20.11 1.86 11.15 3.25 19 8.31 31.15S324.73 454 324.73 454l23.63-6.09s-13.93-59.44-15.19-69.34-3.38-17.57-8.87-26.67-29.71-58.77-29.71-58.77l-38.28 16.09Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M242.57 240.76s7.74.11 8.88.82 5.69 4.21 5.69 5a3 3 0 0 0 .32 1.38s2.86 4 1.57 5.71-2.24 2.27-2.21 2.8-.38 2.37-1.47 2.87a10.78 10.78 0 0 0-2.88 1.84 10.63 10.63 0 0 1-1.73 2.06 10.75 10.75 0 0 1-2.48 1.15 2.92 2.92 0 0 1-2.55 1.57c-1.64-.15-4.63-3.27-5.6-4.43-1.21-1.45-4.29-4-4.69-5.29a14.76 14.76 0 0 0-.92-2.3l-7.5-7.09 11.7-11.75Z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M257.7 248.6a20.35 20.35 0 0 1-2.46-2.31 19.31 19.31 0 0 1-2.41-2.36 11.74 11.74 0 0 1 4.87 4.67Z"
 | 
			
		||||
					style="fill:#ff9a6c"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M238.46 161.84s-8.9 7.54-11 9.43-29.46 24.87-31.29 35.27c-1.16 6.66 4 11.87 6.77 16.24 2 3.26 14.32 25.66 25.83 26.31.68 0 4.15 4.28 4.15 4.28l12.49-10.65a52.68 52.68 0 0 1-4.73-7c-5-9.38-15.89-24-15.55-24.66 2.43-4.55-3.34-3.47-3.34-3.47l23.94-21.95Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M198.89 199.64s-1.73-3.09 0-5.67S225 171.68 225 171.68s1.19.31 2.06 2.27-27 26.62-27 26.62Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M207.41 192.32s14 15.79 16 15.27 11.7-7.84 12.14-10.22-13.33-19-13.33-19ZM187.57 91.83c3.07.73 5.47 3.21 8.28 4.74 5.27 2.86 11.66 2.1 16.68-.23s9.1-6 13.61-9.1 8.09-6.22 14.6-6.81c11.49-1 23 9.11 17.78 17.7l-9.65 3.37a60.17 60.17 0 0 1-25.79 32.32c-2.42 1.47-5.16 2.83-8.17 2.45s-6.07-3.25-5.43-6c-4.78 1-9.7 1.92-14.7 1.17s-10.19-3.54-12.41-8-.26-10.48 4.56-11.35c-4.45-2.73-8.26-7.18-8.54-12s4.18-9.45 9.18-8.26Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M283.46 72.49c-6.22-1.69-12 .21-17.71 3.25a32.78 32.78 0 0 0-16.23 33.58 25.23 25.23 0 0 0 10.86 16.58c7.83 5 18.5 4.54 26.4-.31s13-13.65 14.27-22.84a30.7 30.7 0 0 0-3.05-18.6 23.36 23.36 0 0 0-14.53-11.66"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M263.37 92.26c1.16-2.33 1.06-5.07 1.68-7.6 1.36-5.57 3.94-10.5 9.42-12.19 5.1-1.57 13.45.3 19.16 2.78a22.62 22.62 0 0 1 9.88 7.9 17.84 17.84 0 0 1 1.71 16.26l-28.5-7c-2.92 2-5.92 4-9.32 4.91a6 6 0 0 1-3.07.19 2.7 2.7 0 0 1-2.06-2.07c-.16-1.15.59-2.17 1.1-3.18Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m257.85 145 5-54.65a4.74 4.74 0 0 1 4.85-4.19l25.76-2.89c8.09.25 12 10.79 11.46 18.86-.65 9-1.81 19.89-3.77 26.38-3.94 13.05-14.39 13.35-14.39 13.35s-.06.52-.78 5.35c0 0-1.31 10-12.46 7.81-7.24-1.42-15.67-10.02-15.67-10.02Z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M300.16 112.56a1.72 1.72 0 0 1-1.88 1.5 1.66 1.66 0 0 1-1.56-1.78 1.75 1.75 0 0 1 1.89-1.51 1.65 1.65 0 0 1 1.55 1.79ZM301.8 107.49c-.24.19-1.41-.94-3.25-1.19s-3.32.49-3.49.24.19-.49.84-.84a4.8 4.8 0 0 1 2.83-.46 4.49 4.49 0 0 1 2.54 1.22c.51.54.65.94.53 1.03ZM284.13 110.14a1.73 1.73 0 0 1-1.89 1.5 1.66 1.66 0 0 1-1.55-1.78 1.74 1.74 0 0 1 1.89-1.51 1.65 1.65 0 0 1 1.55 1.79ZM284.19 104.44c-.24.2-1.41-.94-3.25-1.19s-3.32.49-3.49.24.19-.49.84-.83a4.73 4.73 0 0 1 2.83-.47 4.49 4.49 0 0 1 2.54 1.23c.51.51.65.94.53 1.02ZM288.47 119.93a12.12 12.12 0 0 1 3.06-.14c.48 0 .94 0 1.06-.32a2.42 2.42 0 0 0-.12-1.44c-.29-1.2-.59-2.45-.9-3.77-1.24-5.36-2.06-9.74-1.84-9.79s1.39 4.26 2.62 9.62c.29 1.32.58 2.58.84 3.78a2.7 2.7 0 0 1 0 1.88 1.16 1.16 0 0 1-.86.58 3.79 3.79 0 0 1-.82 0 12.7 12.7 0 0 1-3.04-.4Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M286.76 141.88a34.19 34.19 0 0 1-17.21-7.2s3.23 9.85 16.75 10.42ZM287.09 124.26a3.35 3.35 0 0 0-2.81-1.64 3 3 0 0 0-2.24.83 1.93 1.93 0 0 0-.45 2.18 2.26 2.26 0 0 0 2.33.94 7.25 7.25 0 0 0 2.55-1.08 2.35 2.35 0 0 0 .6-.48.64.64 0 0 0 .09-.68"
 | 
			
		||||
					style="fill:#ff9a6c"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M282.88 120.46c.3 0 0 2 1.55 3.67s3.67 1.73 3.65 2c0 .13-.53.32-1.43.22a5 5 0 0 1-3.06-1.56 4.31 4.31 0 0 1-1.17-3.06c.03-.84.32-1.3.46-1.27ZM285.77 97.11c-.25.47-2.06 0-4.25 0s-4 .31-4.25-.17c-.1-.24.27-.67 1-1a7.07 7.07 0 0 1 3.26-.67 7.2 7.2 0 0 1 3.21.83c.79.33 1.14.78 1.03 1.01ZM302.37 99.37c-.39.37-1.59-.24-3.12-.47s-2.85-.1-3.09-.58c-.1-.23.16-.62.78-.94a4.55 4.55 0 0 1 5 .84c.48.51.6.96.43 1.15Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M259.12 87.91c3.37-7.43 15.81-16.79 23.82-15.28l12 9.26a12.54 12.54 0 0 1-9.57 8.54c-2.36.42-5.13.29-6.72 2.09s-1.15 4.86-2.69 6.77c-2.12 2.66-7 1.95-8.69 4.89-.75 1.28-.63 2.94-1.4 4.21-1 1.69-5.82 3.43-7.62 2.61s-1.44-4-1.85-5.74a9 9 0 0 1 .42-6c1.26-3.65.7-7.8 2.3-11.35Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M263.59 113.23c.13-1-.7-3.49-1.66-3.78-2.56-.76-7.31-1.2-8.27 5.45-1.3 9.09 8.09 8.52 8.15 8.26s1.32-6.5 1.78-9.93Z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M259.53 119.32s-.18.09-.46.17a1.64 1.64 0 0 1-1.19-.14c-.91-.49-1.5-2.13-1.31-3.78a5.29 5.29 0 0 1 .77-2.2 1.83 1.83 0 0 1 1.36-1 .82.82 0 0 1 .88.56c.08.28 0 .45 0 .47s.22-.12.2-.51a1 1 0 0 0-.28-.62 1.23 1.23 0 0 0-.84-.36 2.24 2.24 0 0 0-1.85 1.18 5.24 5.24 0 0 0-.91 2.45c-.21 1.84.48 3.69 1.74 4.25a1.68 1.68 0 0 0 1.48 0c.36-.25.43-.45.41-.47Z"
 | 
			
		||||
					style="fill:#ff9a6c"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M292.56 78.87c.48 3.8 5.66 9.12 7.86 12.25 4.19 6 4.95 7.91 3.7 20.24 2.65-4 4.76-8 5.06-12.88a20 20 0 0 0-4-13.6 15.6 15.6 0 0 0-12.61-6"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M254.48 87.34a9.48 9.48 0 0 0 1.51.54c.49.16 1.1.34 1.82.5s1.52.39 2.44.52a43.13 43.13 0 0 0 6.47.7 49.17 49.17 0 0 0 7.95-.43 48.36 48.36 0 0 0 7.76-1.81 44.66 44.66 0 0 0 6-2.47c.85-.39 1.56-.83 2.2-1.19s1.18-.7 1.6-1a10.08 10.08 0 0 0 1.31-.94 9 9 0 0 0-1.42.76c-.44.26-1 .58-1.64.9s-1.37.75-2.21 1.11a49.36 49.36 0 0 1-6 2.34 52.85 52.85 0 0 1-7.69 1.75 53.22 53.22 0 0 1-7.86.47 47.11 47.11 0 0 1-6.42-.58c-.91-.11-1.72-.3-2.43-.44s-1.33-.28-1.83-.41a8.25 8.25 0 0 0-1.56-.32ZM252.89 93.79a3.74 3.74 0 0 0 1 .36 27.49 27.49 0 0 0 2.91.67 36.71 36.71 0 0 0 19.39-1.88 24.88 24.88 0 0 0 2.81-1.21 3.77 3.77 0 0 0 .94-.55c0-.08-1.44.59-3.8 1.4a41.36 41.36 0 0 1-19.2 1.85c-2.52-.34-4.03-.73-4.05-.64Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m155.92 343.89-19.68-8-8.49 15.01 4.13 8.92 18.26 6.26 5.78-22.19z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m133 362.05-1.5-.59s-20.27 21.82-25.21 20.16l20.18-51.36 16.53 7.53c-6.47 6.21-10 14.05-10 24.26Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><g style="opacity:.6000000000000001"
 | 
			
		||||
					><path
 | 
			
		||||
						d="M135.6 340.8a2.25 2.25 0 0 0-1.79-2.4 2.14 2.14 0 0 0-2.36 1.78 2.36 2.36 0 0 0 1.9 2.53 2.23 2.23 0 0 0 2.26-2.12"
 | 
			
		||||
						style="fill:#fff"
 | 
			
		||||
					/></g
 | 
			
		||||
				><g style="opacity:.6000000000000001"
 | 
			
		||||
					><path
 | 
			
		||||
						d="m126.48 330.26 4 1.81-20.81 48.57s-1.87 1.89-3.35 1Z"
 | 
			
		||||
						style="fill:#fff"
 | 
			
		||||
					/></g
 | 
			
		||||
				><path
 | 
			
		||||
					d="M131.3 361.58c-.25-.11.13-1.41-.17-3s-1.09-2.64-.9-2.83 1.46.76 1.8 2.65-.53 3.33-.73 3.18ZM127 366.08c-.26 0-.26-1.3-1-2.57s-1.72-2-1.61-2.22 1.57.16 2.42 1.77.42 3.11.19 3.02ZM119.45 366.56c0-.25 1.4-.09 2.5 1.1s1.17 2.58.93 2.6-.69-1-1.6-2-1.91-1.44-1.83-1.7ZM145.78 339.09s15.36 4.19 25.9 6.15 22.64 4.78 31 7c8.81 2.29 12.72 3.59 12.72 3.59s2 2.61 3.5-7.74 6.1-47.73 6.1-47.73l5.06-16.08 30.38-5.32 16.12 21.27s-14.57 47.3-19.53 57.09-12 38.32-29.14 33.87c-10.14-2.63-89.33-27.5-89.33-27.5Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m223.39 313.78-.13 1.06-.33 2.63 23.27-11.21s.15 10.69 22 19l2.78-4.84v-14.16l-16.89-13.13h-11.68l-18.61 14"
 | 
			
		||||
					style="opacity:.30000000000000004"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M270.41 320.55a72.49 72.49 0 0 1-2.56 8.15 71.37 71.37 0 0 1-3.05 8 70.07 70.07 0 0 1 2.56-8.15 67 67 0 0 1 3.05-8ZM223.88 365.86a37.35 37.35 0 0 0-3.49-5.44 37.33 37.33 0 0 0-4.76-4.36 5.42 5.42 0 0 1 1.7.91 17 17 0 0 1 3.46 3.12 17.56 17.56 0 0 1 2.49 3.93 5.55 5.55 0 0 1 .6 1.84Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M226.55 272.74s-5.06 13.2-4.88 20.65-2.73 22.38-2.73 22.38 27.33-14.67 29.37-16.45c0 0-2 14.72 21.22 21.58l2 .77 7.13-21.5a68.35 68.35 0 0 0 6.19-12c1.88-5.42 4.5-15.41 4.5-15.41L237.47 258Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M289.38 287.64s12.76 17.61 12.94 20.49l-32.44 6.43-3-12.36-1.31-12.87Z"
 | 
			
		||||
					style="fill:#455a64"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M239.86 171.23c.34.94 7 23.93 7 23.93l4 33.37s-6.73 19.75-12.4 26.21c-17.58 20.09-15.68 25.57-15.68 25.57s29.74-8 44.39-5.8 10.66 27.39 32 21.41l-.17-20.54s2.68-18.95 3.35-25.57-.24-30.41-.24-30.41 7.33-23.3 7.33-30.85 1.75-7.88-4.29-18.27-15-24.24-15-24.24l-33.63-2.84-18.64 19.21Z"
 | 
			
		||||
					style="fill:#e0e0e0"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M281.7 292.51c.13 0-1.75 6.24-4.21 13.83s-4.56 13.72-4.7 13.67 1.75-6.23 4.21-13.83 4.56-13.72 4.7-13.67Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M238.71 172.17c-4.43 1.19-8.35 2.56-13.19 0s-8.89-7.36-9-12.84c-.06-4.48-.07-12.74 3.53-15.41s20.16-5.39 29.43-5.36c35.69.11 40.29 6 46 8.93a5.33 5.33 0 0 1-2.15 5.57"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M235.67 165.5s1 19.66 2.47 27.09 7.38 29.67 7.38 29.67-3.28 1.23-2.79 5.68c0 0-3.55 3.09-3.95 9.89s1.57 9.76 1.57 9.76l-3.84 7.72 64.36 12.17 1.08-7.27s4.66 3 5.51-11-.82-24.45-.08-37.3 5.53-17.86 4.3-29.47C311 175.62 289.35 147 289.35 147s-31.46-5.35-34.62-1.91-19.06 20.41-19.06 20.41Z"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M304 226.09c-15.48-8.7-26.72-23.17-37.53-37.27l-1.14-2.82c1.11 5.89 1.42 14.1 2.54 20a72.63 72.63 0 0 0 3.66 14.11 24.45 24.45 0 0 0 8.74 11.42c7 4.61 16.5 3.85 23.85-.1 1.08-.58 2.25-1.45 2.23-2.67s-1.28-2.05-2.35-2.67Z"
 | 
			
		||||
					style="opacity:.30000000000000004"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M234.28 178.25a10.43 10.43 0 0 1 .6 1.69c.35 1.09.82 2.69 1.37 4.67 1.11 4 2.45 9.48 3.87 15.59s2.72 11.64 3.75 15.61c.51 1.93.93 3.51 1.24 4.71a11.9 11.9 0 0 1 .41 1.74 11.81 11.81 0 0 1-.6-1.69c-.37-1.09-.84-2.68-1.4-4.66-1.13-4-2.5-9.48-3.91-15.59s-2.7-11.64-3.71-15.62c-.5-1.92-.91-3.51-1.22-4.71a12.51 12.51 0 0 1-.4-1.74ZM265.5 233.75a26.55 26.55 0 0 1-3.27-1.23c-2-.79-4.76-1.89-7.89-2.86a49.52 49.52 0 0 0-8.18-1.79c-2.13-.26-3.46-.28-3.46-.37a2.82 2.82 0 0 1 .94-.08 20.82 20.82 0 0 1 2.56.09 39.56 39.56 0 0 1 8.29 1.66 69.24 69.24 0 0 1 7.89 3c1 .45 1.77.82 2.31 1.1a3.61 3.61 0 0 1 .81.48ZM305.27 259.42a4.28 4.28 0 0 1-.58.35 10.73 10.73 0 0 1-1.82.75 18.56 18.56 0 0 1-7.17.46 230.88 230.88 0 0 1-23.22-4.74c-9-2.19-17.11-4.25-23-5.61l-7-1.56-1.91-.42a3.36 3.36 0 0 1-.66-.18 3.14 3.14 0 0 1 .68.08l1.93.32c1.67.29 4.07.77 7 1.4 5.93 1.27 14.07 3.28 23.05 5.47s17.16 4 23.14 4.87a19.12 19.12 0 0 0 7.08-.29 23.9 23.9 0 0 0 2.48-.9Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M227.92 172.83c2.38 4 7.5 5.94 12.14 5.33s8.78-3.37 11.94-6.83 5.46-7.59 7.73-11.68l6.17-11.12c-7.32 3.05-9.81 11.52-15.35 17.2s-16.27 9.44-22.63 7.1"
 | 
			
		||||
					style="opacity:.30000000000000004"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M266.52 148.59s-.73.18-2 .74a14.62 14.62 0 0 0-4.52 3.43 41.85 41.85 0 0 0-4.84 7 47 47 0 0 1-2.83 4.21 21.16 21.16 0 0 1-3.91 3.82 30.22 30.22 0 0 1-9.38 4.72 25 25 0 0 1-8.49 1.18 12.17 12.17 0 0 1-5.58-1.48 6.62 6.62 0 0 1-1.27-.93 2.54 2.54 0 0 1-.38-.4 18.82 18.82 0 0 0 1.75 1.16 12.65 12.65 0 0 0 5.5 1.28 25.19 25.19 0 0 0 8.33-1.27 30.59 30.59 0 0 0 9.21-4.67 21.42 21.42 0 0 0 3.82-3.72 47.6 47.6 0 0 0 2.82-4.15 40.4 40.4 0 0 1 5-7 14.13 14.13 0 0 1 4.7-3.36 9.15 9.15 0 0 1 1.5-.47 2 2 0 0 1 .57-.09ZM260.76 225.38c-.07.16-3.33-1.38-7.65-2.23a77 77 0 0 0-7.93-.88 8.27 8.27 0 0 1 2.35-.19 29.41 29.41 0 0 1 5.68.56 28.85 28.85 0 0 1 5.45 1.66 8.68 8.68 0 0 1 2.1 1.08ZM243 227.38c.07.07-.77 1-1.77 2.64a19.77 19.77 0 0 0-1.48 3.08 21.77 21.77 0 0 0-1 4.05 21.32 21.32 0 0 0-.16 4.17 20.22 20.22 0 0 0 .55 3.38c.48 1.89 1 3 .94 3a3.32 3.32 0 0 1-.44-.74 7.08 7.08 0 0 1-.41-.93 12.82 12.82 0 0 1-.45-1.26 17.38 17.38 0 0 1-.66-3.44 19.55 19.55 0 0 1 .11-4.29 20.16 20.16 0 0 1 1.1-4.14 17.53 17.53 0 0 1 1.62-3.11 12.25 12.25 0 0 1 .78-1.08 8.48 8.48 0 0 1 .66-.77c.38-.37.61-.58.61-.56Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M234.59 178.76a19.68 19.68 0 0 0 3.4 13.92c-.99-4.68-2.36-9.22-3.4-13.92Z"
 | 
			
		||||
					style="opacity:.30000000000000004"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M340.48 208.44s5.27-6.17 6.6-6.62 7.22-1.77 7.88-1.22a3.09 3.09 0 0 0 1.33.67s5.11.33 5.66 2.55.34 3.33.78 3.66 1.67 1.89 1.33 3.11a11.48 11.48 0 0 0-.44 3.55 10.77 10.77 0 0 1 .5 2.78 11.39 11.39 0 0 1-.72 2.77 3.08 3.08 0 0 1-.45 3.11c-1.22 1.22-5.73 1.55-7.32 1.56-2 0-6.07.8-7.41.24a16.44 16.44 0 0 0-2.47-.8L335 225.08l-1.66-17.3Z"
 | 
			
		||||
					style="fill:#ffbf9d"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M356.93 201.48a23.19 23.19 0 0 1-3.51.42 22.36 22.36 0 0 1-3.51.38 8.47 8.47 0 0 1 3.45-.9 8.38 8.38 0 0 1 3.57.1Z"
 | 
			
		||||
					style="fill:#ff9a6c"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M264 182.13c4 10.83 7.79 25.38 10.6 33.79 3.81 11.43 15.63 15.68 21.67 16.14 7.73.58 45.4-6.55 45.4-6.55l-1.74-19-4.27.11a5.72 5.72 0 0 0-3-1.44c-7.93-1.87-29.3-.31-29.3-.31s-5.75-23.11-7-34.46c-1-9-1-23.68-15.07-24.53-9.15.43-15.06 3.81-17.29 13.61"
 | 
			
		||||
					style="fill:#5e9cff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M264 159.53s0-.14.08-.4.16-.67.29-1.18a19.72 19.72 0 0 1 1.76-4.37 13.72 13.72 0 0 1 5.28-5.37 22.3 22.3 0 0 1 9.87-2.44l1.53.19a8.4 8.4 0 0 1 1.56.29l1.58.44 1.53.71a12.92 12.92 0 0 1 5 4.92 23.71 23.71 0 0 1 2.63 7.23 74 74 0 0 1 1.13 8.18 161.08 161.08 0 0 0 2.91 17.77c1.3 6.2 2.78 12.67 4.42 19.35l-.26-.18c5.12-.35 10.45-.59 15.92-.62 2.74 0 5.52 0 8.33.23a38.23 38.23 0 0 1 4.22.51 13.37 13.37 0 0 1 2.1.51 5.35 5.35 0 0 1 1.9 1.17l-.19-.07 4.27-.11h.24v.51c.59 6.41 1.17 12.7 1.72 18.7v.23h-.23c-7.65 1.43-15.09 2.73-22.26 3.88q-5.39.87-10.55 1.57c-3.45.47-6.82.88-10.14 1.06a22.36 22.36 0 0 1-9.5-1.65 28.77 28.77 0 0 1-7.87-4.54 23 23 0 0 1-5.39-6.5 36.76 36.76 0 0 1-2.74-7.26c-2.84-9.43-5-17-6.64-22.2-.82-2.6-1.49-4.59-1.95-5.94-.22-.66-.4-1.17-.52-1.53a5.06 5.06 0 0 1-.16-.52s.07.16.2.5.33.86.57 1.52c.49 1.34 1.19 3.32 2 5.91 1.72 5.16 3.91 12.75 6.79 22.16a36.86 36.86 0 0 0 2.75 7.18 22.52 22.52 0 0 0 5.32 6.37 28.72 28.72 0 0 0 7.76 4.45 22.15 22.15 0 0 0 9.33 1.59c3.29-.18 6.66-.6 10.1-1.07s6.95-1 10.53-1.59c7.16-1.16 14.6-2.46 22.24-3.89l-.21.27c-.55-6-1.13-12.28-1.72-18.69v-.26l.26.23-4.26.11h-.11l-.08-.08a6.64 6.64 0 0 0-3.74-1.54 36.32 36.32 0 0 0-4.16-.5c-2.78-.2-5.55-.24-8.28-.24-5.46 0-10.78.26-15.89.6h-.21l-.05-.21a536.51 536.51 0 0 1-4.39-19.37 156.74 156.74 0 0 1-2.88-17.83 74.73 74.73 0 0 0-1.1-8.14 23.43 23.43 0 0 0-2.55-7.13 12.64 12.64 0 0 0-4.89-4.8l-1.48-.7-1.54-.44a7.37 7.37 0 0 0-1.53-.29l-1.51-.2a22.13 22.13 0 0 0-9.77 2.35 13.6 13.6 0 0 0-5.25 5.24 20.45 20.45 0 0 0-1.82 4.31l-.33 1.17Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M298.27 216a19.44 19.44 0 0 0 5-11.61 5.13 5.13 0 0 1 .08 1.92 14.38 14.38 0 0 1-1.12 4.51 14.19 14.19 0 0 1-2.5 3.92 4.86 4.86 0 0 1-1.46 1.26ZM337 226a88.26 88.26 0 0 1-1.07-9.59 91.09 91.09 0 0 1-.54-9.64 87.31 87.31 0 0 1 1.06 9.6A89.66 89.66 0 0 1 337 226Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/></svg
 | 
			
		||||
			>
 | 
			
		||||
			<h3 class="text-2xl font-bold">{$_("done")}</h3>
 | 
			
		||||
			<h4 class="text-xl font-semibold">
 | 
			
		||||
				{cardCode}<br />{runnerinfo.firstname}
 | 
			
		||||
				{runnerinfo.lastname} [#{runnerinfo.id}]
 | 
			
		||||
			</h4>
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					resetAll();
 | 
			
		||||
				}}
 | 
			
		||||
				type="button"
 | 
			
		||||
				class="py-3 px-4 inline-flex items-center gap-x-2 text-sm font-medium rounded-lg border border-transparent bg-blue-100 text-blue-800 hover:bg-blue-200 focus:outline-hidden focus:bg-blue-200 disabled:opacity-50 disabled:pointer-events-none dark:text-blue-500 dark:bg-blue-800/30 dark:hover:bg-blue-800/20 dark:focus:bg-blue-800/20 mt-2"
 | 
			
		||||
			>
 | 
			
		||||
				{$_("next_runner")}
 | 
			
		||||
			</button>
 | 
			
		||||
		</div>
 | 
			
		||||
	{:else if state === "assigning"}
 | 
			
		||||
		<p class="text-center font-semibold">
 | 
			
		||||
			{$_("please_wait_a_moment_while_we_assign_the_card")}<br />{cardCode}
 | 
			
		||||
		</p>
 | 
			
		||||
	{:else if state === "error_runner"}
 | 
			
		||||
		<div class="text-center mx-auto">
 | 
			
		||||
			<svg
 | 
			
		||||
				class="h-64 mx-auto"
 | 
			
		||||
				xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
				viewBox="0 0 500 500"
 | 
			
		||||
				><path
 | 
			
		||||
					d="M298.37 335.5C382 299.85 469.46 233.1 432.31 135 398.6 46 284.74 25.75 219.62 102.47c-28.09 33.09-23.18 77.05-57.16 106.51s-90.4 45.83-75.13 104c23.67 89.93 156 46.02 211.04 22.52Z"
 | 
			
		||||
					style="fill:#407bff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M298.37 335.5C382 299.85 469.46 233.1 432.31 135 398.6 46 284.74 25.75 219.62 102.47c-28.09 33.09-23.18 77.05-57.16 106.51s-90.4 45.83-75.13 104c23.67 89.93 156 46.02 211.04 22.52Z"
 | 
			
		||||
					style="fill:#fff;opacity:.9"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M360.6 263.05h-.36c-26.64-2.18-45-25-45.74-25.92a4.47 4.47 0 0 1 7-5.55c.21.27 15.9 19.61 37.63 22.37 7-7 13-25.48 12.33-31.07v-.16c-.14-1.8-.48-8 1.29-11.65a4.47 4.47 0 0 1 8 3.88c-.44.92-.65 4.23-.44 7 1 9.2-7 32.42-17 40.19a4.47 4.47 0 0 1-2.71.91ZM148.82 238.82a65.8 65.8 0 0 1-48.56-22.28 4.46 4.46 0 0 1-.26-5.64c7.22-9.71 20-32.64 22-40.11a10.91 10.91 0 0 0-4.14-4.33 4.45 4.45 0 0 1-2.55-3.61l-.72-7.32a4.47 4.47 0 0 1 8.89-.88l.5 5.09a22.34 22.34 0 0 1 6.81 8.65 4.48 4.48 0 0 1 .32 2.26c-.92 7.93-13.79 30.9-21.71 42.51 18.49 18.43 40.59 16.75 41.56 16.66a4.47 4.47 0 0 1 .82 8.9c-.26.02-1.29.1-2.96.1ZM292.87 416.09h-12a4.47 4.47 0 0 1-4.31-5.66c3.13-11.24 4.67-20.39 5.82-34.71-4.24-20-8.23-38.21-8.27-38.39a4.47 4.47 0 0 1 8.73-1.91c0 .18 4.12 18.86 8.41 39.08a4.23 4.23 0 0 1 .08 1.28c-1 12.86-2.31 21.75-4.67 31.38h6.18a4.47 4.47 0 0 1 0 8.93ZM200.32 416.09h-6.76a4.45 4.45 0 0 1-4.42-5.08c1.15-8.2 7-23.13 13.3-38.14 2.23-19.8 4.05-36.8 4.07-37a4.47 4.47 0 1 1 8.88 1c0 .17-1.88 17.56-4.15 37.65a4.31 4.31 0 0 1-.32 1.22c-4.43 10.63-9.49 23.15-11.8 31.44h1.2a4.47 4.47 0 1 1 0 8.93Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m204.21 111-52.06 52.07c-2.62 57.71-2.41 118.33 0 181.18h172.16c-3.41-81.1-3.73-159.17 0-233.25Z"
 | 
			
		||||
					style="fill:#fff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M324.31 345.13H152.15a.9.9 0 0 1-.9-.86c-2.49-65.27-2.49-126.27 0-181.27a.9.9 0 0 1 .27-.59l52.06-52.07a.89.89 0 0 1 .63-.26h120.1a.9.9 0 0 1 .65.28.87.87 0 0 1 .24.66c-3.59 71.34-3.59 147.61 0 233.17a.89.89 0 0 1-.25.65.86.86 0 0 1-.64.29ZM153 343.34h170.38c-3.54-84.86-3.55-160.59 0-231.47h-118.8L153 163.43c-2.45 54.64-2.45 115.16 0 179.91Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M214.28 219.19c-.2-4.36-2.67-7.8-5.53-7.7s-5 3.71-4.82 8.07 2.67 7.8 5.53 7.69 5.02-3.71 4.82-8.06ZM274.65 217.82c-.2-4.35-2.67-7.79-5.53-7.69s-5 3.71-4.82 8.07 2.68 7.8 5.53 7.69 5.02-3.71 4.82-8.07ZM229.35 237a36.55 36.55 0 0 1 28.63 1.3 1.27 1.27 0 0 1 .49 1.74 1.3 1.3 0 0 1-1.75.49c-.15-.08-14.4-7.76-31.41 1a1.31 1.31 0 0 1-1.74-.54 1.27 1.27 0 0 1 .55-1.72 41.73 41.73 0 0 1 5.23-2.27ZM205.64 178.34a2.64 2.64 0 0 1 1.26.36 2.58 2.58 0 0 1 .92 3.51A25.29 25.29 0 0 1 188.27 195a2.59 2.59 0 0 1-2.69-2.45 2.55 2.55 0 0 1 2.44-2.66c.39 0 9.62-.58 15.36-10.27a2.52 2.52 0 0 1 2.26-1.28ZM266.05 176.87a2.57 2.57 0 0 1 2.33.72c8 8 17.14 6.39 17.52 6.32a2.6 2.6 0 0 1 3 2 2.54 2.54 0 0 1-2 3c-.5.09-12.14 2.31-22.21-7.75a2.54 2.54 0 0 1 1.31-4.3Z"
 | 
			
		||||
					style="fill:#407bff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m321.72 204.86-7.31.68a5.22 5.22 0 0 1-5.58-4.06L298.7 156.1a5.22 5.22 0 0 1 3.77-6.18l19.59-5.14ZM209 167.69c-5.09-13.89-10.18-36.12-4.81-56.71l-52.06 52.07c14.73 4.95 38.19 7.06 56.87 4.64Z"
 | 
			
		||||
					style="opacity:.2"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M204.21 163.05c-5.71-16.86-3.38-39.78 0-52.07l-52.06 52.07c15.76 2.87 33.37 2.41 52.06 0Z"
 | 
			
		||||
					style="fill:#fff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M176 165.92a133.14 133.14 0 0 1-24-2 .88.88 0 0 1-.47-1.5l52.06-52.07a.89.89 0 0 1 1.49.87c-3.14 11.44-5.75 34.6 0 51.54a.93.93 0 0 1-.09.76.87.87 0 0 1-.64.41 221.85 221.85 0 0 1-28.35 1.99Zm-22-3.46c13.84 2.29 29.91 2.24 49-.16-4.71-14.94-3.64-34.71-.48-48.4Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/></svg
 | 
			
		||||
			>
 | 
			
		||||
			<p class="text-lg font-semibold">{$_("runner_not_found")}</p>
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					resetAll();
 | 
			
		||||
				}}
 | 
			
		||||
				type="button"
 | 
			
		||||
				class="py-3 px-4 inline-flex items-center gap-x-2 text-sm font-medium rounded-lg border border-transparent bg-blue-100 text-blue-800 hover:bg-blue-200 focus:outline-hidden focus:bg-blue-200 disabled:opacity-50 disabled:pointer-events-none dark:text-blue-500 dark:bg-blue-800/30 dark:hover:bg-blue-800/20 dark:focus:bg-blue-800/20 mt-2"
 | 
			
		||||
			>
 | 
			
		||||
				{$_("try_again")}
 | 
			
		||||
			</button>
 | 
			
		||||
		</div>
 | 
			
		||||
	{:else if state === "error_card"}
 | 
			
		||||
		<div class="text-center mx-auto">
 | 
			
		||||
			<svg
 | 
			
		||||
				class="h-64 mx-auto"
 | 
			
		||||
				xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
				viewBox="0 0 500 500"
 | 
			
		||||
				><path
 | 
			
		||||
					d="M298.37 335.5C382 299.85 469.46 233.1 432.31 135 398.6 46 284.74 25.75 219.62 102.47c-28.09 33.09-23.18 77.05-57.16 106.51s-90.4 45.83-75.13 104c23.67 89.93 156 46.02 211.04 22.52Z"
 | 
			
		||||
					style="fill:#407bff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M298.37 335.5C382 299.85 469.46 233.1 432.31 135 398.6 46 284.74 25.75 219.62 102.47c-28.09 33.09-23.18 77.05-57.16 106.51s-90.4 45.83-75.13 104c23.67 89.93 156 46.02 211.04 22.52Z"
 | 
			
		||||
					style="fill:#fff;opacity:.9"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M360.6 263.05h-.36c-26.64-2.18-45-25-45.74-25.92a4.47 4.47 0 0 1 7-5.55c.21.27 15.9 19.61 37.63 22.37 7-7 13-25.48 12.33-31.07v-.16c-.14-1.8-.48-8 1.29-11.65a4.47 4.47 0 0 1 8 3.88c-.44.92-.65 4.23-.44 7 1 9.2-7 32.42-17 40.19a4.47 4.47 0 0 1-2.71.91ZM148.82 238.82a65.8 65.8 0 0 1-48.56-22.28 4.46 4.46 0 0 1-.26-5.64c7.22-9.71 20-32.64 22-40.11a10.91 10.91 0 0 0-4.14-4.33 4.45 4.45 0 0 1-2.55-3.61l-.72-7.32a4.47 4.47 0 0 1 8.89-.88l.5 5.09a22.34 22.34 0 0 1 6.81 8.65 4.48 4.48 0 0 1 .32 2.26c-.92 7.93-13.79 30.9-21.71 42.51 18.49 18.43 40.59 16.75 41.56 16.66a4.47 4.47 0 0 1 .82 8.9c-.26.02-1.29.1-2.96.1ZM292.87 416.09h-12a4.47 4.47 0 0 1-4.31-5.66c3.13-11.24 4.67-20.39 5.82-34.71-4.24-20-8.23-38.21-8.27-38.39a4.47 4.47 0 0 1 8.73-1.91c0 .18 4.12 18.86 8.41 39.08a4.23 4.23 0 0 1 .08 1.28c-1 12.86-2.31 21.75-4.67 31.38h6.18a4.47 4.47 0 0 1 0 8.93ZM200.32 416.09h-6.76a4.45 4.45 0 0 1-4.42-5.08c1.15-8.2 7-23.13 13.3-38.14 2.23-19.8 4.05-36.8 4.07-37a4.47 4.47 0 1 1 8.88 1c0 .17-1.88 17.56-4.15 37.65a4.31 4.31 0 0 1-.32 1.22c-4.43 10.63-9.49 23.15-11.8 31.44h1.2a4.47 4.47 0 1 1 0 8.93Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m204.21 111-52.06 52.07c-2.62 57.71-2.41 118.33 0 181.18h172.16c-3.41-81.1-3.73-159.17 0-233.25Z"
 | 
			
		||||
					style="fill:#fff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M324.31 345.13H152.15a.9.9 0 0 1-.9-.86c-2.49-65.27-2.49-126.27 0-181.27a.9.9 0 0 1 .27-.59l52.06-52.07a.89.89 0 0 1 .63-.26h120.1a.9.9 0 0 1 .65.28.87.87 0 0 1 .24.66c-3.59 71.34-3.59 147.61 0 233.17a.89.89 0 0 1-.25.65.86.86 0 0 1-.64.29ZM153 343.34h170.38c-3.54-84.86-3.55-160.59 0-231.47h-118.8L153 163.43c-2.45 54.64-2.45 115.16 0 179.91Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M214.28 219.19c-.2-4.36-2.67-7.8-5.53-7.7s-5 3.71-4.82 8.07 2.67 7.8 5.53 7.69 5.02-3.71 4.82-8.06ZM274.65 217.82c-.2-4.35-2.67-7.79-5.53-7.69s-5 3.71-4.82 8.07 2.68 7.8 5.53 7.69 5.02-3.71 4.82-8.07ZM229.35 237a36.55 36.55 0 0 1 28.63 1.3 1.27 1.27 0 0 1 .49 1.74 1.3 1.3 0 0 1-1.75.49c-.15-.08-14.4-7.76-31.41 1a1.31 1.31 0 0 1-1.74-.54 1.27 1.27 0 0 1 .55-1.72 41.73 41.73 0 0 1 5.23-2.27ZM205.64 178.34a2.64 2.64 0 0 1 1.26.36 2.58 2.58 0 0 1 .92 3.51A25.29 25.29 0 0 1 188.27 195a2.59 2.59 0 0 1-2.69-2.45 2.55 2.55 0 0 1 2.44-2.66c.39 0 9.62-.58 15.36-10.27a2.52 2.52 0 0 1 2.26-1.28ZM266.05 176.87a2.57 2.57 0 0 1 2.33.72c8 8 17.14 6.39 17.52 6.32a2.6 2.6 0 0 1 3 2 2.54 2.54 0 0 1-2 3c-.5.09-12.14 2.31-22.21-7.75a2.54 2.54 0 0 1 1.31-4.3Z"
 | 
			
		||||
					style="fill:#407bff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="m321.72 204.86-7.31.68a5.22 5.22 0 0 1-5.58-4.06L298.7 156.1a5.22 5.22 0 0 1 3.77-6.18l19.59-5.14ZM209 167.69c-5.09-13.89-10.18-36.12-4.81-56.71l-52.06 52.07c14.73 4.95 38.19 7.06 56.87 4.64Z"
 | 
			
		||||
					style="opacity:.2"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M204.21 163.05c-5.71-16.86-3.38-39.78 0-52.07l-52.06 52.07c15.76 2.87 33.37 2.41 52.06 0Z"
 | 
			
		||||
					style="fill:#fff"
 | 
			
		||||
				/><path
 | 
			
		||||
					d="M176 165.92a133.14 133.14 0 0 1-24-2 .88.88 0 0 1-.47-1.5l52.06-52.07a.89.89 0 0 1 1.49.87c-3.14 11.44-5.75 34.6 0 51.54a.93.93 0 0 1-.09.76.87.87 0 0 1-.64.41 221.85 221.85 0 0 1-28.35 1.99Zm-22-3.46c13.84 2.29 29.91 2.24 49-.16-4.71-14.94-3.64-34.71-.48-48.4Z"
 | 
			
		||||
					style="fill:#263238"
 | 
			
		||||
				/></svg
 | 
			
		||||
			>
 | 
			
		||||
			<p class="text-lg font-semibold">{$_("card_not_found")}</p>
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					state = "scan_card";
 | 
			
		||||
					scannerActive = true;
 | 
			
		||||
				}}
 | 
			
		||||
				type="button"
 | 
			
		||||
				class="py-3 px-4 inline-flex items-center gap-x-2 text-sm font-medium rounded-lg border border-transparent bg-blue-100 text-blue-800 hover:bg-blue-200 focus:outline-hidden focus:bg-blue-200 disabled:opacity-50 disabled:pointer-events-none dark:text-blue-500 dark:bg-blue-800/30 dark:hover:bg-blue-800/20 dark:focus:bg-blue-800/20 mt-2"
 | 
			
		||||
			>
 | 
			
		||||
				{$_("try_again")}
 | 
			
		||||
			</button>
 | 
			
		||||
		</div>
 | 
			
		||||
	{:else}
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		{#if runnerinfo.id === 0}
 | 
			
		||||
			<h3 class="text-2xl font-bold">{$_("scan_runner")}</h3>
 | 
			
		||||
			<h4 class="text-xl font-semibold">
 | 
			
		||||
				{$_("selfservice_qr_registration_barcode")}
 | 
			
		||||
			</h4>
 | 
			
		||||
		{:else}
 | 
			
		||||
			<h3 class="text-2xl font-bold">
 | 
			
		||||
				{runnerinfo.firstname}
 | 
			
		||||
				{runnerinfo.lastname}
 | 
			
		||||
			</h3>
 | 
			
		||||
			<p>
 | 
			
		||||
				ID: #{runnerinfo.id}<br />created_via: {runnerinfo.created_via}<br
 | 
			
		||||
				/>{runnerinfo.group.name}
 | 
			
		||||
			</p>
 | 
			
		||||
		{/if}
 | 
			
		||||
		<!--  -->
 | 
			
		||||
	{/if}
 | 
			
		||||
	{#if state === "scan_card"}
 | 
			
		||||
		<h3 class="text-2xl font-bold">{$_("scan_card")}</h3>
 | 
			
		||||
		<h4 class="text-xl font-semibold">{$_("code_128_barcode")}</h4>
 | 
			
		||||
	{/if}
 | 
			
		||||
	{#if state.includes("scan_")}
 | 
			
		||||
		{#if scannerActive}
 | 
			
		||||
			<QrCodeScanner
 | 
			
		||||
				:paused={!scannerActive}
 | 
			
		||||
				on:detect={(e) => {
 | 
			
		||||
					if (scannerActive) {
 | 
			
		||||
						scannerActive = false;
 | 
			
		||||
						console.log({ type: "DETECT", code: e.detail.decodedText });
 | 
			
		||||
						if (runnerinfo.id === 0) {
 | 
			
		||||
							new Audio("/beep.mp3").play();
 | 
			
		||||
							if (
 | 
			
		||||
								e.detail.decodedText.includes(
 | 
			
		||||
									"https://portal.lauf-fuer-kaya.de/profile/"
 | 
			
		||||
								)
 | 
			
		||||
							) {
 | 
			
		||||
								const runnerID = JSON.parse(
 | 
			
		||||
									atob(
 | 
			
		||||
										e.detail.decodedText
 | 
			
		||||
											.replace("https://portal.lauf-fuer-kaya.de/profile/", "")
 | 
			
		||||
											.split(".")[1]
 | 
			
		||||
									)
 | 
			
		||||
								).id;
 | 
			
		||||
								RunnerService.runnerControllerGetOne(runnerID)
 | 
			
		||||
									.then((runner) => {
 | 
			
		||||
										runnerinfo = runner;
 | 
			
		||||
									})
 | 
			
		||||
									.catch((e) => {
 | 
			
		||||
										console.error(e);
 | 
			
		||||
										state = "error_runner";
 | 
			
		||||
										// resetAll();
 | 
			
		||||
									});
 | 
			
		||||
							} else {
 | 
			
		||||
								const runnerID = parseInt(e.detail.decodedText);
 | 
			
		||||
								RunnerService.runnerControllerGetOne(runnerID)
 | 
			
		||||
									.then((runner) => {
 | 
			
		||||
										runnerinfo = runner;
 | 
			
		||||
									})
 | 
			
		||||
									.catch((e) => {
 | 
			
		||||
										console.error(e);
 | 
			
		||||
										state = "error_runner";
 | 
			
		||||
										// resetAll();
 | 
			
		||||
									});
 | 
			
		||||
							}
 | 
			
		||||
						} else {
 | 
			
		||||
							if (`${e.detail.decodedText}`.length > 10) {
 | 
			
		||||
								cardCode = e.detail.decodedText;
 | 
			
		||||
								new Audio("/beep.mp3").play();
 | 
			
		||||
								state = "assigning";
 | 
			
		||||
								RunnerCardService.runnerCardControllerGetAll()
 | 
			
		||||
									.then((cards) => {
 | 
			
		||||
										console.log(cards);
 | 
			
		||||
										const card = cards.find((c) => c.code === cardCode);
 | 
			
		||||
										if (card) {
 | 
			
		||||
											console.log("card found", card);
 | 
			
		||||
											RunnerCardService.runnerCardControllerPut(card.id, {
 | 
			
		||||
												enabled: true,
 | 
			
		||||
												id: card.id,
 | 
			
		||||
												runner: runnerinfo.id,
 | 
			
		||||
											})
 | 
			
		||||
												.then(() => {
 | 
			
		||||
													state = "done";
 | 
			
		||||
												})
 | 
			
		||||
												.catch(() => {
 | 
			
		||||
													state = "error_card";
 | 
			
		||||
													scannerActive = false;
 | 
			
		||||
												});
 | 
			
		||||
										} else {
 | 
			
		||||
											scannerActive = true;
 | 
			
		||||
										}
 | 
			
		||||
									})
 | 
			
		||||
									.catch(() => {
 | 
			
		||||
										scannerActive = true;
 | 
			
		||||
									});
 | 
			
		||||
							}
 | 
			
		||||
						}
 | 
			
		||||
					}
 | 
			
		||||
				}}
 | 
			
		||||
				width={320}
 | 
			
		||||
				height={320}
 | 
			
		||||
				class="w-full max-w-sm bg-neutral-300 rounded-lg overflow-hidden"
 | 
			
		||||
			/>
 | 
			
		||||
		{/if}
 | 
			
		||||
		{#if runnerinfo.id !== 0 && state !== "scan_card"}
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					state = "scan_card";
 | 
			
		||||
					scannerActive = true;
 | 
			
		||||
				}}
 | 
			
		||||
				type="button"
 | 
			
		||||
				class="py-3 px-4 inline-flex items-center gap-x-2 text-sm font-medium rounded-lg border border-transparent bg-blue-100 text-blue-800 hover:bg-blue-200 focus:outline-hidden focus:bg-blue-200 disabled:opacity-50 disabled:pointer-events-none dark:text-blue-500 dark:bg-blue-800/30 dark:hover:bg-blue-800/20 dark:focus:bg-blue-800/20 w-full mt-2"
 | 
			
		||||
			>
 | 
			
		||||
				{$_("scan_card")}
 | 
			
		||||
			</button>
 | 
			
		||||
		{/if}
 | 
			
		||||
		{#if state === "scan_card" || runnerinfo.id !== 0}
 | 
			
		||||
			<button
 | 
			
		||||
				on:click={() => {
 | 
			
		||||
					state = "scan_runner";
 | 
			
		||||
					scannerActive = true;
 | 
			
		||||
					runnerinfo = { id: 0, firstname: "", lastname: "" };
 | 
			
		||||
					cardCode = "";
 | 
			
		||||
				}}
 | 
			
		||||
				type="button"
 | 
			
		||||
				class="py-3 px-4 inline-flex items-center gap-x-2 text-sm font-medium rounded-lg border border-transparent bg-red-100 text-red-800 hover:bg-red-200 focus:outline-hidden focus:bg-red-200 disabled:opacity-50 disabled:pointer-events-none dark:text-red-500 dark:bg-red-800/30 dark:hover:bg-red-800/20 dark:focus:bg-red-800/20 w-full mt-2"
 | 
			
		||||
			>
 | 
			
		||||
				{$_("cancel")}
 | 
			
		||||
			</button>
 | 
			
		||||
		{/if}
 | 
			
		||||
		<!--  -->
 | 
			
		||||
	{/if}
 | 
			
		||||
</div>
 | 
			
		||||
@@ -20,23 +20,32 @@
 | 
			
		||||
      class="underline"
 | 
			
		||||
      href="https://odit.services"
 | 
			
		||||
      rel="noopener,noreferrer"
 | 
			
		||||
      target="_blank">ODIT.Services</a>
 | 
			
		||||
      target="_blank">ODIT.Services</a
 | 
			
		||||
    >
 | 
			
		||||
  </p>
 | 
			
		||||
  <p class="text-sm text-gray-500 mt-4">
 | 
			
		||||
    <a
 | 
			
		||||
      class="underline"
 | 
			
		||||
      target="_blank"
 | 
			
		||||
      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"
 | 
			
		||||
      target="_blank"
 | 
			
		||||
      rel="noopener, noreferrer"
 | 
			
		||||
      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>
 | 
			
		||||
      href="https://git.odit.services/lfk/frontend/src/tag/{releaseinfo}"
 | 
			
		||||
      >{releaseinfo}</a
 | 
			
		||||
    >
 | 
			
		||||
    -
 | 
			
		||||
    <a class="underline" href="/privacy">{$_('privacy')}</a>
 | 
			
		||||
    <a
 | 
			
		||||
      rel="noopener, noreferrer"
 | 
			
		||||
      class="underline"
 | 
			
		||||
      href="https://docs.lauf-fuer-kaya.de"
 | 
			
		||||
      target="_blank">{$_("documentation")}</a
 | 
			
		||||
    >
 | 
			
		||||
    -
 | 
			
		||||
    <a class="underline" href="/imprint">{$_('imprint')}</a>
 | 
			
		||||
    <a class="underline" href="/privacy">{$_("privacy")}</a>
 | 
			
		||||
    -
 | 
			
		||||
    <a class="underline" href="/imprint">{$_("imprint")}</a>
 | 
			
		||||
  </p>
 | 
			
		||||
</footer>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,20 +1,20 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _, getLocaleFromNavigator } from "svelte-i18n";
 | 
			
		||||
  import marked from "marked";
 | 
			
		||||
  import { parse } from "marked";
 | 
			
		||||
  import Footer from "./Footer.svelte";
 | 
			
		||||
  import * as css from "../base/simple.css";
 | 
			
		||||
  // import * as css from "../base/simple.css";
 | 
			
		||||
  let html = "";
 | 
			
		||||
  async function load() {
 | 
			
		||||
    let md = await fetch("/imprint_" + getLocaleFromNavigator() + ".md");
 | 
			
		||||
    let text = (await md.text()).toString();
 | 
			
		||||
    if(text.includes("<meta")){
 | 
			
		||||
      md.ok=false
 | 
			
		||||
    if (text.includes("<meta")) {
 | 
			
		||||
      md.ok = false;
 | 
			
		||||
    }
 | 
			
		||||
    if (!md.ok) {
 | 
			
		||||
      md = await fetch("/imprint_en.md");
 | 
			
		||||
      text = await md.text();
 | 
			
		||||
    }
 | 
			
		||||
    html = marked(text);
 | 
			
		||||
    html = parse(text);
 | 
			
		||||
  }
 | 
			
		||||
  const promise = load();
 | 
			
		||||
</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="text-center mb-8">
 | 
			
		||||
    <h1
 | 
			
		||||
      class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl">
 | 
			
		||||
      {$_('imprint')}
 | 
			
		||||
      class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl"
 | 
			
		||||
    >
 | 
			
		||||
      {$_("imprint")}
 | 
			
		||||
    </h1>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
@@ -31,16 +32,17 @@
 | 
			
		||||
<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">
 | 
			
		||||
    {#await promise}
 | 
			
		||||
      <p class="text-center w-full">{$_('imprint-loading')}</p>
 | 
			
		||||
      <p class="text-center w-full">{$_("imprint-loading")}</p>
 | 
			
		||||
    {:then}
 | 
			
		||||
      <div class="simplecontent">
 | 
			
		||||
        {@html html}
 | 
			
		||||
      </div>
 | 
			
		||||
    {:catch error}
 | 
			
		||||
      <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">
 | 
			
		||||
          <b class="capitalize">{$_('general_promise_error')}</b>
 | 
			
		||||
          <b class="capitalize">{$_("general_promise_error")}</b>
 | 
			
		||||
          {error}
 | 
			
		||||
        </span>
 | 
			
		||||
      </div>
 | 
			
		||||
 
 | 
			
		||||
@@ -4,19 +4,22 @@
 | 
			
		||||
 | 
			
		||||
<body class="antialiased font-sans">
 | 
			
		||||
  <div class="flex min-h-screen">
 | 
			
		||||
    <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="text-black text-5xl md:text-15xl font-black">
 | 
			
		||||
          {$_('404title')}
 | 
			
		||||
          {$_("404title")}
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="w-16 h-1 bg-purple-light my-3 md:my-6" />
 | 
			
		||||
        <p
 | 
			
		||||
          class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal">
 | 
			
		||||
          {$_('404message')}
 | 
			
		||||
          class="text-grey-darker text-2xl md:text-3xl font-light mb-8 leading-normal"
 | 
			
		||||
        >
 | 
			
		||||
          {$_("404message")}
 | 
			
		||||
        </p>
 | 
			
		||||
        <a
 | 
			
		||||
          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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,20 +1,20 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _, getLocaleFromNavigator } from "svelte-i18n";
 | 
			
		||||
  import marked from "marked";
 | 
			
		||||
  import { parse } from "marked";
 | 
			
		||||
  import Footer from "./Footer.svelte";
 | 
			
		||||
  import * as css from "../base/simple.css";
 | 
			
		||||
  // import * as css from "../base/simple.css?inline";
 | 
			
		||||
  let html = "";
 | 
			
		||||
  async function load() {
 | 
			
		||||
    let md = await fetch("/privacy_" + getLocaleFromNavigator() + ".md");
 | 
			
		||||
    let text = (await md.text()).toString();
 | 
			
		||||
    if(text.includes("<meta")){
 | 
			
		||||
      md.ok=false
 | 
			
		||||
    if (text.includes("<meta")) {
 | 
			
		||||
      md.ok = false;
 | 
			
		||||
    }
 | 
			
		||||
    if (!md.ok) {
 | 
			
		||||
      md = await fetch("/privacy_en.md");
 | 
			
		||||
      text = await md.text();
 | 
			
		||||
    }
 | 
			
		||||
    html = marked(text);
 | 
			
		||||
    html = parse(text);
 | 
			
		||||
  }
 | 
			
		||||
  const promise = load();
 | 
			
		||||
</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="text-center mb-8">
 | 
			
		||||
    <h1
 | 
			
		||||
      class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl">
 | 
			
		||||
      {$_('privacy')}
 | 
			
		||||
      class="mt-9 font-display text-4xl leading-none font-semibold text-white sm:text-5xl lg:text-6xl"
 | 
			
		||||
    >
 | 
			
		||||
      {$_("privacy")}
 | 
			
		||||
    </h1>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
@@ -31,16 +32,17 @@
 | 
			
		||||
<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">
 | 
			
		||||
    {#await promise}
 | 
			
		||||
      <p class="text-center w-full">{$_('privacy-loading')}</p>
 | 
			
		||||
      <p class="text-center w-full">{$_("privacy-loading")}</p>
 | 
			
		||||
    {:then}
 | 
			
		||||
      <div class="simplecontent">
 | 
			
		||||
        {@html html}
 | 
			
		||||
      </div>
 | 
			
		||||
    {:catch error}
 | 
			
		||||
      <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">
 | 
			
		||||
          <b class="capitalize">{$_('general_promise_error')}</b>
 | 
			
		||||
          <b class="capitalize">{$_("general_promise_error")}</b>
 | 
			
		||||
          {error}
 | 
			
		||||
        </span>
 | 
			
		||||
      </div>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										91
									
								
								src/components/general/QrCodeScanner.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								src/components/general/QrCodeScanner.svelte
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,91 @@
 | 
			
		||||
<script>
 | 
			
		||||
	import { onMount, createEventDispatcher } from "svelte";
 | 
			
		||||
	import {
 | 
			
		||||
		Html5QrcodeScanner,
 | 
			
		||||
		Html5QrcodeScanType,
 | 
			
		||||
		Html5QrcodeSupportedFormats,
 | 
			
		||||
		Html5QrcodeScannerState,
 | 
			
		||||
	} from "html5-qrcode";
 | 
			
		||||
 | 
			
		||||
	export let width;
 | 
			
		||||
	export let height;
 | 
			
		||||
	export let paused = false;
 | 
			
		||||
 | 
			
		||||
	const dispatch = createEventDispatcher();
 | 
			
		||||
 | 
			
		||||
	const debounceFunc = (func, delay) => {
 | 
			
		||||
		let timer;
 | 
			
		||||
		return function (...args) {
 | 
			
		||||
			const context = this;
 | 
			
		||||
			clearTimeout(timer);
 | 
			
		||||
			timer = setTimeout(() => {
 | 
			
		||||
				func.apply(context, args);
 | 
			
		||||
			}, delay);
 | 
			
		||||
		};
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const debouncedDispatch = debounceFunc(function (decodedText) {
 | 
			
		||||
		console.log("dispatchEvent");
 | 
			
		||||
		dispatch("detect", { decodedText });
 | 
			
		||||
	}, 500);
 | 
			
		||||
 | 
			
		||||
	function onScanSuccess(decodedText, decodedResult) {
 | 
			
		||||
		if (!paused) {
 | 
			
		||||
			debouncedDispatch(decodedText);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// usually better to ignore and keep scanning
 | 
			
		||||
	function onScanFailure(message) {
 | 
			
		||||
		if (!paused) {
 | 
			
		||||
			dispatch("error", { message });
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	let scanner;
 | 
			
		||||
	onMount(() => {
 | 
			
		||||
		scanner = new Html5QrcodeScanner(
 | 
			
		||||
			"qr-scanner",
 | 
			
		||||
			{
 | 
			
		||||
				fps: 10,
 | 
			
		||||
				showTorchButtonIfSupported: true,
 | 
			
		||||
				rememberLastUsedCamera: true,
 | 
			
		||||
				qrbox: { width, height },
 | 
			
		||||
				aspectRatio: 1,
 | 
			
		||||
				supportedScanTypes: [Html5QrcodeScanType.SCAN_TYPE_CAMERA],
 | 
			
		||||
				formatsToSupport: [
 | 
			
		||||
					Html5QrcodeSupportedFormats.CODE_39,
 | 
			
		||||
					Html5QrcodeSupportedFormats.EAN_8,
 | 
			
		||||
					Html5QrcodeSupportedFormats.EAN_13,
 | 
			
		||||
					Html5QrcodeSupportedFormats.QR_CODE,
 | 
			
		||||
					Html5QrcodeSupportedFormats.CODE_128,
 | 
			
		||||
				],
 | 
			
		||||
			},
 | 
			
		||||
			false // non-verbose
 | 
			
		||||
		);
 | 
			
		||||
		scanner.render(onScanSuccess, onScanFailure);
 | 
			
		||||
	});
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<div id="qr-scanner" class={$$props.class} />
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
	/* Hide unwanted icons */
 | 
			
		||||
	#qr-scanner :global(img[alt="Info icon"]),
 | 
			
		||||
	#qr-scanner :global(img[alt="Camera based scan"]) {
 | 
			
		||||
		display: none;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Change camera permission button text */
 | 
			
		||||
	#qr-scanner :global(#html5-qrcode-button-camera-permission) {
 | 
			
		||||
		visibility: hidden;
 | 
			
		||||
	}
 | 
			
		||||
	#qr-scanner :global(#html5-qrcode-button-camera-permission::after) {
 | 
			
		||||
		position: absolute;
 | 
			
		||||
		inset: auto 0 0;
 | 
			
		||||
		display: block;
 | 
			
		||||
		content: "Allow camera access";
 | 
			
		||||
		visibility: visible;
 | 
			
		||||
		padding: 10px 0;
 | 
			
		||||
	}
 | 
			
		||||
</style>
 | 
			
		||||
@@ -4,26 +4,33 @@
 | 
			
		||||
 | 
			
		||||
<div class="md:flex flex-col md:flex-row h-screen w-full">
 | 
			
		||||
  <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
 | 
			
		||||
      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
 | 
			
		||||
        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
 | 
			
		||||
        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">
 | 
			
		||||
          {#if open}
 | 
			
		||||
            <path
 | 
			
		||||
              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"
 | 
			
		||||
              clip-rule="evenodd" />
 | 
			
		||||
              clip-rule="evenodd"
 | 
			
		||||
            />
 | 
			
		||||
          {/if}
 | 
			
		||||
          {#if !open}
 | 
			
		||||
            <path
 | 
			
		||||
              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"
 | 
			
		||||
              clip-rule="evenodd" />
 | 
			
		||||
              clip-rule="evenodd"
 | 
			
		||||
            />
 | 
			
		||||
          {/if}
 | 
			
		||||
        </svg>
 | 
			
		||||
      </button>
 | 
			
		||||
@@ -31,49 +38,63 @@
 | 
			
		||||
    <nav
 | 
			
		||||
      :class:block={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
 | 
			
		||||
        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
 | 
			
		||||
        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
 | 
			
		||||
        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
 | 
			
		||||
        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">
 | 
			
		||||
        <button
 | 
			
		||||
          on:click={() => {
 | 
			
		||||
            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>
 | 
			
		||||
          <svg
 | 
			
		||||
            fill="currentColor"
 | 
			
		||||
            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"
 | 
			
		||||
              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>
 | 
			
		||||
        <div
 | 
			
		||||
          class:block={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
 | 
			
		||||
            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
 | 
			
		||||
              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
 | 
			
		||||
              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
 | 
			
		||||
              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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,8 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import { focusTrap } from "svelte-focus-trap";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { UserGroupService } from "@odit/lfk-client-js";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  export let current_groups;
 | 
			
		||||
  let description_input_value;
 | 
			
		||||
@@ -32,10 +31,7 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_('group-is-being-added'),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("group-is-being-added"));
 | 
			
		||||
      let postdata = {
 | 
			
		||||
        name: name_input_value,
 | 
			
		||||
        description: description_input_value,
 | 
			
		||||
@@ -46,11 +42,8 @@
 | 
			
		||||
          description_input_value = "";
 | 
			
		||||
          modal_open = false;
 | 
			
		||||
          //
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_('group-added'),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("group-added"));
 | 
			
		||||
          current_groups.push(result);
 | 
			
		||||
          current_groups = current_groups;
 | 
			
		||||
        })
 | 
			
		||||
@@ -59,8 +52,6 @@
 | 
			
		||||
        })
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          //
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -68,105 +59,124 @@
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-hidden"
 | 
			
		||||
    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">
 | 
			
		||||
      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" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"
 | 
			
		||||
            >
 | 
			
		||||
              <svg
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 640 512"
 | 
			
		||||
                class="h-6 w-6 text-blue-600"
 | 
			
		||||
                fill="currentColor"
 | 
			
		||||
                width="24"
 | 
			
		||||
                height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                height="24"
 | 
			
		||||
                ><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                <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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:text-left max-h-[75vh] overflow-y-auto">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-a-new-user-group')}
 | 
			
		||||
                {$_("create-a-new-user-group")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <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>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    use:focus
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('name')}"
 | 
			
		||||
                    placeholder={$_("name")}
 | 
			
		||||
                    class:border-red-500={!isNameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isNameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isNameValid}
 | 
			
		||||
                    bind:value={name_input_value}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('something-about-the-group')}"
 | 
			
		||||
                    placeholder={$_("something-about-the-group")}
 | 
			
		||||
                    bind:value={description_input_value}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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 class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
        <div class="bg-gray-50 px-4 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,220 +1,227 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import {
 | 
			
		||||
    UserGroupService
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  let data_loaded = false;
 | 
			
		||||
  export let params;
 | 
			
		||||
  const promise = UserGroupService.userGroupControllerGetOne(params.groupid);
 | 
			
		||||
  const colors = [
 | 
			
		||||
    "#f3558e",
 | 
			
		||||
    "#17b978",
 | 
			
		||||
    "#3498db",
 | 
			
		||||
    "#3f3b3b",
 | 
			
		||||
    "#775ada",
 | 
			
		||||
    "#7ed6df_#000000",
 | 
			
		||||
    "#000000",
 | 
			
		||||
    "#21e6c1_#000000",
 | 
			
		||||
    "#c0392b",
 | 
			
		||||
    "#d35400",
 | 
			
		||||
    "#7f8c8d",
 | 
			
		||||
    "#6ab04c",
 | 
			
		||||
    "#4834d4",
 | 
			
		||||
    "#ff1f5a",
 | 
			
		||||
    "#eac100",
 | 
			
		||||
  ];
 | 
			
		||||
  let matched_colors = [];
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: search_permission = "";
 | 
			
		||||
  $: original_data = {};
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: changes_performed = !(JSON.stringify(original_data) == JSON.stringify(editable));
 | 
			
		||||
  $: isGroupnameValid = editable.name !== "";
 | 
			
		||||
  $: save_enabled =
 | 
			
		||||
    changes_performed && isGroupnameValid 
 | 
			
		||||
  promise.then((data) => {
 | 
			
		||||
    let current_target = "";
 | 
			
		||||
    let colorindex = -1;
 | 
			
		||||
    data.permissions = data.permissions.sort();
 | 
			
		||||
    data.permissions.forEach((p) => {
 | 
			
		||||
      const target = p.split(":")[0];
 | 
			
		||||
      if (current_target !== p.split(":")[0]) {
 | 
			
		||||
        colorindex++;
 | 
			
		||||
        current_target = p.split(":")[0];
 | 
			
		||||
      }
 | 
			
		||||
      let background = colors[colorindex];
 | 
			
		||||
      let foreground = "#fff";
 | 
			
		||||
      if (background.includes("_")) {
 | 
			
		||||
        foreground = background.split("_")[1];
 | 
			
		||||
        background = background.split("_")[0];
 | 
			
		||||
      }
 | 
			
		||||
      matched_colors[target] = [background, foreground];
 | 
			
		||||
    });
 | 
			
		||||
    data_loaded = true;
 | 
			
		||||
    original_data = Object.assign(original_data, data);
 | 
			
		||||
    editable = Object.assign(editable, original_data);
 | 
			
		||||
  });
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_('updateing-group'),
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      UserGroupService.userGroupControllerPut(original_data.id, editable)
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Object.assign(original_data, editable);
 | 
			
		||||
          original_data = editable;
 | 
			
		||||
          Object.assign(original_data, editable);
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_('group-updated'),
 | 
			
		||||
            duration: 2500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  function deleteGroup() {
 | 
			
		||||
    UserGroupService.userGroupControllerRemove(original_data.id, true)
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {});
 | 
			
		||||
  }
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import { UserGroupService } from "@odit/lfk-client-js";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
 | 
			
		||||
	import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
	let data_loaded = false;
 | 
			
		||||
	export let params;
 | 
			
		||||
	const promise = UserGroupService.userGroupControllerGetOne(params.groupid);
 | 
			
		||||
	const colors = [
 | 
			
		||||
		"#f3558e",
 | 
			
		||||
		"#17b978",
 | 
			
		||||
		"#3498db",
 | 
			
		||||
		"#3f3b3b",
 | 
			
		||||
		"#775ada",
 | 
			
		||||
		"#7ed6df_#000000",
 | 
			
		||||
		"#000000",
 | 
			
		||||
		"#21e6c1_#000000",
 | 
			
		||||
		"#c0392b",
 | 
			
		||||
		"#d35400",
 | 
			
		||||
		"#7f8c8d",
 | 
			
		||||
		"#6ab04c",
 | 
			
		||||
		"#4834d4",
 | 
			
		||||
		"#ff1f5a",
 | 
			
		||||
		"#eac100",
 | 
			
		||||
	];
 | 
			
		||||
	let matched_colors = [];
 | 
			
		||||
	$: delete_triggered = false;
 | 
			
		||||
	$: search_permission = "";
 | 
			
		||||
	$: original_data = {};
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: changes_performed = !(
 | 
			
		||||
		JSON.stringify(original_data) == JSON.stringify(editable)
 | 
			
		||||
	);
 | 
			
		||||
	$: isGroupnameValid = editable.name !== "";
 | 
			
		||||
	$: save_enabled = changes_performed && isGroupnameValid;
 | 
			
		||||
	promise.then((data) => {
 | 
			
		||||
		let current_target = "";
 | 
			
		||||
		let colorindex = -1;
 | 
			
		||||
		data.permissions = data.permissions.sort();
 | 
			
		||||
		data.permissions.forEach((p) => {
 | 
			
		||||
			const target = p.split(":")[0];
 | 
			
		||||
			if (current_target !== p.split(":")[0]) {
 | 
			
		||||
				colorindex++;
 | 
			
		||||
				current_target = p.split(":")[0];
 | 
			
		||||
			}
 | 
			
		||||
			let background = colors[colorindex];
 | 
			
		||||
			let foreground = "#fff";
 | 
			
		||||
			if (background.includes("_")) {
 | 
			
		||||
				foreground = background.split("_")[1];
 | 
			
		||||
				background = background.split("_")[0];
 | 
			
		||||
			}
 | 
			
		||||
			matched_colors[target] = [background, foreground];
 | 
			
		||||
		});
 | 
			
		||||
		data_loaded = true;
 | 
			
		||||
		original_data = Object.assign(original_data, data);
 | 
			
		||||
		editable = Object.assign(editable, original_data);
 | 
			
		||||
	});
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast($_("updating-group"));
 | 
			
		||||
			UserGroupService.userGroupControllerPut(original_data.id, editable)
 | 
			
		||||
				.then((resp) => {
 | 
			
		||||
					Object.assign(original_data, editable);
 | 
			
		||||
					original_data = editable;
 | 
			
		||||
					Object.assign(original_data, editable);
 | 
			
		||||
					toast.success($_("group-updated"));
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		} else {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	function deleteGroup() {
 | 
			
		||||
		UserGroupService.userGroupControllerRemove(original_data.id, true)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#await promise}
 | 
			
		||||
  {$_('loading-group-detail')}
 | 
			
		||||
	{$_("loading-group-detail")}
 | 
			
		||||
{:then}
 | 
			
		||||
  <section class="container p-5 select-none">
 | 
			
		||||
    <div class="flex flex-row mb-4">
 | 
			
		||||
      <div class="w-full">
 | 
			
		||||
        <nav class="w-full flex">
 | 
			
		||||
          <ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <svg 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>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <a class="mr-2" href="../">{$_('groups')}</a><svg
 | 
			
		||||
                stroke="currentColor"
 | 
			
		||||
                fill="none"
 | 
			
		||||
                stroke-width="2"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                class="h-3 w-3 mr-2 stroke-current"
 | 
			
		||||
                height="1em"
 | 
			
		||||
                width="1em"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"><line
 | 
			
		||||
                  x1="5"
 | 
			
		||||
                  y1="12"
 | 
			
		||||
                  x2="19"
 | 
			
		||||
                  y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">{editable.name}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mb-8 text-3xl font-extrabold leading-tight">
 | 
			
		||||
      {original_data.name}
 | 
			
		||||
      <span data-id="group_actions_${editable.id}">
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:DELETE')}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              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>
 | 
			
		||||
            <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-group')}</button>
 | 
			
		||||
          {/if}
 | 
			
		||||
        {/if}
 | 
			
		||||
        {#if !delete_triggered}
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={!save_enabled}
 | 
			
		||||
            class:opacity-50={!save_enabled}
 | 
			
		||||
            type="button"
 | 
			
		||||
            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>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="title"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.name}
 | 
			
		||||
        class:border-red-500={!isGroupnameValid}
 | 
			
		||||
        class:focus:border-red-500={!isGroupnameValid}
 | 
			
		||||
        class:focus:ring-red-500={!isGroupnameValid}
 | 
			
		||||
        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" />
 | 
			
		||||
      {#if !isGroupnameValid}
 | 
			
		||||
        <span
 | 
			
		||||
          class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
          {$_('group-name-is-required')}
 | 
			
		||||
        </span>
 | 
			
		||||
      {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="firstname"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('description')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('description')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.description}
 | 
			
		||||
        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" />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full mt-8">
 | 
			
		||||
      <p class="font-medium mb-4">
 | 
			
		||||
        {$_('permissions')}
 | 
			
		||||
        <a
 | 
			
		||||
          class="px-4 py-2 bg-gray-500 rounded-md text-white"
 | 
			
		||||
          href="/groups/{params.groupid}/permissions/">{$_('edit-permissions')}</a>
 | 
			
		||||
      </p>
 | 
			
		||||
      <div class="w-full sm:my-px sm:px-px sm:w-1/2">
 | 
			
		||||
        <input
 | 
			
		||||
          autocomplete="off"
 | 
			
		||||
          placeholder="{$_('search-for-permission')}"
 | 
			
		||||
          type="text"
 | 
			
		||||
          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" />
 | 
			
		||||
      </div>
 | 
			
		||||
      {#each original_data.permissions as p}
 | 
			
		||||
        {#if p.toLowerCase().includes(search_permission.toLowerCase())}
 | 
			
		||||
          <span
 | 
			
		||||
            style="background:{matched_colors[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}
 | 
			
		||||
      {/each}
 | 
			
		||||
    </div>
 | 
			
		||||
  </section>
 | 
			
		||||
	<section class="container p-5 select-none">
 | 
			
		||||
		<div class="flex flex-row mb-4">
 | 
			
		||||
			<div class="w-full">
 | 
			
		||||
				<nav class="w-full flex">
 | 
			
		||||
					<ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
						<li class="flex items-center"></li>
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="../"
 | 
			
		||||
								><svg
 | 
			
		||||
									xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
									width="24"
 | 
			
		||||
									height="24"
 | 
			
		||||
									viewBox="0 0 24 24"
 | 
			
		||||
									fill="none"
 | 
			
		||||
									stroke="currentColor"
 | 
			
		||||
									stroke-width="2"
 | 
			
		||||
									stroke-linecap="round"
 | 
			
		||||
									stroke-linejoin="round"
 | 
			
		||||
									class="inline-block"
 | 
			
		||||
									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg
 | 
			
		||||
								>
 | 
			
		||||
								{$_("groups")}</a
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
					</ol>
 | 
			
		||||
				</nav>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mb-4 text-3xl font-extrabold leading-tight">
 | 
			
		||||
			{editable.name}
 | 
			
		||||
			<div data-id="group_actions_${editable.id}">
 | 
			
		||||
				{#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:DELETE")}
 | 
			
		||||
					{#if delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							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:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("confirm-deletion")}</button
 | 
			
		||||
						>
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = !delete_triggered;
 | 
			
		||||
							}}
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-400 text-base font-medium text-white sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("cancel")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
					{#if !delete_triggered}
 | 
			
		||||
						<button
 | 
			
		||||
							on:click={() => {
 | 
			
		||||
								delete_triggered = true;
 | 
			
		||||
							}}
 | 
			
		||||
							type="button"
 | 
			
		||||
							class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:w-auto sm:text-sm"
 | 
			
		||||
							>{$_("delete-group")}</button
 | 
			
		||||
						>
 | 
			
		||||
					{/if}
 | 
			
		||||
				{/if}
 | 
			
		||||
				{#if !delete_triggered}
 | 
			
		||||
					<button
 | 
			
		||||
						disabled={!save_enabled}
 | 
			
		||||
						class:opacity-50={!save_enabled}
 | 
			
		||||
						type="button"
 | 
			
		||||
						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:w-auto sm:text-sm mb-1 lg:mb-0"
 | 
			
		||||
						>{$_("save-changes")}</button
 | 
			
		||||
					>
 | 
			
		||||
				{/if}
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<!--  -->
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="title" class="font-semibold text-gray-700">{$_("name")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.name}
 | 
			
		||||
				class:border-red-500={!isGroupnameValid}
 | 
			
		||||
				class:focus:border-red-500={!isGroupnameValid}
 | 
			
		||||
				class:focus:ring-red-500={!isGroupnameValid}
 | 
			
		||||
				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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
			{#if !isGroupnameValid}
 | 
			
		||||
				<span
 | 
			
		||||
					class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
				>
 | 
			
		||||
					{$_("group-name-is-required")}
 | 
			
		||||
				</span>
 | 
			
		||||
			{/if}
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="groupdescription" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("description")}</label
 | 
			
		||||
			>
 | 
			
		||||
			<input
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("description")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.description}
 | 
			
		||||
				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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<p class="font-semibold mb-4">
 | 
			
		||||
				{$_("permissions")}
 | 
			
		||||
			</p>
 | 
			
		||||
			<div>
 | 
			
		||||
				<a
 | 
			
		||||
					class="px-4 py-2 bg-gray-500 rounded-md text-white"
 | 
			
		||||
					href="/groups/{params.groupid}/permissions/"
 | 
			
		||||
					>{$_("edit-permissions")}</a
 | 
			
		||||
				>
 | 
			
		||||
			</div>
 | 
			
		||||
			<div class="w-full sm:my-px sm:px-px sm:w-1/2">
 | 
			
		||||
				<input
 | 
			
		||||
					autocomplete="off"
 | 
			
		||||
					placeholder={$_("search-for-permission")}
 | 
			
		||||
					type="text"
 | 
			
		||||
					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-neutral-800 rounded-md p-2"
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
			{#each original_data.permissions as p}
 | 
			
		||||
				{#if p.toLowerCase().includes(search_permission.toLowerCase())}
 | 
			
		||||
					<span
 | 
			
		||||
						style="background:{matched_colors[
 | 
			
		||||
							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}
 | 
			
		||||
			{/each}
 | 
			
		||||
		</div>
 | 
			
		||||
	</section>
 | 
			
		||||
{:catch error}
 | 
			
		||||
  <PromiseError {error} />
 | 
			
		||||
	<PromiseError {error} />
 | 
			
		||||
{/await}
 | 
			
		||||
 
 | 
			
		||||
@@ -3,9 +3,10 @@
 | 
			
		||||
  import {
 | 
			
		||||
    PermissionService,
 | 
			
		||||
    CreatePermission,
 | 
			
		||||
UserGroupService,
 | 
			
		||||
    UserGroupService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import toast from 'svelte-french-toast'
 | 
			
		||||
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  export let params;
 | 
			
		||||
  let [
 | 
			
		||||
@@ -20,15 +21,14 @@ UserGroupService,
 | 
			
		||||
  $: save_enabled =
 | 
			
		||||
    JSON.stringify(grantedPermissions) ===
 | 
			
		||||
    JSON.stringify(grantedPermissions_initial);
 | 
			
		||||
  const group_promise = UserGroupService.userGroupControllerGetOne(params.groupid);
 | 
			
		||||
  const group_promise = UserGroupService.userGroupControllerGetOne(
 | 
			
		||||
    params.groupid
 | 
			
		||||
  );
 | 
			
		||||
  group_promise.then((data) => {
 | 
			
		||||
    original_data = Object.assign(original_data, data);
 | 
			
		||||
  });
 | 
			
		||||
  function submit() {
 | 
			
		||||
    Toastify({
 | 
			
		||||
      text: $_('updating-permissions'),
 | 
			
		||||
      duration: 2500,
 | 
			
		||||
    }).showToast();
 | 
			
		||||
    toast.loading($_("updating-permissions"));
 | 
			
		||||
    to_delete.forEach((d) => {
 | 
			
		||||
      promises = promises.concat([
 | 
			
		||||
        PermissionService.permissionControllerRemove(d, true),
 | 
			
		||||
@@ -50,11 +50,7 @@ UserGroupService,
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
      grantedPermissions_initial = grantedPermissions;
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("permissions-updated"),
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
        backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.success($_("permissions-updated"));
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  Object.values(CreatePermission.target).forEach((t) => {
 | 
			
		||||
@@ -62,13 +58,15 @@ UserGroupService,
 | 
			
		||||
      allpermissions = allpermissions.concat([{ target: t, action: a }]);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
  UserGroupService.userGroupControllerGetPermissions(params.groupid).then((val) => {
 | 
			
		||||
    val.directlyGranted.forEach((p) => {
 | 
			
		||||
      delete p.responseType;
 | 
			
		||||
      grantedPermissions = grantedPermissions.concat([p]);
 | 
			
		||||
    });
 | 
			
		||||
    grantedPermissions_initial = grantedPermissions;
 | 
			
		||||
  });
 | 
			
		||||
  UserGroupService.userGroupControllerGetPermissions(params.groupid).then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      val.directlyGranted.forEach((p) => {
 | 
			
		||||
        delete p.responseType;
 | 
			
		||||
        grantedPermissions = grantedPermissions.concat([p]);
 | 
			
		||||
      });
 | 
			
		||||
      grantedPermissions_initial = grantedPermissions;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#await group_promise}
 | 
			
		||||
@@ -86,12 +84,15 @@ UserGroupService,
 | 
			
		||||
                width="24"
 | 
			
		||||
                height="24"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 640 512"><path
 | 
			
		||||
                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>
 | 
			
		||||
                  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 class="flex items-center">
 | 
			
		||||
              <a class="mr-2" href="../../">{$_('user-groups')}</a><svg
 | 
			
		||||
              <a class="mr-2" href="../../">{$_("user-groups")}</a><svg
 | 
			
		||||
                stroke="currentColor"
 | 
			
		||||
                fill="none"
 | 
			
		||||
                stroke-width="2"
 | 
			
		||||
@@ -101,12 +102,10 @@ UserGroupService,
 | 
			
		||||
                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>
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                ><line x1="5" y1="12" x2="19" y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg
 | 
			
		||||
              >
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2"><a href="../">{original_data.name}</a></span>
 | 
			
		||||
@@ -122,45 +121,45 @@ UserGroupService,
 | 
			
		||||
                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>
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                ><line x1="5" y1="12" x2="19" y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg
 | 
			
		||||
              >
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">{$_('permissions')}</span>
 | 
			
		||||
              <span class="mr-2">{$_("permissions")}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="mb-8 text-3xl font-extrabold">
 | 
			
		||||
      {$_('permissions')}:
 | 
			
		||||
      {original_data.name}
 | 
			
		||||
      <span>
 | 
			
		||||
    <div class="mb-4 text-3xl font-extrabold">
 | 
			
		||||
      <div>
 | 
			
		||||
        {#if promises.length === 0}
 | 
			
		||||
          <button
 | 
			
		||||
            disabled={save_enabled}
 | 
			
		||||
            class:opacity-50={save_enabled}
 | 
			
		||||
            type="button"
 | 
			
		||||
            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}
 | 
			
		||||
          <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}
 | 
			
		||||
      </span>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
    <div class="flex flex-wrap -mx-1 overflow-hidden">
 | 
			
		||||
      <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2">
 | 
			
		||||
        {$_('verfuegbare')}
 | 
			
		||||
        {$_("available-permissions")}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2">
 | 
			
		||||
        {$_('granted')}
 | 
			
		||||
        {$_("granted")}
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <!--  -->
 | 
			
		||||
@@ -168,12 +167,14 @@ UserGroupService,
 | 
			
		||||
      {#if allpermissions.length > 0}
 | 
			
		||||
        <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2">
 | 
			
		||||
          <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}
 | 
			
		||||
              {#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
 | 
			
		||||
                  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">
 | 
			
		||||
                  {p.target + ':' + p.action}
 | 
			
		||||
                  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}
 | 
			
		||||
                  <button
 | 
			
		||||
                    on:click={() => {
 | 
			
		||||
                      grantedPermissions = grantedPermissions.concat([p]);
 | 
			
		||||
@@ -190,7 +191,9 @@ UserGroupService,
 | 
			
		||||
                      }
 | 
			
		||||
                    }}
 | 
			
		||||
                    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>
 | 
			
		||||
              {/if}
 | 
			
		||||
            {/each}
 | 
			
		||||
@@ -198,22 +201,39 @@ UserGroupService,
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2">
 | 
			
		||||
          <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}
 | 
			
		||||
              <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">
 | 
			
		||||
                {p.target + ':' + p.action}
 | 
			
		||||
                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}
 | 
			
		||||
                <button
 | 
			
		||||
                  on:click={() => {
 | 
			
		||||
                    grantedPermissions = grantedPermissions.filter((o) => 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);
 | 
			
		||||
                    grantedPermissions = grantedPermissions.filter(
 | 
			
		||||
                      (o) =>
 | 
			
		||||
                        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 {
 | 
			
		||||
                      to_delete = to_delete.concat([p.id]);
 | 
			
		||||
                    }
 | 
			
		||||
                  }}
 | 
			
		||||
                  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>
 | 
			
		||||
            {/each}
 | 
			
		||||
          </div>
 | 
			
		||||
 
 | 
			
		||||
@@ -8,22 +8,23 @@
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('user-groups')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('add-user-group')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
  </span>
 | 
			
		||||
  <h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_("user-groups")}
 | 
			
		||||
  </h4>
 | 
			
		||||
  {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP: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"
 | 
			
		||||
    >
 | 
			
		||||
      {$_("add-user-group")}
 | 
			
		||||
    </button>
 | 
			
		||||
  {/if}
 | 
			
		||||
  <UserGroupsOverview bind:current_groups />
 | 
			
		||||
</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 />
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="w-full h-44" src={groups_empty} alt="" />
 | 
			
		||||
    <span class="font-bold">{$_('there-are-no-groups-yet')}.</span><br />
 | 
			
		||||
    <span>{$_('add-your-first-group')}</span>
 | 
			
		||||
    <span class="font-bold">{$_("there-are-no-groups-yet")}.</span><br />
 | 
			
		||||
    <span>{$_("add-your-first-group")}</span>
 | 
			
		||||
  </p>
 | 
			
		||||
</div>
 | 
			
		||||
@@ -13,13 +13,14 @@
 | 
			
		||||
  );
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:GET')}
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:GET")}
 | 
			
		||||
  {#await groups_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">{$_('groups-are-being-loaded')}</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
      role="alert"
 | 
			
		||||
    >
 | 
			
		||||
      <p class="font-bold">{$_("groups-are-being-loaded")}</p>
 | 
			
		||||
      <p class="text-sm">{$_("this-might-take-a-moment")}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_groups.length === 0}
 | 
			
		||||
@@ -28,26 +29,30 @@
 | 
			
		||||
      <input
 | 
			
		||||
        type="search"
 | 
			
		||||
        bind:value={searchvalue}
 | 
			
		||||
        placeholder={$_('datatable.search')}
 | 
			
		||||
        aria-label={$_('datatable.search')}
 | 
			
		||||
        class="gridjs-input gridjs-search-input mb-4" />
 | 
			
		||||
        placeholder={$_("datatable.search")}
 | 
			
		||||
        aria-label={$_("datatable.search")}
 | 
			
		||||
        class="mb-2 w-full sm:w-auto mt-1 sm:mt-0 p-2 rounded-md border"
 | 
			
		||||
      />
 | 
			
		||||
      <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">
 | 
			
		||||
            <tr>
 | 
			
		||||
            <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">
 | 
			
		||||
                {$_('name')}
 | 
			
		||||
                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">
 | 
			
		||||
                {$_('description')}
 | 
			
		||||
                class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"
 | 
			
		||||
              >
 | 
			
		||||
                {$_("description")}
 | 
			
		||||
              </th>
 | 
			
		||||
              <th scope="col" class="relative px-6 py-3">
 | 
			
		||||
                <span class="sr-only">{$_('action')}</span>
 | 
			
		||||
                <span class="sr-only">{$_("action")}</span>
 | 
			
		||||
              </th>
 | 
			
		||||
            </tr>
 | 
			
		||||
          </thead>
 | 
			
		||||
@@ -57,7 +62,10 @@
 | 
			
		||||
                .toString()
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .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">
 | 
			
		||||
                    <div class="flex items-center">
 | 
			
		||||
                      <div class="ml-4">
 | 
			
		||||
@@ -72,39 +80,53 @@
 | 
			
		||||
                  </td>
 | 
			
		||||
                  {#if active_deletes[group.id] === true}
 | 
			
		||||
                    <td
 | 
			
		||||
                      class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
 | 
			
		||||
                      class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"
 | 
			
		||||
                    >
 | 
			
		||||
                      <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          active_deletes[group.id] = false;
 | 
			
		||||
                        }}
 | 
			
		||||
                        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
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                          UserGroupService.userGroupControllerRemove(group.id, true)
 | 
			
		||||
                          UserGroupService.userGroupControllerRemove(
 | 
			
		||||
                            group.id,
 | 
			
		||||
                            true
 | 
			
		||||
                          )
 | 
			
		||||
                            .then((resp) => {
 | 
			
		||||
                              current_groups = current_groups.filter((obj) => obj.id !== group.id);
 | 
			
		||||
                              current_groups = current_groups.filter(
 | 
			
		||||
                                (obj) => obj.id !== group.id
 | 
			
		||||
                              );
 | 
			
		||||
                            })
 | 
			
		||||
                            .catch((err) => {
 | 
			
		||||
                              // error deleting user
 | 
			
		||||
                            });
 | 
			
		||||
                        }}
 | 
			
		||||
                        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>
 | 
			
		||||
                  {:else}
 | 
			
		||||
                    <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
 | 
			
		||||
                        href="./{group.id}"
 | 
			
		||||
                        class="text-indigo-600 hover:text-indigo-900">Details</a>
 | 
			
		||||
                      {#if store.state.jwtinfo.userdetails.permissions.includes('USERGROUP:DELETE')}
 | 
			
		||||
                        class="text-indigo-600 hover:text-indigo-900">Details</a
 | 
			
		||||
                      >
 | 
			
		||||
                      {#if store.state.jwtinfo.userdetails.permissions.includes("USERGROUP:DELETE")}
 | 
			
		||||
                        <button
 | 
			
		||||
                          on:click={() => {
 | 
			
		||||
                            active_deletes[group.id] = true;
 | 
			
		||||
                          }}
 | 
			
		||||
                          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}
 | 
			
		||||
                    </td>
 | 
			
		||||
                  {/if}
 | 
			
		||||
@@ -118,7 +140,7 @@
 | 
			
		||||
  {: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>
 | 
			
		||||
        <b class="capitalize">{$_("general_promise_error")}</b>
 | 
			
		||||
        {error}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,10 @@
 | 
			
		||||
<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 toast from "svelte-french-toast";
 | 
			
		||||
  export let modal_open;
 | 
			
		||||
  export let current_organizations;
 | 
			
		||||
  let name_input_dom;
 | 
			
		||||
@@ -24,7 +25,7 @@
 | 
			
		||||
  $: address_input2_value = "";
 | 
			
		||||
  $: address_zipcode_value = "";
 | 
			
		||||
  $: address_city_value = "";
 | 
			
		||||
  $: address_checked = true;
 | 
			
		||||
  $: address_checked = false;
 | 
			
		||||
 | 
			
		||||
  let address_input1;
 | 
			
		||||
  let address_input2;
 | 
			
		||||
@@ -48,10 +49,7 @@
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (processed_last_submit === true) {
 | 
			
		||||
      processed_last_submit = false;
 | 
			
		||||
      const toast = Toastify({
 | 
			
		||||
        text: $_("organization-is-being-added"),
 | 
			
		||||
        duration: -1,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      toast.loading($_("organization-is-being-added"));
 | 
			
		||||
      let address = {};
 | 
			
		||||
      if (address_checked === true) {
 | 
			
		||||
        address = {
 | 
			
		||||
@@ -70,17 +68,13 @@
 | 
			
		||||
        .then((result) => {
 | 
			
		||||
          name = "";
 | 
			
		||||
          modal_open = false;
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("organization-added"),
 | 
			
		||||
            duration: 500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("organization-added"));
 | 
			
		||||
          current_organizations = current_organizations.concat([result]);
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {})
 | 
			
		||||
        .finally(() => {
 | 
			
		||||
          processed_last_submit = true;
 | 
			
		||||
          toast.hideToast();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -88,58 +82,70 @@
 | 
			
		||||
 | 
			
		||||
{#if modal_open}
 | 
			
		||||
  <div
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-auto"
 | 
			
		||||
    use:focusTrap
 | 
			
		||||
    class="fixed z-10 inset-0 overflow-y-hidden"
 | 
			
		||||
    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">
 | 
			
		||||
      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" />
 | 
			
		||||
          data-id="modal_backdrop"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <span
 | 
			
		||||
        class="hidden sm:inline-block sm:align-middle sm:h-screen"
 | 
			
		||||
        aria-hidden="true">​</span>
 | 
			
		||||
        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"
 | 
			
		||||
        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">
 | 
			
		||||
          <div class="sm:flex sm:items-start">
 | 
			
		||||
        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">
 | 
			
		||||
              class="flex-shrink-0 flex items-center justify-center size-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
 | 
			
		||||
                  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>
 | 
			
		||||
                height="24"
 | 
			
		||||
                ><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 class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
            <div class="mt-3 sm:text-left">
 | 
			
		||||
              <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
                {$_('create-a-new-organization')}
 | 
			
		||||
                {$_("create-a-new-organization")}
 | 
			
		||||
              </h3>
 | 
			
		||||
              <div class="mt-2 mb-6">
 | 
			
		||||
              <div class="mb-6">
 | 
			
		||||
                <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>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="grid grid-cols-6 gap-6">
 | 
			
		||||
              <div class="grid grid-cols-6 gap-2 lg:gap-6 text-left">
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    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
 | 
			
		||||
                    use:focus
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder={$_('name')}
 | 
			
		||||
                    placeholder={$_("name")}
 | 
			
		||||
                    class:border-red-500={!isOrgnameValid}
 | 
			
		||||
                    class:focus:border-red-500={!isOrgnameValid}
 | 
			
		||||
                    class:focus:ring-red-500={!isOrgnameValid}
 | 
			
		||||
@@ -147,11 +153,13 @@
 | 
			
		||||
                    bind:this={name_input_dom}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    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}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('organization-name-is-required')}
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      {$_("organization-name-is-required")}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
@@ -162,115 +170,133 @@
 | 
			
		||||
                      id="comments"
 | 
			
		||||
                      name="comments"
 | 
			
		||||
                      type="checkbox"
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded" />
 | 
			
		||||
                      class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="ml-3 text-sm">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="comments"
 | 
			
		||||
                      class="font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
                    <label for="comments" class="font-semibold text-gray-700"
 | 
			
		||||
                      >{$_("address")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              {#if address_checked === true}
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="address1"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
                  <input
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('address')}"
 | 
			
		||||
                    class:border-red-500={!isAddress1Valid}
 | 
			
		||||
                    class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
                    class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
                    bind:value={address_input1_value}
 | 
			
		||||
                    bind:this={address_input1}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    name="address1"
 | 
			
		||||
                    class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                  {#if !isAddress1Valid}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('address-is-required')}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="address2"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
                  <input
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('apartment-suite-etc')}"
 | 
			
		||||
                    bind:value={address_input2_value}
 | 
			
		||||
                    bind:this={address_input2}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    name="address2"
 | 
			
		||||
                    class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="zipcode"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label>
 | 
			
		||||
                  <input
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('zip-postal-code')}"
 | 
			
		||||
                    class:border-red-500={!iszipcodevalid}
 | 
			
		||||
                    class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
                    class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
                    bind:value={address_zipcode_value}
 | 
			
		||||
                    bind:this={address_zipcode}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    name="zipcode"
 | 
			
		||||
                    class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                  {#if !iszipcodevalid}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="col-span-6">
 | 
			
		||||
                  <label
 | 
			
		||||
                    for="city"
 | 
			
		||||
                    class="block text-sm font-medium text-gray-700">{$_('city')}</label>
 | 
			
		||||
                  <input
 | 
			
		||||
                    autocomplete="off"
 | 
			
		||||
                    placeholder="{$_('city')}"
 | 
			
		||||
                    class:border-red-500={!iscityvalid}
 | 
			
		||||
                    class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
                    class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
                    bind:value={address_city_value}
 | 
			
		||||
                    bind:this={address_city}
 | 
			
		||||
                    type="text"
 | 
			
		||||
                    name="city"
 | 
			
		||||
                    class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
                  {#if !iscityvalid}
 | 
			
		||||
                    <span
 | 
			
		||||
                      class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                      {$_('valid-city-is-required')}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  {/if}
 | 
			
		||||
                </div>
 | 
			
		||||
              {/if}
 | 
			
		||||
            </div>
 | 
			
		||||
                {#if address_checked === true}
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="address1"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("address")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_("address")}
 | 
			
		||||
                      class:border-red-500={!isAddress1Valid}
 | 
			
		||||
                      class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
                      class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
                      bind:value={address_input1_value}
 | 
			
		||||
                      bind:this={address_input1}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address1"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
                    />
 | 
			
		||||
                    {#if !isAddress1Valid}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("address-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-6">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="address2"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("apartment-suite-etc")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
                      bind:value={address_input2_value}
 | 
			
		||||
                      bind:this={address_input2}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="address2"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-2">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="zipcode"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("zip-postal-code")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_("zip-postal-code")}
 | 
			
		||||
                      class:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
                      class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
                      bind:value={address_zipcode_value}
 | 
			
		||||
                      bind:this={address_zipcode}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="zipcode"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
                    />
 | 
			
		||||
                    {#if !iszipcodevalid}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="col-span-4">
 | 
			
		||||
                    <label
 | 
			
		||||
                      for="city"
 | 
			
		||||
                      class="block text-sm font-medium text-gray-700"
 | 
			
		||||
                      >{$_("city")}</label
 | 
			
		||||
                    >
 | 
			
		||||
                    <input
 | 
			
		||||
                      autocomplete="off"
 | 
			
		||||
                      placeholder={$_("city")}
 | 
			
		||||
                      class:border-red-500={!iscityvalid}
 | 
			
		||||
                      class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
                      class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
                      bind:value={address_city_value}
 | 
			
		||||
                      bind:this={address_city}
 | 
			
		||||
                      type="text"
 | 
			
		||||
                      name="city"
 | 
			
		||||
                      class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
                    />
 | 
			
		||||
                    {#if !iscityvalid}
 | 
			
		||||
                      <span
 | 
			
		||||
                        class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
                      >
 | 
			
		||||
                        {$_("valid-city-is-required")}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    {/if}
 | 
			
		||||
                  </div>
 | 
			
		||||
                {/if}
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
          <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')}
 | 
			
		||||
            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="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')}
 | 
			
		||||
            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>
 | 
			
		||||
 
 | 
			
		||||
@@ -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="flex-shrink-0 flex items-center justify-center size-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 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="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 lg:py-3 sm:px-6 grid gap-2 lg:rounded-b-xl pt-3 pb-10">
 | 
			
		||||
					<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}
 | 
			
		||||
@@ -1,473 +1,421 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import {
 | 
			
		||||
    GroupContactService,
 | 
			
		||||
    RunnerOrganizationService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import ConfirmOrgDeletion from "./ConfirmOrgDeletion.svelte";
 | 
			
		||||
  import ImportRunnerModal from "../runners/ImportRunnerModal.svelte";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import { tick } from "svelte";
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: address_valid_or_none =
 | 
			
		||||
    (isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
    editable.address_checked === false;
 | 
			
		||||
  $: save_enabled = data_changed && address_valid_or_none;
 | 
			
		||||
  let original = "";
 | 
			
		||||
  let original_object = {};
 | 
			
		||||
  let contacts = [];
 | 
			
		||||
  let valueCopy = null;
 | 
			
		||||
  let areaDom;
 | 
			
		||||
  let copied = false;
 | 
			
		||||
  export let params;
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: contact = {};
 | 
			
		||||
  $: data_loaded = false;
 | 
			
		||||
  $: data_changed = !(JSON.stringify(editable) === original);
 | 
			
		||||
  $: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
  $: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
  $: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
  $: sponsoring_contracts_show = true;
 | 
			
		||||
  $: cards_show = true;
 | 
			
		||||
  $: generate_orgs = [original_object];
 | 
			
		||||
  $: registrationLink = `${config.baseurl}/selfservice/register/${editable.registrationKey}`;
 | 
			
		||||
  const getContactLabel = (option) =>
 | 
			
		||||
    option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
  const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne(
 | 
			
		||||
    params.orgid
 | 
			
		||||
  ).then((value) => {
 | 
			
		||||
    data_loaded = true;
 | 
			
		||||
    value.address_checked = value.address.address1 !== null;
 | 
			
		||||
    if (value.address_checked === false) {
 | 
			
		||||
      value.address = {
 | 
			
		||||
        address1: "",
 | 
			
		||||
        address2: "",
 | 
			
		||||
        city: "",
 | 
			
		||||
        postalcode: "",
 | 
			
		||||
        country: "",
 | 
			
		||||
      };
 | 
			
		||||
    }
 | 
			
		||||
    editable = Object.assign(editable, value);
 | 
			
		||||
    editable = editable;
 | 
			
		||||
    original_object = Object.assign(editable, value);
 | 
			
		||||
    original = JSON.stringify(value);
 | 
			
		||||
    GroupContactService.groupContactControllerGetAll().then((val) => {
 | 
			
		||||
      contacts = val.map((r) => {
 | 
			
		||||
        return { label: getContactLabel(r), value: r };
 | 
			
		||||
      });
 | 
			
		||||
      if (editable.contact) {
 | 
			
		||||
        contact = contacts.find((g) => g.value.id == editable.contact.id);
 | 
			
		||||
      } else {
 | 
			
		||||
        contact = null;
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
  let modal_open = false;
 | 
			
		||||
  let delete_org = {};
 | 
			
		||||
  function deleteOrganization() {
 | 
			
		||||
    RunnerOrganizationService.runnerOrganizationControllerRemove(
 | 
			
		||||
      original_object.id,
 | 
			
		||||
      false
 | 
			
		||||
    )
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        Toastify({
 | 
			
		||||
          text: $_("organization-deleted"),
 | 
			
		||||
          duration: 500,
 | 
			
		||||
          backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {
 | 
			
		||||
        modal_open = true;
 | 
			
		||||
        delete_org = original_object;
 | 
			
		||||
      });
 | 
			
		||||
  }
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("updating-organization"),
 | 
			
		||||
        duration: 2500,
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      let postdata = Object.assign({}, editable);
 | 
			
		||||
      if (postdata.address_checked === false) {
 | 
			
		||||
        postdata.address = null;
 | 
			
		||||
      }
 | 
			
		||||
      postdata.contact = postdata.contact?.id;
 | 
			
		||||
      RunnerOrganizationService.runnerOrganizationControllerPut(
 | 
			
		||||
        original_object.id,
 | 
			
		||||
        postdata
 | 
			
		||||
      )
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          editable.registrationKey = resp.registrationKey;
 | 
			
		||||
          original_object = Object.assign({}, editable);
 | 
			
		||||
          original = JSON.stringify(original_object);
 | 
			
		||||
          Toastify({
 | 
			
		||||
            text: $_("updated-organization"),
 | 
			
		||||
            duration: 2500,
 | 
			
		||||
            backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
          }).showToast();
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  async function copy() {
 | 
			
		||||
    if(!editable.registrationKey){
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_('you-have-to-save-your-changes-to-generate-a-link'),
 | 
			
		||||
        duration: 500,
 | 
			
		||||
        backgroundColor:
 | 
			
		||||
          "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    valueCopy = registrationLink;
 | 
			
		||||
    await tick();
 | 
			
		||||
    areaDom.focus();
 | 
			
		||||
    areaDom.select();
 | 
			
		||||
    try {
 | 
			
		||||
      const successful = document.execCommand("copy");
 | 
			
		||||
      if (!successful) {
 | 
			
		||||
        throw new Error();
 | 
			
		||||
      }
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("copied-link-to-clipboard"),
 | 
			
		||||
        duration: 500,
 | 
			
		||||
        backgroundColor: "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
      }).showToast();
 | 
			
		||||
      copied = true;
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      Toastify({
 | 
			
		||||
        text: $_("error-whyile-copying-to-clipboard"),
 | 
			
		||||
        duration: 500,
 | 
			
		||||
        backgroundColor:
 | 
			
		||||
          "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
      }).showToast();
 | 
			
		||||
    }
 | 
			
		||||
    // we can notifi by event or storage about copy status
 | 
			
		||||
    valueCopy = null;
 | 
			
		||||
  }
 | 
			
		||||
  export let import_modal_open = false;
 | 
			
		||||
	import {
 | 
			
		||||
		GroupContactService,
 | 
			
		||||
		RunnerOrganizationService,
 | 
			
		||||
	} from "@odit/lfk-client-js";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { tick } from "svelte";
 | 
			
		||||
	import Select from "svelte-select";
 | 
			
		||||
	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 =
 | 
			
		||||
		(isAddress1Valid && iszipcodevalid && iscityvalid) ||
 | 
			
		||||
		editable.address_checked === false;
 | 
			
		||||
	$: save_enabled = data_changed && address_valid_or_none;
 | 
			
		||||
	let original = "";
 | 
			
		||||
	let original_object = {};
 | 
			
		||||
	let contacts = [];
 | 
			
		||||
	let valueCopy = null;
 | 
			
		||||
	let areaDom;
 | 
			
		||||
	export let params;
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: contact = {};
 | 
			
		||||
	$: data_loaded = false;
 | 
			
		||||
	$: data_changed = !(JSON.stringify(editable) === original);
 | 
			
		||||
	$: isAddress1Valid = editable.address?.address1?.trim().length !== 0;
 | 
			
		||||
	$: iszipcodevalid = editable.address?.postalcode?.trim().length !== 0;
 | 
			
		||||
	$: iscityvalid = editable.address?.city?.trim().length !== 0;
 | 
			
		||||
	$: sponsoring_contracts_show = true;
 | 
			
		||||
	$: cards_show = true;
 | 
			
		||||
	$: certificates_show = true;
 | 
			
		||||
	$: generate_orgs = [original_object];
 | 
			
		||||
	$: registrationLink = `${config.baseurl_selfservice}/register/${editable.registrationKey}`;
 | 
			
		||||
	const getContactLabel = (option) =>
 | 
			
		||||
		option.firstname + " " + (option.middlename || "") + " " + option.lastname;
 | 
			
		||||
	const promise = RunnerOrganizationService.runnerOrganizationControllerGetOne(
 | 
			
		||||
		params.orgid
 | 
			
		||||
	).then((value) => {
 | 
			
		||||
		data_loaded = true;
 | 
			
		||||
		value.address_checked = value.address.address1 !== null;
 | 
			
		||||
		if (value.address_checked === false) {
 | 
			
		||||
			value.address = {
 | 
			
		||||
				address1: "",
 | 
			
		||||
				address2: "",
 | 
			
		||||
				city: "",
 | 
			
		||||
				postalcode: "",
 | 
			
		||||
				country: "",
 | 
			
		||||
			};
 | 
			
		||||
		}
 | 
			
		||||
		editable = Object.assign(editable, value);
 | 
			
		||||
		editable = editable;
 | 
			
		||||
		original_object = Object.assign(editable, value);
 | 
			
		||||
		original = JSON.stringify(value);
 | 
			
		||||
		GroupContactService.groupContactControllerGetAll().then((val) => {
 | 
			
		||||
			contacts = val.map((r) => {
 | 
			
		||||
				return { label: getContactLabel(r), value: r };
 | 
			
		||||
			});
 | 
			
		||||
			if (editable.contact) {
 | 
			
		||||
				contact = contacts.find((g) => g.value.id == editable.contact.id);
 | 
			
		||||
			} else {
 | 
			
		||||
				contact = null;
 | 
			
		||||
			}
 | 
			
		||||
		});
 | 
			
		||||
	});
 | 
			
		||||
	let modal_open = false;
 | 
			
		||||
	let delete_org = {};
 | 
			
		||||
	function deleteOrganization() {
 | 
			
		||||
		RunnerOrganizationService.runnerOrganizationControllerRemove(
 | 
			
		||||
			original_object.id,
 | 
			
		||||
			false
 | 
			
		||||
		)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				toast.success($_("organization-deleted"));
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {});
 | 
			
		||||
	}
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast($_("updating-organization"));
 | 
			
		||||
			let postdata = Object.assign({}, editable);
 | 
			
		||||
			if (postdata.address_checked === false) {
 | 
			
		||||
				postdata.address = null;
 | 
			
		||||
			}
 | 
			
		||||
			postdata.contact = postdata.contact?.id;
 | 
			
		||||
			RunnerOrganizationService.runnerOrganizationControllerPut(
 | 
			
		||||
				original_object.id,
 | 
			
		||||
				postdata
 | 
			
		||||
			)
 | 
			
		||||
				.then((resp) => {
 | 
			
		||||
					editable.registrationKey = resp.registrationKey;
 | 
			
		||||
					original_object = Object.assign({}, editable);
 | 
			
		||||
					original = JSON.stringify(original_object);
 | 
			
		||||
					toast.success($_("updated-organization"));
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		} 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;
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if valueCopy != null}<textarea bind:this={areaDom}>{valueCopy}</textarea>{/if}
 | 
			
		||||
<ImportRunnerModal
 | 
			
		||||
  on:cancelDelete={(event) => {
 | 
			
		||||
    import_modal_open = false;
 | 
			
		||||
  }}
 | 
			
		||||
  current_runners={[]}
 | 
			
		||||
  passed_team={{}}
 | 
			
		||||
  passed_orgs={[]}
 | 
			
		||||
  passed_org={editable}
 | 
			
		||||
  opened_from="OrgDetail"
 | 
			
		||||
  bind:import_modal_open />
 | 
			
		||||
<ConfirmOrgDeletion bind:modal_open bind:delete_org />
 | 
			
		||||
	on:cancelDelete={(event) => {
 | 
			
		||||
		import_modal_open = false;
 | 
			
		||||
	}}
 | 
			
		||||
	current_runners={[]}
 | 
			
		||||
	passed_team={{}}
 | 
			
		||||
	passed_orgs={[]}
 | 
			
		||||
	passed_org={editable}
 | 
			
		||||
	opened_from="OrgDetail"
 | 
			
		||||
	bind:import_modal_open
 | 
			
		||||
/>
 | 
			
		||||
<ConfirmOrgDeletionModal bind:modal_open bind:delete_org />
 | 
			
		||||
{#if data_loaded}
 | 
			
		||||
  <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}">
 | 
			
		||||
        <GenerateSponsoringContracts
 | 
			
		||||
          bind:sponsoring_contracts_show
 | 
			
		||||
          bind:generate_orgs />
 | 
			
		||||
        <GenerateRunnerCards bind:cards_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: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="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="mr-2 flex items-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
 | 
			
		||||
                  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"
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                class="h-3 w-3 mr-2 stroke-current"
 | 
			
		||||
                height="1em"
 | 
			
		||||
                width="1em"
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"><line
 | 
			
		||||
                  x1="5"
 | 
			
		||||
                  y1="12"
 | 
			
		||||
                  x2="19"
 | 
			
		||||
                  y2="12" />
 | 
			
		||||
                <polyline points="12 5 19 12 12 19" /></svg>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <span class="mr-2">Org-Details #{params.orgid}</span>
 | 
			
		||||
            </li>
 | 
			
		||||
          </ol>
 | 
			
		||||
        </nav>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label for="name" class="font-medium text-gray-700">{$_('name')}</label>
 | 
			
		||||
      <input
 | 
			
		||||
        autocomplete="off"
 | 
			
		||||
        placeholder={$_('name')}
 | 
			
		||||
        type="text"
 | 
			
		||||
        bind:value={editable.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" />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <label
 | 
			
		||||
        for="contact"
 | 
			
		||||
        class="font-medium text-gray-700">{$_('contact')}</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) => label
 | 
			
		||||
            .toLowerCase()
 | 
			
		||||
            .includes(
 | 
			
		||||
              filterText.toLowerCase()
 | 
			
		||||
            ) || option.value.id
 | 
			
		||||
            .toString()
 | 
			
		||||
            .startsWith(filterText.toLowerCase())}
 | 
			
		||||
        items={contacts}
 | 
			
		||||
        showChevron={true}
 | 
			
		||||
        placeholder={$_('no-contact-selected')}
 | 
			
		||||
        noOptionsMessage={$_('no-contact-found')}
 | 
			
		||||
        bind:selectedValue={contact}
 | 
			
		||||
        on:select={(selectedValue) => (editable.contact = selectedValue.detail.value)}
 | 
			
		||||
        on:clear={() => (editable.contact = null)} />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div>
 | 
			
		||||
      <div class="flex items-start mt-2">
 | 
			
		||||
        <div class="flex items-center h-5">
 | 
			
		||||
          <input
 | 
			
		||||
            bind:checked={editable.registrationEnabled}
 | 
			
		||||
            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">{$_('selfservice-registration')}</label>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div>
 | 
			
		||||
        {#if editable.registrationEnabled}
 | 
			
		||||
          <div class="text-sm w-full">
 | 
			
		||||
            <div on:click={copy} class="inline-flex w-full">
 | 
			
		||||
              <p
 | 
			
		||||
                name="token"
 | 
			
		||||
                class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 p-2">
 | 
			
		||||
                {#if editable.registrationKey}
 | 
			
		||||
                {registrationLink}
 | 
			
		||||
                {:else}
 | 
			
		||||
                {$_('you-have-to-save-your-changes-to-generate-a-link')}
 | 
			
		||||
                {/if}
 | 
			
		||||
              </p>
 | 
			
		||||
              <div
 | 
			
		||||
                class="bg-gray-200 border-gray-300 border-t border-b border-r text-black rounded-r-md sm:text-sm p-2 mt-1 cursor-pointer">
 | 
			
		||||
                <svg
 | 
			
		||||
                  xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                  viewBox="0 0 24 24"
 | 
			
		||||
                  width="24"
 | 
			
		||||
                  height="24"><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
                  <path
 | 
			
		||||
                    fill="currentColor"
 | 
			
		||||
                    d="M7 4V2h10v2h3l1 1v16a1 1 0 01-1 1H4a1 1 0 01-1-1V5l1-1h3zm0 2H5v14h14V6h-2v2H7V6zm2-2v2h6V4H9z" /></svg>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            {#if editable.registrationKey}
 | 
			
		||||
            <p class="text-gray-500 text-xs">
 | 
			
		||||
              {$_('click-to-copy-the-link-into-your-clipboard')}
 | 
			
		||||
            </p>
 | 
			
		||||
            {/if}
 | 
			
		||||
          </div>
 | 
			
		||||
        {/if}
 | 
			
		||||
        <!--  -->
 | 
			
		||||
        <div>
 | 
			
		||||
          <div class="flex items-start mt-2">
 | 
			
		||||
            <div class="flex items-center h-5">
 | 
			
		||||
              <input
 | 
			
		||||
                bind:checked={editable.address_checked}
 | 
			
		||||
                id="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">{$_('address')}</label>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        {#if editable.address_checked === true}
 | 
			
		||||
          <div class="col-span-6">
 | 
			
		||||
            <label
 | 
			
		||||
              for="address1"
 | 
			
		||||
              class="block text-sm font-medium text-gray-700">{$_('address')}</label>
 | 
			
		||||
            <input
 | 
			
		||||
              autocomplete="off"
 | 
			
		||||
              placeholder="Address"
 | 
			
		||||
              class:border-red-500={!isAddress1Valid}
 | 
			
		||||
              class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
              class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
              bind:value={editable.address.address1}
 | 
			
		||||
              type="text"
 | 
			
		||||
              name="address1"
 | 
			
		||||
              class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
            {#if !isAddress1Valid}
 | 
			
		||||
              <span
 | 
			
		||||
                class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                {$_('address-is-required')}
 | 
			
		||||
              </span>
 | 
			
		||||
            {/if}
 | 
			
		||||
          </div>
 | 
			
		||||
          <div class="col-span-6">
 | 
			
		||||
            <label
 | 
			
		||||
              for="address2"
 | 
			
		||||
              class="block text-sm font-medium text-gray-700">{$_('apartment-suite-etc')}</label>
 | 
			
		||||
            <input
 | 
			
		||||
              autocomplete="off"
 | 
			
		||||
              placeholder={$_('apartment-suite-etc')}
 | 
			
		||||
              bind:value={editable.address.address2}
 | 
			
		||||
              type="text"
 | 
			
		||||
              name="address2"
 | 
			
		||||
              class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
          </div>
 | 
			
		||||
          <div class="col-span-6">
 | 
			
		||||
            <label
 | 
			
		||||
              for="zipcode"
 | 
			
		||||
              class="block text-sm font-medium text-gray-700">{$_('zip-postal-code')}</label>
 | 
			
		||||
            <input
 | 
			
		||||
              autocomplete="off"
 | 
			
		||||
              placeholder={$_('zip-postal-code')}
 | 
			
		||||
              class:border-red-500={!iszipcodevalid}
 | 
			
		||||
              class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
              class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
              bind:value={editable.address.postalcode}
 | 
			
		||||
              type="text"
 | 
			
		||||
              name="zipcode"
 | 
			
		||||
              class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
            {#if !iszipcodevalid}
 | 
			
		||||
              <span
 | 
			
		||||
                class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                {$_('valid-zipcode-postal-code-is-required')}
 | 
			
		||||
              </span>
 | 
			
		||||
            {/if}
 | 
			
		||||
          </div>
 | 
			
		||||
          <div class="col-span-6">
 | 
			
		||||
            <label
 | 
			
		||||
              for="city"
 | 
			
		||||
              class="block text-sm font-medium text-gray-700">{$_('city')}</label>
 | 
			
		||||
            <input
 | 
			
		||||
              autocomplete="off"
 | 
			
		||||
              placeholder={$_('city')}
 | 
			
		||||
              class:border-red-500={!iscityvalid}
 | 
			
		||||
              class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
              class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
              bind:value={editable.address.city}
 | 
			
		||||
              type="text"
 | 
			
		||||
              name="city"
 | 
			
		||||
              class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-gray-500 rounded-md p-2" />
 | 
			
		||||
            {#if !iscityvalid}
 | 
			
		||||
              <span
 | 
			
		||||
                class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1">
 | 
			
		||||
                {$_('valid-city-is-required')}
 | 
			
		||||
              </span>
 | 
			
		||||
            {/if}
 | 
			
		||||
          </div>
 | 
			
		||||
        {/if}
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </section>
 | 
			
		||||
	<section class="container p-5">
 | 
			
		||||
		<div class="flex flex-row mb-4">
 | 
			
		||||
			<div class="w-full">
 | 
			
		||||
				<nav class="w-full flex">
 | 
			
		||||
					<ol class="list-none flex flex-row items-center justify-start">
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="./"
 | 
			
		||||
								><svg
 | 
			
		||||
									xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
									width="24"
 | 
			
		||||
									height="24"
 | 
			
		||||
									viewBox="0 0 24 24"
 | 
			
		||||
									fill="none"
 | 
			
		||||
									stroke="currentColor"
 | 
			
		||||
									stroke-width="2"
 | 
			
		||||
									stroke-linecap="round"
 | 
			
		||||
									stroke-linejoin="round"
 | 
			
		||||
									class="inline-block"
 | 
			
		||||
									><path d="m12 19-7-7 7-7" /><path d="M19 12H5" /></svg
 | 
			
		||||
								>
 | 
			
		||||
								{$_("organizations")}</a
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
					</ol>
 | 
			
		||||
				</nav>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="mb-4 text-3xl font-extrabold leading-tight">
 | 
			
		||||
			{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
 | 
			
		||||
				autocomplete="off"
 | 
			
		||||
				placeholder={$_("name")}
 | 
			
		||||
				type="text"
 | 
			
		||||
				bind:value={editable.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-neutral-800 rounded-md p-2"
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full mt-2">
 | 
			
		||||
			<label for="contact" class="font-semibold text-gray-700"
 | 
			
		||||
				>{$_("contact")}</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) =>
 | 
			
		||||
					label.toLowerCase().includes(filterText.toLowerCase()) ||
 | 
			
		||||
					option.value.id.toString().startsWith(filterText.toLowerCase())}
 | 
			
		||||
				items={contacts}
 | 
			
		||||
				showChevron={true}
 | 
			
		||||
				placeholder={$_("no-contact-selected")}
 | 
			
		||||
				noOptionsMessage={$_("no-contact-found")}
 | 
			
		||||
				bind:selectedValue={contact}
 | 
			
		||||
				on:select={(selectedValue) =>
 | 
			
		||||
					(editable.contact = selectedValue.detail.value)}
 | 
			
		||||
				on:clear={() => (editable.contact = null)}
 | 
			
		||||
			/>
 | 
			
		||||
		</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-center h-5">
 | 
			
		||||
							<input
 | 
			
		||||
								bind:checked={editable.address_checked}
 | 
			
		||||
								id="toggle_address_checkbox"
 | 
			
		||||
								name="toggle_address_checkbox"
 | 
			
		||||
								type="checkbox"
 | 
			
		||||
								class="focus:ring-indigo-500 h-4 w-4 text-indigo-600 border-gray-300 rounded"
 | 
			
		||||
							/>
 | 
			
		||||
						</div>
 | 
			
		||||
						<div class="ml-3 text-sm">
 | 
			
		||||
							<label
 | 
			
		||||
								for="toggle_address_checkbox"
 | 
			
		||||
								class="font-semibold text-gray-700">{$_("address")}</label
 | 
			
		||||
							>
 | 
			
		||||
						</div>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
				{#if editable.address_checked === true}
 | 
			
		||||
					<div class="col-span-6">
 | 
			
		||||
						<label
 | 
			
		||||
							for="address1"
 | 
			
		||||
							class="block text-sm font-medium text-gray-700"
 | 
			
		||||
							>{$_("address")}</label
 | 
			
		||||
						>
 | 
			
		||||
						<input
 | 
			
		||||
							autocomplete="off"
 | 
			
		||||
							placeholder="Address"
 | 
			
		||||
							class:border-red-500={!isAddress1Valid}
 | 
			
		||||
							class:focus:border-red-500={!isAddress1Valid}
 | 
			
		||||
							class:focus:ring-red-500={!isAddress1Valid}
 | 
			
		||||
							bind:value={editable.address.address1}
 | 
			
		||||
							type="text"
 | 
			
		||||
							name="address1"
 | 
			
		||||
							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
						/>
 | 
			
		||||
						{#if !isAddress1Valid}
 | 
			
		||||
							<span
 | 
			
		||||
								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
							>
 | 
			
		||||
								{$_("address-is-required")}
 | 
			
		||||
							</span>
 | 
			
		||||
						{/if}
 | 
			
		||||
					</div>
 | 
			
		||||
					<div class="col-span-6">
 | 
			
		||||
						<label
 | 
			
		||||
							for="address2"
 | 
			
		||||
							class="block text-sm font-medium text-gray-700"
 | 
			
		||||
							>{$_("apartment-suite-etc")}</label
 | 
			
		||||
						>
 | 
			
		||||
						<input
 | 
			
		||||
							autocomplete="off"
 | 
			
		||||
							placeholder={$_("apartment-suite-etc")}
 | 
			
		||||
							bind:value={editable.address.address2}
 | 
			
		||||
							type="text"
 | 
			
		||||
							name="address2"
 | 
			
		||||
							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
						/>
 | 
			
		||||
					</div>
 | 
			
		||||
					<div class="col-span-6">
 | 
			
		||||
						<label for="zipcode" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
							>{$_("zip-postal-code")}</label
 | 
			
		||||
						>
 | 
			
		||||
						<input
 | 
			
		||||
							autocomplete="off"
 | 
			
		||||
							placeholder={$_("zip-postal-code")}
 | 
			
		||||
							class:border-red-500={!iszipcodevalid}
 | 
			
		||||
							class:focus:border-red-500={!iszipcodevalid}
 | 
			
		||||
							class:focus:ring-red-500={!iszipcodevalid}
 | 
			
		||||
							bind:value={editable.address.postalcode}
 | 
			
		||||
							type="text"
 | 
			
		||||
							name="zipcode"
 | 
			
		||||
							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
						/>
 | 
			
		||||
						{#if !iszipcodevalid}
 | 
			
		||||
							<span
 | 
			
		||||
								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
							>
 | 
			
		||||
								{$_("valid-zipcode-postal-code-is-required")}
 | 
			
		||||
							</span>
 | 
			
		||||
						{/if}
 | 
			
		||||
					</div>
 | 
			
		||||
					<div class="col-span-6">
 | 
			
		||||
						<label for="city" class="block text-sm font-medium text-gray-700"
 | 
			
		||||
							>{$_("city")}</label
 | 
			
		||||
						>
 | 
			
		||||
						<input
 | 
			
		||||
							autocomplete="off"
 | 
			
		||||
							placeholder={$_("city")}
 | 
			
		||||
							class:border-red-500={!iscityvalid}
 | 
			
		||||
							class:focus:border-red-500={!iscityvalid}
 | 
			
		||||
							class:focus:ring-red-500={!iscityvalid}
 | 
			
		||||
							bind:value={editable.address.city}
 | 
			
		||||
							type="text"
 | 
			
		||||
							name="city"
 | 
			
		||||
							class="mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm rounded-l-md sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2"
 | 
			
		||||
						/>
 | 
			
		||||
						{#if !iscityvalid}
 | 
			
		||||
							<span
 | 
			
		||||
								class="flex items-center font-medium tracking-wide text-red-500 text-xs mt-1 ml-1"
 | 
			
		||||
							>
 | 
			
		||||
								{$_("valid-city-is-required")}
 | 
			
		||||
							</span>
 | 
			
		||||
						{/if}
 | 
			
		||||
					</div>
 | 
			
		||||
				{/if}
 | 
			
		||||
				<div 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>
 | 
			
		||||
{:else}
 | 
			
		||||
  {#await promise}
 | 
			
		||||
    {$_('organization-detail-is-being-loaded')}
 | 
			
		||||
  {:catch error}
 | 
			
		||||
    <PromiseError />
 | 
			
		||||
  {/await}
 | 
			
		||||
	{#await promise}
 | 
			
		||||
		{$_("organization-detail-is-being-loaded")}
 | 
			
		||||
	{:catch error}
 | 
			
		||||
		<PromiseError />
 | 
			
		||||
	{/await}
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,212 +0,0 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  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";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  $: 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);
 | 
			
		||||
  export let current_organizations = [];
 | 
			
		||||
 | 
			
		||||
  const promise = RunnerOrganizationService.runnerOrganizationControllerGetAll().then(
 | 
			
		||||
    (val) => {
 | 
			
		||||
      current_organizations = val;
 | 
			
		||||
    }
 | 
			
		||||
  );
 | 
			
		||||
</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">
 | 
			
		||||
        <GenerateSponsoringContracts
 | 
			
		||||
            bind:sponsoring_contracts_show
 | 
			
		||||
            bind:generate_orgs />
 | 
			
		||||
        <GenerateRunnerCards
 | 
			
		||||
            bind:cards_show
 | 
			
		||||
            bind:generate_orgs />
 | 
			
		||||
      </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,51 +1,253 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import AddOrgModal from "./AddOrgModal.svelte";
 | 
			
		||||
  export let modal_open = false;
 | 
			
		||||
  import OrgOverview from "./OrgOverview.svelte";
 | 
			
		||||
  import ImportRunnerModal from "../runners/ImportRunnerModal.svelte";
 | 
			
		||||
  let current_organizations = [];
 | 
			
		||||
  export let import_modal_open = false;
 | 
			
		||||
	import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
	let delete_org = {};
 | 
			
		||||
	import { RunnerOrganizationService } from "@odit/lfk-client-js";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import OrgsEmptyState from "./OrgsEmptyState.svelte";
 | 
			
		||||
	import ConfirmOrgDeletionModal from "./ConfirmOrgDeletionModal.svelte";
 | 
			
		||||
	import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.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 = [];
 | 
			
		||||
 | 
			
		||||
	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>
 | 
			
		||||
 | 
			
		||||
<section class="container p-5">
 | 
			
		||||
  <span class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
    {$_('organizations')}
 | 
			
		||||
    {#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:CREATE')}
 | 
			
		||||
      <button
 | 
			
		||||
        on:click={() => {
 | 
			
		||||
          modal_open = true;
 | 
			
		||||
        }}
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-blue-600 text-base font-medium text-white hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
        {$_('create-organization')}
 | 
			
		||||
      </button>
 | 
			
		||||
    {/if}
 | 
			
		||||
    {#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}
 | 
			
		||||
  </span>
 | 
			
		||||
  <OrgOverview bind:current_organizations />
 | 
			
		||||
	<h4 class="mb-1 text-3xl font-extrabold leading-tight">
 | 
			
		||||
		{$_("organizations")}
 | 
			
		||||
	</h4>
 | 
			
		||||
	{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:CREATE")}
 | 
			
		||||
		<button
 | 
			
		||||
			on:click={() => {
 | 
			
		||||
				modal_open = true;
 | 
			
		||||
			}}
 | 
			
		||||
			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"
 | 
			
		||||
		>
 | 
			
		||||
			{$_("create-organization")}
 | 
			
		||||
		</button>
 | 
			
		||||
	{/if}
 | 
			
		||||
	{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:IMPORT")}
 | 
			
		||||
		<button
 | 
			
		||||
			on:click={() => {
 | 
			
		||||
				import_modal_open = true;
 | 
			
		||||
			}}
 | 
			
		||||
			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"
 | 
			
		||||
		>
 | 
			
		||||
			{$_("import-runners")}
 | 
			
		||||
		</button>
 | 
			
		||||
	{/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>
 | 
			
		||||
			</div>
 | 
			
		||||
		{/await}
 | 
			
		||||
	{/if}
 | 
			
		||||
</section>
 | 
			
		||||
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('ORGANIZATION:CREATE')}
 | 
			
		||||
  <AddOrgModal bind:current_organizations bind:modal_open />
 | 
			
		||||
  <ImportRunnerModal
 | 
			
		||||
    on:cancelDelete={(event) => {
 | 
			
		||||
      import_modal_open = false;
 | 
			
		||||
    }}
 | 
			
		||||
    passed_team={{}}
 | 
			
		||||
    passed_org={{}}
 | 
			
		||||
    passed_orgs={current_organizations}
 | 
			
		||||
    opened_from="OrgOverview"
 | 
			
		||||
    current_runners={[]}
 | 
			
		||||
    bind:import_modal_open />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes("ORGANIZATION:CREATE")}
 | 
			
		||||
	<AddOrgModal bind:current_organizations bind:modal_open />
 | 
			
		||||
	<ImportRunnerModal
 | 
			
		||||
		on:cancelDelete={(event) => {
 | 
			
		||||
			import_modal_open = false;
 | 
			
		||||
		}}
 | 
			
		||||
		passed_team={{}}
 | 
			
		||||
		passed_org={{}}
 | 
			
		||||
		passed_orgs={current_organizations}
 | 
			
		||||
		opened_from="OrgOverview"
 | 
			
		||||
		bind:import_modal_open
 | 
			
		||||
	/>
 | 
			
		||||
{/if}
 | 
			
		||||
 
 | 
			
		||||
@@ -9,9 +9,9 @@
 | 
			
		||||
<div class="text-center items-center justify-center">
 | 
			
		||||
  <p class="mb-16 text-lg text-gray-500">
 | 
			
		||||
    <img class="w-full h-44" src={org_empty} alt="" />
 | 
			
		||||
    <span
 | 
			
		||||
      class="font-bold">{$_('there-are-no-organizations-added-yet')}</span><br />
 | 
			
		||||
    <span>{$_('add-your-first-organization')}</span>
 | 
			
		||||
    <span class="font-bold">{$_("there-are-no-organizations-added-yet")}</span
 | 
			
		||||
    ><br />
 | 
			
		||||
    <span>{$_("add-your-first-organization")}</span>
 | 
			
		||||
  </p>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										151
									
								
								src/components/pdf_generation/DocumentServer.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								src/components/pdf_generation/DocumentServer.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,151 @@
 | 
			
		||||
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,
 | 
			
		||||
        self_service_link: runners[i].selfserviceLink,
 | 
			
		||||
        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;
 | 
			
		||||
							
								
								
									
										81
									
								
								src/components/pdf_generation/DownloadProgressModal.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								src/components/pdf_generation/DownloadProgressModal.svelte
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,81 @@
 | 
			
		||||
<script>
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
	import { onMount } from "svelte";
 | 
			
		||||
	export let download_details = "";
 | 
			
		||||
	export let modal_open;
 | 
			
		||||
	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();
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
		};
 | 
			
		||||
	});
 | 
			
		||||
</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="flex-shrink-0 flex items-center justify-center size-12 rounded-full bg-blue-100 sm:mx-0 sm:h-10 sm:w-10"
 | 
			
		||||
						>
 | 
			
		||||
							<svg
 | 
			
		||||
								xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
								viewBox="0 0 24 24"
 | 
			
		||||
								class="h-6 w-6 text-blue-600"
 | 
			
		||||
								fill="currentColor"
 | 
			
		||||
								width="24"
 | 
			
		||||
								height="24"
 | 
			
		||||
								><path fill="none" d="M0 0h24v24H0z" />
 | 
			
		||||
								<path
 | 
			
		||||
									d="M9.83 8.79L8 9.456V13H6V8.05h.015l5.268-1.918c.244-.093.51-.14.782-.131a2.616 2.616 0 0 1 2.427 1.82c.186.583.356.977.51 1.182A4.992 4.992 0 0 0 19 11v2a6.986 6.986 0 0 1-5.402-2.547l-.581 3.297L15 15.67V23h-2v-5.986l-2.05-1.987-.947 4.298-6.894-1.215.348-1.97 4.924.868L9.83 8.79zM13.5 5.5a2 2 0 1 1 0-4 2 2 0 0 1 0 4z"
 | 
			
		||||
								/></svg
 | 
			
		||||
							>
 | 
			
		||||
						</div>
 | 
			
		||||
						<div class="mt-3 sm:text-left text-base">
 | 
			
		||||
							<h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
			
		||||
								{$_('download_laeuft')}
 | 
			
		||||
							</h3>
 | 
			
		||||
							<div class="w-full">
 | 
			
		||||
								{download_details}
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
{/if}
 | 
			
		||||
@@ -1,339 +1,197 @@
 | 
			
		||||
<script>
 | 
			
		||||
    import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
    import {
 | 
			
		||||
        RunnerCardService,
 | 
			
		||||
        RunnerOrganizationService,
 | 
			
		||||
        RunnerTeamService,
 | 
			
		||||
    } from "@odit/lfk-client-js";
 | 
			
		||||
    import Toastify from "toastify-js";
 | 
			
		||||
    export let cards_show = false;
 | 
			
		||||
    export let generate_cards = [];
 | 
			
		||||
    export let generate_runners = [];
 | 
			
		||||
    export let generate_orgs = [];
 | 
			
		||||
    export let generate_teams = [];
 | 
			
		||||
    $: cards_dropdown_open = false;
 | 
			
		||||
    document.addEventListener("click", function (e) {
 | 
			
		||||
        if (
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "cards:dropdown" &&
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "cards:dropdown:menu"
 | 
			
		||||
        ) {
 | 
			
		||||
            cards_dropdown_open = false;
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import {
 | 
			
		||||
		RunnerCardService,
 | 
			
		||||
		RunnerOrganizationService,
 | 
			
		||||
		RunnerTeamService,
 | 
			
		||||
	} from "@odit/lfk-client-js";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	import DocumentServer from "./DocumentServer.ts";
 | 
			
		||||
 | 
			
		||||
    function generateRunnerCards(locale) {
 | 
			
		||||
        cards_dropdown_open = false;
 | 
			
		||||
	import { init } from "@paralleldrive/cuid2";
 | 
			
		||||
	const createId = init({ length: 10, fingerprint: "lfk-frontend" });
 | 
			
		||||
	const documentServer = new DocumentServer(
 | 
			
		||||
		config.baseurl_documentserver,
 | 
			
		||||
		config.documentserver_key
 | 
			
		||||
	);
 | 
			
		||||
 | 
			
		||||
        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);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	export let cards_show = false;
 | 
			
		||||
	export let generate_cards = [];
 | 
			
		||||
	export let generate_runners = [];
 | 
			
		||||
	export let generate_orgs = [];
 | 
			
		||||
	export let generate_teams = [];
 | 
			
		||||
 | 
			
		||||
    function generateCards(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        fetch(
 | 
			
		||||
            `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
            {
 | 
			
		||||
                method: "POST",
 | 
			
		||||
                headers: {
 | 
			
		||||
                    "Content-Type": "application/json",
 | 
			
		||||
                },
 | 
			
		||||
                body: JSON.stringify(generate_cards),
 | 
			
		||||
            }
 | 
			
		||||
        )
 | 
			
		||||
            .then((response) => {
 | 
			
		||||
                if (response.status != "200") {
 | 
			
		||||
                    toast.hideToast();
 | 
			
		||||
                    Toastify({
 | 
			
		||||
                        text: $_("pdf-generation-failed"),
 | 
			
		||||
                        duration: 3500,
 | 
			
		||||
                        backgroundColor:
 | 
			
		||||
                            "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                    }).showToast();
 | 
			
		||||
                } else {
 | 
			
		||||
                    return response.blob();
 | 
			
		||||
                }
 | 
			
		||||
            })
 | 
			
		||||
            .then((blob) => {
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Runnercards.pdf";
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
                toast.hideToast();
 | 
			
		||||
                Toastify({
 | 
			
		||||
                    text: $_("pdf-successfully-generated"),
 | 
			
		||||
                    duration: 3500,
 | 
			
		||||
                    backgroundColor:
 | 
			
		||||
                        "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                }).showToast();
 | 
			
		||||
            })
 | 
			
		||||
            .catch((err) => {
 | 
			
		||||
                console.error(err);
 | 
			
		||||
            });
 | 
			
		||||
    }
 | 
			
		||||
	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 generateRunnersCards(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        const current_cards = await RunnerCardService.runnerCardControllerGetAll();
 | 
			
		||||
        let cards = [];
 | 
			
		||||
        for (let runner of generate_runners) {
 | 
			
		||||
            let card = current_cards.find((c) => c.runner?.id == runner.id);
 | 
			
		||||
            if (!card) {
 | 
			
		||||
                card = await RunnerCardService.runnerCardControllerPost({
 | 
			
		||||
                    runner: runner.id,
 | 
			
		||||
                });
 | 
			
		||||
            }
 | 
			
		||||
            cards.push(card);
 | 
			
		||||
        }
 | 
			
		||||
        fetch(
 | 
			
		||||
            `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
            {
 | 
			
		||||
                method: "POST",
 | 
			
		||||
                headers: {
 | 
			
		||||
                    "Content-Type": "application/json",
 | 
			
		||||
                },
 | 
			
		||||
                body: JSON.stringify(cards),
 | 
			
		||||
            }
 | 
			
		||||
        )
 | 
			
		||||
            .then((response) => {
 | 
			
		||||
                if (response.status != "200") {
 | 
			
		||||
                    toast.hideToast();
 | 
			
		||||
                    Toastify({
 | 
			
		||||
                        text: $_("pdf-generation-failed"),
 | 
			
		||||
                        duration: 3500,
 | 
			
		||||
                        backgroundColor:
 | 
			
		||||
                            "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                    }).showToast();
 | 
			
		||||
                } else {
 | 
			
		||||
                    return response.blob();
 | 
			
		||||
                }
 | 
			
		||||
            })
 | 
			
		||||
            .then((blob) => {
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Runnercards.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) => {});
 | 
			
		||||
    }
 | 
			
		||||
	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);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
    async function generateTeamCards(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdfs"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        let count = 0;
 | 
			
		||||
        const current_cards = await RunnerCardService.runnerCardControllerGetAll();
 | 
			
		||||
        for (const t of generate_teams) {
 | 
			
		||||
            const runners = await RunnerTeamService.runnerTeamControllerGetRunners(
 | 
			
		||||
                t.id
 | 
			
		||||
            );
 | 
			
		||||
            let cards = [];
 | 
			
		||||
            for (let runner of runners) {
 | 
			
		||||
                let card = current_cards.find((c) => c.runner?.id == runner.id);
 | 
			
		||||
                if (!card) {
 | 
			
		||||
                    card = await RunnerCardService.runnerCardControllerPost({
 | 
			
		||||
                        runner: runner.id,
 | 
			
		||||
                    });
 | 
			
		||||
                }
 | 
			
		||||
                cards.push(card);
 | 
			
		||||
            }
 | 
			
		||||
            fetch(
 | 
			
		||||
                `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
                {
 | 
			
		||||
                    method: "POST",
 | 
			
		||||
                    headers: {
 | 
			
		||||
                        "Content-Type": "application/json",
 | 
			
		||||
                    },
 | 
			
		||||
                    body: JSON.stringify(cards),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
                .then((response) => {
 | 
			
		||||
                    if (response.status != "200") {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdf-generation-failed"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        return response.blob();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .then((blob) => {
 | 
			
		||||
                    count++;
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + t.name + ".pdf";
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
                    if (count === generate_teams.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	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 generateOrgCards(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        let count = 0;
 | 
			
		||||
        const current_cards = await RunnerCardService.runnerCardControllerGetAll();
 | 
			
		||||
        for (const o of generate_orgs) {
 | 
			
		||||
            const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners(
 | 
			
		||||
                o.id
 | 
			
		||||
            );
 | 
			
		||||
            let cards = [];
 | 
			
		||||
            for (let runner of runners) {
 | 
			
		||||
                let card = current_cards.find((c) => c.runner?.id == runner.id);
 | 
			
		||||
                if (!card) {
 | 
			
		||||
                    card = await RunnerCardService.runnerCardControllerPost({
 | 
			
		||||
                        runner: runner.id,
 | 
			
		||||
                    });
 | 
			
		||||
                }
 | 
			
		||||
                cards.push(card);
 | 
			
		||||
            }
 | 
			
		||||
            fetch(
 | 
			
		||||
                `${config.baseurl}/documents/cards?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
                {
 | 
			
		||||
                    method: "POST",
 | 
			
		||||
                    headers: {
 | 
			
		||||
                        "Content-Type": "application/json",
 | 
			
		||||
                    },
 | 
			
		||||
                    body: JSON.stringify(cards),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
                .then((response) => {
 | 
			
		||||
                    if (response.status != "200") {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdf-generation-failed"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        return response.blob();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .then((blob) => {
 | 
			
		||||
                    count++;
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + o.name + ".pdf";
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
                    if (count === generate_orgs.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	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}
 | 
			
		||||
    <div id="cards:dropdown" class="relative inline-block">
 | 
			
		||||
        <div>
 | 
			
		||||
            <button
 | 
			
		||||
                on:click={() => {
 | 
			
		||||
                    cards_dropdown_open = !cards_dropdown_open;
 | 
			
		||||
                }}
 | 
			
		||||
                type="button"
 | 
			
		||||
                class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex"
 | 
			
		||||
                id="options-menu"
 | 
			
		||||
                aria-haspopup="true"
 | 
			
		||||
                aria-expanded="true">
 | 
			
		||||
                {$_('generate-runnercards')}
 | 
			
		||||
                <svg
 | 
			
		||||
                    xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                    width="24"
 | 
			
		||||
                    height="24"
 | 
			
		||||
                    viewBox="0 0 24 24"
 | 
			
		||||
                    class="-mr-1 ml-2 h-5 w-5"><path
 | 
			
		||||
                        fill="none"
 | 
			
		||||
                        d="M0 0h24v24H0z" />
 | 
			
		||||
                    <path
 | 
			
		||||
                        fill="currentColor"
 | 
			
		||||
                        d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg>
 | 
			
		||||
            </button>
 | 
			
		||||
        </div>
 | 
			
		||||
        {#if cards_dropdown_open}
 | 
			
		||||
            <div
 | 
			
		||||
                class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5"
 | 
			
		||||
                id="cards:dropdown:menu">
 | 
			
		||||
                <div
 | 
			
		||||
                    class="py-1"
 | 
			
		||||
                    role="menu"
 | 
			
		||||
                    aria-orientation="vertical"
 | 
			
		||||
                    aria-labelledby="options-menu">
 | 
			
		||||
                    <span
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateRunnerCards('de');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('german')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateRunnerCards('en');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('english')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
	<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}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,273 +1,180 @@
 | 
			
		||||
<script>
 | 
			
		||||
    import { _ } from "svelte-i18n";
 | 
			
		||||
    import {
 | 
			
		||||
        DonationService,
 | 
			
		||||
        RunnerTeamService,
 | 
			
		||||
        RunnerOrganizationService
 | 
			
		||||
    } from "@odit/lfk-client-js";
 | 
			
		||||
    import Toastify from "toastify-js";
 | 
			
		||||
    export let certificates_show = false;
 | 
			
		||||
    export let generate_runners = [];
 | 
			
		||||
    export let generate_orgs = [];
 | 
			
		||||
    export let generate_teams = [];
 | 
			
		||||
    $: certificates_dropdown_open = false;
 | 
			
		||||
    document.addEventListener("click", function (e) {
 | 
			
		||||
        if (
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "certificates:dropdown" &&
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "certificates:dropdown:menu"
 | 
			
		||||
        ) {
 | 
			
		||||
            certificates_dropdown_open = false;
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import {
 | 
			
		||||
    DonationService,
 | 
			
		||||
    RunnerTeamService,
 | 
			
		||||
    RunnerOrganizationService,
 | 
			
		||||
	RunnerService
 | 
			
		||||
  } 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
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
    function generateCertificates(locale) {
 | 
			
		||||
        certificates_dropdown_open = false;
 | 
			
		||||
  export let certificates_show = false;
 | 
			
		||||
  export let generate_runners = [];
 | 
			
		||||
  export let generate_orgs = [];
 | 
			
		||||
  export let generate_teams = [];
 | 
			
		||||
 | 
			
		||||
        if (generate_orgs.length > 0) {
 | 
			
		||||
            generateOrgCertificates(locale);
 | 
			
		||||
        } else if (generate_teams.length > 0) {
 | 
			
		||||
            generateTeamCertificates(locale);
 | 
			
		||||
        } else {
 | 
			
		||||
            generateRunnerCertificates(locale);
 | 
			
		||||
        }
 | 
			
		||||
  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) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        const current_donations = await DonationService.donationControllerGetAll();
 | 
			
		||||
  async function generateRunnerCertificates(locale) {
 | 
			
		||||
    toast.loading($_("generating-pdf"));
 | 
			
		||||
    const current_donations =
 | 
			
		||||
      (await DonationService.donationControllerGetAll()) || [];
 | 
			
		||||
    let certificateRunners = [];
 | 
			
		||||
    for (let runner of generate_runners) {
 | 
			
		||||
	  const linkRunner = await RunnerService.runnerControllerGetOne(runner.id)
 | 
			
		||||
      linkRunner.distanceDonations =
 | 
			
		||||
        current_donations.filter((d) => d.runner?.id == runner.id) || [];
 | 
			
		||||
      certificateRunners.push(linkRunner);
 | 
			
		||||
    }
 | 
			
		||||
    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,
 | 
			
		||||
        true
 | 
			
		||||
      );
 | 
			
		||||
      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,
 | 
			
		||||
		  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,
 | 
			
		||||
		  true
 | 
			
		||||
        );
 | 
			
		||||
        let certificateRunners = [];
 | 
			
		||||
        for (let runner of generate_runners) {
 | 
			
		||||
            runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
 | 
			
		||||
            certificateRunners.push(runner);
 | 
			
		||||
        for (let runner of runners) {
 | 
			
		||||
          runner.distanceDonations =
 | 
			
		||||
            current_donations.filter((d) => d.runner?.id == runner.id) || [];
 | 
			
		||||
          certificateRunners.push(runner);
 | 
			
		||||
        }
 | 
			
		||||
        fetch(
 | 
			
		||||
            `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
            {
 | 
			
		||||
                method: "POST",
 | 
			
		||||
                headers: {
 | 
			
		||||
                    "Content-Type": "application/json",
 | 
			
		||||
                },
 | 
			
		||||
                body: JSON.stringify(certificateRunners),
 | 
			
		||||
            }
 | 
			
		||||
        )
 | 
			
		||||
            .then((response) => {
 | 
			
		||||
                if (response.status != "200") {
 | 
			
		||||
                    toast.hideToast();
 | 
			
		||||
                    Toastify({
 | 
			
		||||
                        text: $_("pdf-generation-failed"),
 | 
			
		||||
                        duration: 3500,
 | 
			
		||||
                        backgroundColor:
 | 
			
		||||
                            "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                    }).showToast();
 | 
			
		||||
                } else {
 | 
			
		||||
                    return response.blob();
 | 
			
		||||
                }
 | 
			
		||||
            })
 | 
			
		||||
            .then((blob) => {
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Certificates.pdf";
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
                toast.hideToast();
 | 
			
		||||
                Toastify({
 | 
			
		||||
                    text: $_("pdf-successfully-generated"),
 | 
			
		||||
                    duration: 3500,
 | 
			
		||||
                    backgroundColor:
 | 
			
		||||
                        "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                }).showToast();
 | 
			
		||||
            })
 | 
			
		||||
            .catch((err) => {});
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async function generateTeamCertificates(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdfs"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        let count = 0;
 | 
			
		||||
        const current_donations = await DonationService.donationControllerGetAll();
 | 
			
		||||
        for (const t of generate_teams) {
 | 
			
		||||
            const runners = await RunnerTeamService.runnerTeamControllerGetRunners(
 | 
			
		||||
                t.id
 | 
			
		||||
                );
 | 
			
		||||
            let certificateRunners = [];
 | 
			
		||||
            for (let runner of runners) {
 | 
			
		||||
                runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
 | 
			
		||||
                certificateRunners.push(runner);
 | 
			
		||||
            }
 | 
			
		||||
            console.log(certificateRunners)
 | 
			
		||||
            fetch(
 | 
			
		||||
                `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
                {
 | 
			
		||||
                    method: "POST",
 | 
			
		||||
                    headers: {
 | 
			
		||||
                        "Content-Type": "application/json",
 | 
			
		||||
                    },
 | 
			
		||||
                    body: JSON.stringify(certificateRunners),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
                .then((response) => {
 | 
			
		||||
                    if (response.status != "200") {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdf-generation-failed"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        return response.blob();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .then((blob) => {
 | 
			
		||||
                    count++;
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Certificates_" + t.name + ".pdf";
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
                    if (count === generate_teams.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async function generateOrgCertificates(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        let count = 0;
 | 
			
		||||
        const current_donations = await DonationService.donationControllerGetAll();
 | 
			
		||||
        for (const o of generate_orgs) {
 | 
			
		||||
            const runners = await RunnerOrganizationService.runnerOrganizationControllerGetRunners(
 | 
			
		||||
                o.id
 | 
			
		||||
        await documentServer
 | 
			
		||||
          .generateCertificates(certificateRunners, locale)
 | 
			
		||||
          .then((blob) => {
 | 
			
		||||
            download(
 | 
			
		||||
              blob,
 | 
			
		||||
              `${$_("certificates")}_${o.name}_${
 | 
			
		||||
                t.name
 | 
			
		||||
              }-${locale}-${createId()}.pdf`
 | 
			
		||||
            );
 | 
			
		||||
            let certificateRunners = [];
 | 
			
		||||
            for (let runner of runners) {
 | 
			
		||||
                runner.distanceDonations = current_donations.find((d) => d.runner?.id == runner.id) || [];
 | 
			
		||||
                certificateRunners.push(runner);
 | 
			
		||||
            if (
 | 
			
		||||
              count === o.teams.length &&
 | 
			
		||||
              count_orgs === generate_orgs.length
 | 
			
		||||
            ) {
 | 
			
		||||
              toast.dismiss();
 | 
			
		||||
              toast.success($_("pdfs-successfully-generated"));
 | 
			
		||||
            }
 | 
			
		||||
            fetch(
 | 
			
		||||
                `${config.baseurl}/documents/certificates?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
                {
 | 
			
		||||
                    method: "POST",
 | 
			
		||||
                    headers: {
 | 
			
		||||
                        "Content-Type": "application/json",
 | 
			
		||||
                    },
 | 
			
		||||
                    body: JSON.stringify(certificateRunners),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
                .then((response) => {
 | 
			
		||||
                    if (response.status != "200") {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdf-generation-failed"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        return response.blob();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .then((blob) => {
 | 
			
		||||
                    count++;
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Certificates_" + o.name + ".pdf";
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
                    if (count === generate_orgs.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err) => {});
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if certificates_show}
 | 
			
		||||
    <div id="certificates:dropdown" class="relative inline-block">
 | 
			
		||||
        <div>
 | 
			
		||||
            <button
 | 
			
		||||
                on:click={() => {
 | 
			
		||||
                    certificates_dropdown_open = !certificates_dropdown_open;
 | 
			
		||||
                }}
 | 
			
		||||
                type="button"
 | 
			
		||||
                class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex"
 | 
			
		||||
                id="options-menu"
 | 
			
		||||
                aria-haspopup="true"
 | 
			
		||||
                aria-expanded="true">
 | 
			
		||||
                {$_('generate-runner-certificates')}
 | 
			
		||||
                <svg
 | 
			
		||||
                    xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                    width="24"
 | 
			
		||||
                    height="24"
 | 
			
		||||
                    viewBox="0 0 24 24"
 | 
			
		||||
                    class="-mr-1 ml-2 h-5 w-5"><path
 | 
			
		||||
                        fill="none"
 | 
			
		||||
                        d="M0 0h24v24H0z" />
 | 
			
		||||
                    <path
 | 
			
		||||
                        fill="currentColor"
 | 
			
		||||
                        d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg>
 | 
			
		||||
            </button>
 | 
			
		||||
        </div>
 | 
			
		||||
        {#if certificates_dropdown_open}
 | 
			
		||||
            <div
 | 
			
		||||
                class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5"
 | 
			
		||||
                id="certificates:dropdown:menu">
 | 
			
		||||
                <div
 | 
			
		||||
                    class="py-1"
 | 
			
		||||
                    role="menu"
 | 
			
		||||
                    aria-orientation="vertical"
 | 
			
		||||
                    aria-labelledby="options-menu">
 | 
			
		||||
                    <span
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateCertificates('de');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('german')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateCertificates('en');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('english')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
  <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}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,254 +1,163 @@
 | 
			
		||||
<script>
 | 
			
		||||
    import { getLocaleFromNavigator, _ } from "svelte-i18n";
 | 
			
		||||
    import {
 | 
			
		||||
        RunnerOrganizationService,
 | 
			
		||||
        RunnerTeamService,
 | 
			
		||||
    } from "@odit/lfk-client-js";
 | 
			
		||||
    import Toastify from "toastify-js";
 | 
			
		||||
    export let sponsoring_contracts_show = false;
 | 
			
		||||
    export let generate_runners = [];
 | 
			
		||||
    export let generate_orgs = [];
 | 
			
		||||
    export let generate_teams = [];
 | 
			
		||||
    $: sponsoring_contracts_download_open = false;
 | 
			
		||||
    document.addEventListener("click", function (e) {
 | 
			
		||||
        if (
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "sponsoring:dropdown" &&
 | 
			
		||||
            e.target.parentNode?.parentNode?.id != "sponsoring:dropdown:menu"
 | 
			
		||||
        ) {
 | 
			
		||||
            sponsoring_contracts_download_open = false;
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
	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";
 | 
			
		||||
	import DownloadProgressModal from "./DownloadProgressModal.svelte";
 | 
			
		||||
	const createId = init({ length: 10, fingerprint: "lfk-frontend" });
 | 
			
		||||
	const documentServer = new DocumentServer(
 | 
			
		||||
		config.baseurl_documentserver,
 | 
			
		||||
		config.documentserver_key
 | 
			
		||||
	);
 | 
			
		||||
 | 
			
		||||
    function generateSponsoringContract(locale) {
 | 
			
		||||
        sponsoring_contracts_download_open = false;
 | 
			
		||||
	export let sponsoring_contracts_show = false;
 | 
			
		||||
	export let generate_runners = [];
 | 
			
		||||
	export let generate_orgs = [];
 | 
			
		||||
	export let generate_teams = [];
 | 
			
		||||
	//
 | 
			
		||||
	export let download_modal_open = false;
 | 
			
		||||
	export let download_details = "";
 | 
			
		||||
 | 
			
		||||
        if (generate_orgs.length > 0) {
 | 
			
		||||
            generateOrgContracts(locale);
 | 
			
		||||
        } else if (generate_teams.length > 0) {
 | 
			
		||||
            generateTeamContracts(locale);
 | 
			
		||||
        } else {
 | 
			
		||||
            generateRunnerContracts(locale);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	function generateSponsoringContract(locale) {
 | 
			
		||||
		download_modal_open = true;
 | 
			
		||||
		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) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdfs"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        let count = 0;
 | 
			
		||||
        for (const t of generate_teams) {
 | 
			
		||||
            count++;
 | 
			
		||||
            const runners = await RunnerTeamService.runnerTeamControllerGetRunners(
 | 
			
		||||
                t.id
 | 
			
		||||
            );
 | 
			
		||||
            fetch(
 | 
			
		||||
                `${config.baseurl}/documents/contracts?locale=${locale}&download=true&key=${config.documentserver_key}`,
 | 
			
		||||
                {
 | 
			
		||||
                    method: "POST",
 | 
			
		||||
                    headers: {
 | 
			
		||||
                        "Content-Type": "application/json",
 | 
			
		||||
                    },
 | 
			
		||||
                    body: JSON.stringify(runners),
 | 
			
		||||
                }
 | 
			
		||||
            )
 | 
			
		||||
                .then((response) => {
 | 
			
		||||
                    if (response.status != "200") {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdf-generation-failed"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        return response.blob();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .then((blob) => {
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + t.name + ".pdf";
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
                    if (count === generate_teams.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	async function generateTeamContracts(locale) {
 | 
			
		||||
		toast.loading($_("generating-pdfs"));
 | 
			
		||||
		let totalCount = generate_teams.length;
 | 
			
		||||
		let count = 0;
 | 
			
		||||
		for (const t of generate_teams) {
 | 
			
		||||
			count++;
 | 
			
		||||
			download_details = `${t.parentGroup.name} > ${t.name}`;
 | 
			
		||||
			const runners = await RunnerTeamService.runnerTeamControllerGetRunners(
 | 
			
		||||
				t.id
 | 
			
		||||
			);
 | 
			
		||||
			await documentServer
 | 
			
		||||
				.generateContracts(runners, locale)
 | 
			
		||||
				.then((blob) => {
 | 
			
		||||
					download(
 | 
			
		||||
						blob,
 | 
			
		||||
						`${$_("sponsorings")}_${t.name}-${locale}-${createId()}.pdf`
 | 
			
		||||
					);
 | 
			
		||||
					if (count === totalCount) {
 | 
			
		||||
						download_modal_open = false;
 | 
			
		||||
					}
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
    async function generateOrgContracts(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        for (const o of generate_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 === generate_orgs.length) {
 | 
			
		||||
                        toast.hideToast();
 | 
			
		||||
                        Toastify({
 | 
			
		||||
                            text: $_("pdfs-successfully-generated"),
 | 
			
		||||
                            duration: 3500,
 | 
			
		||||
                            backgroundColor:
 | 
			
		||||
                                "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                        }).showToast();
 | 
			
		||||
                    }
 | 
			
		||||
                })
 | 
			
		||||
                .catch((err) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
	async function generateOrgContracts(locale) {
 | 
			
		||||
		toast.loading($_("generating-pdf"));
 | 
			
		||||
		let totalCount = 0;
 | 
			
		||||
		for (const o of generate_orgs) {
 | 
			
		||||
			totalCount++;
 | 
			
		||||
			for (const t of o.teams) {
 | 
			
		||||
				totalCount++;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		console.log({ totalCount });
 | 
			
		||||
		let count = 0;
 | 
			
		||||
		for (const o of generate_orgs) {
 | 
			
		||||
			count++;
 | 
			
		||||
			let runners =
 | 
			
		||||
				await RunnerOrganizationService.runnerOrganizationControllerGetRunners(
 | 
			
		||||
					o.id,
 | 
			
		||||
					true
 | 
			
		||||
				);
 | 
			
		||||
			download_details = o.name;
 | 
			
		||||
			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
 | 
			
		||||
				);
 | 
			
		||||
				download_details = `${o.name} > ${t.name}`;
 | 
			
		||||
				await documentServer
 | 
			
		||||
					.generateContracts(runners, locale)
 | 
			
		||||
					.then((blob) => {
 | 
			
		||||
						download(
 | 
			
		||||
							blob,
 | 
			
		||||
							`${$_("sponsorings")}_${o.name}_${
 | 
			
		||||
								t.name
 | 
			
		||||
							}-${locale}-${createId()}.pdf`
 | 
			
		||||
						);
 | 
			
		||||
						console.log({ count });
 | 
			
		||||
						if (count === totalCount) {
 | 
			
		||||
							download_modal_open = false;
 | 
			
		||||
						}
 | 
			
		||||
					})
 | 
			
		||||
					.catch((err) => {});
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
    function generateRunnerContracts(locale) {
 | 
			
		||||
        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(generate_runners),
 | 
			
		||||
            }
 | 
			
		||||
        )
 | 
			
		||||
            .then((response) => {
 | 
			
		||||
                if (response.status != "200") {
 | 
			
		||||
                    toast.hideToast();
 | 
			
		||||
                    Toastify({
 | 
			
		||||
                        text: $_("pdf-generation-failed"),
 | 
			
		||||
                        duration: 3500,
 | 
			
		||||
                        backgroundColor:
 | 
			
		||||
                            "linear-gradient(90deg, hsla(281, 37%, 45%, 1) 0%, hsla(1, 62%, 48%, 1) 100%)",
 | 
			
		||||
                    }).showToast();
 | 
			
		||||
                } else {
 | 
			
		||||
                    return response.blob();
 | 
			
		||||
                }
 | 
			
		||||
            })
 | 
			
		||||
            .then((blob) => {
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Sponsoring.pdf";
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
                toast.hideToast();
 | 
			
		||||
                Toastify({
 | 
			
		||||
                    text: $_("pdf-successfully-generated"),
 | 
			
		||||
                    duration: 3500,
 | 
			
		||||
                    backgroundColor:
 | 
			
		||||
                        "linear-gradient(to right, #00b09b, #96c93d)",
 | 
			
		||||
                }).showToast();
 | 
			
		||||
            })
 | 
			
		||||
            .catch((err) => {
 | 
			
		||||
                console.error(err);
 | 
			
		||||
            });
 | 
			
		||||
    }
 | 
			
		||||
	async function generateRunnerContracts(locale) {
 | 
			
		||||
		toast.loading($_("generating-pdf"));
 | 
			
		||||
		await 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);
 | 
			
		||||
				download_modal_open = false;
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {
 | 
			
		||||
				console.error(err);
 | 
			
		||||
			});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#if sponsoring_contracts_show}
 | 
			
		||||
    <div id="sponsoring:dropdown" class="relative inline-block">
 | 
			
		||||
        <div>
 | 
			
		||||
            <button
 | 
			
		||||
                on:click={() => {
 | 
			
		||||
                    sponsoring_contracts_download_open = !sponsoring_contracts_download_open;
 | 
			
		||||
                }}
 | 
			
		||||
                type="button"
 | 
			
		||||
                class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-600 text-base font-medium text-white hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 sm:ml-3 sm:w-auto sm:text-sm inline-flex"
 | 
			
		||||
                id="options-menu"
 | 
			
		||||
                aria-haspopup="true"
 | 
			
		||||
                aria-expanded="true">
 | 
			
		||||
                {$_('generate-sponsoring-contracts')}
 | 
			
		||||
                <svg
 | 
			
		||||
                    xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                    width="24"
 | 
			
		||||
                    height="24"
 | 
			
		||||
                    viewBox="0 0 24 24"
 | 
			
		||||
                    class="-mr-1 ml-2 h-5 w-5"><path
 | 
			
		||||
                        fill="none"
 | 
			
		||||
                        d="M0 0h24v24H0z" />
 | 
			
		||||
                    <path
 | 
			
		||||
                        fill="currentColor"
 | 
			
		||||
                        d="M3 19h18v2H3v-2zm10-5.83l6.07-6.07 1.42 1.41L12 17 3.52 8.52l1.4-1.42L11 13.17V2h2v11.17z" /></svg>
 | 
			
		||||
            </button>
 | 
			
		||||
        </div>
 | 
			
		||||
        {#if sponsoring_contracts_download_open}
 | 
			
		||||
            <div
 | 
			
		||||
                class="origin-top-right absolute right-0 mt-2 w-56 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5"
 | 
			
		||||
                id="sponsoring:dropdown:menu">
 | 
			
		||||
                <div
 | 
			
		||||
                    class="py-1"
 | 
			
		||||
                    role="menu"
 | 
			
		||||
                    aria-orientation="vertical"
 | 
			
		||||
                    aria-labelledby="options-menu">
 | 
			
		||||
                    <span
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700">{$_('select-language')}</span>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateSponsoringContract('de');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('german')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                    <button
 | 
			
		||||
                        on:click={() => {
 | 
			
		||||
                            generateSponsoringContract('en');
 | 
			
		||||
                        }}
 | 
			
		||||
                        type="submit"
 | 
			
		||||
                        class="block w-full text-left px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 hover:text-gray-900 focus:outline-none focus:bg-gray-100 focus:text-gray-900"
 | 
			
		||||
                        role="menuitem">
 | 
			
		||||
                        {$_('english')}
 | 
			
		||||
                    </button>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        {/if}
 | 
			
		||||
    </div>
 | 
			
		||||
	<DownloadProgressModal {download_details} modal_open={download_modal_open} />
 | 
			
		||||
	<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}
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user