# @fesjs/plugin-qiankun Fes.js plugin for [qiankun](https://qiankun.umijs.org/),参考[@umijs/plugin-qiankun](https://umijs.org/zh-CN/plugins/plugin-qiankun#MicroApp) 实现,喜欢 React 的同学推荐直接用 Umi。 ## 启用方式 在 `package.json` 中引入依赖: ```json { "dependencies": { "@fesjs/fes": "^2.0.0", "@fesjs/plugin-qiankun": "^2.0.0" }, } ``` ## 介绍 有一种痛叫接手老项目,技术栈老旧,内容多,还要继续维护~ 可能目前迁移、升级老项目最好的解决方案就是微前端。`plugin-qiankun` 是基于 `qiankun` 实现的 Fes.js 微前端解决方案。 ## 主应用配置 ### 第一步:注册子应用 ```js export default { qiankun: { main: { // 注册子应用信息 apps: [ { name: 'app1', // 唯一 id entry: '//localhost:8001', // html entry props: {} // 传递给子应用的数据 }, { name: 'app2', // 唯一 id entry: '//localhost:8002', // html entry }, ], }, }, }; ``` ### 第二步:装载子应用 #### 使用路由绑定的方式 :::warning 主应用和子应用需要自行适配路由路径!!!待完善... ::: 假设我们的系统之前有这样的一些路由: ```js export default { router: { routes: [{ "path": "/", "component": () => import('@/src/.fes/plugin-layout/index.js'), "children": [ { "path": "/onepiece", "component": () => import('@/pages/onepiece'), "name": "onepiece", "meta": { "name": "onepiece", "title": "onepiece" } } ] }] } } ``` 我们现在想在 `/son` 加载子应用 `app1`,只需要增加这样一些配置即可: ```js {16-23} export default { router: { routes: [{ "path": "/", "component": () => import('@/src/.fes/plugin-layout/index.js'), "children": [ { "path": "/onepiece", "component": () => import('@/pages/onepiece'), "name": "onepiece", "meta": { "name": "onepiece", "title": "onepiece" } }, { "path": "/son", "meta": { "name": "son", "title": "子应用", "microApp": "app1" } } ] }] } } ``` 当前我们依然提倡约定路由的方式,在`src/pages` 目录新建 `son.vue`: ```vue { "name": "son", "title": "子应用", "microApp": "app1" } ``` #### 使用 `` 组件的方式 :::tip 建议使用这种方式来引入不带路由的子应用。 否则请自行关注子应用依赖的路由跟当前浏览器 url 是否能正确匹配上,否则很容易出现子应用加载了,但是页面没有渲染出来的情况。 ::: ```vue ``` #### 使用 `` 组件的方式 如果我们的路由使用 `history` 模式,那么在使用乾坤时还算方便,主应用和子应用的路由根据base可以很方便的匹配起来,而且不存在冲突。但是当我们使用 `hash` 模式时,就问题很大,主应用和子应用的路由必须一样才可以匹配上,用起来贼不方便。而且不能在一个页面上同时加载多个子应用,路由存在冲突!这时候,`` 出现了,完美解决上面的问题。 `` 相比 `` ,需要多传入 `url` 参数,用于指定加载子应用什么路由页面。 ```vue ``` ## 子应用配置 ### 第一步:插件注册 ```js export default { qiankun: { micro: {}, } }; ``` ### 第二步:配置运行时生命周期钩子(可选) 插件会自动为你创建好 `qiankun` 子应用需要的生命周期钩子,但是如果你想在生命周期期间加一些自定义逻辑,可以在子应用的 `src/app.js` 里导出 `qiankun` 对象,并实现每一个生命周期钩子,其中钩子函数的入参 `props` 由主应用自动注入。 ```js export const qiankun = { // 应用加载之前 async bootstrap(props) { console.log('app1 bootstrap', props); }, // 应用 render 之前触发 async mount(props) { console.log('app1 mount', props); }, // 当 props 更新时触发 async update(props){ console.log('app1 update', props); }, // 应用卸载之后触发 async unmount(props) { console.log('app1 unmount', props); }, }; ``` ## 父子应用通讯 有两种方式实现 ### 配合 [useModel](./model.md) 使用 确保已经安装了 `@fesjs/plugin-model`: ```json { "dependencies": { "@fesjs/fes": "^2.0.0", "@fesjs/plugin-model": "^2.0.0" }, } ``` #### 主应用传递 props - 如果使用 `MicroApp` 组件模式消费子应用,直接通过 props 传递即可: ```vue ``` - 如果使用路由绑定式消费子应用,那么约定`src/models/qiankunStateForMicro.js` 的模型数据将作为 `props` 船体给子应用,如: ```js import { reactive } from 'vue'; export default () => { const state = reactive({ c: 1 }); return { state }; }; ``` #### 子应用消费 props 子应用中会自动生成一个全局名为 `qiankunStateFromMain` 的 `model`, 可以在任意组件中获取主应用透传的 `props` 的值。 ```vue ``` ### 基于 props 传递 - 主应用使用 props 的模式传递数据(参考主应用装载子应用配置一节) - 子应用在生命周期钩子中获取 props 消费数据(参考子应用运行时配置一节) ### MicroApp | 属性 | 说明 | 类型 | 默认值 | | ---- | ----------- | ------------- | ---------- | | name | 子应用名称,传入`qiankun.main.apps`配置中的`name` | String | - | | settings | 子应用配置信息 | Object | {} | | props | 传入子应用的参数 | Object | {} | | lifeCycles | 子应用生命周期钩子 | Object | {} | | cacheName | 子应用缓存名称,配置后根据`name`+`cacheName`缓存子应用实例 | Object | - | ### MicroAppWithMemoHistory | 属性 | 说明 | 类型 | 默认值 | | ---- | ----------- | ------------- | ---------- | | name | 子应用名称,传入`qiankun.main.apps`配置中的`name` | String | - | | settings | 子应用配置信息 | Object | {} | | props | 传入子应用的参数 | Object | {} | | lifeCycles | 子应用生命周期钩子 | Object | {} | | cacheName | 子应用缓存名称,配置后根据`name`+`cacheName`缓存子应用实例 | Object | - | | url | 子应用的路由地址 | String | - |