SPA項目開發之登陸註冊

1. vue怎麼引入和配置使用element-ui框架
 1.1 使用vue-cli腳手架工具建立一個vue項目
  vue init webpack pro01css

  安裝
  npm install axios -S
  npm install qs -S
  npm install vue-axios -S

 1.2修改main.js文件
  #import axios from 'axios'
  #import qs from 'qs'
  import axios from '@/api/http' #vue項目對axios的全局配置
  import VueAxios from 'vue-axios'
   Vue.use(VueAxios,axios)vue

 

  1.3 在項目中src目錄下找到main.js,並在指定位置添加三行代碼(main.js是入口文件,因此在這裏引入就行,頁面就不用引入了)java

import Vue from 'vue'
import 'element-ui/lib/theme-chalk/index.css'
import App from './App'
import router from './router'
import ElementUI from 'element-ui'
import axios from '@/api/http'              
import VueAxios from 'vue-axios' 
 

1.4 測試
修改HelloWorld.vue添加elementUI組件查看效果webpack

 

2. Vue+ElementUI設計登錄頁面

注1:相關樣式見資料「css.txt」ios

 

注2:<style scoped>
在vue組件中,在style標籤上添加scoped屬性,以表示它的樣式做用於當下的模塊,很好的實現了樣式私有化的目的web

 

注2:auto-complete="off"
autocomplete 屬性是 HTML5 中的新屬性,off-----禁用自動完成ajax

3. 後臺交互(axios/qs/vue-axios)vue-router

3.1 axios
axios是vue2提倡使用的輕量版的ajax。它是基於promise的HTTP庫。它會從瀏覽器中建立XMLHttpRequests,與Vue配合使用很是好。
1.題外話:
vue.js有著名的全家桶系列:vue-router,vuex, vue-resource,再加上構建工具vue-cli,就是一個完整的vue項目的核心構成。
其中vue-resource是Vue.js的一款插件,它能夠經過XMLHttpRequest或JSONP發起請求並處理響應,但在vue更新到2.0以後,
做者就宣告再也不對vue-resource更新,而是推薦的axiosvuex

2.GET提交vue-cli

axios.get('/user', {//注意數據是保存到json對象的params屬性
params: {
ID: 12345
}
}).then(function (response) {
console.log(response);
}).catch(function (error) {
console.log(error);
});

3.POST提交

axios.post('/user', {//注意數據是直接保存到json對象
firstName: 'Fred',
lastName: 'Flintstone'
}).then(function (response) {
console.log(response);
}).catch(function (error) {
console.log(error);
});

注1:axios跨域問題
會一直報錯:「http://127.0.0.1:8848' has been blocked by CORS policy: No 'Access-Control-Allow-Origin' header」
由於使用了先後端分離開發,跨域訪問了,解決方案:須要配置tomcat容許跨域訪問,
tomcat跨域配置方法不少,但最簡單的方式是本身寫一個過濾器CorsFilter實現,添加一個響應頭
Access-Control-Allow-Origin便可
httpResponse.addHeader("Access-Control-Allow-Origin", "*");//*表示任何域名
httpResponse.addHeader("Access-Control-Allow-Origin", "http://localhost:80");

Access-Control-Allow-Origin:* #則容許全部域名的腳本訪問該資源。
Access-Control-Allow-Origin:https://www.fujieace.com #容許特定的域名訪問

注2:axios.get提交沒有問題,axios.post提交後臺接收不到數據
由於POST提交的參數的格式是Request Payload,這樣後臺取不到數據的(詳情見資料「05 Vue中axios踩坑之路-POST傳參 - RainSun - CSDN博客.mht」),
解決方案:使用qs.js庫,將{a:'b',c:'d'}轉換成'a=b&c=d'

注3:爲簡化axios使用,還可使用axios全局配置及攔截器,詳情見資料「api/http.js」
axios.defaults.baseURL = 'https://api.example.com';
//axios.defaults.headers.common['Authorization'] = AUTH_TOKEN;//自定義請求頭,添加認證令牌
axios.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded';

注4:爲方便開發,建議對請求地址進行封裝,詳情見資料「api/action.js」

注5:^_^~~~~爲進一步簡化開發,將action.js的URL地址封裝到axios.urls屬性上

 

3.2 qs
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'

3.3 vue-axios
vue-axios是在axios基礎上擴展的插件,在Vue.prototype原型上擴展了$http等屬性,能夠更加方便的使用axios

JS代碼:

action.js

/**
 * 對後臺請求的地址的封裝,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];
    }
}

http.js

/**
 * vue項目對axios的全局配置
 */
import axios from 'axios'
import qs from 'qs'

//引入action模塊,並添加至axios的類屬性urls上
import action from '@/api/action'
axios.urls = action

// axios默認配置
axios.defaults.timeout = 10000; // 超時時間
// axios.defaults.baseURL = 'http://localhost:8080/j2ee15'; // 默認地址
axios.defaults.baseURL = action.SERVER;


//整理數據
// 只適用於 POST,PUT,PATCH,transformRequest` 容許在向服務器發送前,修改請求數據
axios.defaults.transformRequest = function(data) {
    data = qs.stringify(data);
    return data;
};


// 請求攔截器
axios.interceptors.request.use(function(config) {
    // var jwt = window.vm.$store.getters.getJwt;
    // config.headers['jwt'] = jwt;
    return config;
}, function(error) {
    return Promise.reject(error);
});

// 響應攔截器
axios.interceptors.response.use(function(response) {
    // debugger;
    // var jwt = response.headers['jwt'];
    // if(jwt){
    //     window.vm.$store.commit('setJwt',{jwt:jwt});
    // }
    return response;
}, function(error) {
    return Promise.reject(error);
});

// // 路由請求攔截
// // http request 攔截器
// axios.interceptors.request.use(
//     config => {
//         //config.data = JSON.stringify(config.data);  
//         //config.headers['Content-Type'] = 'application/json;charset=UTF-8';
//         //config.headers['Token'] = 'abcxyz';
//         //判斷是否存在ticket,若是存在的話,則每一個http header都加上ticket
//         // if (cookie.get("token")) {
//         //     //用戶每次操做,都將cookie設置成2小時
//         //     cookie.set("token", cookie.get("token"), 1 / 12)
//         //     cookie.set("name", cookie.get("name"), 1 / 12)
//         //     config.headers.token = cookie.get("token");
//         //     config.headers.name = cookie.get("name");
//         // }
//         return config;
//     },
//     error => {
//         return Promise.reject(error.response);
//     });

// // 路由響應攔截
// // http response 攔截器
// axios.interceptors.response.use(
//     response => {
//         if (response.data.resultCode == "404") {
//             console.log("response.data.resultCode是404")
//             // 返回 錯誤代碼-1 清除ticket信息並跳轉到登陸頁面
//             //      cookie.del("ticket")
//             //      window.location.href='http://login.com'
//             return
//         } else {
//             return response;
//         }
//     },
//     error => {
//         return Promise.reject(error.response) // 返回接口返回的錯誤信息
//     });



export default axios;

index.js

import Vue from 'vue'
import Router from 'vue-router'
import HelloWorld from '@/components/HelloWorld'
import Login from '@/views/Login'
import Reg from '@/views/Reg'

Vue.use(Router)

export default new Router({
      routes: [{
          path: '/',
          name: 'Login',
          component: Login
          },
          {
               path: '/Login',
          name: 'Login',
          component: Login
           },
             {
               path: '/Reg',
          name: 'Reg',
          component: Reg
           }
  ]
})

App.vue

<template>
  <div id="app">
    <!--<img src="./assets/logo.png">-->
    <router-view/>
  </div>
</template>

<script>
export default {
  name: 'App'
}
</script>

<style> 
body {
    width: 100%;
    height: 100%;
    box-sizing: border-box;
    padding: 0px;
    margin: 0px;
}
#app {
    font-family: "Avenir", Helvetica, Arial, sans-serif;
    -webkit-font-smoothing: antialiased;
    -moz-osx-font-smoothing: grayscale;
    color: #2c3e50;
    widows: 100%;
    height: 100%;
}
</style>

Login

<template>
    <div class="login-wrap">
        <el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container">
         <h3 style="text-align: center;">用戶登陸</h3>
             <el-form-item label="用戶名" prop="uname">
                <el-input type="uname" v-model="ruleForm.uname" autocomplete="off"></el-input>
            </el-form-item>
            <el-form-item label="密碼" prop="pwd">
                <el-input type="pwd" v-model="ruleForm.pwd" autocomplete="off"></el-input>
            </el-form-item>
               <el-form-item>
                   <el-row>
                     <el-col :span="24">
                         <div class="grid-content bg-purple-dark">
                             <el-button style="width: 100%;" type="primary" @click="doSub">提交</el-button>
                         </div>
                     </el-col>
                </el-row>
                <el-row>
                    <el-col :span="12">
                         <div class="grid-content bg-purple-dark">
                             <el-link type="success" @click="toReg">用戶註冊</el-link>
                         </div>
                     </el-col>
                     <el-col :span="12">
                         <div class="grid-content bg-purple-dark">
                             <el-link type="success">忘記密碼</el-link>
                         </div>
                     </el-col>
                </el-row>
              </el-form-item>
        </el-form>
    </div>
</template>

<script>
    export default {
        data(){
            return{
                ruleForm:{
                    uname:'',
                    pwd:''
                }
            };
        },
        methods:{
            doSub(){
                let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
                this.axios.post(url, this.ruleForm).then((response)=>{
                    console.log(response);
                    if(response.data.code == 1){
                        this.$message({
                               showClose:true,
                             message: response.data.msg,
                             type: 'success'
                           });
                    }else{
                        this.$message({
                            showClose: true,
                            message: response.data.msg,
                            type: 'error'
                        });
                    }
                }).catch(function(error) {
                    console.log(error);
                });
            },
            toReg(){
                this.$router.push({
                    path:'/Reg'
                });
            }
        }
    }    
</script>

<style scoped>
    .login-wrap {
        box-sizing: border-box;
        width: 100%;
        height: 100%;
        padding-top: 10%;
        background-image: url();
        /* background-color: #112346; */
        background-repeat: no-repeat;
        background-position: center right;
        background-size: 100%;
    }

    .login-container {
        border-radius: 10px;
        margin: 0px auto;
        width: 350px;
        padding: 30px 35px 15px 35px;
        background: #fff;
        border: 1px solid #eaeaea;
        text-align: left;
        box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    }

    .title {
        margin: 0px auto 40px auto;
        text-align: center;
        color: #505458;
    }
</style>

處理CORS跨域的監聽器

CorsFilter.java
package com.zking.vue.util;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * 配置tomcat容許跨域訪問
 * 
 * @author Administrator
 *
 */
public class CorsFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }

    // @Override
    // public void doFilter(ServletRequest servletRequest, ServletResponse
    // servletResponse, FilterChain filterChain)
    // throws IOException, ServletException {
    // HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;
    //
    // // Access-Control-Allow-Origin就是咱們須要設置的域名
    // // Access-Control-Allow-Headers跨域容許包含的頭。
    // // Access-Control-Allow-Methods是容許的請求方式
    // httpResponse.addHeader("Access-Control-Allow-Origin", "*");// *,任何域名
    // httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT,
    // DELETE");
    // // httpResponse.setHeader("Access-Control-Allow-Headers", "Origin,
    // // X-Requested-With, Content-Type, Accept");
    //
    // // 容許請求頭Token
    // httpResponse.setHeader("Access-Control-Allow-Headers",
    // "Origin,X-Requested-With, Content-Type, Accept, Token");
    // HttpServletRequest req = (HttpServletRequest) servletRequest;
    // System.out.println("Token=" + req.getHeader("Token"));
    // if("OPTIONS".equals(req.getMethod())) {
    // return;
    // }
    //
    //
    // filterChain.doFilter(servletRequest, servletResponse);
    // }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
            throws IOException, ServletException {
        HttpServletResponse resp = (HttpServletResponse) servletResponse;
        HttpServletRequest req = (HttpServletRequest) servletRequest;

        // Access-Control-Allow-Origin就是咱們須要設置的域名
        // Access-Control-Allow-Headers跨域容許包含的頭。
        // Access-Control-Allow-Methods是容許的請求方式
        resp.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
        resp.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
        // resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With,
        // Content-Type, Accept");
        // 容許客戶端,發一個新的請求頭jwt
        resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With, Content-Type, Accept, jwt");

        // 容許客戶端,處理一個新的響應頭jwt
        resp.setHeader("Access-Control-Expose-Headers", "jwt");
        // String sss = resp.getHeader("Access-Control-Expose-Headers");
        // System.out.println("sss=" + sss);

        // 容許請求頭Token
        // httpResponse.setHeader("Access-Control-Allow-Headers","Origin,X-Requested-With,
        // Content-Type, Accept, Token");
        // System.out.println("Token=" + req.getHeader("Token"));

        if ("OPTIONS".equals(req.getMethod())) {// axios的ajax會發兩次請求,第一次提交方式爲:option,直接返回便可
            return;
        }
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {

    }
}

 

相關文章
相關標籤/搜索