2026-02-10 17:00:05 +07:00
|
|
|
|
<script>
|
|
|
|
|
|
import { onMount } from 'svelte';
|
|
|
|
|
|
import { fetchSpecialties, createSpecialty, updateSpecialty, deleteSpecialty } from '$lib/api/specialties.js';
|
|
|
|
|
|
import { success as toastSuccess, error as toastError } from '$lib/utils/toast.js';
|
|
|
|
|
|
import DataTable from '$lib/components/DataTable.svelte';
|
|
|
|
|
|
import Modal from '$lib/components/Modal.svelte';
|
|
|
|
|
|
import SelectDropdown from '$lib/components/SelectDropdown.svelte';
|
2026-02-15 17:58:42 +07:00
|
|
|
|
import HelpTooltip from '$lib/components/HelpTooltip.svelte';
|
|
|
|
|
|
import { Plus, Edit2, Trash2, ArrowLeft, Search, Stethoscope, FolderTree } from 'lucide-svelte';
|
2026-02-10 17:00:05 +07:00
|
|
|
|
|
|
|
|
|
|
let loading = $state(false);
|
|
|
|
|
|
let specialties = $state([]);
|
|
|
|
|
|
let modalOpen = $state(false);
|
|
|
|
|
|
let modalMode = $state('create');
|
|
|
|
|
|
let saving = $state(false);
|
|
|
|
|
|
let formData = $state({ SpecialtyID: null, SpecialtyText: '', Title: '', Parent: '' });
|
|
|
|
|
|
let deleteConfirmOpen = $state(false);
|
|
|
|
|
|
let deleteItem = $state(null);
|
2026-02-15 17:58:42 +07:00
|
|
|
|
let deleting = $state(false);
|
|
|
|
|
|
let searchQuery = $state('');
|
2026-02-10 17:00:05 +07:00
|
|
|
|
|
|
|
|
|
|
const columns = [
|
|
|
|
|
|
{ key: 'SpecialtyText', label: 'Specialty Name', class: 'font-medium' },
|
|
|
|
|
|
{ key: 'Title', label: 'Title' },
|
2026-02-15 17:58:42 +07:00
|
|
|
|
{ key: 'ParentLabel', label: 'Parent Specialty' },
|
2026-02-10 17:00:05 +07:00
|
|
|
|
{ key: 'actions', label: 'Actions', class: 'w-32 text-center' },
|
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
|
|
onMount(async () => {
|
|
|
|
|
|
await loadSpecialties();
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
async function loadSpecialties() {
|
|
|
|
|
|
loading = true;
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await fetchSpecialties();
|
|
|
|
|
|
specialties = Array.isArray(response.data) ? response.data : [];
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
toastError(err.message || 'Failed to load specialties');
|
|
|
|
|
|
specialties = [];
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-15 17:58:42 +07:00
|
|
|
|
// Filter specialties based on search query
|
|
|
|
|
|
const filteredSpecialties = $derived(
|
|
|
|
|
|
specialties.filter((s) => {
|
|
|
|
|
|
if (!searchQuery.trim()) return true;
|
|
|
|
|
|
const query = searchQuery.toLowerCase();
|
|
|
|
|
|
const nameMatch = (s.SpecialtyText || '').toLowerCase().includes(query);
|
|
|
|
|
|
const titleMatch = (s.Title || '').toLowerCase().includes(query);
|
|
|
|
|
|
const parentMatch = (s.ParentLabel || '').toLowerCase().includes(query);
|
|
|
|
|
|
return nameMatch || titleMatch || parentMatch;
|
|
|
|
|
|
})
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-02-10 17:00:05 +07:00
|
|
|
|
function openCreateModal() {
|
|
|
|
|
|
modalMode = 'create';
|
|
|
|
|
|
formData = { SpecialtyID: null, SpecialtyText: '', Title: '', Parent: '' };
|
|
|
|
|
|
modalOpen = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function openEditModal(row) {
|
|
|
|
|
|
modalMode = 'edit';
|
|
|
|
|
|
formData = {
|
|
|
|
|
|
SpecialtyID: row.SpecialtyID,
|
|
|
|
|
|
SpecialtyText: row.SpecialtyText || '',
|
|
|
|
|
|
Title: row.Title || '',
|
|
|
|
|
|
Parent: row.Parent || '',
|
|
|
|
|
|
};
|
|
|
|
|
|
modalOpen = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-15 17:58:42 +07:00
|
|
|
|
// Check for duplicate specialty name
|
|
|
|
|
|
function isDuplicateName(name, excludeId = null) {
|
|
|
|
|
|
const normalizedName = name.trim().toLowerCase();
|
|
|
|
|
|
return specialties.some(
|
|
|
|
|
|
(s) =>
|
|
|
|
|
|
s.SpecialtyText.trim().toLowerCase() === normalizedName &&
|
|
|
|
|
|
s.SpecialtyID !== excludeId
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-10 17:00:05 +07:00
|
|
|
|
async function handleSave() {
|
2026-02-15 17:58:42 +07:00
|
|
|
|
// Validate for duplicate names
|
|
|
|
|
|
if (isDuplicateName(formData.SpecialtyText, formData.SpecialtyID)) {
|
|
|
|
|
|
toastError(`A specialty with the name "${formData.SpecialtyText}" already exists`);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-10 17:00:05 +07:00
|
|
|
|
saving = true;
|
|
|
|
|
|
try {
|
|
|
|
|
|
if (modalMode === 'create') {
|
|
|
|
|
|
await createSpecialty(formData);
|
|
|
|
|
|
toastSuccess('Specialty created successfully');
|
|
|
|
|
|
} else {
|
|
|
|
|
|
await updateSpecialty(formData);
|
|
|
|
|
|
toastSuccess('Specialty updated successfully');
|
|
|
|
|
|
}
|
|
|
|
|
|
modalOpen = false;
|
|
|
|
|
|
await loadSpecialties();
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
toastError(err.message || 'Failed to save specialty');
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
saving = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function confirmDelete(row) {
|
|
|
|
|
|
deleteItem = row;
|
|
|
|
|
|
deleteConfirmOpen = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function handleDelete() {
|
2026-02-15 17:58:42 +07:00
|
|
|
|
deleting = true;
|
2026-02-10 17:00:05 +07:00
|
|
|
|
try {
|
|
|
|
|
|
await deleteSpecialty(deleteItem.SpecialtyID);
|
|
|
|
|
|
toastSuccess('Specialty deleted successfully');
|
|
|
|
|
|
deleteConfirmOpen = false;
|
|
|
|
|
|
await loadSpecialties();
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
toastError(err.message || 'Failed to delete specialty');
|
2026-02-15 17:58:42 +07:00
|
|
|
|
} finally {
|
|
|
|
|
|
deleting = false;
|
2026-02-10 17:00:05 +07:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const parentOptions = $derived(
|
|
|
|
|
|
specialties
|
|
|
|
|
|
.filter((s) => s.SpecialtyID !== formData.SpecialtyID)
|
|
|
|
|
|
.map((s) => ({ value: s.SpecialtyID, label: s.SpecialtyText }))
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
const specialtyMap = $derived(
|
|
|
|
|
|
Object.fromEntries(specialties.map((s) => [s.SpecialtyID, s.SpecialtyText]))
|
|
|
|
|
|
);
|
2026-02-15 17:58:42 +07:00
|
|
|
|
|
|
|
|
|
|
// Get parent specialty name
|
|
|
|
|
|
function getParentName(parentId) {
|
|
|
|
|
|
if (!parentId || parentId === '0') return null;
|
|
|
|
|
|
return specialtyMap[parentId] || null;
|
|
|
|
|
|
}
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="p-6">
|
|
|
|
|
|
<div class="flex items-center gap-4 mb-6">
|
|
|
|
|
|
<a href="/master-data" class="btn btn-ghost btn-circle">
|
|
|
|
|
|
<ArrowLeft class="w-5 h-5" />
|
|
|
|
|
|
</a>
|
|
|
|
|
|
<div class="flex-1">
|
|
|
|
|
|
<h1 class="text-3xl font-bold text-gray-800">Medical Specialties</h1>
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<p class="text-gray-600">Manage medical specialty codes and their hierarchical relationships</p>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
<button class="btn btn-primary" onclick={openCreateModal}>
|
|
|
|
|
|
<Plus class="w-4 h-4 mr-2" />
|
|
|
|
|
|
Add Specialty
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<!-- Search Bar -->
|
|
|
|
|
|
<div class="mb-4">
|
|
|
|
|
|
<div class="relative max-w-md">
|
|
|
|
|
|
<Search class="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-gray-400" />
|
|
|
|
|
|
<input
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
class="input input-bordered w-full pl-10"
|
|
|
|
|
|
placeholder="Search by specialty name, title, or parent..."
|
|
|
|
|
|
bind:value={searchQuery}
|
|
|
|
|
|
/>
|
|
|
|
|
|
{#if searchQuery}
|
|
|
|
|
|
<button
|
|
|
|
|
|
class="absolute right-3 top-1/2 -translate-y-1/2 btn btn-xs btn-ghost btn-circle"
|
|
|
|
|
|
onclick={() => (searchQuery = '')}
|
|
|
|
|
|
aria-label="Clear search"
|
|
|
|
|
|
>
|
|
|
|
|
|
×
|
|
|
|
|
|
</button>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-10 17:00:05 +07:00
|
|
|
|
<div class="bg-base-100 rounded-lg shadow border border-base-200">
|
2026-02-15 17:58:42 +07:00
|
|
|
|
{#if filteredSpecialties.length === 0 && !loading}
|
|
|
|
|
|
<!-- Empty State -->
|
|
|
|
|
|
<div class="flex flex-col items-center justify-center py-16 px-4">
|
|
|
|
|
|
<div class="bg-base-200 rounded-full p-6 mb-4">
|
|
|
|
|
|
<Stethoscope class="w-12 h-12 text-base-content/40" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<h3 class="text-lg font-semibold text-base-content mb-2">
|
|
|
|
|
|
{searchQuery ? 'No specialties match your search' : 'No specialties yet'}
|
|
|
|
|
|
</h3>
|
|
|
|
|
|
<p class="text-base-content/60 text-center max-w-sm mb-4">
|
|
|
|
|
|
{searchQuery
|
|
|
|
|
|
? 'Try adjusting your search terms or clear the filter to see all specialties.'
|
|
|
|
|
|
: 'Get started by adding your first medical specialty to the system.'}
|
|
|
|
|
|
</p>
|
|
|
|
|
|
{#if searchQuery}
|
|
|
|
|
|
<button class="btn btn-ghost btn-sm" onclick={() => (searchQuery = '')}>
|
|
|
|
|
|
Clear Search
|
|
|
|
|
|
</button>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
{:else}
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<button class="btn btn-primary" onclick={openCreateModal}>
|
|
|
|
|
|
<Plus class="w-4 h-4 mr-2" />
|
|
|
|
|
|
Add First Specialty
|
|
|
|
|
|
</button>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
{/if}
|
2026-02-15 17:58:42 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
{:else}
|
|
|
|
|
|
<DataTable
|
|
|
|
|
|
{columns}
|
|
|
|
|
|
data={filteredSpecialties.map((s) => ({
|
|
|
|
|
|
...s,
|
|
|
|
|
|
ParentLabel: s.Parent === '0' || !s.Parent ? '-' : specialtyMap[s.Parent] || '-',
|
|
|
|
|
|
hasParent: s.Parent && s.Parent !== '0',
|
|
|
|
|
|
}))}
|
|
|
|
|
|
{loading}
|
|
|
|
|
|
emptyMessage="No specialties found"
|
|
|
|
|
|
hover={true}
|
|
|
|
|
|
bordered={false}
|
|
|
|
|
|
>
|
|
|
|
|
|
{#snippet cell({ column, row, value })}
|
|
|
|
|
|
{#if column.key === 'actions'}
|
|
|
|
|
|
<div class="flex justify-center gap-2">
|
|
|
|
|
|
<button class="btn btn-sm btn-ghost" onclick={() => openEditModal(row)}>
|
|
|
|
|
|
<Edit2 class="w-4 h-4" />
|
|
|
|
|
|
</button>
|
|
|
|
|
|
<button class="btn btn-sm btn-ghost text-error" onclick={() => confirmDelete(row)}>
|
|
|
|
|
|
<Trash2 class="w-4 h-4" />
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
{:else if column.key === 'SpecialtyText'}
|
|
|
|
|
|
<div class="flex items-center gap-2">
|
|
|
|
|
|
{#if row.hasParent}
|
|
|
|
|
|
<span class="text-base-content/40 ml-4">└─</span>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
<span class={row.hasParent ? 'text-base-content/80' : ''}>{value}</span>
|
|
|
|
|
|
{#if row.hasParent}
|
|
|
|
|
|
<span class="badge badge-sm badge-ghost">
|
|
|
|
|
|
<FolderTree class="w-3 h-3 mr-1" />
|
|
|
|
|
|
Child
|
|
|
|
|
|
</span>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
{:else}
|
|
|
|
|
|
{value}
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
{/snippet}
|
|
|
|
|
|
</DataTable>
|
|
|
|
|
|
{/if}
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<Modal bind:open={modalOpen} title={modalMode === 'create' ? 'Add Specialty' : 'Edit Specialty'} size="md">
|
|
|
|
|
|
<form class="space-y-5" onsubmit={(e) => { e.preventDefault(); handleSave(); }}>
|
|
|
|
|
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
|
|
|
|
<div class="form-control">
|
|
|
|
|
|
<label class="label" for="specialtyText">
|
|
|
|
|
|
<span class="label-text font-medium">Specialty Name</span>
|
|
|
|
|
|
<span class="label-text-alt text-error">*</span>
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<input
|
|
|
|
|
|
id="specialtyText"
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
class="input input-bordered w-full"
|
|
|
|
|
|
bind:value={formData.SpecialtyText}
|
2026-02-15 17:58:42 +07:00
|
|
|
|
placeholder="e.g., Cardiology, Internal Medicine"
|
2026-02-10 17:00:05 +07:00
|
|
|
|
required
|
|
|
|
|
|
/>
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<label class="label" for="specialtyText">
|
|
|
|
|
|
<span class="label-text-alt text-base-content/50">Unique name for the specialty</span>
|
|
|
|
|
|
</label>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="form-control">
|
|
|
|
|
|
<label class="label" for="title">
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<span class="label-text font-medium">Professional Title</span>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</label>
|
|
|
|
|
|
<input
|
|
|
|
|
|
id="title"
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
class="input input-bordered w-full"
|
|
|
|
|
|
bind:value={formData.Title}
|
2026-02-15 17:58:42 +07:00
|
|
|
|
placeholder="e.g., Sp. PD, Sp. A, Sp. And"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<label class="label" for="title">
|
|
|
|
|
|
<span class="label-text-alt text-base-content/50">Official abbreviation or title</span>
|
|
|
|
|
|
</label>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="form-control">
|
|
|
|
|
|
<div class="flex items-center gap-2 mb-1">
|
|
|
|
|
|
<label class="label py-0" for="parent">
|
|
|
|
|
|
<span class="label-text font-medium">Parent Specialty</span>
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<HelpTooltip
|
|
|
|
|
|
text="Organize specialties hierarchically. For example, select 'Internal Medicine' as the parent for 'Cardiology' to show it as a subspecialty."
|
|
|
|
|
|
title="Parent Specialty Hierarchy"
|
|
|
|
|
|
position="right"
|
2026-02-10 17:00:05 +07:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<SelectDropdown
|
|
|
|
|
|
label=""
|
|
|
|
|
|
name="parent"
|
|
|
|
|
|
bind:value={formData.Parent}
|
|
|
|
|
|
options={parentOptions}
|
|
|
|
|
|
placeholder="None (Top-level specialty)"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<label class="label" for="parent">
|
|
|
|
|
|
<span class="label-text-alt text-base-content/50">
|
|
|
|
|
|
Optional: Select a parent specialty to create a subspecialty
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</label>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
</form>
|
|
|
|
|
|
{#snippet footer()}
|
|
|
|
|
|
<button class="btn btn-ghost" onclick={() => (modalOpen = false)} type="button">Cancel</button>
|
|
|
|
|
|
<button class="btn btn-primary" onclick={handleSave} disabled={saving} type="button">
|
|
|
|
|
|
{#if saving}
|
|
|
|
|
|
<span class="loading loading-spinner loading-sm mr-2"></span>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
{saving ? 'Saving...' : 'Save'}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
{/snippet}
|
|
|
|
|
|
</Modal>
|
|
|
|
|
|
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<Modal bind:open={deleteConfirmOpen} title="Confirm Deletion" size="md">
|
|
|
|
|
|
<div class="py-4 space-y-4">
|
|
|
|
|
|
<div class="flex items-start gap-3">
|
|
|
|
|
|
<div class="bg-error/10 rounded-full p-2">
|
|
|
|
|
|
<Trash2 class="w-6 h-6 text-error" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<p class="text-base-content font-medium">
|
|
|
|
|
|
Are you sure you want to delete this specialty?
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<p class="text-base-content/70 mt-1">
|
|
|
|
|
|
<strong class="text-base-content">{deleteItem?.SpecialtyText}</strong>
|
|
|
|
|
|
{#if deleteItem?.Title}
|
|
|
|
|
|
<span class="text-base-content/50">({deleteItem?.Title})</span>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="bg-warning/10 border border-warning/20 rounded-lg p-3">
|
|
|
|
|
|
<p class="text-sm text-warning">
|
|
|
|
|
|
<strong>Warning:</strong> This action cannot be undone. If this specialty has child specialties,
|
|
|
|
|
|
they may become orphaned and need to be reassigned.
|
|
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
</div>
|
|
|
|
|
|
{#snippet footer()}
|
|
|
|
|
|
<button class="btn btn-ghost" onclick={() => (deleteConfirmOpen = false)} type="button">Cancel</button>
|
2026-02-15 17:58:42 +07:00
|
|
|
|
<button class="btn btn-error" onclick={handleDelete} disabled={deleting} type="button">
|
|
|
|
|
|
{#if deleting}
|
|
|
|
|
|
<span class="loading loading-spinner loading-sm mr-2"></span>
|
|
|
|
|
|
{/if}
|
|
|
|
|
|
{deleting ? 'Deleting...' : 'Delete Specialty'}
|
|
|
|
|
|
</button>
|
2026-02-10 17:00:05 +07:00
|
|
|
|
{/snippet}
|
|
|
|
|
|
</Modal>
|