mirror of
https://github.com/h44z/wg-portal.git
synced 2025-04-19 08:55:12 +00:00
129 lines
5.2 KiB
Vue
129 lines
5.2 KiB
Vue
<script setup>
|
|
import { RouterLink, RouterView } from 'vue-router';
|
|
import { computed, getCurrentInstance, onMounted, ref } from "vue";
|
|
import { authStore } from "./stores/auth";
|
|
import { securityStore } from "./stores/security";
|
|
import { settingsStore } from "@/stores/settings";
|
|
|
|
const appGlobal = getCurrentInstance().appContext.config.globalProperties
|
|
const auth = authStore()
|
|
const sec = securityStore()
|
|
const settings = settingsStore()
|
|
|
|
onMounted(async () => {
|
|
console.log("Starting WireGuard Portal frontend...");
|
|
|
|
await sec.LoadSecurityProperties();
|
|
await auth.LoadProviders();
|
|
|
|
let wasLoggedIn = auth.IsAuthenticated;
|
|
try {
|
|
await auth.LoadSession();
|
|
await settings.LoadSettings(); // only logs errors, does not throw
|
|
|
|
console.log("WireGuard Portal session is valid");
|
|
} catch (e) {
|
|
if (wasLoggedIn) {
|
|
console.log("WireGuard Portal invalid - logging out");
|
|
await auth.Logout();
|
|
}
|
|
}
|
|
|
|
console.log("WireGuard Portal ready!");
|
|
})
|
|
|
|
const switchLanguage = function (lang) {
|
|
if (appGlobal.$i18n.locale !== lang) {
|
|
localStorage.setItem('wgLang', lang);
|
|
appGlobal.$i18n.locale = lang;
|
|
}
|
|
}
|
|
|
|
const languageFlag = computed(() => {
|
|
// `this` points to the component instance
|
|
let lang = appGlobal.$i18n.locale.toLowerCase();
|
|
if (!appGlobal.$i18n.availableLocales.includes(lang)) {
|
|
lang = appGlobal.$i18n.fallbackLocale;
|
|
}
|
|
if (lang === "en") {
|
|
lang = "us";
|
|
}
|
|
return "fi-" + lang;
|
|
})
|
|
|
|
const companyName = ref(WGPORTAL_SITE_COMPANY_NAME);
|
|
const wgVersion = ref(WGPORTAL_VERSION);
|
|
const currentYear = ref(new Date().getFullYear())
|
|
|
|
</script>
|
|
|
|
<template>
|
|
<notifications :duration="3000" :ignore-duplicates="true" position="top right" />
|
|
|
|
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
|
|
<div class="container-fluid">
|
|
<a class="navbar-brand" href="/"><img :alt="companyName" src="/img/header-logo.png" /></a>
|
|
<button aria-controls="navbarColor01" aria-expanded="false" aria-label="Toggle navigation" class="navbar-toggler"
|
|
data-bs-target="#navbarTop" data-bs-toggle="collapse" type="button">
|
|
<span class="navbar-toggler-icon"></span>
|
|
</button>
|
|
|
|
<div id="navbarTop" class="collapse navbar-collapse">
|
|
<ul class="navbar-nav me-auto">
|
|
<li class="nav-item">
|
|
<RouterLink :to="{ name: 'home' }" class="nav-link">{{ $t('menu.home') }}</RouterLink>
|
|
</li>
|
|
<li v-if="auth.IsAuthenticated && auth.IsAdmin" class="nav-item">
|
|
<RouterLink :to="{ name: 'interfaces' }" class="nav-link">{{ $t('menu.interfaces') }}</RouterLink>
|
|
</li>
|
|
<li v-if="auth.IsAuthenticated && auth.IsAdmin" class="nav-item">
|
|
<RouterLink :to="{ name: 'users' }" class="nav-link">{{ $t('menu.users') }}</RouterLink>
|
|
</li>
|
|
</ul>
|
|
|
|
<div class="navbar-nav d-flex justify-content-end">
|
|
<div v-if="auth.IsAuthenticated" class="nav-item dropdown">
|
|
<a aria-expanded="false" aria-haspopup="true" class="nav-link dropdown-toggle" data-bs-toggle="dropdown"
|
|
href="#" role="button">{{ auth.User.Firstname }} {{ auth.User.Lastname }}</a>
|
|
<div class="dropdown-menu">
|
|
<RouterLink :to="{ name: 'profile' }" class="dropdown-item"><i class="fas fa-user"></i> {{ $t('menu.profile') }}</RouterLink>
|
|
<div class="dropdown-divider"></div>
|
|
<a class="dropdown-item" href="#" @click.prevent="auth.Logout"><i class="fas fa-sign-out-alt"></i> {{ $t('menu.logout') }}</a>
|
|
</div>
|
|
</div>
|
|
<div v-if="!auth.IsAuthenticated" class="nav-item">
|
|
<RouterLink :to="{ name: 'login' }" class="nav-link"><i class="fas fa-sign-in-alt fa-sm fa-fw me-2"></i>{{ $t('menu.login') }}</RouterLink>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
|
|
<div class="container mt-5 flex-shrink-0">
|
|
<RouterView />
|
|
</div>
|
|
|
|
<footer class="page-footer mt-auto">
|
|
<div class="container mt-5">
|
|
<div class="row align-items-center">
|
|
<div class="col-6">Copyright © {{ companyName }} {{ currentYear }} <span v-if="auth.IsAuthenticated"> - version {{ wgVersion }}</span></div>
|
|
<div class="col-6 text-end">
|
|
<div :aria-label="$t('menu.lang')" class="btn-group" role="group">
|
|
<div class="btn-group" role="group">
|
|
<button aria-expanded="false" aria-haspopup="true" class="btn btn btn-secondary pe-0"
|
|
data-bs-toggle="dropdown" type="button"><span :class="languageFlag" class="fi"></span></button>
|
|
<div aria-labelledby="btnGroupDrop3" class="dropdown-menu" style="">
|
|
<a class="dropdown-item" href="#" @click.prevent="switchLanguage('en')"><span class="fi fi-us"></span> English</a>
|
|
<a class="dropdown-item" href="#" @click.prevent="switchLanguage('de')"><span class="fi fi-de"></span> Deutsch</a>
|
|
<a class="dropdown-item" href="#" @click.prevent="switchLanguage('ru')"><span class="fi fi-ru"></span> Русский</a>
|
|
<a class="dropdown-item" href="#" @click.prevent="switchLanguage('vi')"><span class="fi fi-vi"></span> Tiếng Việt</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</footer></template>
|
|
|
|
<style></style>
|