[Vue.js] Getting started with Element to build the login registration interface & GET requests and POST requests in axios & cross-domain issues

1. What is ElementUI?

              Element UI is a desktop component library based on Vue.js. It provides a rich set of UI components for building user interfaces . The goal of Element UI is to provide simple, easy-to-use, and beautiful components while maintaining flexibility and customizability

2. Features and functions of ElementUI (roughly five points)

Element UI features and capabilities include:

Rich components:

           Element UI provides a large number of commonly used components, including buttons, forms, tables, dialog boxes, menus, and more. These components are carefully designed to have a unified style and interactive experience, which is very convenient when building user interfaces.

Responsive layout:

        The components of Element UI can automatically adapt to different screen sizes and support responsive layout, allowing applications to have good display effects on different devices.

Simple and easy to use:

     The components of Element UI are designed concisely and provide a wealth of configuration options. Developers can flexibly customize the appearance and behavior of components through parameters.
 

 Theme customization:

     Element UI provides a set of default theme styles, but also supports custom themes. You can customize your own theme style according to the needs of the project to make the application more consistent with the overall style of the project

International support:

   Element UI has built-in multi-language support, making it easy to implement multi-language applications

2. Preparation for building login and registration of Element

①Build a SPA project   

The SPA project has not been built---》》Click to enter the SPA project construction process

②Import Element dependencies

Use the command npm install element-ui -S to add the Element-UI module

Note : You need to use the cmd terminal command to download in your SPA project workspace (as shown below)

After the download is complete, you will see the Element dependencies required for downloading in the package.json file under your SPA project.

3. Element implements login and registration interface 

Step 1: Create the views directory in the src directory (this directory is used to store vue components)

Step 2: Introduce element-ui module into main.js

Find itmain.js in the src directory of the project and add three lines of code at the specified location :

Note: Add the following three lines of code before the line of import App from './App'

import Vue from 'vue'

// 新添加1
import ElementUI from 'element-ui' 
// 新添加2,避免后期打包样式不同,要放在import App from './App';之前
import 'element-ui/lib/theme-chalk/index.css' 

Vue.jsx mounts Element code (also in main.js文件下添加即可):


Vue.use(ElementUI)   
Vue.config.productionTip = false

Step 3: ①Create user login componentLogin.vue

<template>
	<div class="login-wrap">
		<el-form class="login-container">
			<h1 class="title">用户登录</h1>
			<el-form-item label="">
				<el-input type="text" v-model="username" placeholder="登录账号" autocomplete="off"></el-input>
			</el-form-item>
			<el-form-item label="">
				<el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off"></el-input>
			</el-form-item>
			<el-form-item>
				<el-button type="primary" style="width:100%;" @click="doSubmit()">提交</el-button>
			</el-form-item>
			<el-row style="text-align: center;margin-top:-10px">
				<el-link type="primary">忘记密码</el-link>
				<el-link type="primary" @click="gotoRegister()">用户注册</el-link>
			</el-row>
		</el-form>
	</div>
</template>

In order to make the style more beautiful, import the style in App.vue

App.vue style:

<style> 
html,
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.vue and Register.vue expression:

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

Rendering:

Set the binding routing data relationship in index.js

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


Vue.use(Router)

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

Effect:

4. Data interaction

Use the SSM project to build a Java backend, simulate providing an action address for user login, and Vue requests the specified user login interface.

4.1 Understanding axios and installation

axios is a lightweight version of ajax advocated by vue2. It is a promise-based HTTP library. It creates XMLHttpRequests from the browser and works great with Vue.

Off topic: vue.js has the famous family bucket series: vue-router, vuex, vue-resource, plus the construction tool vue-cli, which is the core component of a complete vue project. Among them, vue-resource is a plug-in for Vue.js. It can initiate requests and process responses through XMLHttpRequest or JSONP. However, after vue was updated to 2.0, the author announced that he would no longer update vue-resource, but recommended axios.

Installation syntax:

npm i axios -S

The default is post request

If data interaction requires a get request, you must download the dependencies related to the get request.

get syntax:

npm i qs -S

Note : In your SPA project workspace, start the cmd window and execute this line of code (below)

4.2 Axios Post request

1 Import the just downloaded axios dependency on the front end

import axios from 'axios'

Front-end writing:

<script>
import axios from 'axios'
export default {
  name: 'Login',
  data () {
    return {
      username:'',
	  password:''
    }
  } 
  ,methods:{
	  gotoRegister(){
		 this.$router.push('/Register'); 
	  },
	 
	   doSubmit(){
		  let url= 'localhost:8080/ssm/user/userLogin';
		   let params = {
			   username:this.username,
			   password:this.password
		   }
		   axios.get(url,{params:params}).then(r=>{
			   console.log(r);
				  if(r.data.success){
					  this.$message({
						  message:r.data.msg,
						  type:'success'
					  });
				  }else{
					  this.$message.error(r.data.msg); 
				  }
		   }).catch(e=>{
			   
		   })
		   
	   }
  }
}
</script>

Next, modify the port number of the SPA project under index.js in the front end, just in case it is consistent with the back-end port number.

In our back-end project, there is a view resolver to handle the requests sent by our front-end.

userController:

package com.zking.ssm.controller;

import com.zking.ssm.service.IUserService;
import com.zking.ssm.util.JsonResponseBody;
import com.zking.ssm.util.PageBean;
import com.zking.ssm.vo.UserVo;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import com.zking.ssm.jwt.*;

@Controller
@RequestMapping("/user")
public class UserController {

    @Autowired
    private IUserService userService;

    @RequestMapping("/userLogin")
    @ResponseBody
    public JsonResponseBody<?> userLogin(UserVo userVo, HttpServletResponse response){
        if(userVo.getUsername().equals("admin")&&userVo.getPassword().equals("123")){
            //私有要求claim
//            Map<String,Object> json=new HashMap<String,Object>();
//            json.put("username", userVo.getUsername());
            //生成JWT,并设置到response响应头中
//            String jwt=JwtUtils.createJwt(json, JwtUtils.JWT_WEB_TTL);
//            response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);
            return new JsonResponseBody<>("用户登陆成功!",true,0,null);
        }else{
            return new JsonResponseBody<>("用户名或密码错误!",false,0,null);
        }
    }

    @RequestMapping("/queryUserPager")
    @ResponseBody
    public JsonResponseBody<List<Map<String,Object>>>
            queryUserPager(UserVo userVo, HttpServletRequest request){
        try {
            PageBean pageBean=new PageBean();
            pageBean.setRequest(request);
            List<Map<String, Object>> users = userService.queryUserPager(userVo, pageBean);
            return new JsonResponseBody<>("OK",true,pageBean.getTotal(),users);
        } catch (Exception e) {
            e.printStackTrace();
            return new JsonResponseBody<>("分页查询用户信息失败!",false,0,null);
        }

    }
}

In the front-end project workspace, enter cmd in the file path and open the cmd window.

Enter the command to start the project: 

npm run dev

And start the maven project in the backend to access it.

4.3  axios Get request

Import dependencies

  import qs from 'qs' 
<template>
	<div class="login-wrap">
		<el-form class="login-container">
			<h1 class="title">用户登录</h1>
			<el-form-item label="">
				<el-input type="text" v-model="username" placeholder="登录账号" autocomplete="off"></el-input>
			</el-form-item>
			<el-form-item label="">
				<el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off"></el-input>
			</el-form-item>
			<el-form-item>
				<el-button type="primary" style="width:100%;" @click="doSubmit()">提交</el-button>
			</el-form-item>
			<el-row style="text-align: center;margin-top:-10px">
				<el-link type="primary">忘记密码</el-link>
				<el-link type="primary" @click="gotoRegister()">用户注册</el-link>
			</el-row>
		</el-form>
	</div>
</template>

<script>
import axios from 'axios'
  import qs from 'qs' //用于post请求
export default {
  name: 'Login',
  data () {
    return {
      username:'',
	  password:''
    }
  } 
  ,methods:{
	  gotoRegister(){
		 this.$router.push('/Register'); 
	  },
	 
	   doSubmit(){
		   let params = {
		             username: this.username,
		             password: this.password
		           };
		           console.log(params);
		           //定义后端都请求地址
		           var url = this.axios.urls.SYSTEM_USER_DOLOGIN;
		   
		   
		  // let url= 'localhost:8080/ssm/user/userLogin';
		  //  let params = {
			 //   username:this.username,
			 //   password:this.password
		  //  }
		  //  axios.get(url,{params:params}).then(r=>{
			 //   console.log(r);
				//   if(r.data.success){
				// 	  this.$message({
				// 		  message:r.data.msg,
				// 		  type:'success'
				// 	  });
				//   }else{
				// 	  this.$message.error(r.data.msg); 
				//   }
		  //  }).catch(e=>{
			   
		  //  })
		   
	   }
  }
}
</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>

Import api files (two):

action.js:

/**
 * 对后台请求的地址的封装,URL格式如下:
 * 模块名_实体名_操作
 */
export default {
	'SERVER': 'http://localhost:8080/ssm', //服务器
	'SYSTEM_USER_DOLOGIN': '/user/userLogin', //登陆
	'SYSTEM_USER_DOREG': '/userAction.action', //注册
	'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) {
	return config;
}, function(error) {
	return Promise.reject(error);
});

// 响应拦截器
axios.interceptors.response.use(function(response) {
	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;

Two js files need to be configured in main.js:

main.js:

import axios from '@/api/http'                 
import VueAxios from 'vue-axios' 
Vue.use(VueAxios,axios)

5. Implementation of registration function 

①Create a registration page by yourself

②Define interface and interface implementation class

Interface class:

int insertSelective(User record);

Implement interface class:

    @Override
    public int insertSelective(User record) {
        return userMapper.insertSelective(record);
    }

③ Controller layer writing:

 @RequestMapping("/userRegister")
    @ResponseBody
    public JsonResponseBody<?> userRegister(UserVo userVo, HttpServletResponse response) {
        //状态新注册默认为0
        userVo.setStatus("0");
        //因为ID为String类型需要手动设置,当然可以根据自己的需要改为Int类型
        userVo.setId("9");
        int i = userService.insertSelective(userVo);
        if (i > 0) {
            return new JsonResponseBody<>("用户注册完成!快去登入吧!", true, 0, null);
        } else {
            return new JsonResponseBody<>("用户注册失败!重新输入。", false, 0, null);
        }
    }

5. Cross-domain issues 

5.1 Concept

      Cross-Origin Resource Sharing (CORS) refers to the fact that on the browser side, due to security policy restrictions, web pages (or Ajax requests) between different sources (domains/protocols/ports) cannot directly interact or access each other. H. The same-origin policy is a browser security mechanism used to protect user information and prevent malicious attacks.

  The same-origin policy requires that web pages can only interact with resources from the same origin. Origin consists of protocol, domain name, and port number. Browsers block cross-domain requests when the origin is inconsistent. For example, if a web page is running in domain A and attempts to send an AJAX request to domain B through JavaScript, a cross-domain issue will be triggered.

5.2 Solving cross-domain issues

web.xml:

  <!--CrosFilter跨域过滤器-->
  <filter>
    <filter-name>corsFilter</filter-name>
    <filter-class>com.zking.ssm.util.CorsFilter2</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>corsFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

Cross-domain tools:

CorsFilter2 filter

The purpose of this code is to configure the Tomcat server to allow cross-domain access from any domain name, and to allow the client to send requests containing specified request headers and specify allowed request methods.

package com.zking.ssm.controller;
 
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.HttpServletResponse;
 
/**
 * 配置tomcat允许跨域访问
 * 
 * @author Administrator
 *
 */
public class CorsFilter2 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.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
		httpResponse.setHeader("Access-Control-Allow-Headers", "responseType,Origin, X-Requested-With, Content-Type, Accept");
		httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
 
		//允许客户端处理一个新的响应头jwt
		//httpResponse.setHeader("Access-Control-Expose-Headers", "jwt,Content-Disposition");
		filterChain.doFilter(servletRequest, servletResponse);
	}
 
	@Override
	public void destroy() {
 
	}
}

Guess you like

Origin blog.csdn.net/m0_73192864/article/details/133179910