preface

The background management system based on Vue2. X + VUE-CLI4.x mainly includes automatic route mounting, theme skin, multi-language internationalization, table operation, chart display, full-screen operation, font icon, permission control, route nesting and data mock and other functions. Part of the content is learned from vue-element-admin and other excellent open source works, and I have made some trade-offs and optimizations. The following content is summarized later, there may be some mistakes and deficiencies, welcome to point out!

  • Demo address: juetan. Making. IO/management
  • Demo account: Admin account: admin-admin; Common account: juetan-juetan

Technology stack

Developed based on NodeJS, VS Code and VUE-CLI, the technology stack mainly consists of VUE, AXIos, Vuex, Vue-Router and Elemet-UI. In addition, the following dependencies or packages are included:

  • Theme changes are based on webpack-theme-color-replacer
  • CSS preprocessing is based on SCSS
  • Internationalization is based on VUE-I18N
  • Data mocks are based on MockJS
  • The chart presentation is based on Echarts
  • Markdown mavon based – editor
  • Full screen display based on ScreenFull
  • Counting dynamic effects are based on countTo
  • Progress loading is based on nProgress
  • Version control is based on Git
  • Static deployment is based on Github Pages and Github Actions

The main function

Topics in the skin

Vue-element-admin provides two ways to change the theme, but only the color of the element-UI is supported, not the variable color of the Vue component. There is a webpack plugin, webpack-theme-color-replacer, which supports these functions perfectly and is smaller. The principle is to separate the color styles involved in the packaging and replace the theme with the regular.

The language of international

Vue-i18n is almost ready to use once installed, but it is important to synchronize with the languages of third-party frameworks such as Element-UI, Echarts and Mavon-Editor. However, these open source packages all provide international functionality, so pay attention to the content added here.

CSS pretreatment

SCSS is used here. Firstly, SCSS functions and uses are more extensive. Secondly, Element-UI is also SCSS, which makes it more convenient to modify the theme. I saw a good article on the planning of CSS files and style files in Zhihu, if you are interested, you can have a look.

State management

I think it is necessary to use Vuex here, not only to share state between VUE components, but also to modify and read state by third party dependencies. Vuex is a modular package, with reads uniformly read from state and updates operated from actions.

Network request

Since it is a pure front-end project, data MOCK is somewhat of a problem. I tried Faskmock before, but sometimes it didn’t feel very stable. In the end, I chose to use pure MOCKJS to generate data. Network requests are encapsulated in AXIos and invoked in a modular fashion, which is more deeply integrated with VuEX.

According to the need to load

There are a number of large libraries in the project that would be wasteful to load completely, so it is necessary to load on demand. The large libraries involved in the project include Element-UI, LoDash, and Echarts.

Static deployment

Github Pages are usually required for personal open source projects. In addition to creating

.github. IO repositories, You can also mount static pages by creating a gh-Pages branch in a normal repository (accessed via

.github. IO /

after uploading). This, combined with Github Actions, helps us build and deploy CI.


Project start

To start, NodeJS and VUE-CLI (version 4.x) need to be installed. The code editor I used was VS Code. I have tried Vue3+TS, but the process is not smooth (mainly tai CAI), here I still use vue of 2.x version, but CLI is 4.x.

Vue-cli4.x has two ways to create projects, one is a GUI interface like Vue UI and the other is a CLI interface like Vue Create. Here I use CLI mode, after all, in VS Code terminal operation is relatively convenient.

// Management is the project name ([Vue 2] Babel, esLint) Vue create ManagementCopy the code

Off-topic: Vue-CLI uses NPM run serve instead of NPM run dev. I was a little surprised why it was serve instead of server, but later I realized that serve is a verb and server is a noun (~ ~ ▽ ~) ~.

The project structure

Vue-cli provides us with a project scaffolding, but the directory structure and whatnot still needs to be set up according to our needs. Here, I put the vue family bucket wrapper in the/SRC directory (which is already the default).

vuex                           # /src/store
axios                          # /src/api
vue-router                     # /src/router
vue-i18n                       # /src/lang
Copy the code

For other third-party libraries, I do not directly import references, but place them in the/SRC /plugins directory and then reference them after secondary encapsulation. Currently, there are as follows:

element-ui                     # /src/plugins/element-ui.js
driver.js                      # /src/plugins/driver.js
nprogress                      # /src/plugins/nprogress.js
vue-count-to                   # /src/plugins/count-to.js
webpack-theme-color-replacer   # /src/plugins/theme-replacer.js
Copy the code

The final project structure is as follows:

├ ─ ─ the lot # making Action configuration directory ├ ─ ─ public # public directory │ │ ─ ─ the favicon. Ico # website icon │ └ ─ ─ index. # HTML page template ├ ─ ─ / SRC # │ ├─ API # Web Request │ ├─ Assets # │ ├─ Components # Public Components │ ├─ Config # │ ├─ Helper # ├─ Lang # Language │ ├─ Pages # Route Page │ ├─ Plugins # │ ├─ ├─ state Management │ ├─ main.js # Import ├─ │ ├─ state Management │ ├── .env. Production # ├──.env. Production # ├──.env. Production # ├──.env ├─ download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txt # Download.txtCopy the code

Project configuration

Vue-cli provides the functions of vue.config.js project configuration and. Env environment variable configuration. It is necessary to do basic configuration for the project before development.

Configure the vue – cli

First, in the project root directory, create a vue.config.js file and write some basic configurations (see here for more configurations).

Exports are commonJS module.exports, not ES Module export default. Node only supports commoJS modules by default, while WebPack only supports ES modules.

<! Module. exports = {-- filename: /vue.config.js --> module.exports = {// publicPath: /vue. process.env.NODE_ENV === "production" ? "/management/" : "/", // reduce file volume in production environment productionSourceMap: false, // chain configuration webPack chainWebpack: Plugin (" HTML ").tap((args) => {// Initialize the title in public/index.html config.plugin(" HTML ").tap((args) => {// This value configures args[0].title = '${process.env.vue_app_name} - in. Env.production and. Env.development ${process.env.VUE_APP_DESCRIPTION}`; return args; }); }};Copy the code

Configuring environment Variables

This is the environment variable functionality provided by vue-CLI to create. Env. development and. Env. production files for development and production environments respectively. For more information on env, click here.

<! -- filename: /.env.development --> # system name VUE_APP_DESCRIPTION = ' '/.env.development --> # system name VUE_APP_DESCRIPTION =' ' VUE_APP_API_BASE_URL = "# Routing mode in development environment VUE_APP_ROUTER_MODE = 'history'Copy the code
<! -- filename: VUE_APP_DESCRIPTION = 'meet the scenery you don't know' /.env.production --> VUE_APP_DESCRIPTION = 'meet the scenery you don't know VUE_APP_API_BASE_URL = 'https://www.fastmock.site/mock/7522085e7af7a7b98bc42530bd1ddfb7/management' # routing patterns of development environment VUE_APP_ROUTER_MODE = 'hash'Copy the code

Website icon

This step can be done or not at present, mainly depends on personal preference, but I like to change the icon so that it looks more pleasing to the eye. Just replace the favicon.ico file in the /public directory.

The local domain name

Localhost :8080/#/home = www.w.com/home

Configuring local Hosts

I use Windows as an example. Open the hosts file in C:\Windows\System32\drivers\etc and add a local domain name mapping at the end of the file. It is recommended to use single-digit or single-letter (non-qxz).com domain names because these domain names are not in use and have no conflict risk.

<! -- filename: C:\Windows\System32\drivers\etc\hosts --> # Separate IP and domain name by a space 127.0.0.1w.comCopy the code

Configure the vue. Config. Js

Once configured, accessing w.com in a browser points directly to local IP127.0.0.1. However, vue-CLI uses port 8080 by default. If you do not want to access the cli in the form of w.com:8080, you need to configure vue.config.js. In addition, you need to set disableHostCheck to true. Otherwise, Invalid Host Header is displayed during access.

<! -- filename: vue.config.js --> module.exports = { configureWebpack: { devServer: {// Add the 127.0.0.1 www.w.com record to the hosts file and use www.w.com instead of localhost:8080 with the following two parameters: Port: "80", // This value must be set to true to avoid Invalid Host header. true, }, } }Copy the code

Configure the vue – the router

This already works, but you can try to optimize it by using the History route to improve the browsing experience in a local development environment and using the Hash route in a production environment.

<! VUE_APP_ROUTER_MODE = 'history' -- filename: /.env.development -->Copy the code
<! Production --> VUE_APP_ROUTER_MODE = 'hash' -- filename: /.env.production --> VUE_APP_ROUTER_MODE = 'hash'Copy the code

Env is used to configure environment variables, which is easier to modify and configure, but do not end in a. Local file name, these files will be ignored by Git.

An automated build

Version control is essential, not to mention collaborative development. In this project, it is mainly to publish to Github, cooperate with Github Action and Github Pages for continuous construction and automatic deployment.

Create a warehouse

Direct 3 even operation, mainly before the transformation of the directory structure to record.

Git add. Git commit -mCopy the code

Associate the Github repository

I’m using Github and it doesn’t have to be Origin, it’s just a name, and I’m using SSH links.

git remote add github [email protected]:juetan/management.git
Copy the code

At first, I used HTTPS links, but due to some mysterious force I encountered the following error:

OpenSSL SSL_read: Connection was reset, errno 10054Copy the code

A Google search also has a solution:

Git config --global http.sslVerify "false"Copy the code

But the actual effect is not very ideal, sometimes I still have this problem, later I thought of using SSH connection, so I switched to this way, so far I have not encountered this problem again.

CI build

Github Pages are usually required for personal open source projects. In addition to creating

.github. IO repositories, You can also mount static pages by creating a gh-Pages branch in a normal repository (accessed via

.github. IO /

after uploading). This, combined with Github Actions, helps us build and deploy CI.


Local configuration

GitHub Action uses the. Yml file as the configuration file. You need to create the. GitHub /workflows directory in the root directory of the project. Some online tutorials may say that you need to set secret key. I checked the documents and found that you can directly use secrets.GITHUB_TOKEN, so this step can be omitted.

<! - filename: /. Making/workflows/main yml - > # name of workflow, customizable name: Use GithubAction to automatically deploy to GithubPages # Event listener to determine what triggers tasks within this workflow on: # Trigger Push: Branches when master branch is pushed to Github: # run -on: Ubuntu -latest # Run -on: Ubuntu -latest # run -on: Ubuntu -latest # use an action - uses: actions/checkout@v2 # Use an action - uses: actions - name: build and deploy run | NPM install NPM run build CD dist # user name custom modify git config - global user. The name "juetan # mailbox" custom modify git config - global User. email "[email protected]" git init git add -a git commit -m "Automatically deploy" git push -f "https://${{via Github Action github.actor }}:${{ secrets.GITHUB_TOKEN }}@github.com/${{ github.repository }}.git" master:gh-pagesCopy the code

In addition, configure vue.config.js to point the public directory to the repository name.

<! -- filename: /vue.config.js --> module.exports = {// publicPath, GitHub pages = {// publicPath, GitHub pages = { process.env.NODE_ENV === "production" ? "/management/" : "/", }Copy the code

Remote configuration

In the GitHub repository setting, switch the branch in Pages to GH-Pages, and you can see the publishing address in the green background box in the image below.

Commit to making

With both local and remote configured, you can push to the GitHub repository.

git push github master
Copy the code

Then from the Actions page in the repository, you can see our CI task. Click there to see the details. After the build is complete (tens of seconds to minutes), it is accessible via

.github. IO /

. In addition, the domain name can be your own domain name and HTTPS support, if you are a personal blog can consider using the domain name, I won’t go into details here. Finally, post a picture of success.

The data simulation

For data simulation, if there is a back end, try to use the API provided by the back end. If there is no back end, the following two ways can be considered to obtain data:

  • Use an online Mock platform such as Easy-Mock or fast-Mock
  • The local environment is implemented using mock.js.

I started with fast-mock, but sometimes it wasn’t very stable, and eventually switched to mock.js. There are two ways to use mock.js:

  • Directly in thesrc\main.js, which overrides XHR and may conflict with other libraries, while in the browser’snetworkNo network request in panel;
  • Through webpackdevServerConfiguration to direct matching requests to mock.js.

Method 1 is currently used because this project is a pure front-end project and involves few requests.

MockJS installation

Install it on the terminal, save it as Dependencies for usage 1 and devDependencies for usage 2.

// If it is mode 2, save as --save-dev NPM install mockjs --saveCopy the code

MockJS configuration

The mock folder is usually placed in the root directory, and the reason I want to do this is that it makes it easier to switch data sources, and it works fine either way. It’s just my guess at this point, but this is the default rule. The configuration and syntax of MockJS are not described here, but can be found on the official website.

<! -- filename: /mock/server.js --> import Mock from "mockjs"; MOCKJS configures mock.setup ({// delay the return of data timeout: 800,}); / / table data Mock. Mock ("/table ", "get", {code: 2000, the message: "data returns success", "data | 80" : [{" id | + 1 ": 1, name:" @ cname ", number: "equip-@first", spec: "ws-@last", type: "@natural(1,3)", status: "@natural(1,3)", manufacturer: "@cname", buyDate: "@date", mark: "no ",},],});Copy the code

The file import

Directly introduced in main.js, you can use axios and other network request libraries for data requests.

<! -- filename: /src/main.js --> import "/mock/server.js";Copy the code

State management

As for the use of VUEX, the official documentation has given a good practice, which can be directly encapsulated in the way of modules.

Vuex installation

Install it directly from the command line.

npm install vuex --save
Copy the code

Instance creation

Create the index.js file in/SRC /store. This file automatically imports all.js files in the./modules directory and exports the instantiated vuex.

<! -- filename: /src/store/index.js --> import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); // Use require.context to automatically import all files in the modules directory const moduleFiles = require.context("./modules", true, /\.js$/); // moduleFiles are both functions and objects, Call const modules = modulefiles.keys ().reduce((modules, modulePath) => { const moduleName = modulePath.replace(/^\.\/(.*)\.\w+$/, "$1"); // This is a function call similar to import const Module = moduleFiles(modulePath); Modules [moduleName] = module.default; // Put all modules in an array. return modules; }, {}); // Create vuex instance const store = new vuex.Store({modules}); export default store;Copy the code

Module to create

Modules are stored in the modules directory under/SRC /store, and each module is named a.js file, such as user.js;

<! - / SRC/store/modules/user. The js - > : const state = {/ / username the username: ' '}; Const mutations = {set_username: (state, name) => {state.username = name; }}; export default { namespaced: true, state, mutations, };Copy the code

Instance is introduced into

/ SRC /main.js;

<! -- filename: /src/main.js --> import store from '@/store' new Vue({ store }).$mount('#app');Copy the code

use

For use in VUE components:

<! -- filetype: .vue --> export default { methods: { xxx() { this.$store.commit('user/set_name',name) } } }Copy the code

For use in non-VUE components:

<! -- filetype: .js --> import store from '@/store' cosnt title = store.state.default.title;Copy the code

Network request

It’s good that Axios has two encapsulation ideas. One is the encapsulation of partial functions, each request module exports a request function, when used as needed on the line; The other is the encapsulation of partial objects, which is exported uniformly for each module, summarized into index file, and finally hung on vue.prototype. The second method is used here, mainly in conjunction with vuex for state management, while mounting the $request method to vue prototype for easy call.

Axios installation

Install AXIos directly from the command line;

npm install axios --save
Copy the code

Instance creation

In the SRC/API directory, create an index.js file, configure the basic parameters, interceptors, and instantiate Axios. Specific code is not posted can go to see the source code, here to post the main code and say the next 2 points:

  • baseURLThrough:process.env.NODE_ENV === 'development' ? 'xx' : 'xx'The configuration is available, but with vue-CLI providedThe environment variableIt’s also good.
  • Loading: This encapsulates a loading and adds some processing logic. For data requests, it is more experience to know that data is being requested.
<! -- filename: / SRC/API /request.js --> Const request = axios.create({baseURL: process.env.vue_app_API_base_URL, timeout: 3000,}); / / set the request interceptor request. Interceptors. Request. Use ((config) = > {/ /... }, (error) => { // ... }); / / set the interceptor request. Interceptors. Response. Use ((response) = > {/ /... }, (error) => { // ... }); export default request;Copy the code

Module to create

In SRC/API /modules, create user.js(each.js file represents a network request for a module). It is possible to do a separate configuration for the request URL, but I don’t think it is necessary for now.

<! -- filename: /src/plugins/api/modules/user.js --> import request from ".. /index"; Export function login_user(data) {return request.post("/login", data); }, export function get_userinfo() {return request.post("/userinfo"); },Copy the code

Examples of the mount

In the SRC \main.js file, import the AXIos instance and mount it on the Vue prototype;

<! -- filename: /src/main.js --> import Vue from "vue"; Import request from "@/ API "; Vue.prototype.$request = request;Copy the code

Request to use

Here, an AXIOS module corresponds to a VUE component or a VUEX module, so there are two usage scenarios of vUE component and VUEX module.

Used within vUE components:

<! -- filename: /src/pages/app/table/index.vue --> import { getTableData } from '@/api/modules/table'; export default { created() { getTableData().then((data)=>{ ... }}}Copy the code

For use in vuEX module:

<! -- filename: /src/store/modules/user.js --> import { login_user } from "@/api/modules/user"; Const actions = {// Login user login_user(context, userData) {return login_user(userData). Then ((data) => {... }); }},Copy the code

The language of international

Let’s start with i18n and L10n in internationalization, these are two words.

  • i18n“Internationalization” is the beginning of internationalizationiAnd at the end of then18 letters apart, short fori18n;
  • l10n“, localizationlAnd at the end of thenTen letters apart, short forl10n.

I first learned about this concept while working with wordpress themes. In wordpress, internationalization uses functions such as __ and _x to translate output from different languages. The function is to switch between multiple languages. Localization, on the other hand, uses plug-ins, POEDITOR and other tools to translate preferred languages into different languages, such as English to Chinese, each representing a.po file.

In wordpress, internationalization and localization are separate, but in Vue-i18n the two are integrated, which is very convenient. In a front-end project, however, there is the question of whether the language in the component and the language in the external dependencies can be reconciled.

However, the good news is that most large open source projects provide international functionality. Here’s an example of how to integrate vuE-I18n and Element-UI.

Vue – i18n installation

Install it directly in the terminal.

npm install vue-i18n --save
Copy the code

Instance creation

Create index.js in/SRC /lang, import all.js files in./modules, create and export vue-i18n instances. For third-party library languages, there are currently three cases involved:

  • Some third-party libraries provide ways to modify the language, such as mavon-Editor;
  • Some third-party libraries have built-in language packs (you need to configure them yourself), such as element-UI.
  • Some third-party libraries, such as Echarts, do not provide a language interface or need to change the language based on the data returned from the API.

In case 1, the call is made directly within the component; For case 2, follow the documentation. For case 3, the language pack is currently placed under./modules, and the component is updated when the data comes back.

<! -- filename: src/lang/index.js--> import Vue from "vue"; import VueI18n from "vue-i18n"; Import locale_element_en from "element-ui/lib/locale/lang/en"; import locale_element_en from "element-ui/lib/locale/lang/en"; import locale_element_zh from "element-ui/lib/locale/lang/zh-CN"; // import locale_zh from "./modules/zh"; import locale_en from "./modules/en"; Vue.use(VueI18n); Const messages = {zh: {... locale_element_zh, ... locale_zh, }, en: { ... locale_element_en, ... locale_en, }, }; / / create an instance const lang = new VueI18n ({locale: localStorage. The getItem (" language ") | | "useful", messages,}); export default lang;Copy the code

Module to create

In the/SRC /lang/modules directory, create a new zh.js file (each.js file represents a language). This module returns an object with a recommended two-level structure: level 1 for the VUE component or system module and level 2 for the concrete content. In the following example, login indicates the VUE login component. The property of this object indicates the specific content. The value can be invoked by $t(‘login.title’).

<! -- filename: / SRC /lang/modules/zh.js --> export default {login: {title: "login ", description: "Hello! , welcome to log in the background management system! , userplaceholder: "please input username ", passplaceholder:" please input password ", uservalidator: "Please check username!" Please check your password slowly. Remember my password. Forget your password. Submit: "submit ", ways:" other login methods ", byMobile: "mobile login ", byemail:" email login ", loginedtip: "login successful, will jump to the home page ",},};Copy the code

Introduction and installation

In addition to vue-I18n’s introduction of the element-UI language pack, element-UI also needs some work. Because I’ve wrapped element-UI separately, I’m currently using method 1. If you don’t want to wrap element-UI separately, you can just use method 2.

Mode 1 (currently used) :

<! -- filename: /src/plugins/element-ui.js --> import Vue from "vue"; import i18n from "@/lang"; Vue.use(ElementUI, { i18n: (key, value) => i18n.t(key, value), });Copy the code
<! -- filename: /src/main.js --> import i18n from "@/lang"; new Vue({ i18n, })Copy the code

Method 2 (conventional method) :

<! -- filename: /src/main.js --> import i18n from "@/lang"; Vue.use(ElementUI, { i18n: (key, value) => i18n.t(key, value), }); new Vue({ i18n, })Copy the code

Note: Either way, you need to pass i18n(vue-i18n instance) in the new Vue option. Vue-i18n needs to read this parameter and mount the $i18n attribute and $t method to vue.

Use and modify the language

After completing the above steps, it is ready for normal use in VUE. Although the calling API is the same, the calling posture is slightly different in vUE components and non-VUE components:

  • in.vueIn the file, language translation is usedthis.$t()Method, switch language modificationthis.$i18n.localeProperties.
  • In other documents, language translations are usedi18n.t()Method, switch language modificationi18n.localeattribute

Pose 1 (used in vUE components) :

{{$t('login.title')}} </div> </template> <script> export Default {methods: {XXX (lang) {// Use this.$i18n.locale = lang}}}; </script>Copy the code

Pose 2 (used in non-VUE components) :

<! -- Filename: /src/store/modules/user.js --> import i18n from "@/lang"; Const actions = {// Switching language switch_language(context, language) {context.com MIT ("set_language", language); i18n.locale = language; return Promise.resolve(); }},Copy the code

Routing management

Common routes need not be encapsulated, but they need to be modified for permission control and automatic menu generation. Route permissions and automatic menu generation are not discussed here and will be explained separately later.

Vue – the router installation

If vue-Router has not been automatically installed in the project, install it first.

npm install vue-router --save
Copy the code

Instance creation

/ SRC /router: create the index.js file directly under/SRC /router. This section involves a lot of content, omit the permission part of the logic, and post the main logic code. Among them, it mainly involves the following aspects:

  • use.envVariable configures the routing mode;
  • Nprogress is used for route transition.
  • Update the site title in the back guard
import Vue from "vue"; import vueRouter from "vue-router"; import normalRoutes from "./modules/normal-routes"; import Nprogress from "@/plugins/nprogress"; import store from "@/store"; import { message } from "@/plugins/element-ui"; import i18n from "@/lang"; Vue.use(vueRouter); Const router = new vueRouter({// History mode is used for development environments, hash mode is used for production environments: VUE_APP_ROUTER_MODE, // routes nested in normalRoutes: normalRoutes,}); BeforeEach (function(to, from, next) {// Start the progress bar nprogress.start (); // Permission handler code... }); Router.aftereach ((to) => {// End the progress bar nprogress.done (); If (to.meta && to.meta. Title) {// Update title document.title = i18n.t("router." + to.meta. Title) + "-" + i18n.t("system.description"); } else { document.title = i18n.t("system.title") + " - " + i18n.t("system.description"); }}); export default router;Copy the code

Module to create

In the SRC /router/modules directory, create two files: normal-routes.js and authed-routes.js. The former saves the routes that do not require permission, and the latter saves the routes that require permission.

<! -- filename: /src/router/modules/normal-routes.js --> export default [ { path: "/login", name: "login", meta: { title: "login", icon: "xxx" }, component: () => import(/* webpackChunkName: "chunk-login" */ "@/pages/login"), }, // ... ]Copy the code

Note import(), which is an asynchronous loading function provided by WebPack. Packing all code into App Chunk can cause first-screen rendering problems; By default, each asynchronously loaded file is packaged into an independent chunk. Using this function, you can pack multiple asynchronously loaded files into the same chunk. This annotation can specify the name of the chunk. This not only achieves the purpose of request optimization, but also facilitates file identification and analysis.

Instance is introduced into

Import the router in/SRC /main.js. Here, delete/SRC /app.vue and directly use rendrer method to generate routing view component. This is level 1 routing view, which is mainly used to render app, login, 404 and other components.

<! -- filename: /src/main.js --> impot router from '@/router' new Vue({ router, render: (h) => h("div", { attrs: { id: "app" } }, [h("router-view")], 1), })Copy the code

Routing view

. In addition to/SRC/main js in grade 1 routing, / SRC/pages/layout/index. The vue routing in view is carrying the whole application level 2 routing, including home, table, markdown and admin – page components.

<! -- filename: /src/pages/layout/index.vue --> <template> ... <! -- Gradient effect, <transition name="fade-transform" mode=" out-of-in "> <keep-alive> <router-view></router-view> </keep-alive> </transition> ... </template>Copy the code

Two built-in components are used:
and

. The former is used for view transitions with the NProgess plug-in. The latter is used to cache components. Currently, there are not many components and there are no restrictions. Both are aimed at improving the user experience.

<! -- filename: /src/router/index.js --> import Nprogress from "@/plugins/nprogress"; BeforeEach (function(to, from, next) {// Start the progress bar nprogress.start (); } router.aftereach ((to) => {// End the progress bar nprogress.done (); }Copy the code

Static resource

The static assets include images, fonts, style files, etc. For now, I put all the static assets in the/SRC /assets directory. For image files, I personally like to place them in the directory of the component. Here is mainly about the processing of style, as well as the introduction and update of icon fonts and other issues.

Style to deal with

The three popular CSS preprocessors (LESS, SCSS, and Stylus) each have their own benefits that are not discussed here. Here are two main reasons I use SCSS here:

  • SCSS fits well with CSS and works well with EMmet (stylus has occasional problems);
  • Make it easy to change the color variables of Elemental-UI (Elemental-UI uses SCSS)

Sass and SCSS are two different generations of versions, the former using indent syntax, the latter using CSS syntax, the earlier sass was written in Ruby and later ported to NodeJS. In nodejs, sass and SCSS are in the same package, and their versions are determined by the. Sass and. SCSS suffixes. This package used Node-sass, which requires Ruby to be installed. Then we use the sass package, which is written using pure NodeJS.

See this article for more information on CSS styles and directory structures.

Install the SCSS

Note the version relationship between SCSS and SCSS-Loader. The latest SCSS and SCSS-Loader may not match each other, and an error may occur if the SCSS is forcibly installed. At present, I use SCSS v1.26.2 and Sass-Loader V8.0.2.

NPM install [email protected] [email protected] --save-devCopy the code

The style name of the class

For CSS style class names, BEM naming is currently used incompletely. For modularity, you can use BEM with file classification, or you can use either of the official moduled and Scoped methods. The scoped method is used here, and it is also easy to use. Vue component style tag to add the scoped attribute.

<! -- filetype: .vue --> <style scoped> ... </style>Copy the code

The directory structure

In addition to styles within the VUE component, we also need to write some other styles involving style resets, common styles, third-party style overrides, and so on. The following is an example, more can be added according to your needs.

<! -- directory: / SRC /assets/styles --> ├─ Base-normalize. SCSS # Normalize. CSS Library ├─ Base-Type.scss # Color Variables ├─ index.scSS # Main File ├─ Layout.app.scSS # Layout Style ├─ Layout.ScrollBar.scSS # ├─ └─ Vendor-nprogress.scss # nprogress.scss # vendor-nprogress.scss # NProgress.scssCopy the code

For third-party styles, the main reason for the rewrite here is to skin the theme with webpack-theme-color-replacer to keep the page style as consistent as possible.

Introduced the global

For styles such as color variables and resets, pre-global injection is required, which can be done using vue.config.js provided by VUe-CLI. Note that the following attributes in the SCSS object are in two cases (official documentation) :

  • Sass – loader < v8, useadditionalDataThe attribute name;
  • Sass – loader > v8, useprependDataAttribute name (currently in use: V8.0.2).
<! -- filename: /vue. Config. js --> module.exports = {// CSS: {loaderOptions: {SCSS: {prependData: `@import "~@/assets/styles/index.scss"; ',},},},}Copy the code

Icon font

Element-ui contains many icon fonts but does not necessarily meet your actual needs. You can use ICONS provided by iconFont, Fontawesome, and other font libraries.

The iconfotn icon is introduced

ICONS on iconfont can be downloaded locally or referenced online. This is referenced directly online, since the ICONS needed at the beginning of the project have not been determined, which is a convenient way to update later. /public/index.html file.

<! -- filename: public/index.html --> <html lang=""> <head> <! <link rel="stylesheet" href="//at.alicdn.com/t/font_2449987_nge8aeefn5.css"> </head> </ HTML >Copy the code

Early treatment

If anyone has noticed, the iconfont and Element-UI ICONS call slightly differently:

  • In element-UI, you usually start withel-icon-xxx(1 class name) in the form of the call icon;
  • In iconfont, it usually begins withiconfont icon-xxx(2 class names) in the form of call icon.

Why is it different? Here, element-UI does this in advance, using property selectors to subtly avoid the problem of overwriting, and the modified code is posted directly below.

<! -- filename: src/assets/fonts/iconfont.scss --> [class*=" icon-"], [class^=icon-] { font-family: iconfont! important; font-style: normal; font-weight: 400; font-variant: normal; text-transform: none; line-height: 1; vertical-align: baseline; display: inline-block; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale }Copy the code

Note: There are two selectors used here, mainly for

and < Element class=”a-class icon-xx”>.

Subject switch

In vue-element-admin, it is mentioned that there are two ways to switch the element-UI theme color:

  • Using color picker, regular replacement of brand color;
  • Use theme-chalk to generate multiple sets of themes, which can be dynamically loaded when switching themes.

However, both methods have limited functionality and do not involve color substitution within vUE components. The webpack-theme-color-replacer plugin is useful for color substitution for vue components and third-party libraries (including Element-UI), as well as support for Element-UI.

Another advantage of this plug-in is that only pulling out styles involving specific colors means that the export volume is small. The full Element-UI style is around 200K, compared to tens of K using plug-ins. Click here to see how plug-ins work.

Plug-in installation

You don’t need to install it under production dependency, you can install it under development dependency.

npm i webpack-theme-color-replacer --save-dev
Copy the code

The topic configuration

This project has preset a set of color variables in/SRC /assets/styles/ base-variable. SCSS, which will be configured as the default theme in the configuration file.

<! -- /src/config/theme.config.js --> const forElementUI = require("webpack-theme-color-replacer/forElementUI"); Module.exports = {default: // CommonJS is used here because the es module cannot be used in vue.config.js. [/ / element - UI theme color... forElementUI getElementUISeries (" # 10 c599 "), / / element - the UI features color - successful color... forElementUI getElementUISeries (" # 3 c9 "), / / element - the UI features color - alarm color... forElementUI getElementUISeries (" # f90 "), / / element - the UI features color - dangerous color... forElementUI getElementUISeries (" # f66 "), / / element - the UI features color - color information... forElementUI getElementUISeries (" # 959595 "), / / the sidebar background color "# 324554"], skyblue: [/ / element - UI theme color... forElementUI getElementUISeries (" # 409 eff "), / / element - the UI features color - successful color... forElementUI getElementUISeries (" # 67 c23a "), / / element - the UI features color - alarm color... forElementUI getElementUISeries (" # E6A23C "), / / element - the UI features color - dangerous color... forElementUI getElementUISeries (" # F56C6C "), / / element - the UI features color - color information... forElementUI getElementUISeries (" # 909399 "), / / the sidebar background color "# 001529"],};Copy the code

Note: The hexadecimal color can be abbreviated as much as possible. For example, # 3C9 should not be written as # 33CC99, otherwise it will cause the inconsistency between the development environment and production environment.

Function to rewrite

In practice, I found that the functional color of the element-UI was abnormal. F12 was found to be a problem with the plug-in changeSelector function, so it was repackaged, the problem was solved, but it is not clear about the side effects.

<! -- filename: /src/helper/change-selector.js --> function changeSelector(selector, util) { switch (selector) { case ".el-button:active": case ".el-button:focus,.el-button:hover": return util.changeEach(selector,".el-button--default:not(.is-plainnot(.el-button--primary)" ); Case ". El-button. is-active,. El-button. is-plain:active": // Note that return "useless"; case ".el-button.is-plain:active": case ".el-button.is-plain:focus,.el-button.is-plain:hover": return util.changeEach(selector, ".el-button--default"); case ".el-pagination button:hover": return selector + ":not(:disabled)"; case ".el-pagination.is-background .el-pager li:not(.disabled):hover": return selector + ":not(.active)"; case ".el-tag": return selector + ":not(.el-tag--dark)"; default: return selector; } } module.exports = changeSelector;Copy the code

Vue configuration

Since vuE-cli4.x is used, you can configure the plugins directly in vue.config.js. Note that vue.config.js cannot use the import statement; you need to use the require statement to import the plug-in. In addition, the plugin provides enhancements to the Element-UI theme colors

<! -- filename: /vue.config.js --> const webpackThemeColorRplacer = require("webpack-theme-color-replacer"); const themeConfig = require("./src/config/theme.config.js"); const changeSelector = require("./src/helper/change-selector"); Module. exports = {// configureWebpack is also available: module.exports = {// configureWebpack is also available: (config) => { config.plugin("webpackThemeColorRplacer").use(webpackThemeColorRplacer, [ { matchColors: themeConfig["default"], fileName: "CSS /chunk-theme-[contenthash:8].css", // Use elemental-UI as a must, otherwise plain and normal button styles will have bugs false, isJsUgly: true, }, ]); }}Copy the code

Function encapsulation

Finish the function of CSS color style above, next write the function of color theme switch.

<! -- filename: src/plugins/theme-replacer.js --> import client from "webpack-theme-color-replacer/client"; import store from "@/store"; const themeConfig = require("@/config/theme.config.js"); Export function replaceThemeColors(color) {const options = {newColors: themeConfig[color]}; return client.changer.changeColor(options, Promise); } / / initializes the theme color export function initThemeColor () {/ / get the current theme const currentTheme = store. State. The default. The theme; if (currentTheme) { document.body.style.display = "none"; / / replace color replaceThemeColors (currentTheme.) finally (() = > {document. Body. Style. The display = ""; }); }}Copy the code

Initialization and usage

/ SRC /main.js for each initialization, the current theme will be saved in localStorage, initialization will read this value for initialization.

<! -- filename: src/main.js --> import { initThemeColor } from "@/plugins/theme-replacer"; // Initialize the theme initThemeColor();Copy the code

The function replaceThemeColors is normally called within a VUE component, but for other reasons, it is combined with vuex and can be called according to personal habits.

<! -- filename: src/store/modules/default.js --> import { replaceThemeColors } from '@/plugins/theme-replacer'; // Switch theme switch_theme(context, theme) {// Switch theme main function, Return replaceThemeColors(theme). Then (() => {// Processing code... }); }},Copy the code

Access control

Access control in a project is not based on roles, but on capabilities or permissions. The main thing is that granular control of permissions is more friendly, and it also works for button control within the page.

Particular way is, in/SRC/config/role. Config. Save all permissions in js configuration in vue – the router meta fields specified capability (or permission) attributes,

Access configuration

In/SRC/config/role. Config. Js will save part (role) and its permissions (capability) configuration, subsequent rights according to the judgment.

<! - filename: / SRC/config/role. Config. Js - > / configuration/user permissions export default {admin: {/ / routing permissions visitAdminPage: True, visitEditorPage: false, // Directive permission publishPost: true, editPost: true, deletePost: true, readPost: true,}, editor: {// Route permission visitAdminPage: false, visitEditorPage: true, // Directive permission publishPost: false, editPost: true, deletePost: false, readPost: true, }, };Copy the code

Permissions function

After logging in, the user will save the role in vuex and sessionStorage. By reading the corresponding permission configuration in role-.config. js, the user can compare it with the permission required to access the target, thus realizing the permission judgment. It is separately encapsulated as a helper function for subsequent calls.

import store from "@/store"; import roleConfig from "@/config/role.config.js"; Function hasCapability(Capability) {// Have permission (Boolean) let isShowedInMenu = false; / / if you need permission to have access to judge the if (capability) {/ / traverse the current user role store. State. The user, role, forEach ((userRole) = > {/ / acquire role permissions table const roleCapabilities = roleConfig[userRole]; If (roleCapabilities[capability] && roleCapabilities[capability] === true) {roleCapabilities[capability] === true isShowedInMenu = true; }}); } else {isShowedInMenu = true;} else {isShowedInMenu = true; } return isShowedInMenu; } export default hasCapability;Copy the code

Permissions page

In the SRC /router/modules directory, create two files: normal-routes.js and authed-routes.js. The former saves the routes that do not require permission, and the latter saves the routes that require permission.

<! -- filename: /src/router/modules/authed-routes.js --> export default [ { path: "/admin-page", name: "adminPage", meta: { title: "adminCapability", icon: "icon-cap", capability: "visitAdminPage", }, component: () => import("@/pages/app/capability"), }, }Copy the code

Compared with common routes, permission routes have a capability attribute added to the meta attribute, indicating the required permission to access the route. When creating a menu, determine whether to render or not.

<! -- filename: /src/pages/layout/components/nav-item/index.vue --> <template> ... <! -- Case [1] : single route --> <el-menu-item v-if="! route.children && ! route.hidden && ! route.meta.link && isCapabilited(route.meta.capability)" :key="route.path" :index="route.path" class="ment-item"> <! < I :class="route.meta.icon"></ I ><! Routing title - - - > < span slot = "title" > {{$t (' the router. '+ route. The meta. Title)}} < / span > < / el menu - item >... </template> <script> export default { methods: IsCapabilited (capability) {return hasCapability(capability)}}} </script>Copy the code

For page routing, it needs to be handled at the global front guard.

<! -- filename: / SRC /router/index.js --> // global front-warder router. BeforeEach (function(to, from, Next) {// Obtain the login token (token) let token = store.state.user.token; If (to.path === "/login") {// A message is displayed indicating that you have logged in and return to the homepage. i18n.t("router.loginedinfo"), type: "success", duration: 2000, onClose() { next({ path: "/" }); }}); } else {/ / check whether has access to the information such as user role const roled = store. State. The user. The role & store. State. The user. The role. The length! = = 0; if (! Store. Dispatch ("user/get_userinfo"). Then (() => {next(); }). Catch (() => {// return to the login page store.com MIT ("user/remove_token"); Nprogress.done(); next("/login"); }); } else {if (hasCapability(to.meta.capability)) {next(); } else { message({ type: "error", message: i18n.t("router.hasNoCapability"), duration: 2000, onClose() { next("/"); }}); }}}} else {// If (whitelist.includes (to.path)) {next(); } else { next({ path: "/login" }); }}});Copy the code

Instruction permissions

For the button permissions in the page, vUE instruction is used to judge the permissions. Here, one permission instruction is encapsulated.

<! -- filename: /src/config/v-cap.js --> import hasCapability from "@/helper/hasCapability"; Inserted (el, bind) {const cap = {inserted(el, bind) {// Get the passed privilege const capability = bind.value; // Return if (! capability) return; const capabilitied = hasCapability(capability); // Remove if (! capabilitied) { el.parentNode && el.parentNode.removeChild(el); }}}; export default cap;Copy the code

Then within the VUE component, pass in the corresponding permissions to decide whether to render the element.

<! -- filename: /src/pages/app/capability/index.vue --> <template> ... <el-button type="primary" v-cap="'publishPost'">{{ $t('capability.publishPost') }}</el-button> ... </template>Copy the code

Navigation menu

In this navigation menu, route is automatically mounted to el-Menu, avoiding manual mounting. Routes here can be divided into three cases:

  • Common routing components, i.epathPoint to the.vueComponents;
  • Nested route: indicates that the current route existschildrenZi lu by;
  • External links, i.epathPointing to external links (e.ghttps://www.juetan.cn).

Routing container

The idea here is to use < el-Menu > as the external container and define the < vue-navItem > component as the recursive component to handle the above situation. The following two things are done here:

  • Fold the variablecollapsedusevuexThis is because this value is used in other components (e.gnav-topComponents).
  • Added support for theme switching, which is used here<el-menu>Provide attribute interface for processing, with pure CSS is also possible, but more trouble.
<! -- filename: /src/pages/layout/components/nav-menu.vue --> <template> <el-scrollbar id="navmenu"> <! -- For routing to work correctly, > <el-menu router :default-active="$route.path" :collapse="collapsed" :unique- Opened ="false" : the collapse - the transition = "false" text - color = "rgba (255255255, 65) :" the active - text - color = "activeColor" :background-color="backgroundColor" class="aside" > <! Router-view --> <vue-navitem :data="routes"></vue-navitem> </el-menu> </el-scrollbar> </template> <script> import vueNavitem from '.. /nav-item' import routes from '@/router/modules/normal-routes' export default { name: "navmenu", data() { return { routes: null } }, computed: { collapsed() { return this.$store.state.default.collapsed }, backgroundColor() { return this.$store.state.default.theme==='skyblue'?'#001529':'#324554' }, activeColor() { return this.$store.state.default.theme==='skyblue'?'#fff':'#ffd04b' } }, created() { this.routes = routes[0].children }, components: { vueNavitem } } </script> <style lang="scss" scoped> .aside.el-menu { border-right: none; } ::v-deep .el-menu-item:not(.is-active):hover { color: $--color-white! important; i { color: $--color-white! important; } } </style>Copy the code

Above, part of the content is theme switching, and style penetration of element-UI components is prefixed with :: V-deep.

Routing component

This is a recursive component that calls its own component in case of child routing, so you need to specify the name attribute. In addition, hasCapability is used for permission determination, visual routing, and mounting. This is a simple implementation, not a complex one.

<! -- filename: /src/pages/layout/components/nav-item.vue --> <template> <div class="navitem"> <! <template v-for="route in data"> <! -- Case [1] : single route --> <el-menu-item v-if="! route.children && ! route.hidden && ! route.meta.link && isCapabilited(route.meta.capability)" :key="route.path" :index="route.path" class="ment-item"> <i :class="route.meta.icon"></i> <span slot="title">{{ $t('router.'+route.meta.title) }}</span> </el-menu-item> <! -- Case [2]: external link --> <a v-else-if="! route.children && ! route.hidden && route.meta.link" :href="route.path" :key="route.path" target="_blank"> <el-menu-item class="ment-item"> <i :class="route.meta.icon"></i> <span slot="title">{{ $t('router.'+route.meta.title) }}</span> </el-menu-item> </a> <! - the situation [3] : <el-submenu v-else-if="route.children" ref=" submenu ":key="route.path" :index="route.path" popper-append-to-body> <template slot="title"> <i :class="route.meta.icon"></i> <span slot="title">{{ $t('router.'+route.meta.title) }}</span> </template> <nav-item :data="route.children"/> </el-submenu> </template> </div>  </template> <script> import hasCapability from '@/helper/hasCapability'; export default { name: "navItem", props:{ data: { type: Array, required: true } }, methods: {// Determine whether the current route requires permissions, IsCapabilited (Capability) {return hasCapability(Capability)}} </script> <style lang=" SCSS "> // Style processing </style>Copy the code

Above, the logic is basically finished, the above CSS style is not posted, mainly want to say two style problems:

  1. Using non-Element-UI ICONS requires a bit of styling (see below). Note: above<style>Label not addedscopedProperty, can also be used::v-deepDo style infiltration.
.el-menu-item [class^=icon-],.el-submenu [class^=icon-] {
    margin-right: 5px;
    width: 24px;
    text-align: center;
    font-size: 18px;
    vertical-align: middle;
}
Copy the code
  1. Nested routines expose the title when folded because the el-Menu subcomponent should be El-Submenu, EL-Menu-item, or el-Menu-item-group. The outermost layer of my child component is div, and there is a style problem when routing is nested, so add the processing style here.
El-menu --collapse. El-submenu__title span{display: none; El-menu --collapse. El-submenu__title. El-submenu__icon-arrow {display: none; }Copy the code

Volume optimization

For some libraries (element-UI, Echarts, etc.), complete packaging is not necessary, and volume optimization can be done by loading on demand.

Volumetric analysis

Analyzer installation

Install to a development dependency.

npm install analyzer-webpack-plugin --save-dev
Copy the code

Webpack configuration

This package is a Webpack plug-in and needs to be configured in Webpack. For vue-cli4, you can configure it in /vue.config.js. One thing to note here: If there is a need to use something like Github Action, you need to unplug the plug-in, otherwise the build will hang.

<! -- filename: /vue. Config. js --> module.exports = {// chain configwebpack chainWebpack: (config) => { config.plugin("BundleAnalyzerPlugin") .use(require("webpack-bundle-analyzer").BundleAnalyzerPlugin, [{// Automatically open openAnalyzer: false,},]); }}Copy the code

Gzip compression

There are two types of Gzip compression: static Gzip and dynamic Gzip. Static Gzip compression, where the Nginx server reads the. Gz file from the website directory and returns it to the browser; Dynamic Gzip compression, in which the Nginx server reads.js,.css, and.html files from the website directory, compresses them into.gz files, and returns them to the browser.

For Github Pages, dynamic Gzip compression is used and files are not read even when compressed into a.gz file. However, if you want to deploy to your own server, using static Gzip compression is still necessary, since it is worth sacrificing a bit of memory for return speed.

Compression will install

If the latest version fails to be installed, you can try to roll back to a previous version.

NPM I [email protected] -d is an error when installing the latest 7.1.2 versionCopy the code

Webpack configuration

Generally used in production environments,

<! -- filename: /vue.config.js --> module.exports = {// webpack config.plugin("compression"). File extension test: [{/ / regular matching/js $|. HTML $|. CSS $/, / / compression threshold for more than 10 KB file: 1, // deleteOriginalAssets: false,},]); }Copy the code

Asynchronous loading

It is not necessary to load all the asynchronous components at once. Instead, you can use the import() function to load them asynchronously: it packs them into a separate chunk and requests them at the time of use.

Used in routing

WebpackChunkName in the comment refers to the packaged Chunk name. If there are multiple small routing components and you do not want to request each component separately, you can package them into the same chunk by specifying webpackChunkName.

component: () => import(/* webpackChunkName: "chunk-home" */ "@/pages/app/home"),
Copy the code

The last

The above content is summarized later, there may be mistakes and deficiencies, welcome to point out! If you think it is useful, you can help to click star(project address). In addition, I take this opportunity to apply for a front-end job. I graduated from an ordinary university in 19 years and worked in a foreign subsidiary for one and a half years (non-front-end, now I have left). Non-zero foundation, I have been in touch with vUE since university (VUE used in graduation), basic is ok, I have certain English reading and writing ability and self-study ability, details can see my blog: www.juetan.cn/65