feat: improve fonts + button positions
This commit is contained in:
		@@ -1,315 +1,318 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { _ } from "svelte-i18n";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
  import store from "../../store";
 | 
			
		||||
  import {
 | 
			
		||||
    RunnerService,
 | 
			
		||||
    RunnerTeamService,
 | 
			
		||||
    RunnerOrganizationService,
 | 
			
		||||
  } from "@odit/lfk-client-js";
 | 
			
		||||
  import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
  import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import toast from "svelte-french-toast";
 | 
			
		||||
  let data_loaded = false;
 | 
			
		||||
  export let params;
 | 
			
		||||
  const runner_promise = RunnerService.runnerControllerGetOne(params.runnerid);
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: original_data_pdf = {};
 | 
			
		||||
  $: original_data = {};
 | 
			
		||||
  $: editable = {};
 | 
			
		||||
  $: group = {};
 | 
			
		||||
  $: 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 &&
 | 
			
		||||
    editable.group != null;
 | 
			
		||||
  $: sponsoring_contracts_show = true;
 | 
			
		||||
  $: cards_show = true;
 | 
			
		||||
  $: certificates_show = true;
 | 
			
		||||
  $: generate_runners = [original_data_pdf];
 | 
			
		||||
  runner_promise.then((data) => {
 | 
			
		||||
    data_loaded = true;
 | 
			
		||||
    original_data_pdf = Object.assign(original_data_pdf, data);
 | 
			
		||||
    data.group = data.group.id;
 | 
			
		||||
    original_data = Object.assign(original_data, data);
 | 
			
		||||
    editable = Object.assign(editable, original_data);
 | 
			
		||||
	import { _ } from "svelte-i18n";
 | 
			
		||||
	import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
	import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
	import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
	import store from "../../store";
 | 
			
		||||
	import {
 | 
			
		||||
		RunnerService,
 | 
			
		||||
		RunnerTeamService,
 | 
			
		||||
		RunnerOrganizationService,
 | 
			
		||||
	} from "@odit/lfk-client-js";
 | 
			
		||||
	import PromiseError from "../base/PromiseError.svelte";
 | 
			
		||||
	import isEmail from "validator/es/lib/isEmail";
 | 
			
		||||
	import Select from "svelte-select";
 | 
			
		||||
	import toast from "svelte-french-toast";
 | 
			
		||||
	let data_loaded = false;
 | 
			
		||||
	export let params;
 | 
			
		||||
	const runner_promise = RunnerService.runnerControllerGetOne(params.runnerid);
 | 
			
		||||
	$: delete_triggered = false;
 | 
			
		||||
	$: original_data_pdf = {};
 | 
			
		||||
	$: original_data = {};
 | 
			
		||||
	$: editable = {};
 | 
			
		||||
	$: group = {};
 | 
			
		||||
	$: 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 &&
 | 
			
		||||
		editable.group != null;
 | 
			
		||||
	$: sponsoring_contracts_show = true;
 | 
			
		||||
	$: cards_show = true;
 | 
			
		||||
	$: certificates_show = true;
 | 
			
		||||
	$: generate_runners = [original_data_pdf];
 | 
			
		||||
	runner_promise.then((data) => {
 | 
			
		||||
		data_loaded = true;
 | 
			
		||||
		original_data_pdf = Object.assign(original_data_pdf, data);
 | 
			
		||||
		data.group = data.group.id;
 | 
			
		||||
		original_data = Object.assign(original_data, data);
 | 
			
		||||
		editable = Object.assign(editable, original_data);
 | 
			
		||||
 | 
			
		||||
    RunnerOrganizationService.runnerOrganizationControllerGetAll().then(
 | 
			
		||||
      (val) => {
 | 
			
		||||
        const orgs = val.map((r) => {
 | 
			
		||||
          return { label: r.name, value: r };
 | 
			
		||||
        });
 | 
			
		||||
        groups = groups.concat(orgs);
 | 
			
		||||
        RunnerTeamService.runnerTeamControllerGetAll().then((val) => {
 | 
			
		||||
          const teams = val.map((r) => {
 | 
			
		||||
            return { label: `${r.parentGroup.name} > ${r.name}`, value: r };
 | 
			
		||||
          });
 | 
			
		||||
          groups = groups.concat(teams);
 | 
			
		||||
          group = groups.find((g) => g.value.id == editable.group);
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
    );
 | 
			
		||||
  });
 | 
			
		||||
  let groups = [];
 | 
			
		||||
  function submit() {
 | 
			
		||||
    if (data_loaded === true && save_enabled) {
 | 
			
		||||
      toast.loading($_("updating-runner"));
 | 
			
		||||
      let postdata = {};
 | 
			
		||||
      postdata = Object.assign(postdata, editable);
 | 
			
		||||
      if (postdata.phone === "") {
 | 
			
		||||
        postdata.phone = null;
 | 
			
		||||
      }
 | 
			
		||||
      RunnerService.runnerControllerPut(original_data.id, postdata)
 | 
			
		||||
        .then((resp) => {
 | 
			
		||||
          Object.assign(original_data, editable);
 | 
			
		||||
          original_data = original_data;
 | 
			
		||||
          toast.dismiss();
 | 
			
		||||
          toast.success($_("runner-updated"));
 | 
			
		||||
        })
 | 
			
		||||
        .catch((err) => {});
 | 
			
		||||
    } else {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  function deleteRunner() {
 | 
			
		||||
    RunnerService.runnerControllerRemove(original_data.id, true)
 | 
			
		||||
      .then((resp) => {
 | 
			
		||||
        location.replace("./");
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err) => {});
 | 
			
		||||
  }
 | 
			
		||||
		RunnerOrganizationService.runnerOrganizationControllerGetAll().then(
 | 
			
		||||
			(val) => {
 | 
			
		||||
				const orgs = val.map((r) => {
 | 
			
		||||
					return { label: r.name, value: r };
 | 
			
		||||
				});
 | 
			
		||||
				groups = groups.concat(orgs);
 | 
			
		||||
				RunnerTeamService.runnerTeamControllerGetAll().then((val) => {
 | 
			
		||||
					const teams = val.map((r) => {
 | 
			
		||||
						return { label: `${r.parentGroup.name} > ${r.name}`, value: r };
 | 
			
		||||
					});
 | 
			
		||||
					groups = groups.concat(teams);
 | 
			
		||||
					group = groups.find((g) => g.value.id == editable.group);
 | 
			
		||||
				});
 | 
			
		||||
			}
 | 
			
		||||
		);
 | 
			
		||||
	});
 | 
			
		||||
	let groups = [];
 | 
			
		||||
	function submit() {
 | 
			
		||||
		if (data_loaded === true && save_enabled) {
 | 
			
		||||
			toast.loading($_("updating-runner"));
 | 
			
		||||
			let postdata = {};
 | 
			
		||||
			postdata = Object.assign(postdata, editable);
 | 
			
		||||
			if (postdata.phone === "") {
 | 
			
		||||
				postdata.phone = null;
 | 
			
		||||
			}
 | 
			
		||||
			RunnerService.runnerControllerPut(original_data.id, postdata)
 | 
			
		||||
				.then((resp) => {
 | 
			
		||||
					Object.assign(original_data, editable);
 | 
			
		||||
					original_data = original_data;
 | 
			
		||||
					toast.dismiss();
 | 
			
		||||
					toast.success($_("runner-updated"));
 | 
			
		||||
				})
 | 
			
		||||
				.catch((err) => {});
 | 
			
		||||
		} else {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	function deleteRunner() {
 | 
			
		||||
		RunnerService.runnerControllerRemove(original_data.id, true)
 | 
			
		||||
			.then((resp) => {
 | 
			
		||||
				location.replace("./");
 | 
			
		||||
			})
 | 
			
		||||
			.catch((err) => {});
 | 
			
		||||
	}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
{#await runner_promise}
 | 
			
		||||
  {$_("loading-runners")}
 | 
			
		||||
	{$_("loading-runners")}
 | 
			
		||||
{: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
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                class="flex-shrink-0 w-5 h-5 mr-2"
 | 
			
		||||
                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
 | 
			
		||||
              >
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="flex items-center">
 | 
			
		||||
              <a class="mr-2" href="./">{$_("runners")}</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="runner_actions_${editable.id}">
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:DELETE")}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              on:click={deleteRunner}
 | 
			
		||||
              class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm: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}
 | 
			
		||||
          <GenerateSponsoringContracts
 | 
			
		||||
            bind:sponsoring_contracts_show
 | 
			
		||||
            bind:generate_runners
 | 
			
		||||
          />
 | 
			
		||||
          <GenerateRunnerCards bind:cards_show bind:generate_runners />
 | 
			
		||||
          <GenerateRunnerCertificates
 | 
			
		||||
            bind:certificates_show
 | 
			
		||||
            bind:generate_runners
 | 
			
		||||
          />
 | 
			
		||||
          {#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-runner")}</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"
 | 
			
		||||
            >{$_("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"
 | 
			
		||||
        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"
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <span class="font-medium text-gray-700">{$_("group")}</span>
 | 
			
		||||
      <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.id.value.toString().startsWith(filterText.toLowerCase())}
 | 
			
		||||
        items={groups}
 | 
			
		||||
        showChevron={true}
 | 
			
		||||
        placeholder={$_("search-for-an-organization-or-team-by-name-or-id")}
 | 
			
		||||
        noOptionsMessage={$_("no-organization-or-team-found")}
 | 
			
		||||
        bind:selectedValue={group}
 | 
			
		||||
        on:select={(selectedValue) => {
 | 
			
		||||
          editable.group = selectedValue.detail.value.id;
 | 
			
		||||
        }}
 | 
			
		||||
        on:clear={() => (editable.group = null)}
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="text-sm w-full">
 | 
			
		||||
      <span class="font-medium text-gray-700">{$_("distance")}</span>
 | 
			
		||||
      <br />
 | 
			
		||||
      <span class="text-gray-700">{original_data.distance / 1000} km</span>
 | 
			
		||||
    </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">
 | 
			
		||||
							<svg
 | 
			
		||||
								xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
								viewBox="0 0 24 24"
 | 
			
		||||
								class="flex-shrink-0 w-5 h-5 mr-2"
 | 
			
		||||
								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
 | 
			
		||||
							>
 | 
			
		||||
						</li>
 | 
			
		||||
						<li class="flex items-center">
 | 
			
		||||
							<a class="mr-2" href="./">{$_("runners")}</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
 | 
			
		||||
				class="grid md:grid-cols-3 gap-1 md:gap-2"
 | 
			
		||||
				data-id="runner_actions_${editable.id}"
 | 
			
		||||
			>
 | 
			
		||||
				{#if store.state.jwtinfo.userdetails.permissions.includes("RUNNER:DELETE")}
 | 
			
		||||
					<GenerateSponsoringContracts
 | 
			
		||||
						bind:sponsoring_contracts_show
 | 
			
		||||
						bind:generate_runners
 | 
			
		||||
					/>
 | 
			
		||||
					<GenerateRunnerCards bind:cards_show bind:generate_runners />
 | 
			
		||||
					<GenerateRunnerCertificates
 | 
			
		||||
						bind:certificates_show
 | 
			
		||||
						bind:generate_runners
 | 
			
		||||
					/>
 | 
			
		||||
					<div>
 | 
			
		||||
						<p class="text-base">Aktionen</p>
 | 
			
		||||
						{#if delete_triggered}
 | 
			
		||||
							<button
 | 
			
		||||
								on:click={deleteRunner}
 | 
			
		||||
								class="w-full justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm: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
 | 
			
		||||
							>
 | 
			
		||||
						{:else}
 | 
			
		||||
							<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-runner")}</button
 | 
			
		||||
							>
 | 
			
		||||
							<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"
 | 
			
		||||
								>{$_("save-changes")}</button
 | 
			
		||||
							>
 | 
			
		||||
						{/if}
 | 
			
		||||
					</div>
 | 
			
		||||
				{/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"
 | 
			
		||||
				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"
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full">
 | 
			
		||||
			<span class="font-medium text-gray-700">{$_("group")}</span>
 | 
			
		||||
			<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.id.value.toString().startsWith(filterText.toLowerCase())}
 | 
			
		||||
				items={groups}
 | 
			
		||||
				showChevron={true}
 | 
			
		||||
				placeholder={$_("search-for-an-organization-or-team-by-name-or-id")}
 | 
			
		||||
				noOptionsMessage={$_("no-organization-or-team-found")}
 | 
			
		||||
				bind:selectedValue={group}
 | 
			
		||||
				on:select={(selectedValue) => {
 | 
			
		||||
					editable.group = selectedValue.detail.value.id;
 | 
			
		||||
				}}
 | 
			
		||||
				on:clear={() => (editable.group = null)}
 | 
			
		||||
			/>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="text-sm w-full">
 | 
			
		||||
			<span class="font-medium text-gray-700">{$_("distance")}</span>
 | 
			
		||||
			<br />
 | 
			
		||||
			<span class="text-gray-700">{original_data.distance / 1000} km</span>
 | 
			
		||||
		</div>
 | 
			
		||||
	</section>
 | 
			
		||||
{:catch error}
 | 
			
		||||
  <PromiseError {error} />
 | 
			
		||||
	<PromiseError {error} />
 | 
			
		||||
{/await}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user