Compare commits
	
		
			No commits in common. "12dbf7227b564f1353de68ec2f9c0caa57003a89" and "5f396ccaf2ab025488039a0e19379776e5b1c0cf" have entirely different histories.
		
	
	
		
			
				12dbf7227b
			
			...
			
				5f396ccaf2
			
		
	
		
					 18 changed files with 133 additions and 153 deletions
				
			
		|  | @ -33,5 +33,4 @@ EXECUTE PROCEDURE update_entry_tsvec (); | ||||||
| ALTER TABLE patients | ALTER TABLE patients | ||||||
| ADD COLUMN full_name TEXT GENERATED ALWAYS AS (first_name || ' ' || last_name) STORED; | ADD COLUMN full_name TEXT GENERATED ALWAYS AS (first_name || ' ' || last_name) STORED; | ||||||
| 
 | 
 | ||||||
| CREATE INDEX entries_tsvec ON entries USING GIN (tsvec); |  | ||||||
| CREATE INDEX patients_full_name ON patients USING gin (full_name gin_trgm_ops); | CREATE INDEX patients_full_name ON patients USING gin (full_name gin_trgm_ops); | ||||||
|  |  | ||||||
|  | @ -6,7 +6,7 @@ | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1" /> |     <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||||
|     %sveltekit.head% |     %sveltekit.head% | ||||||
|   </head> |   </head> | ||||||
|   <body data-sveltekit-preload-data="tap"> |   <body data-sveltekit-preload-data="hover"> | ||||||
|     <div style="display: contents">%sveltekit.body%</div> |     <div style="display: contents">%sveltekit.body%</div> | ||||||
|   </body> |   </body> | ||||||
| </html> | </html> | ||||||
|  |  | ||||||
|  | @ -23,7 +23,7 @@ | ||||||
|   export let placeholder: string | undefined = undefined; |   export let placeholder: string | undefined = undefined; | ||||||
|   export let padding = true; |   export let padding = true; | ||||||
|   export let cls = ""; |   export let cls = ""; | ||||||
|   export let inputCls = "w-full bg-transparent outline-none"; |   export let inputCls = "w-full bg-transparent"; | ||||||
|   export let asTextInput = false; |   export let asTextInput = false; | ||||||
|   export let idInputName: string | undefined = undefined; |   export let idInputName: string | undefined = undefined; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -21,7 +21,6 @@ | ||||||
|   export let onUpdate: (filterData: FilterQdata | undefined) => void = () => {}; |   export let onUpdate: (filterData: FilterQdata | undefined) => void = () => {}; | ||||||
|   /** List of hidden filter IDs, can be specified for prefiltered views (e.g. by patient) */ |   /** List of hidden filter IDs, can be specified for prefiltered views (e.g. by patient) */ | ||||||
|   export let hiddenFilters: string[] = []; |   export let hiddenFilters: string[] = []; | ||||||
|   /** True if a separate search field should be displayed */ |  | ||||||
|   export let search = false; |   export let search = false; | ||||||
| 
 | 
 | ||||||
|   let autocomplete: Autocomplete | undefined; |   let autocomplete: Autocomplete | undefined; | ||||||
|  | @ -60,9 +59,7 @@ | ||||||
|   function updateFromQueryData(filterData: FilterQdata) { |   function updateFromQueryData(filterData: FilterQdata) { | ||||||
|     const filters: FilterData[] = []; |     const filters: FilterData[] = []; | ||||||
|     for (const [id, value] of Object.entries(filterData)) { |     for (const [id, value] of Object.entries(filterData)) { | ||||||
|       // If filter is hidden or undefined, dont display it |       if (hiddenFilters.includes(id)) continue; | ||||||
|       if (hiddenFilters.includes(id) || !FILTERS[id]) continue; |  | ||||||
|       // Extract search parameter if a separate search field is used |  | ||||||
|       if (search && id === "search") { |       if (search && id === "search") { | ||||||
|         searchVal = value.toString(); |         searchVal = value.toString(); | ||||||
|       } else if (Array.isArray(value)) { |       } else if (Array.isArray(value)) { | ||||||
|  | @ -186,8 +183,10 @@ | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <div class="filterbar-outer"> | <div class="flex flex-wrap w-full items-start justify-center gap-2 relative"> | ||||||
|   <div class="filterbar-inner input input-sm input-bordered"> |   <div | ||||||
|  |     class="flex flex-wrap flex-grow items-stretch h-auto p-0 gap-2 input input-sm input-bordered relative" | ||||||
|  |   > | ||||||
|     {#each activeFilters as fdata, i} |     {#each activeFilters as fdata, i} | ||||||
|       <EntryFilterChip |       <EntryFilterChip | ||||||
|         filter={FILTERS[fdata.id]} |         filter={FILTERS[fdata.id]} | ||||||
|  | @ -222,7 +221,6 @@ | ||||||
|       aria-label="Alle Filter entfernen" |       aria-label="Alle Filter entfernen" | ||||||
|       on:click={() => { |       on:click={() => { | ||||||
|         activeFilters = []; |         activeFilters = []; | ||||||
|         searchVal = ""; |  | ||||||
|         updateFilter(); |         updateFilter(); | ||||||
|       }} |       }} | ||||||
|     > |     > | ||||||
|  | @ -241,14 +239,3 @@ | ||||||
|   {/if} |   {/if} | ||||||
|   <slot /> |   <slot /> | ||||||
| </div> | </div> | ||||||
| 
 |  | ||||||
| <style lang="postcss"> |  | ||||||
|   .filterbar-outer { |  | ||||||
|     @apply flex flex-wrap w-full items-start justify-center gap-2 relative; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .filterbar-inner { |  | ||||||
|     @apply flex flex-wrap flex-grow items-stretch h-min p-0 gap-2 relative; |  | ||||||
|     line-height: 30px; |  | ||||||
|   } |  | ||||||
| </style> |  | ||||||
|  |  | ||||||
|  | @ -32,13 +32,9 @@ | ||||||
|     autocomplete.open(); |     autocomplete.open(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const TOFF = " aus"; |  | ||||||
| 
 |  | ||||||
|   $: toggleState = fdata.selection?.toggle !== false; |   $: toggleState = fdata.selection?.toggle !== false; | ||||||
|   $: filterName = toggleState |   $: filterName = toggleState ? filter.name : filter.toggleOff?.name; | ||||||
|     ? filter.name |   $: filterIcon = toggleState ? filter.icon : filter.toggleOff?.icon; | ||||||
|     : filter.toggleOff?.name ?? filter.name + TOFF; |  | ||||||
|   $: filterIcon = toggleState ? filter.icon : filter.toggleOff?.icon ?? filter.icon; |  | ||||||
|   $: hasInputField = filter.inputType !== 0 && filter.inputType !== 3; |   $: hasInputField = filter.inputType !== 0 && filter.inputType !== 3; | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ import { trpc } from "$lib/shared/trpc"; | ||||||
| import { | import { | ||||||
|   mdiAccount, |   mdiAccount, | ||||||
|   mdiAccountInjury, |   mdiAccountInjury, | ||||||
|   mdiAccountMultipleOutline, |  | ||||||
|   mdiAccountRemoveOutline, |   mdiAccountRemoveOutline, | ||||||
|   mdiBedKingOutline, |   mdiBedKingOutline, | ||||||
|   mdiCheckboxBlankOutline, |   mdiCheckboxBlankOutline, | ||||||
|  | @ -119,10 +118,4 @@ export const PATIENT_FILTER: { [key: string]: FilterDef } = { | ||||||
|     icon: mdiAccountRemoveOutline, |     icon: mdiAccountRemoveOutline, | ||||||
|     inputType: 0, |     inputType: 0, | ||||||
|   }, |   }, | ||||||
|   includeHidden: { |  | ||||||
|     id: "includeHidden", |  | ||||||
|     name: "Alle anzeigen", |  | ||||||
|     icon: mdiAccountMultipleOutline, |  | ||||||
|     inputType: 0, |  | ||||||
|   }, |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -6,7 +6,6 @@ | ||||||
|   import { superForm, superValidateSync } from "sveltekit-superforms/client"; |   import { superForm, superValidateSync } from "sveltekit-superforms/client"; | ||||||
|   import type { SuperValidated } from "sveltekit-superforms"; |   import type { SuperValidated } from "sveltekit-superforms"; | ||||||
|   import { ZPatientNew } from "$lib/shared/model/validation"; |   import { ZPatientNew } from "$lib/shared/model/validation"; | ||||||
|   import { browser } from "$app/environment"; |  | ||||||
| 
 | 
 | ||||||
|   export let patient: RouterOutput["patient"]["get"] | null = null; |   export let patient: RouterOutput["patient"]["get"] | null = null; | ||||||
|   export let formData: SuperValidated<typeof ZPatientNew> = |   export let formData: SuperValidated<typeof ZPatientNew> = | ||||||
|  | @ -92,14 +91,11 @@ | ||||||
|     </label> |     </label> | ||||||
|   </div> |   </div> | ||||||
| 
 | 
 | ||||||
|   <div class="flex flex-wrap gap-2"> |   <button | ||||||
|     <button |     class="btn btn-primary max-w-32" | ||||||
|       class="btn btn-primary max-w-32" |     type="submit" | ||||||
|       type="submit" |     disabled={$tainted === undefined} | ||||||
|       disabled={browser && $tainted === undefined} |   > | ||||||
|     > |     Speichern | ||||||
|       Speichern |   </button> | ||||||
|     </button> |  | ||||||
|     <slot /> |  | ||||||
|   </div> |  | ||||||
| </form> | </form> | ||||||
|  |  | ||||||
|  | @ -4,9 +4,11 @@ | ||||||
|   import FilterBar from "$lib/components/filter/FilterBar.svelte"; |   import FilterBar from "$lib/components/filter/FilterBar.svelte"; | ||||||
|   import type { PaginationRequest, SortRequest } from "$lib/shared/model"; |   import type { PaginationRequest, SortRequest } from "$lib/shared/model"; | ||||||
|   import type { FilterQdata } from "$lib/components/filter/types"; |   import type { FilterQdata } from "$lib/components/filter/types"; | ||||||
|   import { type RouterOutput } from "$lib/shared/trpc"; |   import { trpc, type RouterOutput } from "$lib/shared/trpc"; | ||||||
|   import { ENTRY_FILTERS } from "$lib/components/filter/filters"; |   import { ENTRY_FILTERS } from "$lib/components/filter/filters"; | ||||||
|   import { getQueryUrl } from "$lib/shared/util"; |   import { getQueryUrl } from "$lib/shared/util"; | ||||||
|  |   import LoadingBar from "$lib/components/ui/LoadingBar.svelte"; | ||||||
|  |   import ErrorMessage from "$lib/components/ui/ErrorMessage.svelte"; | ||||||
|   import type { ZEntriesQuery } from "$lib/shared/model/validation"; |   import type { ZEntriesQuery } from "$lib/shared/model/validation"; | ||||||
|   import { z } from "zod"; |   import { z } from "zod"; | ||||||
|   import { browser } from "$app/environment"; |   import { browser } from "$app/environment"; | ||||||
|  | @ -17,6 +19,9 @@ | ||||||
|   export let baseUrl: string; |   export let baseUrl: string; | ||||||
|   export let patientId: number | null = null; |   export let patientId: number | null = null; | ||||||
| 
 | 
 | ||||||
|  |   let loadingBar: LoadingBar | undefined; | ||||||
|  |   let loadError: Error | null = null; | ||||||
|  | 
 | ||||||
|   function paginationUpdate(pagination: PaginationRequest) { |   function paginationUpdate(pagination: PaginationRequest) { | ||||||
|     updateQuery({ |     updateQuery({ | ||||||
|       filter: query.filter, |       filter: query.filter, | ||||||
|  | @ -44,6 +49,29 @@ | ||||||
|       // Update page URL |       // Update page URL | ||||||
|       const url = getQueryUrl(q, baseUrl); |       const url = getQueryUrl(q, baseUrl); | ||||||
|       goto(url, { replaceState: true, keepFocus: true }); |       goto(url, { replaceState: true, keepFocus: true }); | ||||||
|  | 
 | ||||||
|  |       // Apply patient filter | ||||||
|  |       if (patientId !== null) { | ||||||
|  |         if (!q.filter) q.filter = {}; | ||||||
|  |         q.filter.patient = [{ id: patientId }]; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       loadingBar?.start(); | ||||||
|  |       trpc() | ||||||
|  |         .entry.list.query(q) | ||||||
|  |         .then((ent) => { | ||||||
|  |           entries = ent; | ||||||
|  |           loadError = null; | ||||||
|  |           loadingBar?.reset(); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           query = q; | ||||||
|  |           loadError = err; | ||||||
|  |           loadingBar?.error(); | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           query = q; | ||||||
|  |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  | @ -57,16 +85,22 @@ | ||||||
|   <slot /> |   <slot /> | ||||||
| </FilterBar> | </FilterBar> | ||||||
| 
 | 
 | ||||||
| <EntryTable | <LoadingBar bind:this={loadingBar} alwaysShown /> | ||||||
|   {entries} |  | ||||||
|   sortData={query.sort} |  | ||||||
|   {sortUpdate} |  | ||||||
|   perPatient={patientId !== null} |  | ||||||
|   {baseUrl} |  | ||||||
| /> |  | ||||||
| 
 | 
 | ||||||
| <PaginationButtons | {#if loadError} | ||||||
|   paginationData={query.pagination} |   <ErrorMessage error={loadError} /> | ||||||
|   data={entries} | {:else} | ||||||
|   onUpdate={paginationUpdate} |   <EntryTable | ||||||
| /> |     {entries} | ||||||
|  |     sortData={query.sort} | ||||||
|  |     {sortUpdate} | ||||||
|  |     perPatient={patientId !== null} | ||||||
|  |     {baseUrl} | ||||||
|  |   /> | ||||||
|  | 
 | ||||||
|  |   <PaginationButtons | ||||||
|  |     paginationData={query.pagination} | ||||||
|  |     data={entries} | ||||||
|  |     onUpdate={paginationUpdate} | ||||||
|  |   /> | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | @ -4,9 +4,11 @@ | ||||||
|   import FilterBar from "$lib/components/filter/FilterBar.svelte"; |   import FilterBar from "$lib/components/filter/FilterBar.svelte"; | ||||||
|   import type { PaginationRequest, SortRequest } from "$lib/shared/model"; |   import type { PaginationRequest, SortRequest } from "$lib/shared/model"; | ||||||
|   import type { FilterQdata } from "$lib/components/filter/types"; |   import type { FilterQdata } from "$lib/components/filter/types"; | ||||||
|   import { type RouterOutput } from "$lib/shared/trpc"; |   import { trpc, type RouterOutput } from "$lib/shared/trpc"; | ||||||
|   import { PATIENT_FILTER } from "$lib/components/filter/filters"; |   import { PATIENT_FILTER } from "$lib/components/filter/filters"; | ||||||
|   import { getQueryUrl } from "$lib/shared/util"; |   import { getQueryUrl } from "$lib/shared/util"; | ||||||
|  |   import LoadingBar from "$lib/components/ui/LoadingBar.svelte"; | ||||||
|  |   import ErrorMessage from "$lib/components/ui/ErrorMessage.svelte"; | ||||||
|   import type { ZPatientsQuery } from "$lib/shared/model/validation"; |   import type { ZPatientsQuery } from "$lib/shared/model/validation"; | ||||||
|   import { z } from "zod"; |   import { z } from "zod"; | ||||||
|   import { browser } from "$app/environment"; |   import { browser } from "$app/environment"; | ||||||
|  | @ -16,6 +18,9 @@ | ||||||
|   export let patients: RouterOutput["patient"]["list"]; |   export let patients: RouterOutput["patient"]["list"]; | ||||||
|   export let baseUrl: string; |   export let baseUrl: string; | ||||||
| 
 | 
 | ||||||
|  |   let loadingBar: LoadingBar | undefined; | ||||||
|  |   let loadError: Error | null = null; | ||||||
|  | 
 | ||||||
|   function paginationUpdate(pagination: PaginationRequest) { |   function paginationUpdate(pagination: PaginationRequest) { | ||||||
|     updateQuery({ |     updateQuery({ | ||||||
|       filter: query.filter, |       filter: query.filter, | ||||||
|  | @ -41,6 +46,22 @@ | ||||||
|       // Update page URL |       // Update page URL | ||||||
|       const url = getQueryUrl(q, baseUrl); |       const url = getQueryUrl(q, baseUrl); | ||||||
|       goto(url, { replaceState: true, keepFocus: true }); |       goto(url, { replaceState: true, keepFocus: true }); | ||||||
|  | 
 | ||||||
|  |       loadingBar?.start(); | ||||||
|  |       trpc() | ||||||
|  |         .patient.list.query(q) | ||||||
|  |         .then((p) => { | ||||||
|  |           patients = p; | ||||||
|  |           loadError = null; | ||||||
|  |           loadingBar?.reset(); | ||||||
|  |         }) | ||||||
|  |         .catch((err) => { | ||||||
|  |           loadError = err; | ||||||
|  |           loadingBar?.error(); | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           query = q; | ||||||
|  |         }); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
|  | @ -54,10 +75,16 @@ | ||||||
|   <slot /> |   <slot /> | ||||||
| </FilterBar> | </FilterBar> | ||||||
| 
 | 
 | ||||||
| <PatientTable {patients} sortData={query.sort} {sortUpdate} {baseUrl} /> | <LoadingBar bind:this={loadingBar} alwaysShown /> | ||||||
| 
 | 
 | ||||||
| <PaginationButtons | {#if loadError} | ||||||
|   paginationData={query.pagination} |   <ErrorMessage error={loadError} /> | ||||||
|   data={patients} | {:else} | ||||||
|   onUpdate={paginationUpdate} |   <PatientTable {patients} sortData={query.sort} {sortUpdate} {baseUrl} /> | ||||||
| /> | 
 | ||||||
|  |   <PaginationButtons | ||||||
|  |     paginationData={query.pagination} | ||||||
|  |     data={patients} | ||||||
|  |     onUpdate={paginationUpdate} | ||||||
|  |   /> | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | @ -2,7 +2,7 @@ | ||||||
|   import type { RouterOutput } from "$lib/shared/trpc"; |   import type { RouterOutput } from "$lib/shared/trpc"; | ||||||
|   import { gotoEntityQuery } from "$lib/shared/util"; |   import { gotoEntityQuery } from "$lib/shared/util"; | ||||||
| 
 | 
 | ||||||
|   export let patient: RouterOutput["patient"]["list"]["items"][0]; |   export let patient: RouterOutput["patient"]["get"]; | ||||||
|   export let baseUrl: string; |   export let baseUrl: string; | ||||||
| 
 | 
 | ||||||
|   function onClick(e: MouseEvent) { |   function onClick(e: MouseEvent) { | ||||||
|  |  | ||||||
|  | @ -1,13 +1,12 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import type { SortRequest } from "$lib/shared/model"; |   import type { SortRequest } from "$lib/shared/model"; | ||||||
|   import type { RouterOutput } from "$lib/shared/trpc"; |   import type { RouterOutput } from "$lib/shared/trpc"; | ||||||
|   import { formatDate, gotoEntityQuery } from "$lib/shared/util"; |   import { formatDate } from "$lib/shared/util"; | ||||||
|   import { mdiClose, mdiFilter } from "@mdi/js"; |   import { mdiClose } from "@mdi/js"; | ||||||
|   import Icon from "$lib/components/ui/Icon.svelte"; |   import Icon from "$lib/components/ui/Icon.svelte"; | ||||||
| 
 | 
 | ||||||
|   import RoomField from "./RoomField.svelte"; |   import RoomField from "./RoomField.svelte"; | ||||||
|   import SortHeader from "./SortHeader.svelte"; |   import SortHeader from "./SortHeader.svelte"; | ||||||
|   import { URL_ENTRIES } from "$lib/shared/constants"; |  | ||||||
| 
 | 
 | ||||||
|   export let patients: RouterOutput["patient"]["list"]; |   export let patients: RouterOutput["patient"]["list"]; | ||||||
|   export let sortData: SortRequest | undefined = undefined; |   export let sortData: SortRequest | undefined = undefined; | ||||||
|  | @ -29,10 +28,9 @@ | ||||||
|     </thead> |     </thead> | ||||||
|     <tbody> |     <tbody> | ||||||
|       {#each patients.items as patient} |       {#each patients.items as patient} | ||||||
|         {@const full_name = patient.first_name + " " + patient.last_name} |  | ||||||
|         <tr |         <tr | ||||||
|           class="transition-colors hover:bg-neutral-content/10" |           class="transition-colors hover:bg-neutral-content/10" | ||||||
|           class:p-hidden={patient.hidden} |           class:hidden={patient.hidden} | ||||||
|         > |         > | ||||||
|           <td |           <td | ||||||
|             ><a |             ><a | ||||||
|  | @ -41,29 +39,18 @@ | ||||||
|               aria-label="Eintrag anzeigen">{patient.id}</a |               aria-label="Eintrag anzeigen">{patient.id}</a | ||||||
|             ></td |             ></td | ||||||
|           > |           > | ||||||
|           <td>{full_name}</td> |           <td>{patient.first_name} {patient.last_name}</td> | ||||||
|           <td>{patient.age ?? ""}</td> |           <td>{patient.age}</td> | ||||||
|           <td> |           <td> | ||||||
|             {#if patient.room} |             {#if patient.room} | ||||||
|               <RoomField room={patient.room} {baseUrl} /> |               <RoomField room={patient.room} {baseUrl} /> | ||||||
|             {/if} |             {/if} | ||||||
|           </td> |           </td> | ||||||
|           <td>{formatDate(patient.created_at, true)}</td> |           <td>{formatDate(patient.created_at, true)}</td> | ||||||
|           <td class="text-right"> |           <td> | ||||||
|             <button |             <button class="btn btn-circle btn-ghost btn-xs" | ||||||
|               class="btn btn-circle btn-ghost btn-xs inline" |               ><Icon size={1.2} path={mdiClose} /></button | ||||||
|               on:click={() => { |  | ||||||
|                 gotoEntityQuery( |  | ||||||
|                   { filter: { patient: [{ id: patient.id, name: full_name }] } }, |  | ||||||
|                   URL_ENTRIES |  | ||||||
|                 ); |  | ||||||
|               }} |  | ||||||
|             > |             > | ||||||
|               <Icon size={1.2} path={mdiFilter} /> |  | ||||||
|             </button> |  | ||||||
|             <button class="btn btn-circle btn-ghost btn-xs inline"> |  | ||||||
|               <Icon size={1.2} path={mdiClose} /> |  | ||||||
|             </button> |  | ||||||
|           </td> |           </td> | ||||||
|         </tr> |         </tr> | ||||||
|       {/each} |       {/each} | ||||||
|  | @ -72,7 +59,7 @@ | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| <style lang="postcss"> | <style lang="postcss"> | ||||||
|   .p-hidden { |   .hidden { | ||||||
|     @apply bg-red-500/20; |     @apply bg-red-500/20; | ||||||
|   } |   } | ||||||
| </style> | </style> | ||||||
|  |  | ||||||
|  | @ -64,10 +64,6 @@ export async function getPatientNames(): Promise<PatientTag[]> { | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function getPatientNEntries(id: number): Promise<number> { |  | ||||||
|   return prisma.entry.count({ where: { patient_id: id } }); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export async function getPatients( | export async function getPatients( | ||||||
|   filter: PatientsFilter = {}, |   filter: PatientsFilter = {}, | ||||||
|   pagination: PaginationRequest = {}, |   pagination: PaginationRequest = {}, | ||||||
|  | @ -90,10 +86,8 @@ export async function getPatients( | ||||||
|     qb.addOrderClause(`similarity(p.full_name, ${qvar}) desc`); |     qb.addOrderClause(`similarity(p.full_name, ${qvar}) desc`); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (filter.hidden) { |   if (filter.hidden !== undefined) { | ||||||
|     qb.addFilter("p.hidden", true); |     qb.addFilter("p.hidden", filter.hidden); | ||||||
|   } else if (!filter.includeHidden) { |  | ||||||
|     qb.addFilter("p.hidden", false); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   qb.addFilterList("r.id", filter.room); |   qb.addFilterList("r.id", filter.room); | ||||||
|  |  | ||||||
|  | @ -87,7 +87,7 @@ class SearchQueryComponents { | ||||||
|  */ |  */ | ||||||
| export function parseSearchQuery(q: string): SearchQueryComponents { | export function parseSearchQuery(q: string): SearchQueryComponents { | ||||||
|   const regexpParts = /(-)?(?:"([^"]*)"|([^"\s]+))(?:\s|$)/g; |   const regexpParts = /(-)?(?:"([^"]*)"|([^"\s]+))(?:\s|$)/g; | ||||||
|   const components = Array.from(q.replaceAll("'", '"').matchAll(regexpParts), (m) => { |   const components = Array.from(q.matchAll(regexpParts), (m) => { | ||||||
|     const negative = m[1] === "-"; |     const negative = m[1] === "-"; | ||||||
|     // Exact
 |     // Exact
 | ||||||
|     if (m[2]) { |     if (m[2]) { | ||||||
|  |  | ||||||
|  | @ -1,7 +1,6 @@ | ||||||
| import { | import { | ||||||
|   deletePatient, |   deletePatient, | ||||||
|   getPatient, |   getPatient, | ||||||
|   getPatientNEntries, |  | ||||||
|   getPatientNames, |   getPatientNames, | ||||||
|   getPatients, |   getPatients, | ||||||
|   hidePatient, |   hidePatient, | ||||||
|  | @ -14,15 +13,9 @@ import { z } from "zod"; | ||||||
| 
 | 
 | ||||||
| export const patientRouter = t.router({ | export const patientRouter = t.router({ | ||||||
|   getNames: t.procedure.query(async () => trpcWrap(getPatientNames)), |   getNames: t.procedure.query(async () => trpcWrap(getPatientNames)), | ||||||
|   get: t.procedure.input(ZEntityId).query(async (opts) => |   get: t.procedure | ||||||
|     trpcWrap(async () => { |     .input(ZEntityId) | ||||||
|       const [patient, n_entries] = await Promise.all([ |     .query(async (opts) => trpcWrap(async () => getPatient(opts.input))), | ||||||
|         getPatient(opts.input), |  | ||||||
|         getPatientNEntries(opts.input), |  | ||||||
|       ]); |  | ||||||
|       return { ...patient, n_entries }; |  | ||||||
|     }) |  | ||||||
|   ), |  | ||||||
|   list: t.procedure.input(ZPatientsQuery).query(async (opts) => { |   list: t.procedure.input(ZPatientsQuery).query(async (opts) => { | ||||||
|     return getPatients( |     return getPatients( | ||||||
|       opts.input.filter ?? {}, |       opts.input.filter ?? {}, | ||||||
|  |  | ||||||
|  | @ -29,14 +29,10 @@ export type EntriesFilter = Partial<{ | ||||||
| }>; | }>; | ||||||
| 
 | 
 | ||||||
| export type PatientsFilter = Partial<{ | export type PatientsFilter = Partial<{ | ||||||
|   /** Search patient names */ |  | ||||||
|   search: string; |   search: string; | ||||||
|   station: FilterList<number>; |   station: FilterList<number>; | ||||||
|   room: FilterList<number>; |   room: FilterList<number>; | ||||||
|   /** Show only hidden patients */ |  | ||||||
|   hidden: boolean; |   hidden: boolean; | ||||||
|   /** Show visible and hidden patients */ |  | ||||||
|   includeHidden: boolean; |  | ||||||
| }>; | }>; | ||||||
| 
 | 
 | ||||||
| export type TRPCErrorResponse = { | export type TRPCErrorResponse = { | ||||||
|  |  | ||||||
|  | @ -107,7 +107,6 @@ export const ZPatientsFilter = z | ||||||
|     room: ZFilterList, |     room: ZFilterList, | ||||||
|     station: ZFilterList, |     station: ZFilterList, | ||||||
|     hidden: z.boolean(), |     hidden: z.boolean(), | ||||||
|     includeHidden: z.boolean(), |  | ||||||
|   }) |   }) | ||||||
|   .partial(); |   .partial(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -8,34 +8,25 @@ import { superValidate } from "sveltekit-superforms/server"; | ||||||
| export const actions = { | export const actions = { | ||||||
|   default: async (event) => { |   default: async (event) => { | ||||||
|     const id = ZUrlEntityId.parse(event.params.id); |     const id = ZUrlEntityId.parse(event.params.id); | ||||||
|     const formData = await event.request.formData(); |     // const form = await event.request.formData();
 | ||||||
| 
 | 
 | ||||||
|     const hide = formData.get("hide"); |     // const schema = zfd.formData(ZPatientNew.partial());
 | ||||||
|     const del = formData.get("delete"); |     // const formData = schema.parse(form);
 | ||||||
|     if (hide) { |     const form = await superValidate(event.request, ZPatientNew); | ||||||
|       await loadWrap(async () => |  | ||||||
|         trpc(event).patient.hide.mutate({ |  | ||||||
|           id, |  | ||||||
|           hidden: Boolean(parseInt(hide.toString())), |  | ||||||
|         }) |  | ||||||
|       ); |  | ||||||
|     } else if (del) { |  | ||||||
|       await loadWrap(async () => trpc(event).patient.delete.mutate(id)); |  | ||||||
|     } else { |  | ||||||
|       const form = await superValidate(formData, ZPatientNew); |  | ||||||
| 
 | 
 | ||||||
|       if (!form.valid) { |     // Convenient validation check:
 | ||||||
|         return fail(400, { form }); |     if (!form.valid) { | ||||||
|       } |       // Again, return { form } and things will just work.
 | ||||||
| 
 |       return fail(400, { form }); | ||||||
|       await loadWrap(async () => |  | ||||||
|         trpc(event).patient.update.mutate({ |  | ||||||
|           id, |  | ||||||
|           patient: form.data, |  | ||||||
|         }) |  | ||||||
|       ); |  | ||||||
| 
 |  | ||||||
|       return { form }; |  | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     await loadWrap(async () => | ||||||
|  |       trpc(event).patient.update.mutate({ | ||||||
|  |         id, | ||||||
|  |         patient: form.data, | ||||||
|  |       }) | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     return { form }; | ||||||
|   }, |   }, | ||||||
| } satisfies Actions; | } satisfies Actions; | ||||||
|  |  | ||||||
|  | @ -4,28 +4,16 @@ | ||||||
|   import PatientForm from "$lib/components/form/PatientForm.svelte"; |   import PatientForm from "$lib/components/form/PatientForm.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let data: PageData; |   export let data: PageData; | ||||||
| 
 |  | ||||||
|   $: hasEntries = data.patient.n_entries > 0; |  | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <svelte:head> | <svelte:head> | ||||||
|   <title>Patient #{data.patient.id}</title> |   <title>Patient #{data.patient.id}</title> | ||||||
| </svelte:head> | </svelte:head> | ||||||
| 
 | 
 | ||||||
| <PatientForm patient={data.patient} formData={data.form}> | <PatientForm patient={data.patient} formData={data.form} /> | ||||||
|   {#if data.patient.hidden} |  | ||||||
|     <button type="submit" name="hide" value="0" class="btn btn-primary" |  | ||||||
|       >Einblenden</button |  | ||||||
|     > |  | ||||||
|   {:else if hasEntries} |  | ||||||
|     <button type="submit" name="hide" value="1" class="btn">Ausblenden</button> |  | ||||||
|   {:else} |  | ||||||
|     <button type="submit" name="delete" value="1" class="btn btn-error">Löschen</button> |  | ||||||
|   {/if} |  | ||||||
| </PatientForm> |  | ||||||
| 
 | 
 | ||||||
| {#if hasEntries} | {#if data.patient.room} | ||||||
|   <h1 class="heading mt-8 mb-4">Einträge ({data.patient.n_entries})</h1> |   <h1 class="heading mt-8 mb-4">Einträge</h1> | ||||||
| 
 | 
 | ||||||
|   <FilteredEntryTable |   <FilteredEntryTable | ||||||
|     baseUrl="/patient/{data.patient.id}" |     baseUrl="/patient/{data.patient.id}" | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue