SPA项目开发之登录注册

1 创建一个新项目

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
修改端口号:修改为8088
在这里插入图片描述
重新启动项目
在这里插入图片描述
在这里插入图片描述

2 spa项目完成登录布局

前端登录表单布局

<template>
  <div class="login-wrap">
    <el-form :model="ruleForm" ref="ruleForm" label-width="100px" class="demo-ruleForm login-container">
      <h2 style="text-align: center;">用户登陆</h2>
      <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 type="primary" style="width: 100%;" @click="dolog">登陆</el-button>
          </div>
          </el-col>
        </el-row>
        <el-row>
          <el-col :span="12"><div class="grid-content bg-purple">
           <el-link type="success" @click="doreg">用户注册</el-link>
          </div>
          </el-col>
          <el-col :span="12"><div class="grid-content bg-purple-light">
           <el-link type="success">忘记密码</el-link>
          </div>
          </el-col>
        </el-row>
      </el-form-item>
    </el-form>
  </div>
</template>

登录样式布局

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

在这里插入图片描述
实现登录功能

<script>
  export default {
    data() {
      return {
          ruleForm:{
            uname:'',
            pwd:''
          }
        };
     },
     methods:{
       doreg(){
         this.$router.push({
           path:'/register'
         })
       },
       dolog(){
         let url=this.axios.urls.SYSTEM_USER_DOLOGIN;
         console.log(url);
         this.axios.post(url, this.ruleForm).then((response)=>{
         					console.log(response);
                  if(response.data.code==1){
                    this.$message({
                      message: response.data.msg,
                      type: 'success'
                    });
                  }else{
                   this.$message({
                       message: response.data.msg,
                       type: 'warning'
                     });
                  }

         				}).catch(function(error) {
         					console.log(error);
         				});
       }
     }
  }
</script>

在这里插入图片描述
后端
处理前端axios请求

package com.zking.vue.web;

import java.util.HashMap;
import java.util.Map;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.opensymphony.xwork2.ModelDriven;
import com.zking.base.web.BaseAction;
import com.zking.vue.biz.UserBiz;
import com.zking.vue.entity.User;
import com.zking.vue.util.JsonData;
import com.zking.vue.util.JwtUtils;
import com.zking.vue.util.PageBean;
import com.zking.vue.util.ResponseUtil;
import com.zking.vue.util.StringUtils;

public class UserAction extends BaseAction implements ModelDriven<User>{

	private UserBiz userBiz;
	private User user = new User();

	public UserBiz getUserBiz() {
		return userBiz;
	}

	public void setUserBiz(UserBiz userBiz) {
		this.userBiz = userBiz;
	}
	 
	public String register() {
		ObjectMapper om=new ObjectMapper();
		JsonData jsonData=null;
		try {
			if(StringUtils.isBlank(user.getUname()) || StringUtils.isBlank(user.getPwd())) {
				jsonData = new JsonData(0, "用户或者密码为空", user);
			}else {
				User u= this.userBiz.register(user);
				Map<String, Object> claims = new HashMap<String, Object>();
				claims.put("uname",user.getUname());
				claims.put("pwd", user.getPwd());
				String jwt = JwtUtils.createJwt(claims, JwtUtils.JWT_WEB_TTL);
				response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);
				jsonData = new JsonData(1, "注册成功", u);
			}
		} catch (Exception e) {
			e.printStackTrace();
			jsonData = new JsonData(0, "注册失败", user);
		}finally{
			try {
				ResponseUtil.write(response, om.writeValueAsString(jsonData));
			} catch (Exception e2) {
				e2.printStackTrace();
			}
		}
		return null;
	}
	public String login() {
		ObjectMapper om = new ObjectMapper();
		JsonData jsonData = null;
		try {
			if(StringUtils.isBlank(user.getUname()) || StringUtils.isBlank(user.getPwd())) {
				jsonData = new JsonData(0, "用户或者密码为空", user);
			}else {
				User u = this.userBiz.login(user);
				jsonData = new JsonData(1, "登录成功", u);
			}
		} catch (Exception e) {
			e.printStackTrace();
			jsonData = new JsonData(0, "用户或者密码错误", user);
		}finally {
			try {
				ResponseUtil.write(response, om.writeValueAsString(jsonData));
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		
		return null;
	}
	
	public String getAsyncData() {
		ObjectMapper om = new ObjectMapper();
		try {
			Thread.sleep(6000);
			ResponseUtil.write(response, om.writeValueAsString("http://www.javaxl.com"));
		} catch (Exception e) {
			e.printStackTrace();
		}
		return null;
	}

	@Override
	public User getModel() {
		return user;
	}
}

配置tomcat允许跨域访问
处理CORS跨域的监听器

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() {

	}
}

3 spa项目完成注册布局

注册表单布局

<template>
  <div class="reg-wrap">
    <el-form :model="ruleForm" status-icon :rules="rules"  ref="ruleForm" label-width="100px" class="demo-ruleForm reg-container">
      <h2 style="text-align: center;">用户注册</h2>
      <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 label="确认密码" prop="checkPass">
          <el-input type="password" v-model="ruleForm.checkPass" autocomplete="off"></el-input>
      </el-form-item>
      <el-form-item>
        <el-row>
          <el-col :span="24">
            <div class="grid-content bg-purple-dark" style="text-align: center;">
          <el-button type="primary" style="width: 100%;"  @click="doRegister">注册</el-button><br />
          <el-button type="primary" style="width: 100%;"  @click="resetForm('ruleForm')">重置</el-button>
          </div>
          </el-col>
        </el-row>
        <el-row>
          <el-col :span="12"><div class="grid-content bg-purple">
           <el-link type="success" @click="dologin">返回登陆</el-link>
          </div>
          </el-col>
          <el-col :span="12"><div class="grid-content bg-purple-light">
           <el-link type="success">注册协议</el-link>
          </div>
          </el-col>
        </el-row>
      </el-form-item>
    </el-form>
  </div>
</template>

注册样式布局

<style scoped>
	.reg-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%;
	}

	.reg-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>

在这里插入图片描述
实现注册表单验证

<script>
  export default {
    data() {
      var varname=(rule,value,callback)=>{
        if(!value){
          return callback(new Error('用户名不能为空'));
        }
      };
      var varpwd=(rule, value, callback) => {
              if (value === '') {
                callback(new Error('请输入密码'));
              } else {
                if (this.ruleForm.checkPass !== '') {
                  this.$refs.ruleForm.validateField('checkPass');
                }
                callback();
              }
            };
            var varpass=(rule, value, callback) => {
              if (value === '') {
                callback(new Error('请再次输入密码'));
              } else if (value !== this.ruleForm.pwd) {
                callback(new Error('两次输入密码不一致!'));
              } else {
                callback();
              }
            };
      return {
          ruleForm:{
            uname:'',
            pwd:'',
            checkPass:''
         },
         rules: {
                 uname: [
                             { validator: varname, trigger: 'blur' }
                           ],
                 pwd: [
                             { validator: varpwd, trigger: 'blur' }
                           ],
                 checkPass: [
                              { validator: varpass, trigger: 'blur' }
                            ]
                 }
         };
            },



    methods:{
      dologin(){
        this.$router.push({
          path:'/login'
        })
      },
      doRegister(){
        let url=this.axios.urls.SYSTEM_USER_DOREG;
        console.log(url);
        this.axios.post(url, this.ruleForm).then((response)=>{
        					console.log(response.data);
                 if(response.data.code==1){
                   this.$message({
                     message: response.data.msg,
                     type: 'success'
                   });
                   }else{
                      this.$message({
                          message: response.data.msg,
                          type: 'warning'
                        });
                     }

                   }).catch(function(error) {
                   	console.log(error);
                   });
      }
    }
  }
</script>

在这里插入图片描述

4 vue怎么引入和配置使用element-ui框架

1 使用vue-cli脚手架工具创建一个vue项目
vue init webpack pro01

2 npm安装elementUI
cd pro01 #进入新建项目的根目录
npm install element-ui -S #安装element-ui模块

重要的事情说三遍:在指定位置!!!在指定位置!!!在指定位置!!!~~~添加三行代码
3 在项目中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

4 测试
修改HelloWorld.vue添加elementUI组件查看效果

5 后台交互(axios/qs/vue-axios)

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更新,而是推荐的axios

  2.GET提交
  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属性上         

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 vue-axios
vue-axios是在axios基础上扩展的插件,在Vue.prototype原型上扩展了$http等属性,可以更加方便的使用axios

4 axios/qs/vue-axios安装及使用步骤
1.安装
npm install axios -S
npm install qs -S
npm install vue-axios -S

  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)

  3.之后就可以直接在Vue组件文件的methods里使用了  
  this.axios.get(url,{params:{id:1,name:'zs'}}).then((response)=>{
    this.newsList=response.data.data;
  }).catch((response)=>{
    console.log(response);
  }); 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值