62 lines
1.6 KiB
Vue
62 lines
1.6 KiB
Vue
<script setup lang="ts">
|
|
import { RouterView, useRouter, useRoute } from 'vue-router'
|
|
import { onMounted, watch } from 'vue'
|
|
import Toast from '@/components/common/Toast.vue'
|
|
import { useAppStore } from '@/stores'
|
|
import { getSetupStatus } from '@/api/setup'
|
|
|
|
const router = useRouter()
|
|
const route = useRoute()
|
|
const appStore = useAppStore()
|
|
|
|
/**
|
|
* Update favicon dynamically
|
|
* @param logoUrl - URL of the logo to use as favicon
|
|
*/
|
|
function updateFavicon(logoUrl: string) {
|
|
// Find existing favicon link or create new one
|
|
let link = document.querySelector<HTMLLinkElement>('link[rel="icon"]')
|
|
if (!link) {
|
|
link = document.createElement('link')
|
|
link.rel = 'icon'
|
|
document.head.appendChild(link)
|
|
}
|
|
link.type = logoUrl.endsWith('.svg') ? 'image/svg+xml' : 'image/x-icon'
|
|
link.href = logoUrl
|
|
}
|
|
|
|
// Watch for site settings changes and update favicon/title
|
|
watch(() => appStore.siteLogo, (newLogo) => {
|
|
if (newLogo) {
|
|
updateFavicon(newLogo)
|
|
}
|
|
}, { immediate: true })
|
|
|
|
watch(() => appStore.siteName, (newName) => {
|
|
if (newName) {
|
|
document.title = `${newName} - AI API Gateway`
|
|
}
|
|
}, { immediate: true })
|
|
|
|
onMounted(async () => {
|
|
// Check if setup is needed
|
|
try {
|
|
const status = await getSetupStatus()
|
|
if (status.needs_setup && route.path !== '/setup') {
|
|
router.replace('/setup')
|
|
return
|
|
}
|
|
} catch {
|
|
// If setup endpoint fails, assume normal mode and continue
|
|
}
|
|
|
|
// Load public settings into appStore (will be cached for other components)
|
|
await appStore.fetchPublicSettings()
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<RouterView />
|
|
<Toast />
|
|
</template>
|