2022-03-28 07:06:36 +00:00
|
|
|
<script setup>
|
2022-11-21 16:33:21 +00:00
|
|
|
import { ref } from 'vue';
|
2022-03-15 09:33:28 +00:00
|
|
|
import { useI18n } from 'vue-i18n';
|
2022-11-24 06:21:45 +00:00
|
|
|
import { useRole } from 'src/composables/useRole';
|
2022-11-24 13:54:38 +00:00
|
|
|
import { useQuasar } from 'quasar';
|
2022-11-24 06:21:45 +00:00
|
|
|
import { useRoute } from 'vue-router';
|
2022-11-29 06:10:04 +00:00
|
|
|
import { useNavigationStore } from 'src/stores/useNavigationStore';
|
2022-11-21 16:33:21 +00:00
|
|
|
import routes from 'src/router/modules';
|
2022-11-29 06:10:04 +00:00
|
|
|
import axios from 'axios';
|
2022-03-15 09:33:28 +00:00
|
|
|
|
|
|
|
const { t } = useI18n();
|
2022-11-24 06:21:45 +00:00
|
|
|
const role = useRole();
|
|
|
|
const route = useRoute();
|
2022-11-24 13:54:38 +00:00
|
|
|
const quasar = useQuasar();
|
2022-11-29 06:10:04 +00:00
|
|
|
const navigation = useNavigationStore();
|
2022-03-15 09:33:28 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
const props = defineProps({
|
2022-11-21 16:33:21 +00:00
|
|
|
source: {
|
|
|
|
type: String,
|
|
|
|
default: 'main',
|
|
|
|
},
|
|
|
|
});
|
2022-03-15 09:33:28 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
function toLowerCamel(value) {
|
|
|
|
return value.charAt(0).toLowerCase() + value.slice(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
function findMatches(search, item) {
|
|
|
|
const matches = [];
|
|
|
|
function findRoute(search, item) {
|
|
|
|
for (const child of item.children) {
|
|
|
|
if (search.indexOf(child.name) > -1) {
|
|
|
|
matches.push(child);
|
|
|
|
} else if (child.children) {
|
|
|
|
findRoute(search, child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
findRoute(search, item);
|
|
|
|
|
|
|
|
return matches;
|
|
|
|
}
|
|
|
|
|
|
|
|
function addChildren(module, route, parent) {
|
|
|
|
if (route.menus) {
|
|
|
|
const mainMenus = route.menus[props.source];
|
|
|
|
const matches = findMatches(mainMenus, route);
|
|
|
|
for (const child of matches) {
|
|
|
|
addMenuItem(module, child, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function addMenuItem(module, route, parent) {
|
|
|
|
const { meta } = route;
|
|
|
|
|
|
|
|
if (meta && meta.roles && role.hasAny(meta.roles) === false) return;
|
2022-03-15 09:33:28 +00:00
|
|
|
|
2022-11-21 16:33:21 +00:00
|
|
|
const item = {
|
2022-11-24 06:21:45 +00:00
|
|
|
name: route.name,
|
2022-11-21 16:33:21 +00:00
|
|
|
};
|
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
if (meta) {
|
|
|
|
item.title = `${module}.pageTitles.${meta.title}`;
|
|
|
|
item.icon = meta.icon;
|
|
|
|
}
|
2022-11-21 16:33:21 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
parent.push(item);
|
2022-11-21 16:33:21 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
const items = ref([]);
|
|
|
|
if (props.source === 'main') {
|
|
|
|
for (const module of navigation.modules) {
|
|
|
|
const moduleDef = routes.find((route) => toLowerCamel(route.name) === module);
|
|
|
|
|
|
|
|
const item = addMenuItem(module, moduleDef, items.value);
|
2022-11-29 06:10:04 +00:00
|
|
|
const pinned = navigation.pinned.value;
|
|
|
|
if (pinned.contains(module)) {
|
|
|
|
item.isPinned = true
|
|
|
|
}
|
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
item.children = [];
|
2022-11-24 13:54:38 +00:00
|
|
|
item.module = module;
|
2022-11-24 06:21:45 +00:00
|
|
|
|
|
|
|
if (!item) continue;
|
|
|
|
addChildren(module, moduleDef, item.children);
|
2022-11-21 16:33:21 +00:00
|
|
|
}
|
2022-11-24 06:21:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (props.source === 'card') {
|
|
|
|
const currentRoute = route.matched[1];
|
|
|
|
const currentModule = toLowerCamel(currentRoute.name);
|
|
|
|
const moduleDef = routes.find((route) => toLowerCamel(route.name) === currentModule);
|
2022-11-21 16:33:21 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
addChildren(currentModule, moduleDef, items.value);
|
2022-03-15 09:33:28 +00:00
|
|
|
}
|
2022-11-24 06:21:45 +00:00
|
|
|
|
2022-11-29 06:10:04 +00:00
|
|
|
async function togglePinned(moduleName, event) {
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
|
|
await axios.post('StarredModules/toggleStarredModule', { moduleName });
|
|
|
|
navigation.togglePinned(moduleName);
|
2022-11-21 16:33:21 +00:00
|
|
|
|
2022-11-24 13:54:38 +00:00
|
|
|
quasar.notify({
|
|
|
|
message: t('globals.dataSaved'),
|
|
|
|
type: 'positive',
|
|
|
|
});
|
|
|
|
}
|
2022-11-24 06:21:45 +00:00
|
|
|
|
|
|
|
function isOpen(name) {
|
|
|
|
const { matched } = route;
|
|
|
|
|
|
|
|
return matched.some((item) => item.name === name);
|
|
|
|
}
|
2022-03-15 09:33:28 +00:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
2022-10-17 14:23:19 +00:00
|
|
|
<q-list padding>
|
2022-11-24 06:21:45 +00:00
|
|
|
<q-item-label v-if="$props.source === 'main'" header>
|
|
|
|
{{ t('globals.favoriteModules') }}
|
|
|
|
</q-item-label>
|
2022-11-24 13:54:38 +00:00
|
|
|
<!-- <template v-for="item in items" :key="item.name">
|
2022-11-24 06:21:45 +00:00
|
|
|
<template v-if="item.children">
|
|
|
|
<q-expansion-item
|
2022-11-24 13:54:38 +00:00
|
|
|
group="itemGroup"
|
2022-11-24 06:21:45 +00:00
|
|
|
class="module"
|
|
|
|
active-class="text-primary"
|
|
|
|
:label="item.title"
|
|
|
|
:to="{ name: item.name }"
|
|
|
|
expand-separator
|
|
|
|
:default-opened="isOpen(item.name)"
|
|
|
|
>
|
|
|
|
<template #header>
|
|
|
|
<q-item-section avatar>
|
|
|
|
<q-icon :name="item.icon"></q-icon>
|
2022-06-08 12:53:10 +00:00
|
|
|
</q-item-section>
|
2022-11-24 06:21:45 +00:00
|
|
|
<q-item-section>{{ t(item.title) }}</q-item-section>
|
2022-11-24 13:54:38 +00:00
|
|
|
<q-item-section side>
|
|
|
|
<div
|
|
|
|
@click="onToggleFavoriteModule(module.name, $event)"
|
|
|
|
class="row items-center"
|
|
|
|
v-if="module.name != 'dashboard'"
|
|
|
|
>
|
|
|
|
<q-icon name="vn:pin"></q-icon>
|
|
|
|
</div>
|
|
|
|
</q-item-section>
|
2022-11-24 06:21:45 +00:00
|
|
|
</template>
|
|
|
|
<template v-for="section in item.children" :key="section.name">
|
2022-11-24 13:54:38 +00:00
|
|
|
<q-item active-class="text-primary" :to="{ name: section.name }" clickable v-ripple>
|
2022-11-24 06:21:45 +00:00
|
|
|
<q-item-section avatar v-if="section.icon">
|
|
|
|
<q-icon :name="section.icon" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section avatar v-if="!item.icon">
|
|
|
|
<q-icon name="disabled_by_default" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section>{{ t(section.title) }}</q-item-section>
|
|
|
|
</q-item>
|
|
|
|
</template>
|
|
|
|
</q-expansion-item>
|
|
|
|
</template>
|
|
|
|
<template v-if="!item.children">
|
2022-11-24 13:54:38 +00:00
|
|
|
<q-item active-class="text-primary" :to="{ name: item.name }" clickable v-ripple>
|
2022-11-24 06:21:45 +00:00
|
|
|
<q-item-section avatar v-if="item.icon">
|
|
|
|
<q-icon :name="item.icon" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section avatar v-if="!item.icon">
|
|
|
|
<q-icon name="disabled_by_default" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section>{{ t(item.title) }}</q-item-section>
|
|
|
|
</q-item>
|
|
|
|
</template>
|
2022-11-24 13:54:38 +00:00
|
|
|
</template> -->
|
|
|
|
<q-separator />
|
|
|
|
<q-expansion-item :label="t('moduleIndex.allModules')">
|
|
|
|
<q-list padding>
|
|
|
|
<template v-for="item in items" :key="item.name">
|
|
|
|
<template v-if="item.children">
|
|
|
|
<q-expansion-item
|
|
|
|
group="itemGroup"
|
|
|
|
class="module"
|
|
|
|
active-class="text-primary"
|
|
|
|
:label="item.title"
|
|
|
|
:to="{ name: item.name }"
|
|
|
|
expand-separator
|
|
|
|
:default-opened="isOpen(item.name)"
|
|
|
|
>
|
|
|
|
<template #header>
|
|
|
|
<q-item-section avatar>
|
|
|
|
<q-icon :name="item.icon"></q-icon>
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section>{{ t(item.title) }}</q-item-section>
|
|
|
|
<q-item-section side>
|
|
|
|
<q-btn
|
2022-11-29 06:10:04 +00:00
|
|
|
@click="togglePinned(item.module, $event)"
|
2022-11-24 13:54:38 +00:00
|
|
|
icon="vn:pin"
|
|
|
|
size="xs"
|
|
|
|
flat
|
|
|
|
round
|
|
|
|
>
|
|
|
|
<q-tooltip>Pin this module to favorites</q-tooltip>
|
|
|
|
</q-btn>
|
|
|
|
</q-item-section>
|
|
|
|
</template>
|
|
|
|
<template v-for="section in item.children" :key="section.name">
|
|
|
|
<q-item active-class="text-primary" :to="{ name: section.name }" clickable v-ripple>
|
|
|
|
<q-item-section avatar v-if="section.icon">
|
|
|
|
<q-icon :name="section.icon" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section avatar v-if="!item.icon">
|
|
|
|
<q-icon name="disabled_by_default" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section>{{ t(section.title) }}</q-item-section>
|
|
|
|
</q-item>
|
|
|
|
</template>
|
|
|
|
</q-expansion-item>
|
|
|
|
</template>
|
|
|
|
<template v-if="!item.children">
|
|
|
|
<q-item active-class="text-primary" :to="{ name: item.name }" clickable v-ripple>
|
|
|
|
<q-item-section avatar v-if="item.icon">
|
|
|
|
<q-icon :name="item.icon" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section avatar v-if="!item.icon">
|
|
|
|
<q-icon name="disabled_by_default" />
|
|
|
|
</q-item-section>
|
|
|
|
<q-item-section>{{ t(item.title) }}</q-item-section>
|
|
|
|
</q-item>
|
|
|
|
</template>
|
|
|
|
</template>
|
|
|
|
</q-list>
|
|
|
|
</q-expansion-item>
|
|
|
|
<q-separator />
|
2022-11-21 16:33:21 +00:00
|
|
|
<!-- <template v-for="module in navigation.favorites.value" :key="module.title">-->
|
|
|
|
<!-- <div class="module" v-if="!module.children">-->
|
|
|
|
<!-- <q-item-->
|
|
|
|
<!-- clickable-->
|
|
|
|
<!-- v-ripple-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :key="module.title"-->
|
|
|
|
<!-- :to="{ name: module.stateName }"-->
|
|
|
|
<!-- v-if="!module.roles || !module.roles.length || hasAny(module.roles)"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-item-section avatar :if="module.icon">-->
|
|
|
|
<!-- <q-icon :name="module.icon" />-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${module.title}`) }}</q-item-section>-->
|
|
|
|
<!-- <q-item-section side>-->
|
|
|
|
<!-- <div @click="onToggleFavoriteModule(module.name, $event)" class="row items-center">-->
|
|
|
|
<!-- <q-icon name="vn:pin_off"></q-icon>-->
|
|
|
|
<!-- </div>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- </q-item>-->
|
|
|
|
<!-- </div>-->
|
|
|
|
|
|
|
|
<!-- <template v-if="module.children">-->
|
|
|
|
<!-- <q-expansion-item-->
|
|
|
|
<!-- class="module"-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :label="t(`${module.name}.pageTitles.${module.title}`)"-->
|
|
|
|
<!-- v-if="!module.roles || !module.roles.length || hasAny(module.roles)"-->
|
|
|
|
<!-- :to="{ name: module.stateName }"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <template #header>-->
|
|
|
|
<!-- <q-item-section avatar>-->
|
|
|
|
<!-- <q-icon :name="module.icon"></q-icon>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${module.title}`) }}</q-item-section>-->
|
|
|
|
<!-- <q-item-section side>-->
|
|
|
|
<!-- <div @click="onToggleFavoriteModule(module.name, $event)" class="row items-center">-->
|
|
|
|
<!-- <q-icon name="vn:pin_off"></q-icon>-->
|
|
|
|
<!-- </div>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- <template v-for="section in module.children" :key="section.title">-->
|
|
|
|
<!-- <q-item-->
|
|
|
|
<!-- clickable-->
|
|
|
|
<!-- v-ripple-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :to="{ name: section.stateName }"-->
|
|
|
|
<!-- v-if="!section.roles || !section.roles.length || hasAny(section.roles)"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-item-section avatar :if="section.icon">-->
|
|
|
|
<!-- <q-icon :name="section.icon" />-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${section.title}`) }}</q-item-section>-->
|
|
|
|
<!-- </q-item>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- </q-expansion-item>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- </template>-->
|
2022-10-17 14:23:19 +00:00
|
|
|
</q-list>
|
2022-06-08 12:53:10 +00:00
|
|
|
|
2022-11-21 16:33:21 +00:00
|
|
|
<!-- <q-expansion-item :label="t('moduleIndex.allModules')">-->
|
|
|
|
<!-- <q-list padding>-->
|
|
|
|
<!-- <template v-for="module in navigation.modules.value" :key="module.title">-->
|
|
|
|
<!-- <div class="module" v-if="!module.children">-->
|
|
|
|
<!-- <q-item-->
|
|
|
|
<!-- class="module"-->
|
|
|
|
<!-- clickable-->
|
|
|
|
<!-- v-ripple-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :key="module.title"-->
|
|
|
|
<!-- :to="{ name: module.stateName }"-->
|
|
|
|
<!-- v-if="!module.roles || !module.roles.length || hasAny(module.roles)"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-item-section avatar :if="module.icon">-->
|
|
|
|
<!-- <q-icon :name="module.icon" />-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${module.title}`) }}</q-item-section>-->
|
|
|
|
<!-- <q-item-section side>-->
|
|
|
|
<!-- <div-->
|
|
|
|
<!-- @click="onToggleFavoriteModule(module.name, $event)"-->
|
|
|
|
<!-- class="row items-center"-->
|
|
|
|
<!-- v-if="module.name != 'dashboard'"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-icon name="vn:pin"></q-icon>-->
|
|
|
|
<!-- </div>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- </q-item>-->
|
|
|
|
<!-- </div>-->
|
2022-06-08 12:53:10 +00:00
|
|
|
|
2022-11-21 16:33:21 +00:00
|
|
|
<!-- <template v-if="module.children">-->
|
|
|
|
<!-- <q-expansion-item-->
|
|
|
|
<!-- class="module"-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :label="t(`${module.name}.pageTitles.${module.title}`)"-->
|
|
|
|
<!-- v-if="!module.roles || !module.roles.length || hasAny(module.roles)"-->
|
|
|
|
<!-- :to="{ name: module.stateName }"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <template #header>-->
|
|
|
|
<!-- <q-item-section avatar>-->
|
|
|
|
<!-- <q-icon :name="module.icon"></q-icon>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${module.title}`) }}</q-item-section>-->
|
|
|
|
<!-- <q-item-section side>-->
|
|
|
|
<!-- <div-->
|
|
|
|
<!-- @click="onToggleFavoriteModule(module.name, $event)"-->
|
|
|
|
<!-- class="row items-center"-->
|
|
|
|
<!-- v-if="module.name != 'dashboard'"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-icon name="vn:pin"></q-icon>-->
|
|
|
|
<!-- </div>-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- <template v-for="section in module.children" :key="section.title">-->
|
|
|
|
<!-- <q-item-->
|
|
|
|
<!-- clickable-->
|
|
|
|
<!-- v-ripple-->
|
|
|
|
<!-- active-class="text-primary"-->
|
|
|
|
<!-- :to="{ name: section.stateName }"-->
|
|
|
|
<!-- v-if="!section.roles || !section.roles.length || hasAny(section.roles)"-->
|
|
|
|
<!-- >-->
|
|
|
|
<!-- <q-item-section avatar :if="section.icon">-->
|
|
|
|
<!-- <q-icon :name="section.icon" />-->
|
|
|
|
<!-- </q-item-section>-->
|
|
|
|
<!-- <q-item-section>{{ t(`${module.name}.pageTitles.${section.title}`) }}</q-item-section>-->
|
|
|
|
<!-- </q-item>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- </q-expansion-item>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- </template>-->
|
|
|
|
<!-- </q-list>-->
|
|
|
|
<!-- </q-expansion-item>-->
|
2022-05-17 12:58:01 +00:00
|
|
|
</template>
|
2022-06-08 12:53:10 +00:00
|
|
|
|
2022-11-24 06:21:45 +00:00
|
|
|
<!--<style>-->
|
|
|
|
<!--.module .icon-pin,-->
|
|
|
|
<!--.module .icon-pin_off {-->
|
|
|
|
<!-- visibility: hidden;-->
|
|
|
|
<!--}-->
|
|
|
|
<!--.module:hover .icon-pin,-->
|
|
|
|
<!--.module:hover .icon-pin_off {-->
|
|
|
|
<!-- visibility: visible;-->
|
|
|
|
<!--}-->
|
|
|
|
<!--</style>-->
|