All checks were successful
		
		
	
	Build Latest and dev images / build-container (push) Successful in 1m0s
				
			
		
			
				
	
	
		
			414 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Svelte
		
	
	
	
	
	
			
		
		
	
	
			414 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Svelte
		
	
	
	
	
	
| <script>
 | |
| 	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>
 | |
| 
 | |
| {#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">
 | |
| 							<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} />
 | |
| {/await}
 |