• SPA项目开发之登录注册


    目录

    一、前置工作

    准备步骤:

    1、确保后台项目T216_SSH能够运行起来,标志: http://localhost:8080/T216_SSH/vue/treeNodeAction.action 能够返回数据

    2、要将api的文件夹导入src的目录

    3、下载js依赖

            axios:前端向后台发送请求

            qs:解决发送post请求代码冗余的问题

            ElementUI:快速布局

            vue-axios:将axios依赖整合进vue中

    二、用户登录界面排版

    三、Vue之数据交互登录功能

    this.$router.push(.....);

    作用:试用获取后台数据,做局部刷新

    语法:

    this.axios.post(url,this.param)

    .then(function(resp){ })

    .catch(function(error){ })

    四、this指针污染

    五、CORS跨域

     跨域产生的因素:协议、ip、端口、项目有任意一处发生改变,都会产生跨域的现象

    六、工具类的作用&get和post的区别

    引入main.js配置

    解决

    axios、qs的作用

    api/http.js:

    get/post


    一、前置工作

    准备步骤:

    1、确保后台项目T216_SSH能够运行起来,标志: http://localhost:8080/T216_SSH/vue/treeNodeAction.action 能够返回数据

    2、要将api的文件夹导入src的目录

    3、下载js依赖

            axios:前端向后台发送请求

            qs:解决发送post请求代码冗余的问题

            ElementUI:快速布局

            vue-axios:将axios依赖整合进vue中

    eclipse导入项目:

    选择maven工程:

                 

     选择项目所在路径:

     

    导入成功后再运行:

    要注意Tomact的配置问题!) 

    之后测试能不能访问到数据,如下: 

     

     返回了数据:

    紧接着去黑窗口下载四个js依赖:

    你们可以在项目所在的目录下去执行: 

     

    由于本人下载了WebStorm

    因此我直接在那里面输入:

     下载成功后注意package.json有没有生成 element-ui:

     成功后继续下载:

     

     将以下四个依赖下载完之后,

     npm install element-ui -S 

    npm install axios -S

    npm install qs -S  

    npm install vue-axios -S

    对应的 package.json 里面也生成了四个依赖:

    前置工作已都准备完毕!

    二、用户登录界面排版

     复制到App.vue中:

    1. <script>
    2. export default {
    3. name: 'App'
    4. }
    5. script>
    6. <style>
    7. #app {
    8. font-family: 'Avenir', Helvetica, Arial, sans-serif;
    9. -webkit-font-smoothing: antialiased;
    10. -moz-osx-font-smoothing: grayscale;
    11. text-align: center;
    12. color: #2c3e50;
    13. margin-top: 60px;
    14. }
    15. style>

    npm run dev 运行:

     目前页面效果这样:

    接下来写我们的登录界面:

     复制到新建的views文件夹内的Login.vue中:

    然后其中进行调整代码即样式:

    Login.vue:

    1. <template>
    2. <div class="login-wrap">
    3. <el-row>
    4. <el-col :span="24">
    5. <div style="text-align: center" class="grid-content bg-purple-dark"><h3>用户登录h3>
    6. div>el-col>
    7. el-row>
    8. <el-form :model="ruleForm" status-icon label-width="100px" class="demo-ruleForm login-container">
    9. <el-form-item label="用户名" prop="uname">
    10. <el-input type="text" v-model="ruleForm.uname" autocomplete="off">el-input>
    11. el-form-item>
    12. <el-form-item label="密码" prop="pwd">
    13. <el-input type="password" v-model="ruleForm.pwd" autocomplete="off">el-input>
    14. el-form-item>
    15. <el-form-item>
    16. <el-row>
    17. <el-col :span="24">
    18. <div style="text-align: center" class="grid-content bg-purple-dark">
    19. <el-button style="width: 100%" type="primary" @click="submitForm">提交el-button>
    20. div>
    21. el-col>
    22. el-row>
    23. el-form-item>
    24. <el-form-item>
    25. <el-row>
    26. <el-col :span="12">
    27. <div style="text-align: center" class="grid-content bg-purple-dark">
    28. <el-link type="success">用户注册el-link>
    29. div>
    30. el-col>
    31. <el-col :span="12">
    32. <div style="text-align: center" class="grid-content bg-purple-dark">
    33. <el-link type="warning">忘记密码el-link>
    34. div>
    35. el-col>
    36. el-row>
    37. el-form-item>
    38. el-form>
    39. div>
    40. template>
    41. <script>
    42. export default {
    43. name: 'Login',
    44. data () {
    45. return {
    46. ruleForm: {}
    47. }
    48. },
    49. methods:{
    50. submitForm(){
    51. },
    52. resetForm(){
    53. }
    54. }
    55. }
    56. script>
    57. <style scoped>
    58. .login-wrap {
    59. box-sizing: border-box;
    60. width: 100%;
    61. height: 100%;
    62. padding-top: 10%;
    63. background-image: url();
    64. /* background-color: #112346; */
    65. background-repeat: no-repeat;
    66. background-position: center right;
    67. background-size: 100%;
    68. }
    69. .login-container {
    70. border-radius: 10px;
    71. margin: 0px auto;
    72. width: 350px;
    73. padding: 30px 35px 15px 35px;
    74. background: #fff;
    75. border: 1px solid #eaeaea;
    76. text-align: left;
    77. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    78. }
    79. .title {
    80. margin: 0px auto 40px auto;
    81. text-align: center;
    82. color: #505458;
    83. }
    84. style>

    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. Vue.use(Router)
    6. export default new Router({
    7. routes: [
    8. {
    9. path: '/',
    10. name: 'Login',
    11. component: Login
    12. },
    13. {
    14. path: '/Login',
    15. name: 'Login',
    16. component: Login
    17. },
    18. ]
    19. })

    App.vue:

    1. <template>
    2. <div id="app">
    3. <router-view/>
    4. div>
    5. template>
    6. <script>
    7. export default {
    8. name: 'App'
    9. }
    10. script>
    11. <style>
    12. html,
    13. body {
    14. width: 100%;
    15. height: 100%;
    16. box-sizing: border-box;
    17. padding: 0px;
    18. margin: 0px;
    19. }
    20. #app {
    21. font-family: "Avenir", Helvetica, Arial, sans-serif;
    22. -webkit-font-smoothing: antialiased;
    23. -moz-osx-font-smoothing: grayscale;
    24. color: #2c3e50;
    25. widows: 100%;
    26. height: 100%;
    27. }
    28. style>

     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 ElementUI from 'element-ui' // 新添加 1
    5. import 'element-ui/lib/theme-chalk/index.css' // 新添加 2 ,避免后期打包样式
    6. import App from './App'
    7. import router from './router'
    8. Vue.use(ElementUI)//新添加 3
    9. Vue.config.productionTip = false
    10. /* eslint-disable no-new */
    11. new Vue({
    12. el: '#app',
    13. router,
    14. components: { App },
    15. template: ''
    16. })

    页面运行效果:

     用户登录界面排版完成~

    三、Vue之数据交互登录功能

    this.$router.push(.....);

    axios相当于jQuery中的ajax

    作用:试用获取后台数据,做局部刷新

    语法:

    this.axios.post(url,this.param)

    .then(function(resp){ })

    .catch(function(error){ })

     首先再添加一个用户注册,实现登录时假若没有注册就可以跳转注册页面:

    Reg.vue:

    1. <template>
    2. <div class="login-wrap">
    3. <el-row>
    4. <el-col :span="24">
    5. <div style="text-align: center" class="grid-content bg-purple-dark"><h3>用户注册h3>
    6. div>el-col>
    7. el-row>
    8. <el-form :model="ruleForm" status-icon label-width="100px" class="demo-ruleForm login-container">
    9. <el-form-item label="用户名" prop="uname">
    10. <el-input type="text" v-model="ruleForm.uname" autocomplete="off">el-input>
    11. el-form-item>
    12. <el-form-item label="密码" prop="pwd">
    13. <el-input type="password" v-model="ruleForm.pwd" autocomplete="off">el-input>
    14. el-form-item>
    15. <el-form-item>
    16. <el-row>
    17. <el-col :span="24">
    18. <div style="text-align: center" class="grid-content bg-purple-dark">
    19. <el-button style="width: 100%" type="primary" @click="submitForm">提交el-button>
    20. div>
    21. el-col>
    22. el-row>
    23. el-form-item>
    24. <el-form-item>
    25. <el-row>
    26. <el-col :span="12">
    27. <div style="text-align: center" class="grid-content bg-purple-dark">
    28. <el-link type="success" @click="toLogin">用户登录el-link>
    29. div>
    30. el-col>
    31. <el-col :span="12">
    32. <div style="text-align: center" class="grid-content bg-purple-dark">
    33. <el-link type="warning">忘记密码el-link>
    34. div>
    35. el-col>
    36. el-row>
    37. el-form-item>
    38. el-form>
    39. div>
    40. template>
    41. <script>
    42. export default {
    43. name: 'Reg',
    44. data () {
    45. return {
    46. ruleForm: {}
    47. }
    48. },
    49. methods:{
    50. submitForm(){
    51. },
    52. resetForm(){
    53. },
    54. toLogin(){
    55. this.$router.push({path:'/Login'})
    56. }
    57. }
    58. }
    59. script>
    60. <style scoped>
    61. .login-wrap {
    62. box-sizing: border-box;
    63. width: 100%;
    64. height: 100%;
    65. padding-top: 10%;
    66. background-image: url();
    67. /* background-color: #112346; */
    68. background-repeat: no-repeat;
    69. background-position: center right;
    70. background-size: 100%;
    71. }
    72. .login-container {
    73. border-radius: 10px;
    74. margin: 0px auto;
    75. width: 350px;
    76. padding: 30px 35px 15px 35px;
    77. background: #fff;
    78. border: 1px solid #eaeaea;
    79. text-align: left;
    80. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    81. }
    82. .title {
    83. margin: 0px auto 40px auto;
    84. text-align: center;
    85. color: #505458;
    86. }
    87. style>

    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. Vue.use(Router)
    7. export default new Router({
    8. routes: [
    9. {
    10. path: '/',
    11. name: 'Login',
    12. component: Login
    13. },
    14. {
    15. path: '/Login',
    16. name: 'Login',
    17. component: Login
    18. },
    19. {
    20. path: '/Reg',
    21. name: 'Reg',
    22. component: Reg
    23. }
    24. ]
    25. })

    效果: 

    进行数据交互登录功能的实现:

    1. <template>
    2. <div class="login-wrap">
    3. <el-row>
    4. <el-col :span="24">
    5. <div style="text-align: center" class="grid-content bg-purple-dark"><h3>用户登录h3>
    6. div>el-col>
    7. el-row>
    8. <el-form :model="ruleForm" status-icon label-width="100px" class="demo-ruleForm login-container">
    9. <el-form-item label="用户名" prop="uname">
    10. <el-input type="text" v-model="ruleForm.uname" autocomplete="off">el-input>
    11. el-form-item>
    12. <el-form-item label="密码" prop="pwd">
    13. <el-input type="password" v-model="ruleForm.pwd" autocomplete="off">el-input>
    14. el-form-item>
    15. <el-form-item>
    16. <el-row>
    17. <el-col :span="24">
    18. <div style="text-align: center" class="grid-content bg-purple-dark">
    19. <el-button style="width: 100%" type="primary" @click="submitForm">提交el-button>
    20. div>
    21. el-col>
    22. el-row>
    23. el-form-item>
    24. <el-form-item>
    25. <el-row>
    26. <el-col :span="12">
    27. <div style="text-align: center" class="grid-content bg-purple-dark">
    28. <el-link type="success" @click="toReg">用户注册el-link>
    29. div>
    30. el-col>
    31. <el-col :span="12">
    32. <div style="text-align: center" class="grid-content bg-purple-dark">
    33. <el-link type="warning">忘记密码el-link>
    34. div>
    35. el-col>
    36. el-row>
    37. el-form-item>
    38. el-form>
    39. div>
    40. template>
    41. <script>
    42. export default {
    43. name: 'Login',
    44. data () {
    45. return {
    46. ruleForm: {}
    47. }
    48. },
    49. methods:{
    50. submitForm(){
    51. //this.axios找到的是 /api/http.js文件
    52. //xios.urls——>this.axios/urls找到了 /api/action.js
    53. //action.js是一个JSON对象,那么就可以取到请求的值
    54. let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
    55. // alert(url);
    56. // {uname:'zs',pwd:'123'}
    57. this.axios.post(url,this.ruleForm)
    58. .then(function (resp){//代表成功
    59. console.log(resp);
    60. }).catch(function (){//代表失败
    61. });
    62. },
    63. resetForm(){
    64. },
    65. toReg(){
    66. this.$router.push({path:'/Reg'})
    67. }
    68. }
    69. }
    70. script>
    71. <style scoped>
    72. .login-wrap {
    73. box-sizing: border-box;
    74. width: 100%;
    75. height: 100%;
    76. padding-top: 10%;
    77. background-image: url();
    78. /* background-color: #112346; */
    79. background-repeat: no-repeat;
    80. background-position: center right;
    81. background-size: 100%;
    82. }
    83. .login-container {
    84. border-radius: 10px;
    85. margin: 0px auto;
    86. width: 350px;
    87. padding: 30px 35px 15px 35px;
    88. background: #fff;
    89. border: 1px solid #eaeaea;
    90. text-align: left;
    91. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    92. }
    93. .title {
    94. margin: 0px auto 40px auto;
    95. text-align: center;
    96. color: #505458;
    97. }
    98. style>

    直接点击提交:

     输错密码:

     输入正确:

     到这里呢,那么我们的Vue的数据交互登录功能就已实现啦!

    四、this指针污染

    当我们登录时 密码错误/密码正确/账户密码为空 时,要出现相应的消息提示框,所以我们可去element官网上进行查找我们需要的提示框源代码:

     我们就以成功和错误的消息提示来做案例:

     进行代码调整:

    Login.vue:

     未填任何数据,直接提交:

     填写错误用户或者密码:

     填写正确用户或者密码:

    五、CORS跨域

    在T216_SSH项目中的web.xml有个解决cors跨域问题的过滤器:

     再去运行项目,直接点提交:

    跨域的现象:

    请求地址由协议+ip+端口+项目+....等构成

     跨域产生的因素:协议、ip、端口、项目有任意一处发生改变,都会产生跨域的现象

    解决方案: 

    1. package com.javaxl.util;
    2. import java.io.IOException;
    3. import javax.servlet.Filter;
    4. import javax.servlet.FilterChain;
    5. import javax.servlet.FilterConfig;
    6. import javax.servlet.ServletException;
    7. import javax.servlet.ServletRequest;
    8. import javax.servlet.ServletResponse;
    9. import javax.servlet.http.HttpServletResponse;
    10. /**
    11. * 配置tomcat允许跨域访问
    12. *
    13. * @author Administrator
    14. *
    15. */
    16. public class CorsFilter implements Filter {
    17. @Override
    18. public void init(FilterConfig filterConfig) throws ServletException {
    19. }
    20. @Override
    21. public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
    22. throws IOException, ServletException {
    23. HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;
    24. // Access-Control-Allow-Origin就是我们需要设置的域名
    25. // Access-Control-Allow-Headers跨域允许包含的头。
    26. // Access-Control-Allow-Methods是允许的请求方式
    27. httpResponse.addHeader("Access-Control-Allow-Origin", "*");// *,任何域名
    28. httpResponse.setHeader("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
    29. httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
    30. filterChain.doFilter(servletRequest, servletResponse);
    31. }
    32. @Override
    33. public void destroy() {
    34. }
    35. }

    六、工具类的作用&get和post的区别

    引入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' // 新添加 2
    5. import App from './App'
    6. import router from './router'
    7. import ElementUI from 'element-ui' // 新添加 1
    8. // import axios from '@/api/http'
    9. import axios from 'axios'
    10. import VueAxios from 'vue-axios'
    11. Vue.use(ElementUI)
    12. Vue.use(VueAxios,axios)
    13. Vue.config.productionTip = false
    14. /* eslint-disable no-new */
    15. new Vue({
    16. el: '#app',
    17. router,
    18. components: { App },
    19. template: ''
    20. })
    1. let url = 'http://localhost:8080/T216_SSH/vue/userAction_login.action';
    2. this.axios.post(url, this.ruleForm).then(function(response) {
    3. console.log(response);
    4. }).catch(function(error) {
    5. console.log(error);
    6. });

     

    解决

    import qs from 'qs'
    1. let url = 'http://localhost:8080/T216_SSH/vue/userAction_login.action';
    2. this.axios.post(url, qs.stringify(this.ruleForm)).then(function(response) {
    3. console.log(response);
    4. }).catch(function(error) {
    5. console.log(error);
    6. });

    axios、qs的作用

    api/http.js:

            1、将原生的json对象转换成字符串,通过拦截器转

            2、将本项目中的所有接口地址,进行统一管理

    get/post

            get请求:

    1. let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
    2. this.axios.get(url, { //注意数据是保存到json对象的params属性
    3. params: this.ruleForm
    4. }).then(function(response) {
    5. console.log(response);
    6. }).catch(function(error) {
    7. console.log(error);
    8. });

            post请求:

    1. let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
    2. this.axios.post(url, this.ruleForm).then(function(response) {
    3. console.log(response);
    4. }).catch(function(error) {
    5. console.log(error);
    6. });

    今日分享就结束啦!如果觉得对您有帮助的话可以点点赞好嘛~

  • 相关阅读:
    Python学习之CSDN21天学习挑战赛计划之16
    一个计算密集小程序在不同CPU下的表现
    leetcode-11. 盛最多水的容器(双指针)
    maltab datenum函数与正则表达式巧用:逐日数据转为逐月数据、日序转月序
    初级篇—第八章精讲MySQL数据类型
    前端面试问题汇总 - 工程管理工具篇
    2022年全国最新消防设施操作员(初级消防设施操作员)模拟题及答案
    C#取两个集合的交集、并集和差集
    jmeter单接口和多接口测试
    大聪明教你学Java | 面试管:谈谈如何解决 RabbitMQ 消息丢失与消息积压
  • 原文地址:https://blog.csdn.net/weixin_65808248/article/details/126762413