• SPA项目开发之登录注册


    CMD安装所需要的pom依赖

    npm install element-ui -S  

    npm install axios -S

    npm install qs -S  

    npm install vue-axios -S

    package.json中查看安装的依赖

    引入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'

    Vue.use(ElementUI) //新添加3
    Vue.config.productionTip = false

    main.js

     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组件查看效果

    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      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      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      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=" 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=" 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      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      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>

     

  • 相关阅读:
    Shortest path of the king
    二分查找c++简单模板
    2017广东工业大学程序设竞赛B题占点游戏
    2017广东工业大学程序设竞赛C题爬楼梯
    2017广东工业大学程序设竞赛E题(倒水)
    p1250 种树 贪心
    P1248 加工生产调度 贪心
    P1209 [USACO1.3]修理牛棚 Barn Repair 贪心
    P1607 [USACO09FEB]庙会班车Fair Shuttle 贪心
    P2602 [ZJOI2010]数字计数 数位dp
  • 原文地址:https://www.cnblogs.com/xcn123/p/11426922.html
Copyright © 2020-2023  润新知