CMD安裝所須要的pom依賴
css
npm install element-ui -S html
npm install axios -Svue
npm install qs -S java
npm install vue-axios -Swebpack
package.json中查看安裝的依賴ios
在項目中src目錄下找到main.js,並在指定位置添加三行代碼(main.js是入口文件,因此在這裏引入就行,頁面就不用引入了)
import Vue from 'vue'
import ElementUI from 'element-ui' //新添加1
import 'element-ui/lib/theme-chalk/index.css' //新添加2,避免後期打包樣式不一樣,要放在import App from './App';以前
import App from './App'
import router from './router'web
Vue.use(ElementUI) //新添加3
Vue.config.productionTip = falseajax
main.jsvue-router
1 // The Vue build version to load with the `import` command 2 // (runtime-only or standalone) has been set in webpack.base.conf with an alias. 3 import Vue from 'vue' 4 import 'element-ui/lib/theme-chalk/index.css' 5 import App from './App' 6 import router from './router' 7 import ElementUI from 'element-ui' 8 import axios from '@/api/http' 9 import VueAxios from 'vue-axios' 10 11 Vue.use(VueAxios,axios) 12 Vue.use(ElementUI) 13 Vue.config.productionTip = false 14 15 /* eslint-disable no-new */ 16 new Vue({ 17 el: '#app', 18 router, 19 components: { App }, 20 template: '<App/>' 21 })
修改HelloWorld.vue添加elementUI組件查看效果vuex
ElementUI官網:https://element.eleme.cn/#/zh-CN
Vue+ElementUI設計登錄頁面
在vue組件中,在style標籤上添加scoped屬性,以表示它的樣式做用於當下的模塊,很好的實現了樣式私有化的目的
auto-complete="off"
autocomplete 屬性是 HTML5 中的新屬性,off-----禁用自動完成
App.vue
1 <style> 2 html, 3 body { 4 width: 100%; 5 height: 100%; 6 box-sizing: border-box; 7 padding: 0px; 8 margin: 0px; 9 } 10 #app { 11 font-family: "Avenir", Helvetica, Arial, sans-serif; 12 -webkit-font-smoothing: antialiased; 13 -moz-osx-font-smoothing: grayscale; 14 color: #2c3e50; 15 widows: 100%; 16 height: 100%; 17 } 18 </style>
login.vue
1 <style scoped> 2 .login-wrap { 3 box-sizing: border-box; 4 width: 100%; 5 height: 100%; 6 padding-top: 10%; 7 background-image: url(); 8 /* background-color: #112346; */ 9 background-repeat: no-repeat; 10 background-position: center right; 11 background-size: 100%; 12 } 13 .login-container { 14 border-radius: 10px; 15 margin: 0px auto; 16 width: 350px; 17 padding: 30px 35px 15px 35px; 18 background: #fff; 19 border: 1px solid #eaeaea; 20 text-align: left; 21 box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1); 22 } 23 .title { 24 margin: 0px auto 40px auto; 25 text-align: center; 26 color: #505458; 27 } 28 </style>
大概界面
跳轉
login.vue
1 <template> 2 <div class=".login-wrap"> 3 <el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container"> 4 <h3 style="text-align: center;">用戶登錄</h3> 5 <el-form-item label="用戶名" prop="uname"> 6 <el-input type="uname" v-model="ruleForm.pass" autocomplete="off"></el-input> 7 </el-form-item> 8 <el-form-item label="密碼" prop="pwd"> 9 <el-input type="pwd" v-model="ruleForm.checkPass" autocomplete="off"></el-input> 10 </el-form-item> 11 <el-form-item> 12 <el-row> 13 <el-col :span="24"> 14 <div class="grid-content bg-purple-dark"></div> 15 </el-col> 16 <el-button style="width: 100%;" type="primary" @click='doSub'>提交</el-button> 17 </el-row> 18 <el-row> 19 <el-col :span="12"> 20 <div class="grid-content bg-purple-dark"></div> 21 <el-link type="success" @click='toLogin'>用戶註冊</el-link> 22 </el-col> 23 <el-col :span="12"> 24 <div class="grid-content bg-purple-dark"></div> 25 <el-link type="success">找回密碼</el-link> 26 </el-col> 27 </el-row> 28 29 </el-form-item> 30 </el-form> 31 </div> 32 </template> 33 34 <script> 35 export default { 36 data() { 37 return { 38 ruleForm: { 39 uname: ',', 40 pwd: '' 41 } 42 }; 43 }, 44 methods:{ 45 doSub(){ 46 47 }, 48 toReg(){ 49 this.$router.push({ 50 path:'/Reg' 51 }) 52 } 53 } 54 } 55 </script>
index.js
1 import Vue from 'vue' 2 import Router from 'vue-router' 3 import HelloWorld from '@/components/HelloWorld' 4 import Login from '@/views/Login' 5 import Reg from '@/views/Reg' 6 7 Vue.use(Router) 8 9 export default new Router({ 10 routes: [ 11 { 12 path: '/', 13 name: 'Login', 14 component: Login 15 }, 16 { 17 path: '/Login', 18 name: 'Login', 19 component: Login 20 }, 21 { 22 path: '/Reg', 23 name: 'Reg', 24 component: Reg 25 } 26 ] 27 })
後臺交互(axios/qs/vue-axios)
axios
axios是vue2提倡使用的輕量版的ajax。它是基於promise的HTTP庫。
vue.js有著名的全家桶系列:vue-router,vuex, vue-resource,再加上構建工具vue-cli,就是一個完整的vue項目的核心構成。
其中vue-resource是Vue.js的一款插件,它能夠經過XMLHttpRequest或JSONP發起請求並處理響應,但在vue更新到2.0以後,
做者就宣告再也不對vue-resource更新,而是推薦的axios
axios跨域問題
會一直報錯:「http://127.0.0.1:8848' has been blocked by CORS policy: No 'Access-Control-Allow-Origin' header」
由於使用了先後端分離開發,跨域訪問了,解決方案:須要配置tomcat容許跨域訪問,
tomcat跨域配置方法不少,但最簡單的方式是本身寫一個過濾器CorsFilter實現,添加一個響應頭
CorsFilter
1 package com.zking.vue.util; 2 3 import java.io.IOException; 4 5 import javax.servlet.Filter; 6 import javax.servlet.FilterChain; 7 import javax.servlet.FilterConfig; 8 import javax.servlet.ServletException; 9 import javax.servlet.ServletRequest; 10 import javax.servlet.ServletResponse; 11 import javax.servlet.http.HttpServletRequest; 12 import javax.servlet.http.HttpServletResponse; 13 14 /** 15 * 配置tomcat容許跨域訪問 16 * 17 * @author Administrator 18 * 19 */ 20 public class CorsFilter implements Filter { 21 22 @Override 23 public void init(FilterConfig filterConfig) throws ServletException { 24 } 25 26 // @Override 27 // public void doFilter(ServletRequest servletRequest, ServletResponse 28 // servletResponse, FilterChain filterChain) 29 // throws IOException, ServletException { 30 // HttpServletResponse httpResponse = (HttpServletResponse) servletResponse; 31 // 32 // // Access-Control-Allow-Origin就是咱們須要設置的域名 33 // // Access-Control-Allow-Headers跨域容許包含的頭。 34 // // Access-Control-Allow-Methods是容許的請求方式 35 // httpResponse.addHeader("Access-Control-Allow-Origin", "*");// *,任何域名 36 // httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, 37 // DELETE"); 38 // // httpResponse.setHeader("Access-Control-Allow-Headers", "Origin, 39 // // X-Requested-With, Content-Type, Accept"); 40 // 41 // // 容許請求頭Token 42 // httpResponse.setHeader("Access-Control-Allow-Headers", 43 // "Origin,X-Requested-With, Content-Type, Accept, Token"); 44 // HttpServletRequest req = (HttpServletRequest) servletRequest; 45 // System.out.println("Token=" + req.getHeader("Token")); 46 // if("OPTIONS".equals(req.getMethod())) { 47 // return; 48 // } 49 // 50 // 51 // filterChain.doFilter(servletRequest, servletResponse); 52 // } 53 54 @Override 55 public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) 56 throws IOException, ServletException { 57 HttpServletResponse resp = (HttpServletResponse) servletResponse; 58 HttpServletRequest req = (HttpServletRequest) servletRequest; 59 60 // Access-Control-Allow-Origin就是咱們須要設置的域名 61 // Access-Control-Allow-Headers跨域容許包含的頭。 62 // Access-Control-Allow-Methods是容許的請求方式 63 resp.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名 64 resp.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE"); 65 // resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With, 66 // Content-Type, Accept"); 67 // 容許客戶端,發一個新的請求頭jwt 68 resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With, Content-Type, Accept, jwt"); 69 70 // 容許客戶端,處理一個新的響應頭jwt 71 resp.setHeader("Access-Control-Expose-Headers", "jwt"); 72 // String sss = resp.getHeader("Access-Control-Expose-Headers"); 73 // System.out.println("sss=" + sss); 74 75 // 容許請求頭Token 76 // httpResponse.setHeader("Access-Control-Allow-Headers","Origin,X-Requested-With, 77 // Content-Type, Accept, Token"); 78 // System.out.println("Token=" + req.getHeader("Token")); 79 80 if ("OPTIONS".equals(req.getMethod())) {// axios的ajax會發兩次請求,第一次提交方式爲:option,直接返回便可 81 return; 82 } 83 filterChain.doFilter(servletRequest, servletResponse); 84 } 85 86 @Override 87 public void destroy() { 88 89 } 90 }
axios 的 get/post 的區別( qs )
GET提交
1 let url = this.axios.urls.SYSTEM_USER_DOLOGIN; 2 3 this.axios.get(url, { //注意數據是保存到json對象的params屬性 4 params: this.ruleForm 5 }).then(function(response) { 6 console.log(response); 7 }).catch(function(error) { 8 console.log(error); 9 });
POST提交
1 let url = this.axios.urls.SYSTEM_USER_DOLOGIN; 2 3 this.axios.post(url, this.ruleForm).then(function(response) { 4 console.log(response); 5 }).catch(function(error) { 6 console.log(error); 7 });
axios.get提交沒有問題,axios.post提交後臺接收不到數據
由於POST提交的參數的格式是Request Payload,這樣後臺取不到數據的
解決方案:使用qs.js庫
1 let url = 'http://localhost:8080/T216_SSH/vue/userAction_login.action'; 2 3 this.axios.post(url, qs.stringify(this.ruleForm)).then(function(response) { 4 console.log(response); 5 }).catch(function(error) { 6 console.log(error); 7 });
qs.js它是一個url參數轉化的js庫。
用法就兩個:
var obj = qs.parse('a=b&c=d'); //將URL解析成對象的形式:{a:'b',c:'d'}
var str = qs.stringify(obj); //將對象 序列化成URL的形式,以&進行拼接:a=b&c=d'
T his 指針帶來的變量污染
解決方案
this.axios.post(url, this.ruleForm).then((response) =>
1 methods:{ 2 doSub(){ 3 let url = this.axios.urls.SYSTEM_USER_DOLOGIN; 4 this.axios.post(url, this.ruleForm).then((response) => { 5 console.log(response); 6 if (response.data.code == 1) { 7 this.$message({ 8 showClose: true, 9 message: response.data.msg, 10 type: 'success' 11 }); 12 }else{ 13 this.$message({ 14 showClose: true, 15 message: response.data.msg, 16 type: 'error' 17 }); 18 } 19 }).catch(function(error) { 20 console.log(error); 21 }); 22 },
爲簡化axios使用,還可使用axios全局配置及攔截器。
http.js
1 /** 2 * vue項目對axios的全局配置 3 */ 4 import axios from 'axios' 5 import qs from 'qs' 6 7 //引入action模塊,並添加至axios的類屬性urls上 8 import action from '@/api/action' 9 axios.urls = action 10 11 // axios默認配置 12 axios.defaults.timeout = 10000; // 超時時間 13 // axios.defaults.baseURL = 'http://localhost:8080/j2ee15'; // 默認地址 14 axios.defaults.baseURL = action.SERVER; 15 16 //整理數據 17 // 只適用於 POST,PUT,PATCH,transformRequest` 容許在向服務器發送前,修改請求數據 18 axios.defaults.transformRequest = function(data) { 19 data = qs.stringify(data); 20 return data; 21 }; 22 23 24 // 請求攔截器 25 axios.interceptors.request.use(function(config) { 26 // var jwt = window.vm.$store.getters.getJwt; 27 // config.headers['jwt'] = jwt; 28 return config; 29 }, function(error) { 30 return Promise.reject(error); 31 }); 32 33 // 響應攔截器 34 axios.interceptors.response.use(function(response) { 35 // debugger; 36 // var jwt = response.headers['jwt']; 37 // if(jwt){ 38 // window.vm.$store.commit('setJwt',{jwt:jwt}); 39 // } 40 return response; 41 }, function(error) { 42 return Promise.reject(error); 43 }); 44 45 // // 路由請求攔截 46 // // http request 攔截器 47 // axios.interceptors.request.use( 48 // config => { 49 // //config.data = JSON.stringify(config.data); 50 // //config.headers['Content-Type'] = 'application/json;charset=UTF-8'; 51 // //config.headers['Token'] = 'abcxyz'; 52 // //判斷是否存在ticket,若是存在的話,則每一個http header都加上ticket 53 // // if (cookie.get("token")) { 54 // // //用戶每次操做,都將cookie設置成2小時 55 // // cookie.set("token", cookie.get("token"), 1 / 12) 56 // // cookie.set("name", cookie.get("name"), 1 / 12) 57 // // config.headers.token = cookie.get("token"); 58 // // config.headers.name = cookie.get("name"); 59 // // } 60 // return config; 61 // }, 62 // error => { 63 // return Promise.reject(error.response); 64 // }); 65 66 // // 路由響應攔截 67 // // http response 攔截器 68 // axios.interceptors.response.use( 69 // response => { 70 // if (response.data.resultCode == "404") { 71 // console.log("response.data.resultCode是404") 72 // // 返回 錯誤代碼-1 清除ticket信息並跳轉到登陸頁面 73 // // cookie.del("ticket") 74 // // window.location.href='http://login.com' 75 // return 76 // } else { 77 // return response; 78 // } 79 // }, 80 // error => { 81 // return Promise.reject(error.response) // 返回接口返回的錯誤信息 82 // }); 83 84 85 86 export default axios;
建議對請求地址進行封裝
/** * 對後臺請求的地址的封裝,URL格式以下: * 模塊名_實體名_操做 */ export default { 'SERVER': 'http://localhost:8080/T216_SSH', //服務器 'SYSTEM_USER_DOLOGIN': '/vue/userAction_login.action', //用戶登錄 'SYSTEM_USER_DOREG': '/vue/userAction_reg.action', //用戶註冊 'SYSTEM_MENU_TREE': '/vue/treeNodeAction.action', //左側樹形菜單加載 'SYSTEM_ARTICLE_LIST': '/vue/articleAction_list.action', //文章列表 'SYSTEM_ARTICLE_ADD': '/vue/articleAction_add.action', //文章新增 'SYSTEM_ARTICLE_EDIT': '/vue/articleAction_edit.action', //文章修改 'SYSTEM_ARTICLE_DEL': '/vue/articleAction_del.action', //文章刪除 'SYSTEM_USER_GETASYNCDATA': '/vue/userAction_getAsyncData.action', //vuex中的異步加載數據 'getFullPath': k => { //得到請求的完整地址,用於mockjs測試時使用 return this.SERVER + this[k]; } }
Login.vue
1 <template> 2 <div class=".login-wrap"> 3 <el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container"> 4 <h3 style="text-align: center;">用戶登陸</h3> 5 <el-form-item label="用戶名" prop="uname"> 6 <el-input type="uname" v-model="ruleForm.uname" autocomplete="off"></el-input> 7 </el-form-item> 8 <el-form-item label="密碼" prop="pwd"> 9 <el-input type="pwd" v-model="ruleForm.pwd" autocomplete="off"></el-input> 10 </el-form-item> 11 <el-form-item> 12 <el-row> 13 <el-col :span="24"> 14 <div class="grid-content bg-purple-dark"></div> 15 </el-col> 16 <el-button style="width: 100%;" type="primary" @click='doSub'>提交</el-button> 17 </el-row> 18 <el-row> 19 <el-col :span="12"> 20 <div class="grid-content bg-purple-dark"></div> 21 <el-link type="success" @click='toReg'>用戶註冊</el-link> 22 </el-col> 23 <el-col :span="12"> 24 <div class="grid-content bg-purple-dark"></div> 25 <el-link type="success">找回密碼</el-link> 26 </el-col> 27 </el-row> 28 29 </el-form-item> 30 </el-form> 31 </div> 32 </template> 33 <script> 34 export default { 35 data() { 36 return { 37 ruleForm: { 38 uname: '', 39 pwd: '' 40 } 41 }; 42 }, 43 methods:{ 44 doSub(){ 45 let url = this.axios.urls.SYSTEM_USER_DOLOGIN; 46 this.axios.post(url, this.ruleForm).then((response) => { 47 console.log(response); 48 if (response.data.code == 1) { 49 this.$message({ 50 showClose: true, 51 message: response.data.msg, 52 type: 'success' 53 }); 54 }else{ 55 this.$message({ 56 showClose: true, 57 message: response.data.msg, 58 type: 'error' 59 }); 60 } 61 }).catch(function(error) { 62 console.log(error); 63 }); 64 }, 65 toReg(){ 66 this.$router.push({ 67 path:'/Reg' 68 }) 69 } 70 } 71 } 72 </script> 73 <style scoped> 74 .login-wrap { 75 box-sizing: border-box; 76 width: 100%; 77 height: 100%; 78 padding-top: 10%; 79 background-image: url(); 80 /* background-color: #112346; */ 81 background-repeat: no-repeat; 82 background-position: center right; 83 background-size: 100%; 84 } 85 .login-container { 86 border-radius: 10px; 87 margin: 0px auto; 88 width: 350px; 89 padding: 30px 35px 15px 35px; 90 background: #fff; 91 border: 1px solid #eaeaea; 92 text-align: left; 93 box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1); 94 } 95 .title { 96 margin: 0px auto 40px auto; 97 text-align: center; 98 color: #505458; 99 } 100 </style>