SPA項目開發之登陸註冊

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

引入main.js配置

在項目中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>

 

 

相關文章
相關標籤/搜索