由于最近公司要開發(fā)一個后臺管理系統(tǒng),查閱了很多vue框架,本人覺得element簡潔,方便,于是選擇它作為我們的首選框架,并分享給大家,如果您覺得有需要改進的地方可以提出來一起探討,Github地址。本文篇幅比較長,希望同學們可以耐心的讀下去,如有不懂可以下方留言
成都創(chuàng)新互聯(lián)公司網(wǎng)站建設(shè)公司一直秉承“誠信做人,踏實做事”的原則,不欺瞞客戶,是我們最起碼的底線! 以服務(wù)為基礎(chǔ),以質(zhì)量求生存,以技術(shù)求發(fā)展,成交一個客戶多一個朋友!專注中小微企業(yè)官網(wǎng)定制,網(wǎng)站設(shè)計、做網(wǎng)站,塑造企業(yè)網(wǎng)絡(luò)形象打造互聯(lián)網(wǎng)企業(yè)效應(yīng)。
首先全局安裝的vue框架,這里是用的npm包管理工具來安裝的,如果你的網(wǎng)不是很好的話可以先安裝淘寶鏡像 npm install -g cnpm -registry=https://registry.npm.taobao.org
,然后通過cnpm來安裝
cnpm install -g @vue/cli or npm install -g @vue/cli
其次開始安裝vue腳手架,當前是第三版本vue-cli 3.x
cnpm install -g @vue/cli
安裝完成后,你還可以用這個命令來檢查其版本是否正確 (3.x):
vue --version
安裝腳手架后開始創(chuàng)建我們的項目
vue create vue-admin-project
隨后會出現(xiàn)兩個選項
選擇第二項并繼續(xù),并選擇自己需要配置的功能,完成后并繼續(xù),然后開始生成項目
項目初始化成功
接下來按照上面的提示運行 cd app
以及啟動本地服務(wù)器 npm run serve
,當運行完成之后會提示你打來本地端口 http://localhost:8080
,會出現(xiàn)歡迎頁面,此時代表你的vue項目初始化完成。
整理前的初始目錄
|-- vue-admin-project |-- .gitignore //git項目忽視文件 |-- babel.config.js //babel 配置文件 |-- package-lock.json //記錄安裝包的具體版本號 |-- package.json //包的類型 |-- README.md |-- public //項目打包后的目錄 | |-- favicon.ico | |-- index.html |-- src //項目開發(fā)目錄 |-- App.vue //主入口文件 |-- main.js //主入口文件 |-- router.js //vue-router文件 |-- store.js //vuex |-- assets //靜態(tài)文件 |-- logo.png |-- components //組件存放目錄 |-- HelloWorld.vue |-- views //視圖目錄 |-- About.vue |-- Home.vue
整理后的目錄,主要更改 /src
文件夾下的目錄
|-- vue-admin-project |-- .gitignore |-- babel.config.js |-- package-lock.json |-- package.json |-- README.md |-- public |-- favicon.ico |-- index.html |-- src |-- App.vue |-- main.js |-- assets |-- logo.png |-- components |-- HelloWorld.vue |-- router //路由配置文件夾 |-- router.js |-- store //狀態(tài)管理文件夾 |-- store.js |-- views |-- About.vue |-- Home.vue
vue-cli 3.0x與vue-cli 2.0x最主要的區(qū)別是項目結(jié)構(gòu)目錄精簡化,這也帶來了許多問題,很多配置需要自己配置,由于2.0x版本中直接在 cofig/
文件夾下面配置開發(fā)環(huán)境與線上環(huán)境,3.0x則需要自己配置。
首先配置開發(fā)環(huán)境,在項目根目錄下新建一個文件 .env
文件。
NODE_ENV="development" //開發(fā)環(huán)境 BASE_URL="http://localhost:3000/" //開發(fā)環(huán)境接口地址
接下來我們配置線上環(huán)境,同樣在項目根目錄新建一個文件 .env.prod
這就表明是生產(chǎn)環(huán)境。
NODE_ENV="production" //生產(chǎn)環(huán)境 BASE_URL="url" //生產(chǎn)環(huán)境的地址
現(xiàn)在我們?nèi)绾卧陧椖恐信袛喈斍碍h(huán)境呢?
我們可以根據(jù) process.env.BASE_URL
來獲取它是線上環(huán)境還是開發(fā)環(huán)境,后面會有運用
if(process.env.NODE_ENV='development'){ console.log( process.env.BASE_URL) //http://localhost:3000/ }else{ console.log( process.env.BASE_URL) //url }
至此,我們成功的配置好了開發(fā)環(huán)境與線上環(huán)境。
講到 vue.config.js
項目配置文件,又不得不說下3.x和2.x的區(qū)別,2.x里面webpack相關(guān)的配置項直接在項目的 build/webpack.base.conf.js
里面配置,而3.x完全在 vue.config.js
中配置,這使得整個項目看起來更加簡潔明了,項目運行速度更快。
由于項目初始化的時候沒有 vue.config.js
配置文件,因此我們需要在項目根目錄下新建一個 vue.config.js
配置項。
在這個配置項里面,本項目主要是配置三個東西,第一個就是目錄別名 alias
,另一個是項目啟動時自動打開瀏覽器,最后一個就是處理引入的全局scss文件。當然有 vue.config.js的配置遠遠不止這幾項,有興趣的同學可以去看看vue.config.js具體配置,具體代碼如下。
let path=require('path'); function resolve(dir){ return path.join(__dirname,dir) } module.exports = { chainWebpack: config => { //設(shè)置別名 config.resolve.alias .set('@',resolve('src')) }, devServer: { open:true //打開瀏覽器窗口 }, //定義scss全局變量 css: { loaderOptions: { sass: { data: `@import "@/assets/scss/global.scss";` } } } }
開始安裝ElementUI
vue add element
接下來兩個選項,第一個是全部引入,第二個是按需引入,我選擇第一個 Fully import
,大家可以按照自己的項目而定。接下來會詢問是否引入scss,這里選擇是,語言選擇zh-cn。
接下來會提示安裝成功,并在項目首頁有一個element樣式的按鈕。
路由管理也是本項目核心部分。
1.引入文件
import Vue from 'vue' import Router from 'vue-router' import store from '../store/store' //引入狀態(tài)管理 import NProgress from 'nprogress' //引入進度條組件 cnpm install nprogress --save import 'nprogress/nprogress.css' Vue.use(Router)
2.路由懶加載
/** *@parma {String} name 文件夾名稱 *@parma {String} component 視圖組件名稱 */ const getComponent = (name,component) => () => import(`@/views/${name}/${component}.vue`);
3.路由配置
const myRouter=new Router({ routes: [ { path: '/', redirect: '/home', component: getComponent('login','index') }, { path: '/login', name: 'login', component: getComponent('login','index') }, { path: '/', component:getComponent('layout','Layout'), children:[{ path:'/home', name:'home', component: getComponent('home','index'), meta:{title:'首頁'} }, { path:'/icon', component: getComponent('icons','index'), name:'icon', meta:{title:'自定義圖標'} }, { path:'/editor', component: getComponent('component','editor'), name:'editor', meta:{title:'富文本編譯器'} }, { path:'/countTo', component: getComponent('component','countTo'), name:'countTo', meta:{title:'數(shù)字滾動'} }, { path:'/tree', component: getComponent('component','tree'), name:'tree', meta:{title:'自定義樹'} }, { path:'/treeTable', component: getComponent('component','treeTable'), name:'treeTable', meta:{title:'表格樹'} }, { path:'/treeSelect', component: getComponent('component','treeSelect'), name:'treeSelect', meta:{title:'下拉樹'} }, { path:'/draglist', component: getComponent('draggable','draglist'), name:'draglist', meta:{title:'拖拽列表'} }, { path:'/dragtable', component: getComponent('draggable','dragtable'), name:'dragtable', meta:{title:'拖拽表格'} }, { path:'/cricle', component: getComponent('charts','cricle'), name:'cricle', meta:{title:'餅圖'} }, ] } ] })
4.本項目存在一個token,來驗證權(quán)限問題,因此進入頁面的時候需要判斷是否存在token,如果不存在則跳轉(zhuǎn)到登陸頁面
//判斷是否存在token myRouter.beforeEach((to,from,next)=>{ NProgress.start() if (to.path !== '/login' && !store.state.token) { next('/login') //跳轉(zhuǎn)登錄 NProgress.done() // 結(jié)束Progress } next() }) myRouter.afterEach(() => { NProgress.done() // 結(jié)束Progress })
5.導出路由
export default myRouter
1.接口處理我選擇的是axios,由于它遵循promise規(guī)范,能很好的避免回調(diào)地獄?,F(xiàn)在我們開始安裝
cnpm install axios -S
2.在 src
目錄下新建文件夾命名為 api
,里面新建兩個文件,一個是 api.js
,用于接口的整合,另一個是 request.js
,根據(jù)相關(guān)業(yè)務(wù)封裝axios請求。
request.js
1.引入依賴
import axios from "axios"; import router from "../router/router"; import { Loading } from "element-ui"; import {messages} from '../assets/js/common.js' //封裝的提示文件 import store from '../store/store' //引入vuex
2.編寫axios基本設(shè)置
axios.defaults.timeout = 60000; //設(shè)置接口超時時間 axios.defaults.baseURL = process.env.BASE_URL; //根據(jù)環(huán)境設(shè)置基礎(chǔ)路徑 axios.defaults.headers.post["Content-Type"] = "application/x-www-form-urlencoded;charset=UTF-8"; //設(shè)置編碼 let loading = null; //初始化loading
3.編寫請求攔截,也就是說在請求接口前要做的事情
/* *請求前攔截 *用于處理需要請求前的操作 */ axios.interceptors.request.use( config => { loading = Loading.service({ text: "正在加載中......", fullscreen: true }); if (store.state.token) { config.headers["Authorization"] = "Bearer " + store.state.token; } return config; }, error => { return Promise.reject(error); } );
4.編寫請求響應(yīng)攔截,用于處理數(shù)據(jù)返回操作
/* *請求響應(yīng)攔截 *用于處理數(shù)據(jù)返回后的操作 */ axios.interceptors.response.use( response => { return new Promise((resolve, reject) => { //請求成功后關(guān)閉加載框 if (loading) { loading.close(); } const res = response.data; if (res.err_code === 0) { resolve(res) } else{ reject(res) } }) }, error => { console.log(error) //請求成功后關(guān)閉加載框 if (loading) { loading.close(); } //斷網(wǎng)處理或者請求超時 if (!error.response) { //請求超時 if (error.message.includes("timeout")) { console.log("超時了"); messages("error", "請求超時,請檢查互聯(lián)網(wǎng)連接"); } else { //斷網(wǎng),可以展示斷網(wǎng)組件 console.log("斷網(wǎng)了"); messages("error", "請檢查網(wǎng)絡(luò)是否已連接"); } return; } const status = error.response.status; switch (status) { case 500: messages("error", "服務(wù)器內(nèi)部錯誤"); break; case 404: messages( "error", "未找到遠程服務(wù)器" ); break; case 401: messages("warning", "用戶登陸過期,請重新登陸"); localStorage.removeItem("token"); setTimeout(() => { router.replace({ path: "/login", query: { redirect: router.currentRoute.fullPath } }); }, 1000); break; case 400: messages("error", "數(shù)據(jù)異常"); break; default: messages("error", error.response.data.message); } return Promise.reject(error); } );
5.請求相關(guān)的事情已經(jīng)完成,現(xiàn)在開始封裝get,post請求
/* *get方法,對應(yīng)get請求 *@param {String} url [請求的url地址] *@param {Object} params [請求時候攜帶的參數(shù)] */ export function get(url, params) { return new Promise((resolve, reject) => { axios .get(url, { params }) .then(res => { resolve(res); }) .catch(err => { reject(err); }); }); } /* *post方法,對應(yīng)post請求 *@param {String} url [請求的url地址] *@param {Object} params [請求時候攜帶的參數(shù)] */ export function post(url, params) { return new Promise((resolve, reject) => { axios .post(url, params) .then(res => { resolve(res); }) .catch(err => { reject(err); }); }); }
api.js
封裝好axios的業(yè)務(wù)邏輯之后自然要開始,運用,首先引入 get
以及 post
方法
import {get,post} from './request';
接下來開始封裝接口,并導出
//登陸 export const login=(login)=>post('/api/post/user/login',login) //上傳 export const upload=(upload)=>get('/api/get/upload',upload)
那我們?nèi)绾握{(diào)用接口呢?以登陸頁面為例。
import { login } from "@/api/api.js"; //引入login
/** * @oarma {Object} login 接口傳遞的參數(shù) */ login(login) .then(res => { //成功之后要做的事情 }) .catch(err => { //出錯時要做的事情 });
接口相關(guān)的邏輯已經(jīng)處理完。
由于vue項目中組件之間傳遞數(shù)據(jù)比較復雜,因此官方引入了一個全局狀態(tài)管理的東東,也就是現(xiàn)在要說的vuex,vuex能更好的管理數(shù)據(jù),方便組件之間的通信。
現(xiàn)在在store文件夾下面新建四個文件 state.js
, mutations.js
, getter.js
, action.js
。
state.js
state就是Vuex中的公共的狀態(tài), 我是將state看作是所有組件的data, 用于保存所有組件的公共數(shù)據(jù).
const state = { token: '',//權(quán)限驗證 tagsList: [], //打開的標簽頁個數(shù), isCollapse: false, //側(cè)邊導航是否折疊 } export default state //導出
mutations.js
我將mutaions理解為store中的methods, mutations對象中保存著更改數(shù)據(jù)的回調(diào)函數(shù),該函數(shù)名官方規(guī)定叫type, 第一個參數(shù)是state, 第二參數(shù)是payload, 也就是自定義的參數(shù).改變state的值必須經(jīng)過mutations
const mutations = { //保存token COMMIT_TOKEN(state, object) { state.token = object.token; }, //保存標簽 TAGES_LIST(state, arr) { state.tagsList = arr; }, IS_COLLAPSE(state, bool) { state.isCollapse = bool; } } export default mutations
getter.js
我將getters屬性理解為所有組件的computed屬性,也就是計算屬性。vuex的官方文檔也是說到可以將getter理解為store的計算屬性, getters的返回值會根據(jù)它的依賴被緩存起來,且只有當它的依賴值發(fā)生了改變才會被重新計算。
const getters={ //你要計算的屬性 } export default getters
action.js
actions 類似于 mutations,不同在于:
1.actions提交的是mutations而不是直接變更狀態(tài)
2.actions中可以包含異步操作, mutations中絕對不允許出現(xiàn)異步
3.actions中的回調(diào)函數(shù)的第一個參數(shù)是context, 是一個與store實例具有相同屬性和方法的對象
const actions={ } export default actions
store.js
store.js是vuex模塊整合文件,由于刷新頁面會造成vuex數(shù)據(jù)丟失,所以這里引入了一個vuex數(shù)據(jù)持久話插件,將state里面的數(shù)據(jù)保存到localstorage。
安裝 vuex-persistedstate
npm install vuex-persistedstate --save
import Vue from 'vue' import Vuex from 'vuex' import state from "./state"; import mutations from "./mutations"; import actions from "./actions"; import getters from "./getters"; //引入vuex 數(shù)據(jù)持久化插件 import createPersistedState from "vuex-persistedstate" Vue.use(Vuex) export default new Vuex.Store({ state, mutations, actions, getters, plugins: [createPersistedState()] })
至此vuex引入完畢,如同學們還有不明白的可以去翻閱vuex文檔。
現(xiàn)在我們開始進行頁面的布局。首先我們來分析下首頁的情況
首先我們在 view
文件夾下面新建一個 layout
文件夾,里面再添加一個 layout.vue
,以及 compentents
文件夾。
側(cè)邊欄
在compentents文件夾下面新建一個 Aside.vue
文件,實現(xiàn)路由跳轉(zhuǎn)相關(guān)的邏輯,運用了element導航菜單的路由模式,如有不明白的可以去ElementUI導航菜單去看看。
{{ item.title }} {{ subItem.title }} {{ threeItem.title }} {{ subItem.title }} {{ item.title }}
import { mapState } from "vuex"; export default { data() { return { //配置目錄 items: [ { icon: "el-icon-edit-outline", index: "home", title: "系統(tǒng)首頁" }, { icon: "el-icon-edit-outline", index: "icon", title: "自定義圖標" }, { icon: "el-icon-edit-outline", index: "component", title: "組件", subs: [ { index: "editor", title: "富文本編譯器" }, { index: "countTo", title: "數(shù)字滾動" }, { index: "trees", title: "樹形控件", subs: [ { index: "tree", title: "自定義樹" }, { index: "treeSelect", title: "下拉樹" } // ,{ // index:'treeTable', // title:'表格樹', // } ] }, ] }, { icon: "el-icon-edit-outline", index: "draggable", title: "拖拽", subs: [ { index: "draglist", title: "拖拽列表" }, { index: "dragtable", title: "拖拽表格" } ] }, { icon: "el-icon-edit-outline", index: "charts", title: "圖表", subs: [ { index: "cricle", title: "餅圖" }, ] }, { icon: "el-icon-edit-outline", index: "7", title: "錯誤處理", subs: [ { index: "permission", title: "權(quán)限測試" }, { index: "404", title: "404頁面" } ] }, ] }; }, computed: { onRoutes() { return this.$route.path.replace("/", ""); }, ...mapState(["isCollapse"]) //從vuex里面獲取菜單是否折疊 }, methods: { //下拉展開 handleOpen(key, keyPath) { console.log(key, keyPath); }, //下來關(guān)閉 handleClose(key, keyPath) { console.log(key, keyPath); } } };
頂部欄
在 view/compentents
文件夾下面新建一個 Header.vue
{{username }}首頁 個人設(shè)置 退出登陸
import showAside from "@/components/showAside.vue";//引入了一個側(cè)邊欄是否折疊的組件 export default { // name:'header', components: { showAside }, data() { return { fullscreen: false, name: "linxin", message: 2, username: "zyh" }; }, computed: { isCollapse: { get: function() { return this.$store.state.isCollapse; }, set: function(newValue) { console.log(newValue); this.$store.commit("IS_COLLAPSE", newValue);//提交到vuex } } }, methods: { toggleClick() { this.isCollapse = !this.isCollapse; }, // 用戶名下拉菜單選擇事件 logout(command) { this.$router.push("/login"); }, // 全屏事件 handleFullScreen() { let element = document.documentElement; if (this.fullscreen) { if (document.exitFullscreen) { document.exitFullscreen(); } else if (document.webkitCancelFullScreen) { document.webkitCancelFullScreen(); } else if (document.mozCancelFullScreen) { document.mozCancelFullScreen(); } else if (document.msExitFullscreen) { document.msExitFullscreen(); } } else { if (element.requestFullscreen) { element.requestFullscreen(); } else if (element.webkitRequestFullScreen) { element.webkitRequestFullScreen(); } else if (element.mozRequestFullScreen) { element.mozRequestFullScreen(); } else if (element.msRequestFullscreen) { // IE11 element.msRequestFullscreen(); } } this.fullscreen = !this.fullscreen; } } };
現(xiàn)在在 src/components
文件夾下面新建一個 showAside.vue
組件
export default { name: "showAside", props: { toggleClick: { type: Function, default: null } } };
頂部導航欄標簽組件
在 view/compentents
文件夾下面新建一個 Tags.vue
標簽選項 關(guān)閉其他
import { messages } from "@/assets/js/common.js"; export default { created() { //判斷標簽里面是否有值 有的話直接加載 if (this.tagsList.length == 0) { this.setTags(this.$route); } }, computed: { //computed 方法里面沒有set方法因此不能使用mapState,需要重新定義set方法 tagsList: { get: function() { return this.$store.state.tagsList; }, set: function(newValue) { this.$store.commit("TAGES_LIST", newValue); // this.$store.state.tagsList = newValue; } } }, watch: { //監(jiān)聽路由變化 $route(newValue, oldValue) { this.setTags(newValue); } }, methods: { //選中的高亮 isActive(path) { return path === this.$route.fullPath; }, handleCommand(command) { if (command == "closeOther") { // 關(guān)閉其他標簽 const curItem = this.tagsList.filter(item => { return item.path === this.$route.fullPath; }); this.tagsList = curItem; } }, //添加標簽 setTags(route) { let isIn = this.tagsList.some(item => { //判斷標簽是否存在 return item.path === route.fullPath; }); //不存在 if (!isIn) { // 判斷當前的標簽個數(shù) if (this.tagsList.length >= 10) { messages("warning", "當標簽大于10個,請關(guān)閉后再打開"); } else { this.tagsList.push({ title: route.meta.title, path: route.fullPath, name: route.name }); //存到vuex this.$store.commit("TAGES_LIST", this.tagsList); } } }, closeTags(index) { console.log(this.tagsList.length); if (this.tagsList.length == 1) { messages("warning", "不可全都關(guān)閉"); } else { //刪除當前 let tags = this.tagsList.splice(index, 1); this.$store.commit("TAGES_LIST", this.tagsList); } } } };
接下來在 view/compentents
文件夾下面新建一個 Main.vue
,主要是將頂部導航標簽欄以及內(nèi)容部分結(jié)合起來。
import Tags from './Tags.vue' export default { components:{ Tags } }
相關(guān)組件寫好,在layout組件中匯總
import Aside from "./components/Aside.vue"; import Header from "./components/Header.vue"; import Main from "./components/Main.vue"; import { mapState } from "vuex"; export default { name: "Layout", components: { Aside, Header, Main }, computed: { ...mapState(["isCollapse"]) } };
至此首頁布局已經(jīng)規(guī)劃完成,如有不太清楚的可以查看項目地址
管理系統(tǒng)是多種多樣的,每家公司都有不同的業(yè)務(wù)邏輯,本篇文章也只是拋磚引玉,還有許多需要修正改進的地方,如果同學們有更好的想法可以提出來希望大家一起完善本項目。
|-- vue-admin-project |-- .env |-- .env.prod |-- .env.test |-- .gitignore |-- babel.config.js |-- package-lock.json |-- package.json |-- README.md |-- vue.config.js |-- public | |-- favicon.ico | |-- index.html |-- src |-- App.vue |-- element-variables.scss |-- main.js |-- api | |-- api.js | |-- request.js |-- assets | |-- logo.png | |-- css | | |-- normalize.css | | |-- public.css | |-- icon | | |-- demo.css | | |-- demo_index.html | | |-- iconfont.css | | |-- iconfont.eot | | |-- iconfont.js | | |-- iconfont.svg | | |-- iconfont.ttf | | |-- iconfont.woff | | |-- iconfont.woff2 | |-- img | | |-- tou.jpg | |-- js | | |-- common.js | |-- scss | |-- global.scss |-- components | |-- showAside.vue |-- plugins | |-- element.js |-- router | |-- router.js |-- store | |-- actions.js | |-- getters.js | |-- mutations.js | |-- state.js | |-- store.js |-- views |-- layout | |-- Layout.vue | |-- components | |-- Aside.vue | |-- Header.vue | |-- Main.vue | |-- Tags.vue
最后項目目錄文件結(jié)構(gòu)
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。