mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-06 03:59:53 +08:00
feat: qiankun支持多页签keepalive (#117)
* fix: 判断页面权限的path统一从match中拿路由path * fix: qiankun主应用不改rootContainer * fix: 消除initialState为空时的warn * refactor: modifyCreateHistroy更改为modifyCreateHistory * fix: qiankun支持多页签keepalive
This commit is contained in:
parent
7632e12785
commit
9181bf50ac
@ -4,10 +4,10 @@
|
|||||||
对于前端应用来说,权限就是页面、页面元素是否可见。
|
对于前端应用来说,权限就是页面、页面元素是否可见。
|
||||||
|
|
||||||
### 资源
|
### 资源
|
||||||
Fes.js 把页面、页面元素统一叫做资源,每个资源都有 `accessId`:
|
Fes.js 把页面、页面元素统一叫做资源,用资源 ID 来识别区分他们:
|
||||||
- 页面的 `accessId` 默认是页面的路由 `path` 。比如页面 `pages/a.vue` 的路由 `path` 是 `/a`。当页面访问 `/a` 时会渲染当前页面,`/a` 也就是页面的 `accessId`。
|
- 页面的资源 ID 默认是页面的路由 `path` 。比如页面 `pages/a.vue` 的路由 `path` 是 `/a`。当页面访问 `/a` 时会渲染当前页面,`/a` 也就是页面的 `accessId`。
|
||||||
|
|
||||||
- 页面元素的 `accessId` 没有默认值,由我们自定义。
|
- 页面元素的资源 ID 没有默认值,需要自定义。
|
||||||
```vue
|
```vue
|
||||||
<template>
|
<template>
|
||||||
<access :id="accessId"> accessOnepicess1 <input /> </access>
|
<access :id="accessId"> accessOnepicess1 <input /> </access>
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { plugin, ApplyPluginsType } from '@@/core/coreExports';
|
import { plugin, ApplyPluginsType } from '@@/core/coreExports';
|
||||||
|
// eslint-disable-next-line import/extensions
|
||||||
import { access, install } from './core';
|
import { access, install } from './core';
|
||||||
|
|
||||||
export function onRouterCreated({ router }) {
|
export function onRouterCreated({ router }) {
|
||||||
@ -9,25 +10,35 @@ export function onRouterCreated({ router }) {
|
|||||||
initialValue: {}
|
initialValue: {}
|
||||||
});
|
});
|
||||||
if (to.matched.length === 0) {
|
if (to.matched.length === 0) {
|
||||||
if (runtimeConfig.noFoundHandler && typeof runtimeConfig.noFoundHandler === 'function') {
|
if (
|
||||||
|
runtimeConfig.noFoundHandler
|
||||||
|
&& typeof runtimeConfig.noFoundHandler === 'function'
|
||||||
|
) {
|
||||||
return runtimeConfig.noFoundHandler({
|
return runtimeConfig.noFoundHandler({
|
||||||
router, to, from, next
|
router,
|
||||||
|
to,
|
||||||
|
from,
|
||||||
|
next
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
return next(false);
|
||||||
}
|
}
|
||||||
let path;
|
// path是匹配路由的path,不是页面hash
|
||||||
if (to.matched.length === 1) {
|
const canRoute = await access.hasAccess(
|
||||||
path = to.matched[0].path;
|
to.matched[to.matched.length - 1].path
|
||||||
} else {
|
);
|
||||||
path = to.path;
|
|
||||||
}
|
|
||||||
const canRoute = await access.hasAccess(path);
|
|
||||||
if (canRoute) {
|
if (canRoute) {
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
if (runtimeConfig.unAccessHandler && typeof runtimeConfig.unAccessHandler === 'function') {
|
if (
|
||||||
|
runtimeConfig.unAccessHandler
|
||||||
|
&& typeof runtimeConfig.unAccessHandler === 'function'
|
||||||
|
) {
|
||||||
return runtimeConfig.unAccessHandler({
|
return runtimeConfig.unAccessHandler({
|
||||||
router, to, from, next
|
router,
|
||||||
|
to,
|
||||||
|
from,
|
||||||
|
next
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
next(false);
|
next(false);
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
main
|
main
|
||||||
<FTabs v-model="activeKey">
|
<FTabs v-model="activeKey">
|
||||||
<FTabPane name="Tab 1" value="1">
|
<FTabPane name="Tab 1" value="1">
|
||||||
<MicroAppWithMemoHistory key="1" name="app1" url="/app1" />
|
<MicroAppWithMemoHistory key="1" name="app1" url="/app1" a="1" />
|
||||||
</FTabPane>
|
</FTabPane>
|
||||||
<FTabPane name="Tab 2" value="2">
|
<FTabPane name="Tab 2" value="2">
|
||||||
<MicroAppWithMemoHistory key="2" name="app1" url="/app1/test" />
|
<MicroAppWithMemoHistory key="2" name="app1" url="/app1/test" />
|
||||||
|
@ -32,7 +32,7 @@
|
|||||||
"@fesjs/utils": "^2.0.4",
|
"@fesjs/utils": "^2.0.4",
|
||||||
"address": "^1.1.2",
|
"address": "^1.1.2",
|
||||||
"lodash-es": "^4.17.15",
|
"lodash-es": "^4.17.15",
|
||||||
"qiankun": "^2.4.4"
|
"qiankun": "^2.7.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"npm-run-all": "^4.1.5"
|
"npm-run-all": "^4.1.5"
|
||||||
|
@ -26,6 +26,7 @@ export default function (api) {
|
|||||||
enableBy: () => isMasterEnable(api)
|
enableBy: () => isMasterEnable(api)
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 避免跟子应用冲突
|
||||||
api.modifyDefaultConfig(config => ({
|
api.modifyDefaultConfig(config => ({
|
||||||
...config,
|
...config,
|
||||||
mountElementId: defaultMainRootId
|
mountElementId: defaultMainRootId
|
||||||
|
@ -11,26 +11,26 @@ import { loadMicroApp } from "{{{QIANKUN}}}";
|
|||||||
import {mergeWith} from "{{{LODASH_ES}}}";
|
import {mergeWith} from "{{{LODASH_ES}}}";
|
||||||
// eslint-disable-next-line import/extensions
|
// eslint-disable-next-line import/extensions
|
||||||
import { getMasterOptions } from "./masterOptions";
|
import { getMasterOptions } from "./masterOptions";
|
||||||
import { onBeforeRouteLeave } from "@@/core/coreExports";
|
|
||||||
|
|
||||||
async function unmountMicroApp(microApp) {
|
function unmountMicroApp(microApp) {
|
||||||
if (microApp) {
|
if (!microApp) {
|
||||||
const status = microApp.getStatus();
|
return;
|
||||||
if(status === 'MOUNTED'){
|
}
|
||||||
await microApp.unmount();
|
const status = microApp.getStatus();
|
||||||
}
|
if (status === 'MOUNTED') {
|
||||||
|
microApp.unmount();
|
||||||
}
|
}
|
||||||
return Promise.resolve();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const MicroApp = defineComponent({
|
export const MicroApp = defineComponent({
|
||||||
props: {
|
props: {
|
||||||
name: {
|
name: {
|
||||||
type: String,
|
type: String,
|
||||||
required: true,
|
required: true
|
||||||
},
|
},
|
||||||
settings: Object,
|
settings: Object,
|
||||||
lifeCycles: Object,
|
props: Object,
|
||||||
|
lifeCycles: Object
|
||||||
},
|
},
|
||||||
setup(props, { attrs }) {
|
setup(props, { attrs }) {
|
||||||
const {
|
const {
|
||||||
@ -40,8 +40,6 @@ export const MicroApp = defineComponent({
|
|||||||
...globalSettings
|
...globalSettings
|
||||||
} = getMasterOptions();
|
} = getMasterOptions();
|
||||||
|
|
||||||
const stateForSlave = reactive({});
|
|
||||||
|
|
||||||
// 挂载节点
|
// 挂载节点
|
||||||
const containerRef = ref(null);
|
const containerRef = ref(null);
|
||||||
const microAppRef = ref();
|
const microAppRef = ref();
|
||||||
@ -68,6 +66,14 @@ export const MicroApp = defineComponent({
|
|||||||
|
|
||||||
const propsFromParams = attrs;
|
const propsFromParams = attrs;
|
||||||
|
|
||||||
|
const propsConfigRef = computed(() => {
|
||||||
|
return {
|
||||||
|
...propsFromConfigRef.value,
|
||||||
|
...props.props,
|
||||||
|
...propsFromParams
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
// 只有当name变化时才重新加载新的子应用
|
// 只有当name变化时才重新加载新的子应用
|
||||||
const loadApp = () => {
|
const loadApp = () => {
|
||||||
const appConfig = appConfigRef.value;
|
const appConfig = appConfigRef.value;
|
||||||
@ -75,21 +81,21 @@ export const MicroApp = defineComponent({
|
|||||||
// 加载新的
|
// 加载新的
|
||||||
microAppRef.value = loadMicroApp(
|
microAppRef.value = loadMicroApp(
|
||||||
{
|
{
|
||||||
name: name,
|
// 保证唯一
|
||||||
|
name: `${name}_${Date.now()}`,
|
||||||
entry: entry,
|
entry: entry,
|
||||||
container: containerRef.value,
|
container: containerRef.value,
|
||||||
props: {
|
props: propsConfigRef.value
|
||||||
...propsFromConfigRef.value,
|
|
||||||
...stateForSlave,
|
|
||||||
...propsFromParams,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
...globalSettings,
|
...globalSettings,
|
||||||
...(props.settings || {}),
|
...(props.settings || {})
|
||||||
},
|
},
|
||||||
mergeWith({}, globalLifeCycles || {}, props.lifeCycles || {}, (v1, v2) =>
|
mergeWith(
|
||||||
concat(v1 ?? [], v2 ?? [])
|
{},
|
||||||
|
globalLifeCycles || {},
|
||||||
|
props.lifeCycles || {},
|
||||||
|
(v1, v2) => concat(v1 ?? [], v2 ?? [])
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@ -106,9 +112,9 @@ export const MicroApp = defineComponent({
|
|||||||
updatingPromiseRef.value = updatingPromiseRef.value.then(
|
updatingPromiseRef.value = updatingPromiseRef.value.then(
|
||||||
() => {
|
() => {
|
||||||
const canUpdate = (app) =>
|
const canUpdate = (app) =>
|
||||||
app?.update && app.getStatus() === "MOUNTED";
|
app?.update && app.getStatus() === 'MOUNTED';
|
||||||
if (canUpdate(microApp)) {
|
if (canUpdate(microApp)) {
|
||||||
if (process.env.NODE_ENV === "development") {
|
if (process.env.NODE_ENV === 'development') {
|
||||||
if (
|
if (
|
||||||
Date.now() -
|
Date.now() -
|
||||||
updatingTimestampRef.value <
|
updatingTimestampRef.value <
|
||||||
@ -127,11 +133,7 @@ export const MicroApp = defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 返回 microApp.update 形成链式调用
|
// 返回 microApp.update 形成链式调用
|
||||||
return microApp.update({
|
return microApp.update(propsConfigRef.value);
|
||||||
...propsFromConfigRef.value,
|
|
||||||
...stateForSlave,
|
|
||||||
...propsFromParams,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -152,16 +154,10 @@ export const MicroApp = defineComponent({
|
|||||||
loadApp();
|
loadApp();
|
||||||
});
|
});
|
||||||
|
|
||||||
onBeforeRouteLeave(async () => {
|
watch(propsConfigRef, () => {
|
||||||
return await unmountMicroApp(microAppRef.value);
|
|
||||||
});
|
|
||||||
|
|
||||||
watch(()=>{
|
|
||||||
return {...{}, ...propsFromConfigRef.value, ...stateForSlave, ...propsFromParams}
|
|
||||||
}, () => {
|
|
||||||
updateApp();
|
updateApp();
|
||||||
});
|
});
|
||||||
|
|
||||||
return () => <div ref={containerRef}></div>;
|
return () => <div ref={containerRef}></div>;
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
@ -15,6 +15,7 @@ export const MicroAppWithMemoHistory = defineComponent({
|
|||||||
required: true
|
required: true
|
||||||
},
|
},
|
||||||
settings: Object,
|
settings: Object,
|
||||||
|
props: Object,
|
||||||
lifeCycles: Object,
|
lifeCycles: Object,
|
||||||
url: String
|
url: String
|
||||||
},
|
},
|
||||||
|
@ -110,11 +110,7 @@ export function genUpdate() {
|
|||||||
export function genUnmount() {
|
export function genUnmount() {
|
||||||
return async (props) => {
|
return async (props) => {
|
||||||
const history = getHistory();
|
const history = getHistory();
|
||||||
history.destroy();
|
history.destroy(); // 会触发app.unmount
|
||||||
if (cacheAppPromise) {
|
|
||||||
const app = await cacheAppPromise;
|
|
||||||
app.unmount();
|
|
||||||
}
|
|
||||||
destroyRouter();
|
destroyRouter();
|
||||||
const slaveRuntime = getSlaveRuntime();
|
const slaveRuntime = getSlaveRuntime();
|
||||||
if (slaveRuntime.unmount) {
|
if (slaveRuntime.unmount) {
|
||||||
|
@ -14,7 +14,7 @@ export function modifyClientRenderOpts(memo) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function modifyCreateHistroy(memo) {
|
export function modifyCreateHistory(memo) {
|
||||||
if (history.url) {
|
if (history.url) {
|
||||||
return createMemoryHistory
|
return createMemoryHistory
|
||||||
}
|
}
|
||||||
|
@ -29,8 +29,8 @@ export default function (api) {
|
|||||||
'render',
|
'render',
|
||||||
// 修改路由
|
// 修改路由
|
||||||
'patchRoutes',
|
'patchRoutes',
|
||||||
// 修改histror
|
// 修改history
|
||||||
'modifyCreateHistroy',
|
'modifyCreateHistory',
|
||||||
// 生成router时触发
|
// 生成router时触发
|
||||||
'onRouterCreated'
|
'onRouterCreated'
|
||||||
]
|
]
|
||||||
|
@ -27,7 +27,7 @@ const renderClient = (opts = {}) => {
|
|||||||
|
|
||||||
const app = createApp(rootContainer);
|
const app = createApp(rootContainer);
|
||||||
// initialState是响应式的,后期可以更改
|
// initialState是响应式的,后期可以更改
|
||||||
app.provide("initialState", reactive(initialState));
|
app.provide("initialState", reactive(initialState ?? {}));
|
||||||
|
|
||||||
plugin.applyPlugins({
|
plugin.applyPlugins({
|
||||||
key: 'onAppCreated',
|
key: 'onAppCreated',
|
||||||
|
@ -14,7 +14,7 @@ export const createRouter = (routes) => {
|
|||||||
return router;
|
return router;
|
||||||
}
|
}
|
||||||
const createHistory = plugin.applyPlugins({
|
const createHistory = plugin.applyPlugins({
|
||||||
key: 'modifyCreateHistroy',
|
key: 'modifyCreateHistory',
|
||||||
type: ApplyPluginsType.modify,
|
type: ApplyPluginsType.modify,
|
||||||
args: {
|
args: {
|
||||||
base: ROUTER_BASE
|
base: ROUTER_BASE
|
||||||
|
Loading…
x
Reference in New Issue
Block a user