-
Notifications
You must be signed in to change notification settings - Fork 223
《Vue和React项目如何互相远程调用》
使用 emp-cli 的 init 命令: npx @efox/emp-cli init
- react
- vue2
- vue3
- react-base
- react-project
- vue3-base
- vue3-project (陆续会支持到所有主流技术栈)
使用 emp-cli , npx @efox/emp-cli init
,选择 React 模板
写一个简单 React 组件
新建 /src/components/Hello.tsx
import React from 'react'
import './common.scss'
import './common.less'
import './common.css'
const Hello = ({title}: {title: string}) => (
<>
<h1>{title}</h1>
</>
)
export default Hello
修改项目里的 emp.config.js
(emp.config.js 是 EMP 项目的配置文件) :
- 暴露这个 React 组件,以供远程调用
- 引入远程的 Vue 组件(下面会写 Vue )
const path = require('path')
const packagePath = path.join(path.resolve('./'), 'package.json')
const {dependencies} = require(packagePath)
console.log(packagePath)
module.exports = ({config, env}) => {
const port = 8001
const projectName = 'ReactComponents'
const publicPath = `http://localhost:${port}/`
config.plugin('mf').tap(args => {
args[0] = {
...args[0],
...{
// 项目名称
name: projectName,
// 暴露项目的全局变量名
library: {type: 'var', name: projectName},
// 被远程引入的文件名
filename: 'emp.js',
// 远程项目别名:远程引入的项目名
remotes: {
'@emp/vueComponents': 'vueComponents',
},
// 需要暴露的东西
exposes: {
// 别名:组件的路径
'./components/Hello': 'src/components/Hello',
},
// shared: ['react', 'react-dom'],
shared: {...dependencies},
},
}
return args
})
config.output.publicPath(publicPath)
config.devServer.port(port)
config.plugin('html').tap(args => {
args[0] = {
...args[0],
...{
files: {
js: ['http://localhost:8006/emp.js'],
},
},
}
return args
})
}
在 /src/bootstrap.tsx
引入远程 Vue
组件,引入 vuera
,使用 VueInReact
包裹远程 Vue
组件进行使用
安装 vuera , yarn add vuera
import * as React from 'react'
import * as ReactDOM from 'react-dom'
import Hello from 'src/components/Hello'
import Content from '@emp/vueComponents/Content.vue'
import {VueInReact} from 'vuera'
const VueComponent = VueInReact(Content)
ReactDOM.render(
<>
<Hello title="I am React Project" />
<div style={{backgroundColor: '#eee', padding: '20px'}}>
<VueComponent title="React use Remote Vue Component" />
</div>
</>,
document.getElementById('emp-root'),
)
启动项目 yarn dev
,可以看到本项目的组件和远程引用的 Vue 组件
使用 emp-cli , npx @efox/emp-cli init
,选择 Vue2 模板
写一个简单的 Vue 组件 /src/components/Content.vue
<template>
<div style="color: red">{{ title }}</div>
</template>
<script>
export default {
name:'Content',
props:['title'],
data() {
return {
};
},
};
</script>
修改项目里的 emp.config.js
(emp.config.js 是 EMP 项目的配置文件) :
- 暴露这个 Vue 组件,以供远程调用
- 引入远程的 React 组件
const path = require('path')
const {VueLoaderPlugin} = require('vue-loader')
//
const ProjectRootPath = path.resolve('./')
// const packagePath = path.join(ProjectRootPath, 'package.json')
// const {dependencies} = require(packagePath)
//
const {getConfig} = require(path.join(ProjectRootPath, './src/config'))
//
module.exports = ({config, env, empEnv}) => {
const confEnv = env === 'production' ? 'prod' : 'dev'
const conf = getConfig(empEnv || confEnv)
console.log('config', conf)
//
const srcPath = path.resolve('./src')
config.entry('index').clear().add(path.join(srcPath, 'main.js'))
//
config.resolve.alias.set('vue', '@vue/runtime-dom')
config.plugin('vue').use(VueLoaderPlugin, [])
config.module
.rule('vue')
.test(/\.vue$/)
.use('vue-loader')
.loader('vue-loader')
//
const host = conf.host
const port = conf.port
const projectName = 'vueComponents'
const publicPath = conf.publicPath
config.output.publicPath(publicPath)
config.devServer.port(port)
//
config.plugin('mf').tap(args => {
args[0] = {
...args[0],
...{
name: projectName,
library: {type: 'var', name: projectName},
filename: 'emp.js',
remotes: {
ReactComponents: 'ReactComponents',
},
exposes: {
'./Content.vue': './src/components/Content',
},
/* shared: {
...dependencies,
}, */
},
}
return args
})
config.resolve.alias
.set('vue$', 'vue/dist/vue.esm.js')
.clear()
//
config.plugin('html').tap(args => {
args[0] = {
...args[0],
...{
title: 'EMP Vue Components',
files: {
js: ['http://localhost:8001/emp.js'],
},
},
}
return args
})
}
安装 vuera , yarn add vuera
在 /src/App.vue
引入远程 React
组件,引入 vuera
,使用 ReactInVue
包裹远程 React
组件进行使用
<template>
<div>
<Content title="I am Vue Project" />
<hello-react title="Vue use Remote React Component" />
</div>
</template>
<script>
import { ReactInVue } from "vuera";
import Content from "./components/Content";
import Vue from "vue";
const HelloReact = () => ({
// 需要加载的组件 (应该是一个 `Promise` 对象)
component: import('ReactComponents/components/Hello').then(res=>{
return ReactInVue(res.default)
}),
// 展示加载时组件的延时时间。默认值是 200 (毫秒)
delay: 0,
// 如果提供了超时时间且组件加载也超时了,
// 则使用加载失败时使用的组件。默认值是:`Infinity`
timeout: 3000
})
export default {
name: "APP",
components: {
Content,
"hello-react":HelloReact
},
data() {
return {};
},
created() {
},
};
</script>
<style scoped>
img {
width: 200px;
}
h1 {
font-family: Arial, Helvetica, sans-serif;
}
</style>
启动项目 yarn dev
,可以看到本项目的组件和远程引用的 React 组件
EMP 根据 emp.config.js 的 exposes 字段配置将组件编译成一个单独的闭包,然后将组件单独打包成一个js,最后以 emp.js 的形式作为引用索引,按需加载。
上一节 React 的 Hello 组件编译后的代码如下:
(self["webpackChunk_empreactvue_react"] = self["webpackChunk_empreactvue_react"] || []).push([["src_components_Hello_tsx"],{
/***/ "./src/components/Hello.tsx":
/*!**********************************!*\
!*** ./src/components/Hello.tsx ***!
\**********************************/
/*! namespace exports */
/*! export default [provided] [maybe used in ReactComponents (runtime-defined); used in index] [usage prevents renaming] */
/*! other exports [not provided] [maybe used in ReactComponents (runtime-defined)] */
/*! runtime requirements: __webpack_require__, __webpack_require__.n, __webpack_exports__, __webpack_require__.r, __webpack_require__.* */
/***/ (function(__unused_webpack_module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var react__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! react */ "webpack/sharing/consume/default/react/react");
/* harmony import */ var react__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(react__WEBPACK_IMPORTED_MODULE_0__);
/* harmony import */ var _common_scss__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./common.scss */ "./src/components/common.scss");
/* harmony import */ var _common_scss__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_common_scss__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _common_less__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./common.less */ "./src/components/common.less");
/* harmony import */ var _common_less__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_common_less__WEBPACK_IMPORTED_MODULE_2__);
/* harmony import */ var _common_css__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./common.css */ "./src/components/common.css");
/* harmony import */ var _common_css__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(_common_css__WEBPACK_IMPORTED_MODULE_3__);
var Hello = function Hello(_ref) {
var title = _ref.title;
return /*#__PURE__*/react__WEBPACK_IMPORTED_MODULE_0___default().createElement((react__WEBPACK_IMPORTED_MODULE_0___default().Fragment), null, /*#__PURE__*/react__WEBPACK_IMPORTED_MODULE_0___default().createElement("h1", null, title));
};
/* harmony default export */ __webpack_exports__["default"] = (Hello);
/***/ })
}]);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9AZW1wcmVhY3R2dWUvcmVhY3QvLi9zcmMvY29tcG9uZW50cy9IZWxsby50c3giXSwibmFtZXMiOlsiSGVsbG8iLCJ0aXRsZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBOztBQUNBLElBQU1BLEtBQUssR0FBRyxTQUFSQSxLQUFRO0FBQUEsTUFBRUMsS0FBRixRQUFFQSxLQUFGO0FBQUEsc0JBQ1osdUlBQ0UsdUVBQUtBLEtBQUwsQ0FERixDQURZO0FBQUEsQ0FBZDs7QUFNQSwrREFBZUQsS0FBZixFIiwiZmlsZSI6ImpzL3NyY19jb21wb25lbnRzX0hlbGxvX3RzeC5lMzAzYzRjNC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCAnLi9jb21tb24uc2NzcydcbmltcG9ydCAnLi9jb21tb24ubGVzcydcbmltcG9ydCAnLi9jb21tb24uY3NzJ1xuY29uc3QgSGVsbG8gPSAoe3RpdGxlfToge3RpdGxlOiBzdHJpbmd9KSA9PiAoXG4gIDw+XG4gICAgPGgxPnt0aXRsZX08L2gxPlxuICA8Lz5cbilcblxuZXhwb3J0IGRlZmF1bHQgSGVsbG9cbiJdLCJzb3VyY2VSb290IjoiIn0=
在调用时编译其他框架的组件需要用到 vuera 这个库,帮助我们把其他框架的组件编译成当前所用的框架。
- 状态方面。qiankun 所做的微前端不能把基站项目和子项目过度隔离导致上下文不一致,共享状态等等需要通过总线方式传递,十分麻烦。而 EMP 通过把调用远程的状态管理使得状态共享十分方便。
- 跨框架调用实现。qiankun 通过 dom 隔离的方式,使得跨框架实现十分容易,但是不能互相调用,粒度只能渲染在规定的 dom 区域。EMP 实现的跨框架调用粒度到了 function ,而且使用十分方便。
- 体积方面。qiankun 因为是通过 dom 隔离方式实现,所以依赖共享并不完善,需要依赖于 systemjs,而且共享不方便,导致依赖可能会出现重复,使得出现体积变大。EMP 通过 module federation 实现依赖共享,使得依赖不会重新重复(依赖变成全局变量,相同依赖只会留下一个),所以体积会相对 qiankun 更小。
- 状态方面。iframe 的微前端,无真正意义上的状态管理,通过 postMessage 进行通信。
- 跨框架调用方面。iframe 的微前端不能跨框架调用。
- 体积方面。 iframe 的微前端并不能共享依赖。