2026-03-11 23:33:10 +00:00
|
|
|
<script setup lang="ts">
|
2026-03-18 00:12:50 +00:00
|
|
|
import { usePage } from '@inertiajs/vue3';
|
|
|
|
|
import { CheckCircle, XCircle, X } from 'lucide-vue-next';
|
|
|
|
|
import { computed, ref, watch } from 'vue';
|
2026-03-11 23:33:10 +00:00
|
|
|
import AppContent from '@/components/AppContent.vue';
|
|
|
|
|
import AppShell from '@/components/AppShell.vue';
|
|
|
|
|
import AppSidebar from '@/components/AppSidebar.vue';
|
|
|
|
|
import AppSidebarHeader from '@/components/AppSidebarHeader.vue';
|
|
|
|
|
import type { BreadcrumbItem } from '@/types';
|
|
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
|
breadcrumbs?: BreadcrumbItem[];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
withDefaults(defineProps<Props>(), {
|
|
|
|
|
breadcrumbs: () => [],
|
|
|
|
|
});
|
2026-03-18 00:12:50 +00:00
|
|
|
|
|
|
|
|
const page = usePage<{
|
|
|
|
|
flash: { success?: string; error?: string };
|
|
|
|
|
}>();
|
|
|
|
|
|
|
|
|
|
const flashMessage = ref<{ type: 'success' | 'error'; text: string } | null>(
|
|
|
|
|
null,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const flash = computed(() => page.props.flash);
|
|
|
|
|
|
|
|
|
|
watch(
|
|
|
|
|
flash,
|
|
|
|
|
(val) => {
|
|
|
|
|
if (val?.success) {
|
|
|
|
|
flashMessage.value = { type: 'success', text: val.success };
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
flashMessage.value = null;
|
|
|
|
|
}, 4000);
|
|
|
|
|
} else if (val?.error) {
|
|
|
|
|
flashMessage.value = { type: 'error', text: val.error };
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
flashMessage.value = null;
|
|
|
|
|
}, 4000);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{ immediate: true },
|
|
|
|
|
);
|
2026-03-11 23:33:10 +00:00
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<template>
|
|
|
|
|
<AppShell variant="sidebar">
|
|
|
|
|
<AppSidebar />
|
|
|
|
|
<AppContent variant="sidebar" class="overflow-x-hidden">
|
|
|
|
|
<AppSidebarHeader :breadcrumbs="breadcrumbs" />
|
|
|
|
|
<slot />
|
|
|
|
|
</AppContent>
|
2026-03-18 00:12:50 +00:00
|
|
|
|
|
|
|
|
<!-- Flash Messages -->
|
|
|
|
|
<Teleport to="body">
|
|
|
|
|
<Transition
|
|
|
|
|
enter-active-class="transition duration-300 ease-out"
|
|
|
|
|
enter-from-class="translate-y-2 opacity-0"
|
|
|
|
|
enter-to-class="translate-y-0 opacity-100"
|
|
|
|
|
leave-active-class="transition duration-200 ease-in"
|
|
|
|
|
leave-from-class="translate-y-0 opacity-100"
|
|
|
|
|
leave-to-class="translate-y-2 opacity-0"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
v-if="flashMessage"
|
|
|
|
|
class="fixed right-4 bottom-4 z-50 flex max-w-sm items-center gap-3 rounded-lg border bg-background px-4 py-3 shadow-lg"
|
|
|
|
|
:class="
|
|
|
|
|
flashMessage.type === 'success'
|
|
|
|
|
? 'border-green-200 dark:border-green-800'
|
|
|
|
|
: 'border-red-200 dark:border-red-800'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<CheckCircle
|
|
|
|
|
v-if="flashMessage.type === 'success'"
|
|
|
|
|
class="h-5 w-5 shrink-0 text-green-600"
|
|
|
|
|
/>
|
|
|
|
|
<XCircle v-else class="h-5 w-5 shrink-0 text-red-600" />
|
|
|
|
|
<span class="text-sm">{{ flashMessage.text }}</span>
|
|
|
|
|
<button
|
|
|
|
|
class="ml-auto shrink-0 text-muted-foreground hover:text-foreground"
|
|
|
|
|
@click="flashMessage = null"
|
|
|
|
|
>
|
|
|
|
|
<X class="h-4 w-4" />
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</Transition>
|
|
|
|
|
</Teleport>
|
2026-03-11 23:33:10 +00:00
|
|
|
</AppShell>
|
|
|
|
|
</template>
|