@umijs/plugin-qiankun

Umi plugin for qiankun.

Examples

导航是主应用,App1 和 App2 是子应用,

@umijs-plugin-qiankun - 图1

Features

  • ✔︎ 基于 qiankun
  • ✔︎ 支持主应用和子应用都用 umi
  • ✔︎ 支持主子应用 browser、hash 等多种 history 模式
  • ✔︎ 父子应用通讯
  • ✔︎ 子应用运行时配置自定义 bootstrap()mount()unmount()
  • ✔︎ 主应用、子应用联调

Usage

主应用

构建期

  1. export default {
  2. qiankun: {
  3. master: {
  4. // 注册子应用信息
  5. apps: [
  6. {
  7. name: 'app1', // 唯一 id
  8. entry: '//localhost:7001', // html entry
  9. base: '/app1', // app1 的路由前缀,通过这个前缀判断是否要启动该应用,通常跟子应用的 base 保持一致
  10. history: 'browser', // 子应用的 history 配置,默认为当前主应用 history 配置
  11. },
  12. {
  13. name: 'app2',
  14. entry: {
  15. // TODO 支持 config entry
  16. scripts: [],
  17. styles: [],
  18. },
  19. base: '/app2',
  20. },
  21. ],
  22. jsSandbox: true, // 是否启用 js 沙箱,默认为 false
  23. prefetch: true, // 是否启用 prefetch 特性,默认为 true
  24. },
  25. },
  26. };

Note: 当主应用跟子应用的 history 模式一致时(比如都是 browser 或 hash),插件会自动创建一些空路由来避免 404,所以建议主应用跟子应用使用相同的 history mode,否则需要自己处理这些情况。

运行时

1. src/app.js 里配置
  1. // 从接口中获取子应用配置,export 出的 qiankun 变量是一个 promise
  2. export const qiankun = fetch('/config').then(({ apps }) => ({
  3. // 注册子应用信息
  4. apps,
  5. jsSandbox: true, // 是否启用 js 沙箱,默认为 false
  6. prefetch: true, // 是否启用 prefetch 特性,默认为 true
  7. lifeCycles: {
  8. // see https://github.com/umijs/qiankun#registermicroapps
  9. afterMount: props => {
  10. console.log(props);
  11. },
  12. },
  13. // ...even more options qiankun start() supported, see https://github.com/umijs/qiankun#start
  14. }));
2. 添加子应用路由

由于 umi 不支持在应用 render 之后修改路由,故运行时配置方式插件无法自动生成子应用相关路由,(见代码)。所以这里需要用户手动添加子应用相关路由配置避免 404 情况:

2.1 主应用新建 pages/subAppContainer.js
  1. import React from 'react';
  2. export default function() {
  3. return <div />;
  4. }
2.2 新建 pages/subAppContainer.js
  1. // 主应用 config.js 中添加子应用相关路由配置
  2. export default {
  3. routes: [
  4. { path: '/app1', exact: true, component: 'subAppContainer' },
  5. { path: '/app2', exact: true, component: 'subAppContainer' },
  6. ],
  7. };

配置列表

配置 说明 类型 是否必填 默认值
apps 子应用配置 App[]
jsSandbox 是否启用 js 沙箱 boolean false
prefetch 是否启用 prefetch 特性 boolean true
defer 是否异步渲染,比如子应用的渲染容器依赖主应用生成的节点,而主应用生成该节点的过程是异步的。当该配置开启的时候,需要使用 import { qiankunStart } from 'umi' api 通知 qiankun 启动。参考 example boolean false

qiankun start 方法其他可接收的参数在这里也都可以配置

App

配置 说明 类型 是否必填 默认值
name 子应用唯一 id string
entry 子应用 html 地址 string \ { script: string[], styles: [] }
base 子应用路由前缀,通常跟子应用的 base 配置 一致,框架会以这个配置作为前缀判断是否激活当前应用,支持配置一组前缀 string \ string[]
history umi history mode string 主应用 history 配置
mountElementId 子应用挂载到主应用的哪个 id 节点上(注意不要跟子应用的 mountElementId 一致) string root-subapp
props 主应用传递给子应用的数据 object {}

子应用

  1. export default {
  2. base: `/${appName}`, // 子应用的 base,默认为 package.json 中的 name 字段
  3. plugins: ['@umijs/plugin-qiankun', { slave: {} }],
  4. };

如果子应用配置项为空,即 slave: {},则可以省略配置:

  1. export default {
  2. base: `/${appName}`, // 子应用的 base,默认为 package.json 中的 name 字段
  3. plugins: ['@umijs/plugin-qiankun'],
  4. };

环境变量配置

为了获得更好地本地开发及调试体验,我们建议您提前在子应用中指定应用启动的具体端口号,如通过.env指定

  1. PORT=8081

详细配置参考:https://umijs.org/zh/guide/env-variables.html#port

父子应用通讯

有两种方式可以实现

基于 props 传递

类似 react 中组件间通信的方案

  1. 主应用中配置 apps 时以 props 将数据传递下去(参考主应用运行时配置一节)

    1. // src/app.js
    2. export const qiankun = fetch('/config').then(config => {
    3. return {
    4. apps: [
    5. {
    6. name: 'app1',
    7. entry: '//localhost:2222',
    8. base: '/app1',
    9. props: {
    10. onClick: event => console.log(event),
    11. ...config,
    12. },
    13. },
    14. ],
    15. };
    16. });
  2. 子应用在生命周期钩子中获取 props 消费数据(参考子应用运行时配置一节)

基于 Hooks 共享数据

由于方案基于 react hook,所以只能在 functional component 中使用相关 api,无法在 class component 中使用。

  1. 约定父应用中在 src/rootExports.js 里 export 内容
  2. 子应用中通过 import { useRootExports } from 'umi'; const rootExports = useRootExports(); 取到

子应用运行时配置

在子应用的 src/app.js 里输出 qiankunprops 由主应用注册子应用时提供

  1. export const qiankun = {
  2. // 应用加载之前
  3. async bootstrap(props) {
  4. console.log('app1 bootstrap', props);
  5. },
  6. // 应用 render 之前触发
  7. async mount(props) {
  8. console.log('app1 mount', props);
  9. },
  10. // 应用卸载之后触发
  11. async unmount(props) {
  12. console.log('app1 unmount', props);
  13. },
  14. };

相关