目录
一、前置准备工作
1.1 导入后端代码
我们找一个后台有增删改查的简单项目,导入我们的编码器,Leaf这里用的是eclipse;
然后我们还需要把两个js工具包组成的一个文件夹导入前端SPA项目中,用以与后台代码相关联,调用后台接口。
这里把两个js文件放到这里,需要的朋友把它们放到同一文件夹【api】中,然后再导入前端项目中:
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;
然后我们这个时候一定要对我们找来的项目进行测试,确认与你本机数据库连接没有问题后再进行后面的开发。
1.2 下载js依赖
我们需要分别下载四个依赖:
axios:前端新后台发送请求
qs:解决发送post请求代码冗余的问题
ElementUI:快速布局
vue-axios:将axios依赖整合到vue中
而下载这个的话就必须去到我们前端项目的根目录下运行下载指令,进行下载!
下载指令:
npm install element-ui -S
npm install axios -S
npm install qs -S
npm install vue-axios -S
第一个指令
第二个指令
第三个指令
第四个指令
当我们成功下载好四个依赖后,就可以在前端项目的package.json文件的dependencies中看到多出来的四个依赖:
到这一步后,就代表,我们的前置准备工作完成啦!
二、登录页面的排版
2.1 导入Element-UI的依赖
我们需要用到ElementUI来完成登录页面的制作,所以就需要先引入它的依赖到main.js中;
main.js
// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import ElementUI from 'element-ui' // 新添加 1
// 避免后期打包样式不同,所以放在import App from './App'; 之前
import 'element-ui/lib/theme-chalk/index.css' // 新添加 2
import App from './App'
import router from './router'
Vue.use(ElementUI) // 新添加 3
Vue.config.productionTip = false
/* eslint-disable no-new */
new Vue({
el: '#app',
router,
components: { App },
template: '<App/>'
})
然后我们把上篇文章写的那些组件、触发事件、组件和路由的关系定义全部删除,让它们回到最初的模样,准备测试今天内容。
测试ElementUI是否能正常使用:
① 我们去到ElementUI的官网,随便找一些组件按钮代码,粘贴到App.vue文件中;
官网:Element - The world's most popular Vue UI framework
② 在项目根目录运行npm run dev
欧克,我们测试ElementUI组件能用,没有问题。
2.2 编写登录页面展示内容
我们看看项目src目录下文件夹的一些参考定义:
assets:静态资源
components:放自定义组件
router:路由对应关系
views:展示内容
而登录的业务代码就属于views里面;
我们在views中建立一个vue文件:Login.vue
并且在ElementUI官网中粘贴一份表单代码过来。
参考代码:Login.vue
<template>
<div class="login-wrap">
<el-form :model="ruleForm" status-icon label-width="100px" class="demo-ruleForm login-container">
<el-row>
<el-col :span="24">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<h3>用户登录</h3>
</div>
</el-col>
</el-row>
<el-form-item label="用户名" prop="uname">
<el-input v-model.number="ruleForm.uname"></el-input>
</el-form-item>
<el-form-item label="密码" prop="pwd">
<el-input type="password" v-model="ruleForm.pwd" autocomplete="off"></el-input>
</el-form-item>
<el-form-item>
<el-row>
<el-col :span="24">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-button style="width: 100%;" type="primary" @click="submitForm">提交</el-button>
</div>
</el-col>
</el-row>
</el-form-item>
<el-form-item>
<el-row>
<el-col :span="12">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-link type="success" @click="">用户注册</el-link>
</div>
</el-col>
<el-col :span="12">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-link type="warning" @click="">忘记密码</el-link>
</div>
</el-col>
</el-row>
</el-form-item>
</el-form>
</div>
</template>
<script>
export default {
name: 'Login',
data() {
return {
ruleForm: {}
}
},
methods: {
submitForm() {
},
resetForm() {
}
}
}
</script>
<style>
</style>
接下来就一起改一下我们的样式,让这个登录页面好看一些嘿嘿嘿。
① 修改App.vue中的style;
② 修改Login.vue中的style;
App.vue
<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
<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>
我们一起来看看运行效果图:
三、Vue数据交互 - 实现登录
3.1 添加点击事件
我们给Login.vue中的用户注册添加一个点击事件,定义方法;
点击事件:
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-link type="success" @click="toReg">用户注册</el-link>
</div>
跳转注册的方法:
toReg() {
this.$router.push({
path: '/Reg'
});
}
然后我们再复制一份登录页面,修改一下,改成注册页面,并且一样的写好跳转到登录页面的方法;
Reg.vue
<template>
<div class="login-wrap">
<el-form :model="ruleForm" status-icon label-width="100px" class="demo-ruleForm login-container">
<el-row>
<el-col :span="24">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<h3>用户注册</h3>
</div>
</el-col>
</el-row>
<el-form-item label="用户名" prop="uname">
<el-input v-model.number="ruleForm.uname"></el-input>
</el-form-item>
<el-form-item label="密码" prop="pwd">
<el-input type="password" v-model="ruleForm.pwd" autocomplete="off"></el-input>
</el-form-item>
<el-form-item>
<el-row>
<el-col :span="24">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-button style="width: 100%;" type="primary" @click="submitForm">提交</el-button>
</div>
</el-col>
</el-row>
</el-form-item>
<el-form-item>
<el-row>
<el-col :span="12">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-link type="success" @click="toLogin">用户登录</el-link>
</div>
</el-col>
<el-col :span="12">
<div style="text-align: center;" class="grid-content bg-purple-dark">
<el-link type="warning" @click="">忘记密码</el-link>
</div>
</el-col>
</el-row>
</el-form-item>
</el-form>
</div>
</template>
<script>
export default {
name: 'Reg',
data() {
return {
ruleForm: {}
}
},
methods: {
submitForm() {
},
resetForm() {
},
toLogin() {
this.$router.push({
path: '/Login'
});
}
}
}
</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>
3.2 导入axios的依赖
axios相当于jQuery中的ajax
作用:使用获取后台数据,做局部刷新
import axios from '@/api/http' #vue项目对axios的全局配置
import VueAxios from 'vue-axios'
Vue.use(VueAxios,axios);
编写登录的提交方法:
submitForm() {
//this.axios找到的是/api/http.js文件
//http又配置了axios.urls = action ---> this.axios.urls找到了action.js
//action.js是一个JSON对象,那么就可以取到请求的路径值
let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
//alert(url);
//url就是请求后台的某一个接口
this.axios.post(url,this.ruleForm)
.then(function(resp){//成功回调函数
console.log(resp);
}).catch(function(){//失败回调函数
});
},
然后我们测试一下:
不输入密码:
输入错误信息:
输入正确信息:
最后我们总结一下:
语法:
this.axios.post(请求地址,参数)
.then(function(resp) {})
.catch(function(error) {})
四、this指针污染
4.1 箭头函数
我们发现我们把ElementUI中的消息提示语句,写到axios.post提交的方法内是没有效果的,原因就是因为这个post提交方法的内部已经不是vue实例,而是http.js,这就是this的指针污染。
而现在我们的解决方法就是利用jdk8的箭头函数语法。
//这个this指的是vue实例
//let obj = this;
this.axios.post(url,this.ruleForm)
.then(resp => {//成功回调函数 箭头函数 jdk8语法
//console.log(resp);
if(resp.data.code == 1){
this.$message({
message: '恭喜你,登录成功!!!',
type: 'success'
});
}
else if(resp.data.code == 0){
this.$message.error('密码或用户名输入有误,登录失败!!!');
}
}).catch(function(){//失败回调函数
});
然后我们来测试一下;
登录成功
登录失败
五、CORS跨域
5.1 CORS跨域体现
5.2 跨域产生的因素
我们先来分析一下我们访问网址是由哪些元素组成的;
例如:
http://localhost:8080/Leaf_0909/ --->
协议+ip+端口号+项目
跨域产生原因:只要请求地址中的元素有任意一处发生改变,就会产生跨域的问题。
5.3 解决跨域问题办法
要解决跨域的问题,我们就可以在后台项目的web.xml中添加一个过滤器就ok了;
过滤器代码:
<!-- 解决cors跨域问题过滤器 -->
<filter>
<filter-name>corsFilter</filter-name>
<filter-class>com.zking.vue.util.CorsFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>corsFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
六、原始axios的使用&get - post的区别
6.1 原始axios请求的使用
更换在main.js文件中导入的依赖,注释原本导入的vue项目对axios的全局配置;
// import axios from '@/api/http' //vue项目对axios的全局配置
import axios from 'axios'
改变导入依赖后,一定要重启项目:
然后我们测试一下登录:
这个时候我们只用了原始的axios提交,没有用到工具包,所以返回的是一个对象,不能验证。
所以我们就需要用到工具包:
我们在Login.vue中单独使用一下上篇文章下载的“qs”,在这个页面中导入qs依赖;
import qs from 'qs'
然后在登录的方法上使用:
this.axios.post(url,qs.stringify(this.ruleForm))
并且写死请求路径值:
let url = "http://localhost:8080/T216_SSH/vue/treeNodeAction";
然后我们再次测试:
因此,我们可以发现前面我们用的工具包http.js、action.js的几个作用:
1、将原生的json对象转成字符串,通过拦截器转换;
2、将本项目中俄所有接口地址,进行统一管理;
6.2 get - post的区别
这个Leaf就直接放上代码,看代码参考咯;
get请求
let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
this.axios.post(url, this.ruleForm).then(function(response) {
console.log(response);
}).catch(function(error) {
console.log(error);
});
post请求
let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
this.axios.get(url, { //注意数据是保存到json对象的params属性
params: this.ruleForm
}).then(function(response) {
console.log(response);
}).catch(function(error) {
console.log(error);
});