5e4021bc60
Deploy to LXC / deploy (push) Successful in 17s
Settings page shows account info and a change password form. Validates current password, minimum 8 chars, confirmation match. Added Settings link in sidebar. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
120 lines
4.8 KiB
Svelte
120 lines
4.8 KiB
Svelte
<script lang="ts">
|
|
import { page } from '$app/stores';
|
|
|
|
interface Props {
|
|
open: boolean;
|
|
onToggle: () => void;
|
|
counts: { devices: number; components: number; needsRepair: number };
|
|
}
|
|
|
|
let { open, onToggle, counts }: Props = $props();
|
|
|
|
const navItems = $derived([
|
|
{
|
|
href: '/',
|
|
label: 'Dashboard',
|
|
icon: 'M3 12l2-2m0 0l7-7 7 7M5 10v10a1 1 0 001 1h3m10-11l2 2m-2-2v10a1 1 0 01-1 1h-3m-6 0a1 1 0 001-1v-4a1 1 0 011-1h2a1 1 0 011 1v4a1 1 0 001 1m-6 0h6'
|
|
},
|
|
{
|
|
href: '/devices',
|
|
label: 'Devices',
|
|
badge: counts.devices,
|
|
icon: 'M9 17V7m0 10a2 2 0 01-2 2H5a2 2 0 01-2-2V7a2 2 0 012-2h2a2 2 0 012 2m0 10a2 2 0 002 2h2a2 2 0 002-2M9 7a2 2 0 012-2h2a2 2 0 012 2m0 10V7m0 10a2 2 0 002 2h2a2 2 0 002-2V7a2 2 0 00-2-2h-2a2 2 0 00-2 2'
|
|
},
|
|
{
|
|
href: '/components',
|
|
label: 'Components',
|
|
badge: counts.components,
|
|
icon: 'M19.428 15.428a2 2 0 00-1.022-.547l-2.387-.477a6 6 0 00-3.86.517l-.318.158a6 6 0 01-3.86.517L6.05 15.21a2 2 0 00-1.806.547M8 4h8l-1 1v5.172a2 2 0 00.586 1.414l5 5c1.26 1.26.367 3.414-1.415 3.414H4.828c-1.782 0-2.674-2.154-1.414-3.414l5-5A2 2 0 009 10.172V5L8 4z'
|
|
},
|
|
{
|
|
href: '/installations',
|
|
label: 'Install Log',
|
|
icon: 'M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-3 7h3m-3 4h3m-6-4h.01M9 16h.01'
|
|
},
|
|
{
|
|
href: '/locations',
|
|
label: 'Locations',
|
|
icon: 'M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z M15 11a3 3 0 11-6 0 3 3 0 016 0z'
|
|
},
|
|
{
|
|
href: '/todos',
|
|
label: 'Todos',
|
|
icon: 'M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z'
|
|
},
|
|
{
|
|
href: '/checklists',
|
|
label: 'Checklists',
|
|
icon: 'M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4'
|
|
},
|
|
{
|
|
href: '/gallery',
|
|
label: 'Gallery',
|
|
icon: 'M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z'
|
|
},
|
|
{
|
|
href: '/settings',
|
|
label: 'Settings',
|
|
icon: 'M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.066 2.573c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.573 1.066c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.066-2.573c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z M15 12a3 3 0 11-6 0 3 3 0 016 0z'
|
|
}
|
|
]);
|
|
</script>
|
|
|
|
<aside
|
|
class="flex w-64 flex-col border-r border-gray-200 bg-white transition-transform duration-200 dark:border-gray-700 dark:bg-gray-800 {open
|
|
? 'translate-x-0'
|
|
: '-translate-x-full'} fixed inset-y-0 left-0 z-30 lg:relative lg:translate-x-0"
|
|
>
|
|
<!-- Logo -->
|
|
<div class="flex h-14 items-center border-b border-gray-200 px-4 dark:border-gray-700">
|
|
<a href="/" class="text-lg font-bold text-gray-900 dark:text-white">B4L Repair</a>
|
|
</div>
|
|
|
|
<!-- Navigation -->
|
|
<nav class="flex-1 overflow-y-auto px-3 py-4">
|
|
{#each navItems as item}
|
|
<a
|
|
href={item.href}
|
|
class="mb-1 flex items-center gap-2 rounded-md px-3 py-2 text-sm font-medium text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-gray-700"
|
|
>
|
|
<svg class="h-4 w-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d={item.icon} />
|
|
</svg>
|
|
{item.label}
|
|
{#if item.badge}
|
|
<span class="ml-auto rounded-full bg-gray-100 px-2 py-0.5 text-xs text-gray-600 dark:bg-gray-700 dark:text-gray-400">
|
|
{item.badge}
|
|
</span>
|
|
{/if}
|
|
</a>
|
|
{/each}
|
|
|
|
{#if counts.needsRepair > 0}
|
|
<div class="mt-4 mb-2 px-3 text-xs font-semibold uppercase tracking-wider text-gray-400 dark:text-gray-500">
|
|
Attention
|
|
</div>
|
|
<a
|
|
href="/devices?condition=needs-repair"
|
|
class="mb-0.5 flex items-center gap-2 rounded-md px-3 py-2 text-sm text-amber-700 hover:bg-amber-50 dark:text-amber-400 dark:hover:bg-amber-900/20"
|
|
>
|
|
<svg class="h-4 w-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-2.5L13.732 4c-.77-.833-1.964-.833-2.732 0L3.34 16.5c-.77.833.192 2.5 1.732 2.5z" />
|
|
</svg>
|
|
Needs Repair
|
|
<span class="ml-auto rounded-full bg-amber-100 px-2 py-0.5 text-xs font-medium text-amber-700 dark:bg-amber-900/40 dark:text-amber-300">
|
|
{counts.needsRepair}
|
|
</span>
|
|
</a>
|
|
{/if}
|
|
</nav>
|
|
</aside>
|
|
|
|
<!-- Backdrop for mobile -->
|
|
{#if open}
|
|
<button
|
|
class="fixed inset-0 z-20 bg-black/30 lg:hidden"
|
|
onclick={onToggle}
|
|
aria-label="Close sidebar"
|
|
></button>
|
|
{/if}
|