chahinebrini 790b5e5c24 feat(admin): surface Lyra-Posts page in nav + dashboard grid
The pages/lyra.vue page (create community posts as Lyra/ReBreak, AI-generated
or manual) existed but wasn't linked anywhere. Adds it to the sidebar +
mobile bottom-tab (grid-cols-5→6) and the dashboard quick-links grid
(lg:grid-cols-4→5). Admin app stays team-internal (stats / users / domain
approval / social posts / moderation) — no relation to the RN app.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-11 15:06:12 +02:00

127 lines
4.1 KiB
Vue

<template>
<div class="min-h-screen bg-gray-950 text-gray-100">
<!-- Topbar (Desktop + Mobile) -->
<header class="border-b border-gray-800 bg-gray-900 px-4 py-3 flex items-center justify-between lg:px-6">
<div class="flex items-center gap-3">
<span class="text-sm font-semibold tracking-wide text-gray-400 uppercase">rebreak</span>
<span class="text-gray-600">/</span>
<span class="text-sm font-medium text-white">Admin</span>
</div>
<!-- Desktop: Email + Logout-Button sichtbar -->
<div class="hidden lg:flex items-center gap-4">
<span class="text-xs text-gray-500">{{ adminEmail }}</span>
<UButton
size="xs"
color="neutral"
variant="ghost"
label="Logout"
@click="logout"
/>
</div>
<!-- Mobile: Dropdown-Menu fuer Logout/Email -->
<UDropdownMenu :items="mobileMenuItems" :popper="{ placement: 'bottom-end' }">
<UButton
icon="heroicons:bars-3"
size="sm"
color="neutral"
variant="ghost"
class="lg:hidden"
aria-label="Menu"
/>
</UDropdownMenu>
</header>
<!-- Sidebar (Desktop) + Content -->
<div class="flex flex-col lg:flex-row">
<!-- Sidebar nur Desktop -->
<aside class="hidden lg:block w-56 min-h-[calc(100vh-49px)] border-r border-gray-800 bg-gray-900 py-4">
<nav class="flex flex-col gap-1 px-3">
<NuxtLink
v-for="item in nav"
:key="item.to"
:to="item.to"
class="flex items-center gap-2 rounded px-3 py-2 text-sm text-gray-400 hover:bg-gray-800 hover:text-white transition-colors"
active-class="bg-gray-800 text-white"
>
<UIcon :name="item.icon" class="h-4 w-4 shrink-0" />
{{ item.label }}
</NuxtLink>
</nav>
</aside>
<!-- Content -->
<main class="flex-1 p-4 lg:p-6 pb-24 lg:pb-6">
<slot />
</main>
</div>
<!-- Bottom-Tab-Bar (Mobile only) -->
<nav
class="lg:hidden fixed bottom-0 inset-x-0 z-40 border-t border-gray-800 bg-gray-950/95 backdrop-blur"
:style="{ paddingBottom: 'env(safe-area-inset-bottom)' }"
>
<ul class="grid grid-cols-6">
<li v-for="item in nav" :key="item.to">
<NuxtLink
:to="item.to"
class="flex flex-col items-center justify-center gap-1 py-2 text-gray-500 transition-colors"
:class="isActive(item.to)
? 'text-white bg-gray-800'
: 'hover:text-gray-300'"
>
<UIcon :name="item.icon" class="h-5 w-5" />
<span class="text-[10px] font-medium leading-none">{{ item.shortLabel }}</span>
</NuxtLink>
</li>
</ul>
</nav>
</div>
</template>
<script setup lang="ts">
const { adminEmail, logout } = useAdminAuth()
const route = useRoute()
interface NavItem {
to: string
label: string
shortLabel: string
icon: string
}
const nav: NavItem[] = [
{ to: "/", label: "Dashboard", shortLabel: "Home", icon: "heroicons:home" },
{ to: "/domains", label: "Domain-Approval", shortLabel: "Domains", icon: "heroicons:globe-alt" },
{ to: "/users", label: "User-Management", shortLabel: "Users", icon: "heroicons:users" },
{ to: "/stats", label: "Statistiken", shortLabel: "Stats", icon: "heroicons:chart-bar" },
{ to: "/moderation", label: "Moderation", shortLabel: "Mod", icon: "heroicons:flag" },
{ to: "/lyra", label: "Lyra-Posts", shortLabel: "Lyra", icon: "heroicons:sparkles" },
]
// Active-State: exact match fuer "/", startsWith fuer Sub-Routes
function isActive(to: string): boolean {
if (to === "/") return route.path === "/"
return route.path === to || route.path.startsWith(`${to}/`)
}
// Mobile-Dropdown-Menu (Header oben rechts)
const mobileMenuItems = computed(() => [
[
{
label: adminEmail.value || "Admin",
icon: "heroicons:user-circle",
disabled: true,
},
],
[
{
label: "Logout",
icon: "heroicons:arrow-right-on-rectangle",
onSelect: () => logout(),
},
],
])
</script>