213 lines
5.5 KiB
Vue
213 lines
5.5 KiB
Vue
<script setup lang="ts">
|
|
import { useRouter, useRoute } from 'vue-router'
|
|
import { computed, ref, onMounted } from 'vue'
|
|
import LoginButton from '../components/LoginButton.vue'
|
|
|
|
const router = useRouter()
|
|
const route = useRoute()
|
|
|
|
const handleBack = () => {
|
|
if (window.history.length > 1) {
|
|
router.back()
|
|
} else {
|
|
router.push('/child')
|
|
}
|
|
}
|
|
|
|
const showBack = computed(
|
|
() =>
|
|
!(
|
|
route.path === '/parent' ||
|
|
route.name === 'TaskView' ||
|
|
route.name === 'RewardView' ||
|
|
route.name === 'NotificationView'
|
|
),
|
|
)
|
|
|
|
// Version fetching
|
|
const appVersion = ref('')
|
|
onMounted(async () => {
|
|
try {
|
|
const resp = await fetch('/api/version')
|
|
if (resp.ok) {
|
|
const data = await resp.json()
|
|
appVersion.value = data.version || ''
|
|
}
|
|
} catch (e) {
|
|
appVersion.value = ''
|
|
}
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div class="layout-root">
|
|
<header class="topbar">
|
|
<div class="back-btn-container">
|
|
<button v-show="showBack" class="back-btn" @click="handleBack" tabindex="0">← Back</button>
|
|
</div>
|
|
<nav class="view-selector">
|
|
<button
|
|
:class="{
|
|
active: [
|
|
'ParentChildrenListView',
|
|
'ParentView',
|
|
'ChildEditView',
|
|
'CreateChild',
|
|
'TaskAssignView',
|
|
'RewardAssignView',
|
|
].includes(String(route.name)),
|
|
}"
|
|
@click="router.push({ name: 'ParentChildrenListView' })"
|
|
aria-label="Children"
|
|
title="Children"
|
|
>
|
|
<!-- Children Icon -->
|
|
<svg
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="1.7"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
>
|
|
<circle cx="8" cy="10" r="3" />
|
|
<circle cx="16" cy="10" r="3" />
|
|
<path d="M2 20c0-2.5 3-4.5 6-4.5s6 2 6 4.5" />
|
|
<path d="M10 20c0-2 2-3.5 6-3.5s6 1.5 6 3.5" />
|
|
</svg>
|
|
</button>
|
|
<button
|
|
:class="{ active: ['TaskView', 'EditTask', 'CreateTask'].includes(String(route.name)) }"
|
|
@click="router.push({ name: 'TaskView' })"
|
|
aria-label="Tasks"
|
|
title="Tasks"
|
|
>
|
|
<!-- Book Icon -->
|
|
<svg
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="1.7"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
>
|
|
<path d="M4 19.5A2.5 2.5 0 0 1 6.5 17H20" />
|
|
<path d="M20 22V6a2 2 0 0 0-2-2H6.5A2.5 2.5 0 0 0 4 6.5v13" />
|
|
<path d="M16 2v4" />
|
|
</svg>
|
|
</button>
|
|
<button
|
|
:class="{
|
|
active: ['RewardView', 'EditReward', 'CreateReward'].includes(String(route.name)),
|
|
}"
|
|
@click="router.push({ name: 'RewardView' })"
|
|
aria-label="Rewards"
|
|
title="Rewards"
|
|
>
|
|
<!-- Trophy Icon -->
|
|
<svg
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="1.7"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
>
|
|
<path d="M8 21h8" />
|
|
<path d="M12 17v4" />
|
|
<path d="M17 17a5 5 0 0 0 5-5V7h-4" />
|
|
<path d="M7 17a5 5 0 0 1-5-5V7h4" />
|
|
<rect x="7" y="2" width="10" height="15" rx="5" />
|
|
</svg>
|
|
</button>
|
|
<button
|
|
:class="{ active: ['NotificationView'].includes(String(route.name)) }"
|
|
@click="router.push({ name: 'NotificationView' })"
|
|
aria-label="Notifications"
|
|
title="Notifications"
|
|
>
|
|
<!-- Notification/Bell Icon -->
|
|
<svg
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="1.7"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
>
|
|
<path d="M18 16v-5a6 6 0 1 0-12 0v5" />
|
|
<path d="M2 16h20" />
|
|
<path d="M8 20a4 4 0 0 0 8 0" />
|
|
<circle cx="19" cy="7" r="2" fill="#ef4444" stroke="none" />
|
|
</svg>
|
|
</button>
|
|
</nav>
|
|
<LoginButton class="login-btn-container" />
|
|
</header>
|
|
|
|
<main class="main-content">
|
|
<router-view />
|
|
</main>
|
|
|
|
<div v-if="appVersion" class="app-version">v{{ appVersion }}</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
/* Only keep styles unique to ParentLayout */
|
|
|
|
.view-selector {
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: stretch;
|
|
flex: 2 1 0;
|
|
justify-content: center;
|
|
}
|
|
|
|
.view-selector button {
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: center;
|
|
background: var(--button-bg);
|
|
color: var(--button-text);
|
|
border: 0;
|
|
border-radius: 8px 8px 0 0;
|
|
padding: 0.6rem 1.2rem;
|
|
font-weight: 600;
|
|
cursor: pointer;
|
|
transition:
|
|
background 0.18s,
|
|
color 0.18s;
|
|
font-size: 1rem;
|
|
box-shadow: 0 2px 8px rgba(102, 126, 234, 0.08);
|
|
}
|
|
|
|
.view-selector button.active {
|
|
background: var(--button-active-bg);
|
|
color: var(--button-active-text);
|
|
}
|
|
|
|
.view-selector button.active svg {
|
|
stroke: var(--button-active-text);
|
|
}
|
|
|
|
.view-selector button:hover:not(.active) {
|
|
background: var(--button-hover-bg);
|
|
}
|
|
|
|
@media (max-width: 480px) {
|
|
.view-selector button {
|
|
padding: 0.45rem 0.75rem;
|
|
font-size: 0.85rem;
|
|
}
|
|
}
|
|
</style>
|