mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-06 03:59:53 +08:00
fix: 修复layout菜单折叠不更新状态问题 (#184)
* fix: 修复layout菜单折叠不更新状态问题 * fix: 修复accessApi.setAccess['/403']执行多次问题 * feat: 手动切换当前路由,菜单也会自动展开 * fix: 修复menu问题
This commit is contained in:
parent
eb4c2d92f9
commit
a5d422ea04
@ -3,8 +3,6 @@ import createDirective from "./createDirective";
|
|||||||
import createComponent from "./createComponent";
|
import createComponent from "./createComponent";
|
||||||
import {isPlainObject} from "{{{ lodashPath }}}";
|
import {isPlainObject} from "{{{ lodashPath }}}";
|
||||||
|
|
||||||
const accessKey = Symbol("plugin-access");
|
|
||||||
|
|
||||||
function isPromise(obj) {
|
function isPromise(obj) {
|
||||||
return (
|
return (
|
||||||
!!obj &&
|
!!obj &&
|
||||||
@ -132,14 +130,6 @@ const hasAccess = async (path) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const install = (app) => {
|
export const install = (app) => {
|
||||||
const allowPageIds = computed(getAllowAccessIds);
|
|
||||||
const useAccess = (path) => {
|
|
||||||
const result = computed(() => {
|
|
||||||
return match(unref(path), allowPageIds.value);
|
|
||||||
});
|
|
||||||
return result;
|
|
||||||
};
|
|
||||||
app.provide(accessKey, useAccess);
|
|
||||||
app.directive("access", createDirective(useAccess));
|
app.directive("access", createDirective(useAccess));
|
||||||
app.component("Access", createComponent(useAccess));
|
app.component("Access", createComponent(useAccess));
|
||||||
};
|
};
|
||||||
@ -153,6 +143,14 @@ export const access = {
|
|||||||
getAccess: getAllowAccessIds,
|
getAccess: getAllowAccessIds,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const hasAccessSync = (path) => {
|
||||||
|
return match(unref(path), getAllowAccessIds());
|
||||||
|
}
|
||||||
|
|
||||||
export const useAccess = (path) => {
|
export const useAccess = (path) => {
|
||||||
return inject(accessKey)(path);
|
const allowPageIds = computed(getAllowAccessIds);
|
||||||
|
const result = computed(() => {
|
||||||
|
return match(unref(path), allowPageIds.value);
|
||||||
|
});
|
||||||
|
return result;
|
||||||
};
|
};
|
||||||
|
@ -1,32 +1,29 @@
|
|||||||
import { computed, ref } from 'vue';
|
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
import { useAccess } from '../../plugin-access/core';
|
import { hasAccessSync } from '../../plugin-access/core';
|
||||||
|
|
||||||
if (!useAccess) {
|
if (!hasAccessSync) {
|
||||||
throw new Error('[plugin-layout]: pLugin-layout depends on plugin-access,please install plugin-access first!');
|
throw new Error('[plugin-layout]: pLugin-layout depends on plugin-access,please install plugin-access first!');
|
||||||
}
|
}
|
||||||
|
|
||||||
export const hasAccessByMenuItem = (item) => {
|
export const hasAccessByMenuItem = (item) => {
|
||||||
const hasChild = item.children && item.children.length;
|
const hasChild = item.children && item.children.length;
|
||||||
if (item.path && !hasChild) {
|
if (item.path && !hasChild) {
|
||||||
return useAccess(item.path);
|
return hasAccessSync(item.path);
|
||||||
}
|
}
|
||||||
if (hasChild) {
|
if (hasChild) {
|
||||||
return computed(() =>
|
return item.children.some((child) => {
|
||||||
item.children.some((child) => {
|
|
||||||
const rst = hasAccessByMenuItem(child);
|
const rst = hasAccessByMenuItem(child);
|
||||||
return rst && rst.value;
|
return rst;
|
||||||
}),
|
});
|
||||||
);
|
|
||||||
}
|
}
|
||||||
return ref(true);
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const transform = (menus) =>
|
export const transform = (menus) =>
|
||||||
menus
|
menus
|
||||||
.map((menu) => {
|
.map((menu) => {
|
||||||
const hasAccess = hasAccessByMenuItem(menu);
|
const hasAccess = hasAccessByMenuItem(menu);
|
||||||
if (!hasAccess.value) {
|
if (!hasAccess) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (menu.children) {
|
if (menu.children) {
|
||||||
|
@ -6,7 +6,7 @@ export const transTitle = (name) => {
|
|||||||
}
|
}
|
||||||
const sharedLocale = plugin.getShared('locale');
|
const sharedLocale = plugin.getShared('locale');
|
||||||
if (sharedLocale) {
|
if (sharedLocale) {
|
||||||
const { t } = sharedLocale.useI18n();
|
const { t } = sharedLocale.locale;
|
||||||
return t(name.slice(1));
|
return t(name.slice(1));
|
||||||
}
|
}
|
||||||
return name;
|
return name;
|
||||||
|
@ -7,10 +7,11 @@ if (!accessApi) {
|
|||||||
throw new Error('[plugin-layout]: plugin-layout depends on plugin-access,please install plugin-access first!');
|
throw new Error('[plugin-layout]: plugin-layout depends on plugin-access,please install plugin-access first!');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const accessIds = accessApi.getAccess();
|
||||||
|
accessApi.setAccess(accessIds.concat(['/403', '/404']));
|
||||||
|
|
||||||
export const access = (memo) => {
|
export const access = (memo) => {
|
||||||
const runtimeConfig = getConfig();
|
const runtimeConfig = getConfig();
|
||||||
const accessIds = accessApi.getAccess();
|
|
||||||
accessApi.setAccess(accessIds.concat(['/403', '/404']));
|
|
||||||
return {
|
return {
|
||||||
unAccessHandler({ router, to, from, next }) {
|
unAccessHandler({ router, to, from, next }) {
|
||||||
if (runtimeConfig.unAccessHandler && typeof runtimeConfig.unAccessHandler === 'function') {
|
if (runtimeConfig.unAccessHandler && typeof runtimeConfig.unAccessHandler === 'function') {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<f-menu
|
<f-menu
|
||||||
|
v-model:expandedKeys="expandedKeysRef"
|
||||||
:modelValue="activePath"
|
:modelValue="activePath"
|
||||||
:expandedKeys="defaultExpandMenu"
|
|
||||||
:inverted="inverted"
|
:inverted="inverted"
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
:options="transformedMenus"
|
:options="transformedMenus"
|
||||||
@ -12,7 +12,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { computed, h } from 'vue';
|
import { computed, h, ref, watch } from 'vue';
|
||||||
import { FMenu } from '@fesjs/fes-design';
|
import { FMenu } from '@fesjs/fes-design';
|
||||||
import { useRoute, useRouter } from '@@/core/coreExports';
|
import { useRoute, useRouter } from '@@/core/coreExports';
|
||||||
import { transform as transformByAccess } from '../helpers/pluginAccess';
|
import { transform as transformByAccess } from '../helpers/pluginAccess';
|
||||||
@ -20,12 +20,12 @@ import { transform as transformByLocale } from '../helpers/pluginLocale';
|
|||||||
import { flatNodes } from '../helpers/utils';
|
import { flatNodes } from '../helpers/utils';
|
||||||
import MenuIcon from './MenuIcon.vue';
|
import MenuIcon from './MenuIcon.vue';
|
||||||
|
|
||||||
const transform = (menus) =>
|
const transform = (menus, level = 1) =>
|
||||||
menus.map((menu) => {
|
menus.map((menu, index) => {
|
||||||
const copy = {
|
const copy = {
|
||||||
...menu,
|
...menu,
|
||||||
label: menu.title,
|
label: menu.title,
|
||||||
value: menu.path || Date.now(),
|
value: menu.path || `${level}_${index}`,
|
||||||
};
|
};
|
||||||
if (menu.icon) {
|
if (menu.icon) {
|
||||||
copy.icon = () =>
|
copy.icon = () =>
|
||||||
@ -34,7 +34,7 @@ const transform = (menus) =>
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (menu.children) {
|
if (menu.children) {
|
||||||
copy.children = transform(menu.children);
|
copy.children = transform(menu.children, level + 1);
|
||||||
}
|
}
|
||||||
return copy;
|
return copy;
|
||||||
});
|
});
|
||||||
@ -94,10 +94,15 @@ export default {
|
|||||||
}
|
}
|
||||||
return matchMenus[0].path;
|
return matchMenus[0].path;
|
||||||
});
|
});
|
||||||
const defaultExpandMenu = computed(() => {
|
|
||||||
|
const expandedKeysRef = ref(props.expandedKeys);
|
||||||
|
|
||||||
|
watch(
|
||||||
|
[menuArray, activePath],
|
||||||
|
() => {
|
||||||
let index = menuArray.value.findIndex((item) => item.value === activePath.value);
|
let index = menuArray.value.findIndex((item) => item.value === activePath.value);
|
||||||
if (index === -1) {
|
if (index === -1) {
|
||||||
return props.expandedKeys;
|
return;
|
||||||
}
|
}
|
||||||
const activeMenu = menuArray.value[index];
|
const activeMenu = menuArray.value[index];
|
||||||
const arr = [activeMenu];
|
const arr = [activeMenu];
|
||||||
@ -108,8 +113,15 @@ export default {
|
|||||||
arr.push(lastMenu);
|
arr.push(lastMenu);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return props.expandedKeys.concat(arr.map((item) => item.value));
|
expandedKeysRef.value = expandedKeysRef.value.concat(
|
||||||
});
|
arr.filter((item) => !expandedKeysRef.value.includes(item.value)).map((item) => item.value),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
{
|
||||||
|
immediate: true,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
const onMenuClick = (e) => {
|
const onMenuClick = (e) => {
|
||||||
const path = e.value;
|
const path = e.value;
|
||||||
if (/^https?:\/\//.test(path)) {
|
if (/^https?:\/\//.test(path)) {
|
||||||
@ -120,9 +132,10 @@ export default {
|
|||||||
console.warn('[plugin-layout]: 菜单的path只能使以http(s)开头的网址或者路由地址');
|
console.warn('[plugin-layout]: 菜单的path只能使以http(s)开头的网址或者路由地址');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
activePath,
|
activePath,
|
||||||
defaultExpandMenu,
|
expandedKeysRef,
|
||||||
transformedMenus,
|
transformedMenus,
|
||||||
onMenuClick,
|
onMenuClick,
|
||||||
};
|
};
|
||||||
|
@ -85,19 +85,19 @@ const getAllLocales = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const install = (app) => {
|
const install = (app) => {
|
||||||
const runtimeConfig = plugin.applyPlugins({
|
|
||||||
key: "locale",
|
|
||||||
type: ApplyPluginsType.modify,
|
|
||||||
initialValue: {},
|
|
||||||
});
|
|
||||||
app.use(i18n);
|
app.use(i18n);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const t = (key) => {
|
||||||
|
return i18n.global.t(key)
|
||||||
|
}
|
||||||
|
|
||||||
const locale = {
|
const locale = {
|
||||||
setLocale,
|
setLocale,
|
||||||
addLocale,
|
addLocale,
|
||||||
getAllLocales,
|
getAllLocales,
|
||||||
messages,
|
messages,
|
||||||
|
t
|
||||||
};
|
};
|
||||||
|
|
||||||
export { useI18n, locale, install };
|
export { useI18n, locale, install };
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import { plugin } from '@@/core/coreExports';
|
import { plugin } from '@@/core/coreExports';
|
||||||
// eslint-disable-next-line import/extensions
|
// eslint-disable-next-line import/extensions
|
||||||
import { useI18n, install } from './core';
|
import { useI18n, locale, install } from './core';
|
||||||
import SelectLang from './views/SelectLang.vue';
|
import SelectLang from './views/SelectLang.vue';
|
||||||
|
|
||||||
// 共享出去
|
// 共享出去
|
||||||
plugin.share('locale', { useI18n, SelectLang });
|
plugin.share('locale', { useI18n, locale, SelectLang });
|
||||||
|
|
||||||
export function onAppCreated({ app }) {
|
export function onAppCreated({ app }) {
|
||||||
install(app);
|
install(app);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user