Merge branch 'dev' into feature/110-virtual_list
This commit is contained in:
		@@ -12,6 +12,7 @@
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
  import { tick } from "svelte";
 | 
			
		||||
  $: delete_triggered = false;
 | 
			
		||||
  $: address_valid_or_none =
 | 
			
		||||
@@ -34,6 +35,7 @@
 | 
			
		||||
  $: 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) =>
 | 
			
		||||
@@ -176,6 +178,7 @@
 | 
			
		||||
          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={() => {
 | 
			
		||||
@@ -186,7 +189,7 @@
 | 
			
		||||
            {$_('import-runners')}
 | 
			
		||||
          </button>
 | 
			
		||||
        {/if}
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('USER:DELETE')}
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:DELETE')}
 | 
			
		||||
          {#if delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
              on:click={deleteOrganization}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,212 +1,219 @@
 | 
			
		||||
<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}
 | 
			
		||||
<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";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.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.some((r) => r.is_selected === true);
 | 
			
		||||
  $: certificates_show = current_organizations.some(
 | 
			
		||||
    (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 />
 | 
			
		||||
        <GenerateRunnerCertificates
 | 
			
		||||
            bind:certificates_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}
 | 
			
		||||
 
 | 
			
		||||
@@ -67,7 +67,7 @@
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Runnercards.pdf";
 | 
			
		||||
                a.download = `${$_('runnercards')}-${locale}.pdf`;
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
@@ -127,7 +127,12 @@
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Runnercards.pdf";
 | 
			
		||||
                if(generate_runners.length == 1){
 | 
			
		||||
                    a.download = `${$_('runnercards')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
 | 
			
		||||
                }
 | 
			
		||||
                else{
 | 
			
		||||
                    a.download = `Runnercards-${locale}.pdf`;
 | 
			
		||||
                }
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
@@ -191,7 +196,7 @@
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + t.name + ".pdf";
 | 
			
		||||
                    a.download = `${$_('runnercards')}_${t.name}-${locale}.pdf`;
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
@@ -258,7 +263,7 @@
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + o.name + ".pdf";
 | 
			
		||||
                    a.download = `${$_('runnercards')}_${o.name}-${locale}.pdf`;
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										277
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										277
									
								
								src/components/pdf_generation/GenerateRunnerCertificates.svelte
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,277 @@
 | 
			
		||||
<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;
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    function generateCertificates(locale) {
 | 
			
		||||
        certificates_dropdown_open = false;
 | 
			
		||||
 | 
			
		||||
        if (generate_orgs.length > 0) {
 | 
			
		||||
            generateOrgCertificates(locale);
 | 
			
		||||
        } else if (generate_teams.length > 0) {
 | 
			
		||||
            generateTeamCertificates(locale);
 | 
			
		||||
        } else {
 | 
			
		||||
            generateRunnerCertificates(locale);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async function generateRunnerCertificates(locale) {
 | 
			
		||||
        const toast = Toastify({
 | 
			
		||||
            text: $_("generating-pdf"),
 | 
			
		||||
            duration: -1,
 | 
			
		||||
        }).showToast();
 | 
			
		||||
        const current_donations = await DonationService.donationControllerGetAll();
 | 
			
		||||
        let certificateRunners = [];
 | 
			
		||||
        for (let runner of generate_runners) {
 | 
			
		||||
            runner.distanceDonations = current_donations.find((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;
 | 
			
		||||
                if(generate_runners.length == 1){
 | 
			
		||||
                    a.download = `${$_('certificates')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
 | 
			
		||||
                }
 | 
			
		||||
                else{
 | 
			
		||||
                    a.download = `${$_('certificates')}-${locale}.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);
 | 
			
		||||
            }
 | 
			
		||||
            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}-${locale}.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
 | 
			
		||||
            );
 | 
			
		||||
            let certificateRunners = [];
 | 
			
		||||
            for (let runner of runners) {
 | 
			
		||||
                runner.distanceDonations = current_donations.find((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) => {
 | 
			
		||||
                    count++;
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = `${$_('certificates')}_${o.name}-${locale}.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) => {});
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
</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>
 | 
			
		||||
{/if}
 | 
			
		||||
@@ -69,7 +69,7 @@
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + t.name + ".pdf";
 | 
			
		||||
                    a.download = `${$_('sponsorings')}_${t.name}-${locale}.pdf`;
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
@@ -124,7 +124,7 @@
 | 
			
		||||
                    const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                    let a = document.createElement("a");
 | 
			
		||||
                    a.href = url;
 | 
			
		||||
                    a.download = "Sponsorings_" + o.name + ".pdf";
 | 
			
		||||
                    a.download = `${$_('sponsorings')}_${o.name}-${locale}.pdf`;
 | 
			
		||||
                    document.body.appendChild(a);
 | 
			
		||||
                    a.click();
 | 
			
		||||
                    a.remove();
 | 
			
		||||
@@ -174,7 +174,10 @@
 | 
			
		||||
                const url = window.URL.createObjectURL(blob);
 | 
			
		||||
                let a = document.createElement("a");
 | 
			
		||||
                a.href = url;
 | 
			
		||||
                a.download = "Sponsoring.pdf";
 | 
			
		||||
                if(generate_runners.length == 1){
 | 
			
		||||
                    a.download = `${$_('sponsorings')}_${generate_runners[0].firstname}_${generate_runners[0].lastname}-${locale}.pdf`;
 | 
			
		||||
                }
 | 
			
		||||
                a.download = `${$_('sponsorings')}-${locale}.pdf`;
 | 
			
		||||
                document.body.appendChild(a);
 | 
			
		||||
                a.click();
 | 
			
		||||
                a.remove();
 | 
			
		||||
 
 | 
			
		||||
@@ -34,7 +34,7 @@
 | 
			
		||||
    document.onkeydown = (e) => {
 | 
			
		||||
      e = e || window.event;
 | 
			
		||||
      if (e.key === "Escape") {
 | 
			
		||||
        import_modal_open = false;
 | 
			
		||||
        cancelModal();
 | 
			
		||||
      }
 | 
			
		||||
      if (e.keyCode === 13) {
 | 
			
		||||
        //
 | 
			
		||||
@@ -281,6 +281,16 @@
 | 
			
		||||
                  bind:files
 | 
			
		||||
                  type="file" />
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="overflow-hidden relative mt-4 mb-4">
 | 
			
		||||
                <button
 | 
			
		||||
                  on:click={() => {
 | 
			
		||||
                    cancelModal();
 | 
			
		||||
                  }}
 | 
			
		||||
                  type="button"
 | 
			
		||||
                  class="w-full rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 md:ml-40 mr-0 sm:ml-0 sm:w-auto sm:text-sm">
 | 
			
		||||
                  {$_('cancel')}
 | 
			
		||||
                </button>
 | 
			
		||||
              </div>
 | 
			
		||||
            {/if}
 | 
			
		||||
            {#if json_output.length > 0}
 | 
			
		||||
              {#if opened_from === 'OrgOverview'}
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,7 @@
 | 
			
		||||
  import { getLocaleFromNavigator, _ } 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,
 | 
			
		||||
@@ -36,6 +37,7 @@
 | 
			
		||||
    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;
 | 
			
		||||
@@ -158,7 +160,10 @@
 | 
			
		||||
            bind:sponsoring_contracts_show
 | 
			
		||||
            bind:generate_runners />
 | 
			
		||||
          <GenerateRunnerCards
 | 
			
		||||
            bind:sponsoring_contracts_show
 | 
			
		||||
            bind:cards_show
 | 
			
		||||
            bind:generate_runners />
 | 
			
		||||
          <GenerateRunnerCertificates
 | 
			
		||||
            bind:certificates_show
 | 
			
		||||
            bind:generate_runners />
 | 
			
		||||
          {#if !delete_triggered}
 | 
			
		||||
            <button
 | 
			
		||||
 
 | 
			
		||||
@@ -10,6 +10,7 @@
 | 
			
		||||
  import Select from "svelte-select";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
  $: searchvalue = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  export let current_runners = [];
 | 
			
		||||
@@ -27,6 +28,9 @@
 | 
			
		||||
  $: cards_show = current_runners.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: certificates_show = current_runners.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: generate_runners = current_runners.filter((r) => r.is_selected === true);
 | 
			
		||||
  $: teams = [];
 | 
			
		||||
  $: orgs = [];
 | 
			
		||||
@@ -92,6 +96,9 @@
 | 
			
		||||
        <GenerateRunnerCards
 | 
			
		||||
          bind:cards_show
 | 
			
		||||
          bind:generate_runners />
 | 
			
		||||
        <GenerateRunnerCertificates
 | 
			
		||||
          bind:certificates_show
 | 
			
		||||
          bind:generate_runners />
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="shadow border-b border-gray-200 sm:rounded-lg overflow-x-scroll">
 | 
			
		||||
 
 | 
			
		||||
@@ -14,6 +14,7 @@
 | 
			
		||||
  import Teams from "./Teams.svelte";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
  let [teamdata, original, delete_team, orgs, contacts, modal_open] = [
 | 
			
		||||
    {},
 | 
			
		||||
    {},
 | 
			
		||||
@@ -30,6 +31,7 @@
 | 
			
		||||
  $: data_changed = JSON.stringify(teamdata) === JSON.stringify(original);
 | 
			
		||||
  $: sponsoring_contracts_show = true;
 | 
			
		||||
  $: cards_show = true;
 | 
			
		||||
  $: certificates_show = true;
 | 
			
		||||
  $: generate_teams = [original];
 | 
			
		||||
  $: group = {};
 | 
			
		||||
  $: contact = {};
 | 
			
		||||
@@ -122,6 +124,9 @@
 | 
			
		||||
        <GenerateRunnerCards
 | 
			
		||||
          bind:cards_show
 | 
			
		||||
          bind:generate_teams />
 | 
			
		||||
        <GenerateRunnerCertificates
 | 
			
		||||
          bind:certificates_show
 | 
			
		||||
          bind:generate_teams />
 | 
			
		||||
        {#if store.state.jwtinfo.userdetails.permissions.includes('RUNNER:IMPORT')}
 | 
			
		||||
          <button
 | 
			
		||||
            on:click={() => {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,217 +1,224 @@
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, t, _ } from "svelte-i18n";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { RunnerTeamService } from "@odit/lfk-client-js";
 | 
			
		||||
  const teams_promise = RunnerTeamService.runnerTeamControllerGetAll();
 | 
			
		||||
  import store, { users as usersstore } from "../../store.js";
 | 
			
		||||
  import TeamsEmptyState from "./TeamsEmptyState.svelte";
 | 
			
		||||
  import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  $: searchvalue = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  $: sponsoring_contracts_show = current_teams.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: cards_show = current_teams.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: generate_teams = current_teams.filter((r) => r.is_selected === true);
 | 
			
		||||
  export let current_teams = [];
 | 
			
		||||
  let modal_open = false;
 | 
			
		||||
  let delete_team = {};
 | 
			
		||||
  usersstore.subscribe((val) => {
 | 
			
		||||
    current_teams = val;
 | 
			
		||||
  });
 | 
			
		||||
  teams_promise.then((data) => {
 | 
			
		||||
    usersstore.set(data);
 | 
			
		||||
  });
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<ConfirmTeamDeletion
 | 
			
		||||
  on:cancelDelete={(event) => {
 | 
			
		||||
    modal_open = false;
 | 
			
		||||
    active_deletes[event.detail.id] = false;
 | 
			
		||||
  }}
 | 
			
		||||
  bind:modal_open
 | 
			
		||||
  bind:delete_team />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')}
 | 
			
		||||
  {#await teams_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">{$_('teams-are-being-loaded')}</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_teams.length === 0}
 | 
			
		||||
      <TeamsEmptyState />
 | 
			
		||||
    {: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_teams />
 | 
			
		||||
        <GenerateRunnerCards
 | 
			
		||||
          bind:cards_show
 | 
			
		||||
          bind:generate_teams />
 | 
			
		||||
      </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_teams.some((r) => r.is_selected === true);
 | 
			
		||||
                    current_teams = current_teams.map((r) => {
 | 
			
		||||
                      r.is_selected = newstate;
 | 
			
		||||
                      return r;
 | 
			
		||||
                    });
 | 
			
		||||
                  }}
 | 
			
		||||
                  class="underline cursor-pointer select-none">{#if current_teams.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">
 | 
			
		||||
                {$_('organization')}
 | 
			
		||||
              </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_teams as t}
 | 
			
		||||
              {#if Object.values(t)
 | 
			
		||||
                .toString()
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .includes(searchvalue)}
 | 
			
		||||
                <tr data-rowid="team_{t.id}">
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    <input
 | 
			
		||||
                      bind:checked={t.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">
 | 
			
		||||
                          {t.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 t.parentGroup}
 | 
			
		||||
                            <a
 | 
			
		||||
                              href="../orgs/{t.parentGroup.id}"
 | 
			
		||||
                              class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a>
 | 
			
		||||
                          {:else}{$_('no-organization-specified')}{/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.contact}
 | 
			
		||||
                            <a
 | 
			
		||||
                              href="../contacts/{t.contact.id}"
 | 
			
		||||
                              class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname}
 | 
			
		||||
                              {t.contact.middlename || ''}
 | 
			
		||||
                              {t.contact.lastname}</a>
 | 
			
		||||
                          {:else}{$_('no-contact-specified')}{/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={() => {
 | 
			
		||||
                          RunnerTeamService.runnerTeamControllerRemove(t.id, false)
 | 
			
		||||
                            .then((resp) => {
 | 
			
		||||
                              current_teams = current_teams.filter((obj) => obj.id !== t.id);
 | 
			
		||||
                              Toastify({
 | 
			
		||||
                                text: $_('organization-deleted'),
 | 
			
		||||
                                duration: 500,
 | 
			
		||||
                                backgroundColor:
 | 
			
		||||
                                  'linear-gradient(to right, #00b09b, #96c93d)',
 | 
			
		||||
                              }).showToast();
 | 
			
		||||
                            })
 | 
			
		||||
                            .catch((err) => {
 | 
			
		||||
                              modal_open = true;
 | 
			
		||||
                              delete_team = t;
 | 
			
		||||
                            });
 | 
			
		||||
                        }}
 | 
			
		||||
                        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}
 | 
			
		||||
<script>
 | 
			
		||||
  import { getLocaleFromNavigator, t, _ } from "svelte-i18n";
 | 
			
		||||
  import Toastify from "toastify-js";
 | 
			
		||||
  import { RunnerTeamService } from "@odit/lfk-client-js";
 | 
			
		||||
  const teams_promise = RunnerTeamService.runnerTeamControllerGetAll();
 | 
			
		||||
  import store, { users as usersstore } from "../../store.js";
 | 
			
		||||
  import TeamsEmptyState from "./TeamsEmptyState.svelte";
 | 
			
		||||
  import ConfirmTeamDeletion from "./ConfirmTeamDeletion.svelte";
 | 
			
		||||
  import { clickOutside } from "../base/outsideclick";
 | 
			
		||||
  import GenerateSponsoringContracts from "../pdf_generation/GenerateSponsoringContracts.svelte";
 | 
			
		||||
  import GenerateRunnerCards from "../pdf_generation/GenerateRunnerCards.svelte";
 | 
			
		||||
  import GenerateRunnerCertificates from "../pdf_generation/GenerateRunnerCertificates.svelte";
 | 
			
		||||
  $: searchvalue = "";
 | 
			
		||||
  $: active_deletes = [];
 | 
			
		||||
  $: sponsoring_contracts_show = current_teams.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: cards_show = current_teams.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: certificates_show = current_teams.some(
 | 
			
		||||
    (r) => r.is_selected === true
 | 
			
		||||
  );
 | 
			
		||||
  $: generate_teams = current_teams.filter((r) => r.is_selected === true);
 | 
			
		||||
  export let current_teams = [];
 | 
			
		||||
  let modal_open = false;
 | 
			
		||||
  let delete_team = {};
 | 
			
		||||
  usersstore.subscribe((val) => {
 | 
			
		||||
    current_teams = val;
 | 
			
		||||
  });
 | 
			
		||||
  teams_promise.then((data) => {
 | 
			
		||||
    usersstore.set(data);
 | 
			
		||||
  });
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<ConfirmTeamDeletion
 | 
			
		||||
  on:cancelDelete={(event) => {
 | 
			
		||||
    modal_open = false;
 | 
			
		||||
    active_deletes[event.detail.id] = false;
 | 
			
		||||
  }}
 | 
			
		||||
  bind:modal_open
 | 
			
		||||
  bind:delete_team />
 | 
			
		||||
{#if store.state.jwtinfo.userdetails.permissions.includes('TEAM:GET')}
 | 
			
		||||
  {#await teams_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">{$_('teams-are-being-loaded')}</p>
 | 
			
		||||
      <p class="text-sm">{$_('this-might-take-a-moment')}</p>
 | 
			
		||||
    </div>
 | 
			
		||||
  {:then}
 | 
			
		||||
    {#if current_teams.length === 0}
 | 
			
		||||
      <TeamsEmptyState />
 | 
			
		||||
    {: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_teams />
 | 
			
		||||
        <GenerateRunnerCards
 | 
			
		||||
          bind:cards_show
 | 
			
		||||
          bind:generate_teams />
 | 
			
		||||
        <GenerateRunnerCertificates
 | 
			
		||||
          bind:certificates_show
 | 
			
		||||
          bind:generate_teams />
 | 
			
		||||
      </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_teams.some((r) => r.is_selected === true);
 | 
			
		||||
                    current_teams = current_teams.map((r) => {
 | 
			
		||||
                      r.is_selected = newstate;
 | 
			
		||||
                      return r;
 | 
			
		||||
                    });
 | 
			
		||||
                  }}
 | 
			
		||||
                  class="underline cursor-pointer select-none">{#if current_teams.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">
 | 
			
		||||
                {$_('organization')}
 | 
			
		||||
              </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_teams as t}
 | 
			
		||||
              {#if Object.values(t)
 | 
			
		||||
                .toString()
 | 
			
		||||
                .toLowerCase()
 | 
			
		||||
                .includes(searchvalue)}
 | 
			
		||||
                <tr data-rowid="team_{t.id}">
 | 
			
		||||
                  <td class="px-6 py-4 whitespace-nowrap">
 | 
			
		||||
                    <input
 | 
			
		||||
                      bind:checked={t.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">
 | 
			
		||||
                          {t.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 t.parentGroup}
 | 
			
		||||
                            <a
 | 
			
		||||
                              href="../orgs/{t.parentGroup.id}"
 | 
			
		||||
                              class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.parentGroup.name}</a>
 | 
			
		||||
                          {:else}{$_('no-organization-specified')}{/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.contact}
 | 
			
		||||
                            <a
 | 
			
		||||
                              href="../contacts/{t.contact.id}"
 | 
			
		||||
                              class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">{t.contact.firstname}
 | 
			
		||||
                              {t.contact.middlename || ''}
 | 
			
		||||
                              {t.contact.lastname}</a>
 | 
			
		||||
                          {:else}{$_('no-contact-specified')}{/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={() => {
 | 
			
		||||
                          RunnerTeamService.runnerTeamControllerRemove(t.id, false)
 | 
			
		||||
                            .then((resp) => {
 | 
			
		||||
                              current_teams = current_teams.filter((obj) => obj.id !== t.id);
 | 
			
		||||
                              Toastify({
 | 
			
		||||
                                text: $_('organization-deleted'),
 | 
			
		||||
                                duration: 500,
 | 
			
		||||
                                backgroundColor:
 | 
			
		||||
                                  'linear-gradient(to right, #00b09b, #96c93d)',
 | 
			
		||||
                              }).showToast();
 | 
			
		||||
                            })
 | 
			
		||||
                            .catch((err) => {
 | 
			
		||||
                              modal_open = true;
 | 
			
		||||
                              delete_team = t;
 | 
			
		||||
                            });
 | 
			
		||||
                        }}
 | 
			
		||||
                        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}
 | 
			
		||||
 
 | 
			
		||||
@@ -50,6 +50,7 @@
 | 
			
		||||
    "card-deleted": "Karte gelöscht",
 | 
			
		||||
    "card-updated": "Karte aktualisiert",
 | 
			
		||||
    "cards": "Läuferkarten",
 | 
			
		||||
    "certificates": "Urkunden",
 | 
			
		||||
    "change-your-password-here": "Hier kannst du dein Passwort ändern",
 | 
			
		||||
    "changing-your-password": "Passwort wird geändert",
 | 
			
		||||
    "city": "Stadt",
 | 
			
		||||
@@ -191,10 +192,12 @@
 | 
			
		||||
    "geerbte": "geerbte",
 | 
			
		||||
    "general-stats": "Allgemeine Statistiken",
 | 
			
		||||
    "general_promise_error": "😢 Ein unbekannter Fehler ist aufgetreten",
 | 
			
		||||
    "generate-runner-certificate": "Urkunde generieren",
 | 
			
		||||
    "generate-runner-certificates": "Urkunden generieren",
 | 
			
		||||
    "generate-runnercards": "Läuferkarten generieren",
 | 
			
		||||
    "generate-sponsoring-contract": "Sponsoringvertrag generieren",
 | 
			
		||||
    "generate-sponsoring-contracts": "Sponsoringverträge generieren",
 | 
			
		||||
    "generating-pdf": "Pdf wird generiert...",
 | 
			
		||||
    "generating-pdf": "PDF wird generiert...",
 | 
			
		||||
    "generating-pdfs": "PDFs werden generiert...",
 | 
			
		||||
    "generic-ui-logic-error": "Etwas ist in der Benutzeroberfläche schiefgelaufen.",
 | 
			
		||||
    "german": "Deutsch",
 | 
			
		||||
@@ -321,6 +324,7 @@
 | 
			
		||||
    "runner-import": "Läufer:innen Import",
 | 
			
		||||
    "runner-is-being-added": "Läufer:in wird hinzugefügt...",
 | 
			
		||||
    "runner-updated": "Läufer:in aktualisiert!",
 | 
			
		||||
    "runnercards": "Laeuferkarten",
 | 
			
		||||
    "runnerimport_verify_runners_org": "Bitte die Läufer:innen für den Import in die Organisation \"{org_name}\" bestätigen",
 | 
			
		||||
    "runners": "Läufer",
 | 
			
		||||
    "runners-are-being-imported": "Läufer:innen werden importiert ...",
 | 
			
		||||
@@ -350,6 +354,7 @@
 | 
			
		||||
    "settings": "Einstellungen",
 | 
			
		||||
    "settings-for-your-profile": "Die Einstellungen deines Accounts",
 | 
			
		||||
    "something-about-the-group": "Infos zur Gruppe",
 | 
			
		||||
    "sponsorings": "Sponsoringerklaerungen",
 | 
			
		||||
    "stats-are-being-loaded": "Die Statistiken werden geladen...",
 | 
			
		||||
    "status": "Status",
 | 
			
		||||
    "stuff-that-could-harm-your-profile": "Einstellungen, die deinem Profil nachhaltig schaden können",
 | 
			
		||||
 
 | 
			
		||||
@@ -50,6 +50,7 @@
 | 
			
		||||
    "card-deleted": "Card deleted",
 | 
			
		||||
    "card-updated": "Card updated",
 | 
			
		||||
    "cards": "Cards",
 | 
			
		||||
    "certificates": "Certificates",
 | 
			
		||||
    "change-your-password-here": "Change your password here",
 | 
			
		||||
    "changing-your-password": "Changing your password",
 | 
			
		||||
    "city": "City",
 | 
			
		||||
@@ -191,6 +192,8 @@
 | 
			
		||||
    "geerbte": "inherited",
 | 
			
		||||
    "general-stats": "General Stats",
 | 
			
		||||
    "general_promise_error": "😢 Error",
 | 
			
		||||
    "generate-runner-certificate": "Generate runner certificate",
 | 
			
		||||
    "generate-runner-certificates": "Generate runner certificates",
 | 
			
		||||
    "generate-runnercards": "Generate Runnercards",
 | 
			
		||||
    "generate-sponsoring-contract": "generate sponsoring contract",
 | 
			
		||||
    "generate-sponsoring-contracts": "generate sponsoring contracts",
 | 
			
		||||
@@ -321,6 +324,7 @@
 | 
			
		||||
    "runner-import": "Runner Import",
 | 
			
		||||
    "runner-is-being-added": "Runner is being added...",
 | 
			
		||||
    "runner-updated": "Runner updated!",
 | 
			
		||||
    "runnercards": "Runnercards",
 | 
			
		||||
    "runnerimport_verify_runners_org": "Please confirm these runners for import into the organization \"{org_name}\"",
 | 
			
		||||
    "runners": "Runners",
 | 
			
		||||
    "runners-are-being-imported": "Runners are being imported...",
 | 
			
		||||
@@ -350,6 +354,7 @@
 | 
			
		||||
    "settings": "Settings",
 | 
			
		||||
    "settings-for-your-profile": "Settings for your profile",
 | 
			
		||||
    "something-about-the-group": "Something about the group...",
 | 
			
		||||
    "sponsorings": "Sponsorings",
 | 
			
		||||
    "stats-are-being-loaded": "stats are being loaded...",
 | 
			
		||||
    "status": "Status",
 | 
			
		||||
    "stuff-that-could-harm-your-profile": "Stuff that could harm your profile",
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user