npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

v-micro-app-plugin

v1.0.72

Published

v-micro-app-plugin是一款基于MicroApp的微前端插件,实现快速安装使用。

Downloads

363

Readme

v-micro-app-plugin 使用指南

v-micro-app-plugin 是一款基于京东MicroApp框架的微前端插件,旨在帮助开发者快速地将微应用集成到不同的系统中,实现高效、灵活的前端模块化开发。以下是详细的使用指南,帮助你快速上手。

特性

快速集成:通过简单的配置,即可将MicroApp微应用作为插件快速集成到现有系统中。

灵活部署:支持动态加载和卸载微应用,便于按需加载,提升页面加载速度和用户体验。

无缝通信:内置跨应用通信机制,支持不同微应用间的数据交换和事件传递。

易于扩展:插件架构清晰,易于根据业务需求进行定制和扩展。

为什么要封装 v-micro-app-plugin?

京东 micro-app 框架本身已经非常完善,但在使用方面,其对主应用和子应用的适配以及 API 有些差异,导致在开发过程中需要做很多重复性的工作。比如:

  • 发送消息 API:
    • 主应用:microApp.setData(appName, data, callback)
    • 子应用:window.microApp.dispatch(data, callback)
  • 应用初始化:
    • 主应用需要调用: microApp.start();
    • 子应用需要注册相关方法:
      window.unmount = () => {
                app.unmount()
                router = null
                store = null
            }

为了屏蔽主应用和子应用的 API 差异,向上层使用者提供统一的 API,我们封装了 v-micro-app-plugin 插件。

安装

​ 你可以通过 npm、pnpm 或 yarn 来安装 v-micro-app-plugin

  • 通过 npm 安装
npm i v-micro-app-plugin --save
  • 通过 pnpm 安装
pnpm i v-micro-app-plugin --save
  • 通过 yarn 安装
yarn add v-micro-app-plugin

参数

initMyMicroApp

调用参数:

| 参数名 | 介绍 | 类型 | | ------- | ------------ | -------------- | | app | 应用实例 | object | | options | 配置参数 | microAppConfig | | router | Router 实例 | any | | store | 状态管理实例 | any |

返回值:microApp 实例

注意:microAppConfig的类型声明如下:

export interface microAppConfig {
    projectName?: string; // 项目名称
    subAppConfigs?: Object; // 子应用配置
    isBaseApp?: boolean; // 是否为 micro-app 主应用
    basePath?: string; // 打包路径
    disableSandbox?: boolean; // 是否禁用沙箱
    iframe?: boolean; // 是否使用 iframe
}

options 参数介绍

​ 在配置 v-micro-app-plugin 时,你需要准备一个符合条件的 options 对象,该对象包含以下参数:

| 参数名 | 介绍 | 类型 | | -------------- | --------------------------------------- | ------- | | projectName | 项目名称 | string | | subAppConfigs | 子应用配置对象,包含多个子应用的配置 | Object | | isBaseApp | 标记当前应用是否为主应用(默认为 true) | boolean | | basePath | 打包路径或其他基础路径 | string | | disableSandbox | 是否禁用沙箱(默认为 false) | boolean | | iframe | 是否使用 iframe(默认为 true) | boolean |

注意:subAppConfigs 对象中每个子应用的配置包括:

  • name:子应用名称
  • url:子应用的运行地址

使用

引入插件

​ 在你的主应用或目标系统的入口文件中,引入 v-micro-app-plugin

import initMyMicroApp from 'v-micro-app-plugin'

配置和启动

​ 你需要准备一个配置对象 options,包含项目名称、子应用配置、是否为MicroApp主应用以及打包路径等信息。然后,使用 initMyMicroApp 函数进行初始化。这里假设你已经有了一个 Vue 应用实例 app , Vue Router 实例 router,以及状态管理实例 store。

const options = {  
  projectName: 'v-micro-app-plugin',  
  subAppConfigs: {  
    'appFirst': {  
      name: 'appFirst',  
      url: 'http://localhost:4000/#/' // 微应用的运行地址  
    },  
    'appSecond': {  
      name: 'appSecond',  
      url: 'http://localhost:3000/#/' // 另一个微应用的运行地址  
    }  
  },  
  isBaseApp: true, // 当前应用是否为主应用
  basePath: '/', // 打包路径或其他基础路径 
  disableSandbox: false, // 是否禁用沙箱
  iframe: true, // 是否使用 iframe
};  

// 初始化微前端插件  
await initMyMicroApp(app, options, router, store); 

环境变量配置

​ 为了在不同环境下(如开发、测试、生产)使用不同的微应用地址,你可以使用环境变量来动态设置 url。这里提供一个示例:

// 假设你已经在项目中配置了环境变量,例如使用 Vite 的 .env 文件   import.meta.env.MODE
const env = import.meta.env.MODE; // 这里使用的是 Vite 默认的 MODE 变量 
const microAppUrl = {  
  appFirst: {  
    development: 'http://localhost:3000/#/',  
    test: 'https://test.example.com/vivien/appFirst/',  
    production: 'https://www.example.com/vivien/appFirst/'  
  },  
  appSecond: {  
    development: 'http://localhost:4000/#/',  
    test: 'https://test.example.com/vivien/appSecond/',  
    production: 'https://www.example.com/vivien/appSecond/'  
  },  
};  
  
const options = {  
  projectName: 'micro-app-Name',  
  subAppConfigs: {  
    'appFirst': {  
      name: 'appFirst',  
      url: microAppUrl['appFirst'][env]  
    },  
    'appSecond': {  
      name: 'appSecond',  
      url: microAppUrl['appSecond'][env]  
    }  
  },  
  isBaseApp: true, // 是否为主应用  
  basePath: '/', // 打包路径或其他基础路径  
  disableSandbox: false, // 是否禁用沙箱
  iframe: true, // 是否使用 iframe
};  
  
await initMyMicroApp(app, options, router, store);

注意

不管是主应用还是子应用,都必须安装插件,根据需要进行不同的配置。

对象和方法

v-micro-app-plugin 提供了一系列可供使用的方法和对象:

microAppUtils 对象

​ 该对象用于获取当前应用的一些基本信息,提供多个方法。具体如下:

| 方法名 | 介绍 | 参数 | | --------------------- | -------------------------- | ------------- | | isMicroApp | 判断应用是否在微前端环境中 | - | | isBaseApp | 判断应用是否是主应用 | - | | getMicroAppBaseRoute | 获取子应用的基础路由 | - | | getMicroAppPublicPath | 获取子应用的publicPath目录 | - | | getMicroAppName | 获取当前应用名称 | name?: string | | getMicroApp | 返回当前应用实例 | - |

使用示例:

import { microAppUtils } from "v-micro-app-plugin";

const { isMicroApp, isBaseApp } = microAppUtils;
import { onBeforeMount } from "vue";

onBeforeMount(() => {
  	console.log('是否在微前端环境中:',isMicroApp(), '是否为主应用:'isBaseApp())
})

getMicroAppMessage 方法

​ 此方法直接返回一个通信实例对象,该对象用于实现应用之间的通信,提供一系列 API 方便使用。具体如下:

| 方法名 | 介绍 | 参数 | | ------------------ | ---------------------------------- | ---------------------------------------------- | | sendMessage | 发送数据 | { data, appName, callback }: MessageParamsType | | sendGlobal | 全局发送数据 | { data, appName, callback }: MessageParamsType | | forceSend | 强制发送数据,无论数据是否变化 | { data, appName, callback }: MessageParamsType | | forceSendGlobal | 强制全局发送数据,无论数据是否变化 | { data, appName, callback }: MessageParamsType | | getMessage | 获取数据 | appName?: string | | getGlobalMessage | 获取全局数据 | - | | clearMessage | 清除数据 | appName: string | | clearGlobalMessage | 清除全局数据 | - |

注意:MessageParamsType的类型声明如下:

interface MessageParamsType {
  data: object // 发送的数据内容
  appName?: string // 接收数据应用名称,当且仅当主应用发送数据时需要传入
  callback?: Function // 回调函数
}

使用示例:

在子应用 appFirst 中,使用 sendMessage 给子应用发出数据,使用发出 sendGlobal 全局数据:

import { getMicroAppMessage } from "v-micro-app-plugin";

function testSendMessage(){
    const microAppMessage = getMicroAppMessage()
    microAppMessage.sendMessage({
     data: { type: 'sendMessage', value: 'appFirst给主应用发送数据~' },
     callback: () => {
       console.log('使用sendMessage发送数据成功,执行回调!')
     }
    })
    microAppMessage.sendGlobal({
     data: { fun: 'sendGlobal', text: 'appFirst给全局发送数据~' },
     callback: () => {
       console.log('使用sendGlobal发送数据成功,执行回调!')
     }
    })
    setTimeout(() => {
       console.log("getGlobalMessage:", microAppMessage.getGlobalMessage(),"getMessage:", microAppMessage.getMessage())
    },3000)
}
// 结果发现:子应用 => 可以接收到全局信息,但接收不到自己发给主应用的信息。

在主应用中,使用 getMessage 接收子应用发来的数据,使用 getGlobalMessage 接收全局数据:

import { getMicroAppMessage } from "v-micro-app-plugin";

function testGetMessage() {
    const microAppMessage = getMicroAppMessage()
    setTimeout(() => {
     console.log('getGlobalMessage:',microAppMessage.getGlobalMessage(),'getMessage:',microAppMessage.getMessage('appFirst'))
    }, 3000)
}
// 结果发现:主应用 => 可以接收到全局信息,也可以收到 appFirst 发来的信息。

Tip:其它通信 API 方法的使用方式同上。

注意应用发送数据给主应用时,无需传递appName参数;而应用发送数据给子应用时,则通过appName参数来指定某个具体子应用名称。同理,清空当前应用发送给主应用的数据时,无需传递appName参数;而清空应用发送给子应用的数据时,则通过appName参数来指定某个具体子应用名称。

可直接引入的方法

| 方法名 | 介绍 | 参数 | | ------------------ | ---------------------- | ---- | | getMainAppConfigs | 获取主应用配置项 | - | | getSubAppConfigs | 获取子应用配置项 | - | | getRounterInstance | 获取 Microapp 路由实例 | - | | renderAllSubApp | 渲染所有子应用 | - |

使用示例:

import { getMainAppConfigs, getSubAppConfigs } from "v-micro-app-plugin";
import { onBeforeMount } from "vue";

onBeforeMount(() => {
	console.log('使用示例:',getMainAppConfigs(), getSubAppConfigs())
})
// 将会输出当前主应用配置信息,以及其包含的子应用配置信息

其它

​ 为了照顾到一些微前端小白,在这里,我附上一些必要的、可能会有所帮助的信息。

资源地址

微前端插件 v-micro-app-plugin 源码地址:https://github.com/yoguoer/v-micro-app-plugin.git

用该插件搭建的的示例项目 vMicroVerseHub 源码地址:https://github.com/yoguoer/vMicroVerseHub.git

配置路由信息

有了主子应用之后,我们就需要在主应用中给子应用配置路由信息,这里一共有 2 个子应用,我们为它们分别进行配置。

  • appFirst:
import microAppSetting from '@/settings/microAppSetting'

export default {
  path: '/appFirst',
  name: 'appFirst',
  component: Layout,
  order: 1,
  hidden: false,
  meta: {
    title: 'appFirst',
    hideBreadcrumb: false,
    icon: Document,
    microAppOptions: microAppSetting.subAppConfigs!['appFirst']
  }
}
  • appSecond:
import microAppSetting from '@/settings/microAppSetting'

export default {
  path: '/appSecond',
  name: 'appSecond',
  component: Layout,
  order: 2,
  hidden: false,
  
  meta: {
    title: 'appSecond',
    hideBreadcrumb: false,
    icon: Document,
    microAppOptions: microAppSetting.subAppConfigs!['appSecond'],
  }
}

封装 MicroAppContainer

众所周知,路由切换时,可以给<router-view />填充上对应路径的内容,同理,microApp中的<micro-app></micro-app>也有同样的功能。我们可以对其进行二次封装,结合v-if,以便于根据是路由指向的是子应用,还是本系统自由模块,来判断究竟是渲染微应用视图,还是渲染普通视图。

为了达到这个目的,我们可以新建一个 MicroAppContainer 文件夹,在其中创建一个index.vue,然后键入以下内容:

<template>
  <div :class="[`${prefixCls}-container`]">
    <!-- name:应用名称, url:应用地址 -->
    <micro-app v-bind="options" :name="options.name" keep-alive></micro-app>
  </div>
</template>
<script setup lang="ts">
import { watch } from "vue";

const props = defineProps<{
  options: {
    [key: string]: any;
  };
}>();

let prefixCls = props.options.name

watch(
  () => props.options,
  (newValue) => {
    prefixCls = newValue.name
  },
  { immediate: true, deep: true }
);
</script>
<style></style>

**⚠注意:**keep-alive 属性可根据需要决定是否设置。

区分是否微应用视图

  • 在你需要加载子应用页面的地方:
        <div :class="[`${prefixCls}-viewer-microapp`]" v-if="isMicroAppView">
          <MicroAppContainer :options="microAppOptions" />
        </div>
        <div v-else>
          <router-view />
        </div>
  • 一些必要的逻辑语句:
import { watchEffect, ref } from 'vue'
import { useRoute } from 'vue-router'
const route = useRoute()

let isMicroAppView: Ref = ref(false)
let microAppOptions: Ref = ref({})
watchEffect(async () => {
  microAppOptions.value = route.meta.microAppOptions
  isMicroAppView.value = !isNullOrUnDef(microAppOptions.value) && !isEmpty(microAppOptions.value)
})

通信功能

​ 完成了基础功能之后,我们还需要确保应用之间能够相互通信,由于主应用和子应用的通信 API 有一点差别,用的时候容易混淆,不够简便,所以我们对其进行了二次封装,提供了统一的通信 API。

对于具体的使用方法,我们通过几个简单的例子来说明:

准备工作

首先,要引入我们的 getMicroAppMessage() 方法,获取一个通信对象

import { getMicroAppMessage } from "v-micro-app-plugin";

const microAppMessage = getMicroAppMessage();
  • 发出全局信息:用法一致
microAppMessage.sendGlobal({
    data: { fun: "sendGlobal", text: "给全局发送数据~sendGlobal" },
    callback: () => {
        console.log("使用sendGlobal发送数据成功,执行回调!");
    },
});
  • 子应用给主应用发出信息:无需 appName 参数
microAppMessage.sendMessage({
    data: { app: "appSecond", value: "子应用给主应用发送数据~sendMessage" },
    callback: () => {
        console.log("子应用使用sendMessage发送数据成功,执行回调!");
    },
});
  • 主应用给子应用发出信息:需要 appName 参数
microAppMessage.sendMessage({
    data: { app: "mainApp", value: "主应用给appFirst发送数据~sendMessage" },
    appName: "appFirst",
    callback: () => {
        console.log("主应用使用sendMessage发送数据成功,执行回调!");
    },
});
  • 接收全局信息: 用法一致
setTimeout(() => {
    console.log("接收到的全局信息getGlobalMessage:", microAppMessage.getGlobalMessage());
}, 1000);
  • 子应用接收主应用发来的信息:无需 appName 参数
setTimeout(() => {
    console.log(
        "子应用接收到主应用发来的非全局信息getMessage:",
        microAppMessage.getMessage()
    );
}, 1000);
  • 主应用接收子应用发来的信息:需要 appName 参数
setTimeout(() => {
    console.log(
        "主应用收到appFirst发来的信息getMessage:", microAppMessage.getMessage('appFirst'),
        "主应用收到appSecond发来的信息getMessage:", microAppMessage.getMessage('appSecond')
    );
}, 1000);
  • 控制台信息:

image-20240815102734720.png

image-20240815101949378.png

示例

微前端插件 v-micro-app-plugin 源码地址:https://github.com/yoguoer/v-micro-app-plugin.git

用该插件搭建的的示例项目 vMicroVerseHub 源码地址:https://github.com/yoguoer/vMicroVerseHub.git

具体使用方法和效果可以查看仓库中的 readme 文档,或者运行 vMicroVerseHub 进行更直观的体验,项目中含有各项功能的测试 Demo 以及关键节点的 console.log,更好地呈现效果。

效果如下:

  • 主应用视图:

main -original-original

  • sub-app-first视图:

first -original-original

  • sub-app-second视图:

seond -original-original