introduce
The Qiankun is based on single-SPA. What is a micro front end? Personal understanding is similar to back-end microservices; I’m not saying that some code level error will cause the whole thing to go down. This is also compared with the introduction of qiankun’s features on the official website.
-
Independent development and deployment: the micro-application warehouse is independent, and the front and back ends can be independently developed. After the deployment, the main framework automatically completes synchronous update.
-
Run independently: you can run only one child application during development;
-
Stack independent: any front-end technology can be accessed;
-
Incremental update: Similar to standalone deployment, you can update a child application separately;
Quick learning
Before starting, amway daily youxian team micro front-end practice; Absolute best practices. The following examples all use the Vue technology stack;
The main application
After the project was created, the main application was installed.
npm install qiankun -s
Copy the code
Register child applications in main.js;
import Vue from 'vue';
import App from './App.vue';
import router from './router';
import { registerMicroApps, start } from 'qiankun';
Vue.config.productionTip = false;
new Vue({
router,
render: h= > h(App)
}).$mount('#app');
/ * * *@param {*} App configuration object *@param {*} Lifecycle hooks */
registerMicroApps([
{
name: 'vueApp'.entry: '//localhost:3000'.container: '#container'.activeRule: '/vueApp'}]); start();Copy the code
Then put a container in app.vue for the child app; The id of this container is the container in the first parameter configuration object in registerMicroApps; Other parameters are described later.
<template>
<div id="app">
<div id="nav">
<router-link to="/">Home</router-link> |
<router-link to="/about">About</router-link>
</div>
<router-view/>
<div id="container"></div>
</div>
</template>
Copy the code
The child application
The sub-application does not need to download the qiankun sub-application.
1. Add public-path.js to project SRC
if (window.__POWERED_BY_QIANKUN__) {
__webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__;
}
Copy the code
2. Introduce public-path.js at the top of the entry file and modify and export the three lifecycle functions
import './public-path';
import Vue from 'vue';
import App from './App.vue';
import router from './router';
Vue.config.productionTip = false;
let instance = null;
function render(props = {}) {
const { container } = props;
instance = new Vue({
router,
render: h= > h(App)
}).$mount(container ? container.querySelector('#app') : '#app');
}
// Independent runtime
if (!window.__POWERED_BY_QIANKUN__) {
render();
}
// The lifecycle hooks must be exported
export async function bootstrap() {
console.log('[vue] vue app bootstraped');
}
export async function mount(props) {
console.log('[vue] props from main framework', props);
render(props);
}
export async function unmount() {
instance.$destroy();
instance.$el.innerHTML = ' ';
instance = null;
}
Copy the code
4. The microapplication recommends using the route in history mode. The route base should be set to the same value as its activeRule
const router = new VueRouter({
mode: 'history'.// The latter is used for standalone and the first is used for standalone
// Base should be the same as activeRule in registerMicroApps in the main application
base: window.__POWERED_BY_QIANKUN__ ? '/vueApp/' : '/child/vue-app/',
routes
});
export default router;
Copy the code
5. Modify webPack packaging to allow cross-domain and UMD packaging for development environments. New vue. Config. Js
const { name } = require('./package');
module.exports = {
// Package the application into a specified folder so that it can be easily identified during deployment
outputDir: 'sub-vue'.// Set the base route for the standalone runtime
publicPath: '/child/vue-app/'.devServer: {
port: 9099.headers: {
'Access-Control-Allow-Origin': The '*'}},configureWebpack: {
output: {
library: `${name}-[name]`.libraryTarget: 'umd'.// Package microapplications into umD library format
jsonpFunction: `webpackJsonp_${name}`}}};Copy the code
In the last step, we modify the entry in the registerMicroApps configuration of the main application and then add a route jump;
<! -- Main app app.vue -->
<template>
<div id="app">
<div id="nav">
<router-link to="/">Home</router-link> |
<router-link to="/about">About</router-link> |
<router-link to="/appVue">goToSubVue</router-link>
</div>
<router-view/>
<div id="container"></div>
</div>
</template>
Copy the code
// The main application main.js
registerMicroApps([
{
name: 'vueApp'.entry: '//localhost:9099/child/vue-app/'.container: '#container'.activeRule: '/vueApp'}]);Copy the code
preview
Then start the project preview effect; Remember to test the launcher alone first and don’t capsize in the gutter!
The life cycle
Main application life cycle
Mention of the main application’s declaration cycle is inseparable from the regsterMicroApps method;
registerMicroApps(apps, lifeCycles)
;
registerMicroApps(
[
{
name: ' '.entry: ' '.container: ' '.activeRule: ' '.loader(loading) {},
props: {}}])Copy the code
name
: Indicates the name of a subapplication. Each micro-application must be uniqueentry
: Entry of the child application, which is consistent with the startup service of the child application, such as after the child application is started192.168.10.1:8088 / child/app - vue /
thenentry
You can make//localhost:8088/child/app-vue/
or/child/app-vue/
.container
: Which DOM container should be loaded into the child application and configured in the parent application.activeRule
: the activation entry of the micro-application, which is routed to the sub-applicationwindow.__POWERED_BY_QIANKUN__
If the value is true, the configurations are consistent. For example, subapplications/vueApp/
So the master appliedactiveRule
for/vueApp
.loader
: Optional parameter Loading effect of the child application.props
: this value is available in the child application’s mount hook,Note: the object obtained in the props child application is not an empty object. Note the key name conflicts
registerMicroApps(
[{}],
{
beforeLoad(app) {},
beforeMount(app) {},
afterMount(app) {},
beforeUnmount(app) {},
afterUnmount: [app= > console.log(app)]
}
)
Copy the code
App in the above declaration cycle hook is an optional parameter to obtain the information of the currently activated sub-application. BeforeLoad, beforeMount, and afterMount are triggered when loading the child application, and beforeUnmount, afterUnmount are triggered when leaving the child application.
Subapplication declaration cycle
bootstrap
: This command is triggered only when the child application is initialized.mount
: Receive aprops
That’s passed in the parent applicationprops
This hook is called every time a child application is entered. Ps: Use this hook to mount react or Vue applications.unmount
: Receive aprops
That’s passed in the parent applicationprops
, the application triggers this hook every time it leaves; Ps: You can clear the data recovery status.update
: Receive aprops
That’s passed in the parent applicationprops
Is valid only when the microapplication is loaded in loadMicroApp mode. This method is optional;
Note: The hooks of the child application are separatePromise
Function, need to call separately, below to Vue projectmain.js
As an example.
Parent-child application communication
Want to know more about the communication of Qiankun
According to the official way of Qiankun; There are two kinds of communication;
1. Communicate based on initGlobalState method; In this way we first apply a slight modification to the master. Using this approach, I personally understand that communication between VUE components is similar; The master app is responsible for maintaining data, while the child app is only responsible for consuming it.
InitGlobalState: This method receives one state and returns three communication methods.
-
OnGlobalStateChange (cb, fireImmediately) : The current application listens for global status, and if there is a change, the callback will be triggered. FireImmediately = true Will trigger the callback immediately
-
SetGlobalState (state) : Sets global status by level 1 attribute. Only existing level 1 attributes can be modified in microapplications
-
OffGlobalStateChange: Removes status listening for the current application. It is called by default when the microapplication is umount
// Main application: shared/actions.js
import { initGlobalState } from 'qiankun';
const state = {};
const actions = initGlobalState(state);
export default actions;
Copy the code
// Child application: shared/actions.js
function emptyAction () {
// Indicates that an empty Action is currently in use
console.warn('Current execute action is empty! ');
}
class Actions {
// The default value is empty action
actions = {
onGlobalStateChange: emptyAction,
setGlobalState: emptyAction,
};
/** * sets actions */
setActions(actions) {
this.actions = actions;
}
/** * mapping */
onGlobalStateChange(. args) {
return this.actions.onGlobalStateChange(... args); }/** * mapping */
setGlobalState(. args) {
return this.actions.setGlobalState(...args);
}
};
const actions = new Actions();
export default actions;
Copy the code
To complete the above configuration, set the actions in the child application’s mount hook.
import actions from '.. /shared/actions';
/ /...
export async function mount (props) {
console.log('[vue] props from main framework', props);
actions.setActions(props);
render(props);
}
Copy the code
With that done, let’s write an example
// Main application home.vue<template>
<div>
<fieldset>
<legend>formData</legend>
username: <input type="text" v-model="formData.username" /><br />
password: <input type="text" v-model="formData.password" /> <br />
<button @click="submit">submit</button>
</fieldset>
</div>
</template>
<script>
import actions from '.. /.. /shared/actions';
export default {
data() {
return {
formData: {
username: ' '.password: ' '}}},methods: {
submit() {
actions.setGlobalState(this.formData)
}
}
}
</script>
Copy the code
// The child home.vue<template>
<div>{{ form }}</div>
</template>
<script>
export default {
data() {
return {
form: {}}; },mounted() {
// Since we are jumping to the child app in the parent app's Home;
// Actually do/jump to /vueApp, but need to be able to get the content set in /; Although our setup was successful, the child application was mounted without the change event;
// Add the second parameter true, similar to the immediate in watch.
actions.onGlobalStateChange((state, prev) = > {
this.form = state;
}, true); }}</script>
Copy the code
You see there’s a little bit of a problem up here if we want to return/from /vueApp we still have the value that we typed in the input box; You need to listen for the change event again in home.vue of the main application
export default {
/ /...
mounted() {
// Add true again
actions.onGlobalStateChange((state, prev) = > {
this.formData = state;
}, true)}}Copy the code
While trying this communication, I found that Qiankun had another mechanism. There were multiple change events in an application, and only the last one would take effect to prevent memory explosion.
Summary: practical operation down; Feel inadequate place, of course, the probability is my dish.
- How to update state; That is to say through
setGlobalState
But the official website describes this method as updating a level 1 attribute; Wouldn’t updating my example overwrite the original state every time? Assuming we have more than two child applications, how should state be maintained? I hope someone answers this question in the comments section; - Suppose the above question is true; I now have two child applications, one needs to update the information of userInfo in state, one needs to update the information of systemInfo in state; How can we ensure that we can update the data without changing the entire state?
2. Communicate with props in the APP configuration object in registerMicroApps; Refer to the communication link above; Don’t write. But it could be a better solution.
The deployment of
Nginx can be installed through yum
yum install nginx
Copy the code
The nginx configuration file is installed in /etc/nginx/nginx.conf. Static resources in /usr/local/html;
The website provides two deployment modes;
1. Put microapps in a special folder (not the same name as microapps), which is also recommended;
└ ─ ─ HTML / # root folder | ├ ─ ─ child / # for all the application of micro folder | ├ ─ ─ vue - app / # deposit application vue micro - app folder packed by outputDir specified ├ ─ ─ index. The HTML # Index.html ├─ CSS / # ├─ JS / # Js folder for Main applicationCopy the code
server { listen 80; listen [::]:80; server_name _; add_header Access-Control-Allow-Origin *; add_header Access-Control-Allow-Methods *; add_header Access-Control-Allow-Headers *; Location / {root /usr/local/html/dist; index index.html index.htm; try_files $uri $uri/ /index.html; } # call the publickPath of the child application. Location /child/appVue {root /user/local/ HTML /dist; index index.html index.htm; try_files $uri $uri/ /child/app-vue/index.html; } # Load configuration files for the default server block. include /etc/nginx/default.d/*.conf; }Copy the code
The registerMicroApps configuration in the main application could look like this;
But there’s a puzzle here; //localhost:9099 //localhost:9099 //localhost:9099 //localhost:9099
registerMicroApps([
{
name: 'vueApp'.// entry: '//localhost:9099/child/vue-app/',
entry: '/child/vue-app/'.container: '#container'.activeRule: '/vueApp'}]);Copy the code
2, it is directly placed in the secondary directory, no need to create the child folder in dist, the operation is basically the same, just need to change the Linux configuration
└ ─ ─ HTML / # root folder | ├ ─ ─ vue - app / # deposit application vue micro - app folder ├ ─ ─ index. The HTML # main application of the index. The HTML ├ ─ ─ CSS / # main application CSS folder ├ ─ ─ js / # The js folder of the main applicationCopy the code
#This is the publickPath of the child application, which must be the same as entry in the registerMicroApps configuration object of the master applicationlocation /child/appVue { root /user/local/html/dist; index index.html index.htm; Try_files $uri $uri/ /app-vue/index.html; }Copy the code
3, there is a case is the child application and the main application is not a server, this really can not test, no money in your pocket 😂; Microapplications are deployed on different servers