一、项目环境搭建
1.环境要求:node.js、python3+、vscode、git
(1)git安装 ——下载地址
(2)生成SSH的公钥私钥
右键更多——
然后输入 ssh-keygen -t rsa;一直回车之后就会发现在c盘用户当中多了一个.ssh文件夹。
即可得到公钥私钥。
(3)vue-Element-admin
目录结构可以在vue-Element-admin官网查看意义
(3)TortoiseGit
设置汉化:安装完成之后,文件任意位置右键,按下面操作,即可设置汉化。注意语言包要和我们下载的tortoisegit版本保持一致,才能够正确汉化 。
扩:
(4)关闭ESLint
二、展示改动部分代码
1.改动登陆页面
(1)placeholder 属性提供可描述输入字段预期值的提示信息(hint)。该提示会在输入字段为空时显示,并会在字段获得焦点时消失。
注释:placeholder 属性适用于以下的 <input> 类型:text, search, url, telephone, email 以及 password。
(2)
index.vue
<el-button :loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">登录</el-button>
</div>
</template>
<script>
export default {
name: 'Login',
data() {
return {
//登陆表单对象
loginForm: {
username: '',
password: ''
},
//表单验证规则;tigger:触发器
loginRules: {
username: [{ required: true, trigger: 'blur',message:"请输入用户名"}],
password: [{ required: true, trigger: 'blur', message:"请输入密码"}]
},
passwordType: 'password',
capsTooltip: false,
loading: false,
showDialog: false,
redirect: undefined,
otherQuery: {}
}
},
watch: {
$route: {
handler: function(route) {
const query = route.query
if (query) {
this.redirect = query.redirect
this.otherQuery = this.getOtherQuery(query)
}
},
immediate: true
}
},
mounted() {
//如果用户名为空,则聚焦到用户名输入框
if (this.loginForm.username === '') {
this.$refs.username.focus()
//如果密码为空,则聚焦到密码输入框
} else if (this.loginForm.password === '') {
this.$refs.password.focus()
}
},
methods: {
//显示密码
showPwd() {
if (this.passwordType === 'password') {
this.passwordType = ''
} else {
this.passwordType = 'password'
}
this.$nextTick(() => {
this.$refs.password.focus()
})
},
//处理登录请求
handleLogin() {
//进行表单验证
this.$refs.loginForm.validate(valid => {
//如果表单验证通过
if (valid) {
this.loading = true//显示加载动画(开启进度条)
//发送登录请求,调用vuex中的user/login方法(src/store/modules/user.js)
this.$store.dispatch('user/login', this.loginForm)
.then(() => {
//由路由发送到指定位置
this.$router.push({ path: this.redirect || '/', query: this.otherQuery })
//关闭进度条
this.loading = false
})
.catch(() => {
this.loading = false
})
} else {
console.log('error submit!!')
return false
}
})
},
getOtherQuery(query) {
return Object.keys(query).reduce((acc, cur) => {
if (cur !== 'redirect') {
acc[cur] = query[cur]
}
return acc
}, {})
}
}
}
2.登陆前后端联调
vue.config.js
devServer: {
port: port,
open: true,
overlay: {
warnings: false,
errors: true
},
proxy:{
[process.env.VUE_APP_BASE_API]:{
target:"http://localhost:8089/api",
changeOrigin:true,
pathRewrite:{
'^/api':''
}
}
}
},
这一步由于是我在前端自己测试,所以按照源代码形式,没有改动
main.js
//注释掉以下代码
// if (process.env.NODE_ENV === 'production') {
// const { mockXHR } = require('../mock')
// mockXHR()
// }
这一步由于是我在前端自己测试,所以按照源代码形式,没有改动
src/utils/request.js
先执行:npm install qs
import axios from 'axios'
import { MessageBox, Message } from 'element-ui'
import store from '@/store'
import { getToken } from '@/utils/auth'
import qs from 'qs'
// create an axios instance
//创建axios异步请求实例
const service = axios.create({
baseURL: process.env.VUE_APP_BASE_API, // url = base url + request url
// withCredentials: true, // send cookies when cross-domain requests
timeout: 15000 // request timeout
})
// request interceptor
//请求前拦截
service.interceptors.request.use(
config => {
// do something before request is sent
//判断store中是否存在token
if (store.getters.token) {
//读取token信息,并将token添加到headers头部信息中
// let each request carry token
// ['X-Token'] is a custom headers key
// please modify it according to the actual situation
config.headers['X-Token'] = getToken()
}
return config
},
error => {
// do something with request error
console.log(error) // for debug
return Promise.reject(error)
}
)
// response interceptor
//响应时拦截
service.interceptors.response.use(
/**
* If you want to get http information such as headers or status
* Please return response => response
*/
/**
* Determine the request status by custom code
* Here is just an example
* You can also judge the status by HTTP Status Code
*/
response => {
//获取后端返回的数据,如果后端返回的状态码不是200,则提示错误信息
const res = response.data
// if the custom code is not 20000, it is judged as an error.
if (res.code !== 200) {
Message({
message: res.message || 'Error',
type: 'error',
duration: 5 * 1000
})
// 50008: Illegal token; 50012: Other clients logged in; 50014: Token expired;
if (res.code === 50008 || res.code === 50012 || res.code === 50014) {
// to re-login。重新登录
MessageBox.confirm('用户登录信息过期,请重新登录', '系统提示', {
confirmButtonText: '登录',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
store.dispatch('user/resetToken').then(() => {
location.reload()
})
})
}
return Promise.reject(new Error(res.message || 'Error'))
} else {
return res
}
},
error => {
console.log('err' + error) // for debug
Message({
message: error.message,
type: 'error',
duration: 5 * 1000
})
return Promise.reject(error)
}
)
//请求方法
const http={
//post请求提交
post(url,params){
return serrvice.post(url,params,{
transformRequser:[(params)=>{
return JSON.stringify(params)
}],
header:{
'Content-Type':'application/json'
}
})
},
//put请求发送
put(url,params){
return service.put(url,params,{
transformRequser:[(params)=>{
return JSON.stringify(params)
}],
header:{
'Content-Type':'application/json'
}
})
},
//get请求
get(url,params){
return service.get(url,{
params:params,
paramsSerializer:(params)=>{
return qs.stringify(params)
}
})
},
//rest风格的get请求
getRestApi(url,params){
let_params
if(Object.is(params,undefined||null)){
_params=''
}else{
_params='/'
for(const key in params){
console.log(params[key])
if(params.hasOwnProperty(key)&¶ms[key]!==null&¶ms[key]!==''){
_params+=`${[params[key]]}/`
}
}
_params=_params.substr(0,_paras.length-1)
}
if(_params){
return service.get(`${url}${_params}`)
}else{
return service.get(url)
}
},
//删除请求
delete(url,params){
let_params
if(Object.is(params,undefined||null)){
_params=''
}else{
_params='/'
for(const key in params){
console.log(params[key])
if(params.hasOwnProperty(key)&¶ms[key]!==null&¶ms[key]!==''){
_params+=`${[params[key]]}/`
}
}
_params=_params.substr(0,_paras.length-1)
}
if(_params){
return service.get(`${url}${_params}`).catch(err=>{
message.error(err.mes)
return Promise.reject(err)
})
}else{
return service.get(url).catch(err=>{
message.error(err.mes)
return Promise.reject(err)
})
}
},
//文件上传请求
upload(url,params){
return service.post(url,params,{
headers:{
'Content-Type':'multipart/form-data'
}
})
},
//登录请求
login(url,params){
return service.post(url,params,{
transformRequest:[(params)=>{
return qs.stringify(params)
}],
headers:{
'Content-Type':'appliciation/x-www-form-urlencoded'
}
})
}
}
export default http
src/api/user.js
import http from '@/utils/request'
/*
*用户登录
*@returns
*/
export async function login(data) {
return await http.login("/api/user/login",data)
}
/**
* 获取用户信息和权限信息
* @returns
*/
export async function getInfo() {
return await http.get("/api/sysUser/getInfo")
}
/**
* 退出登录
* @returns
*/
export async function logout() {
return await http.post("/api/sysUser/loginOut",param)
}
.env.development和.env.production
VUE_APP_BASE_API = 'http://localhost:9999/'
//端口号不一定是9999,要修改的!!!
这一步由于是我在前端自己测试,所以按照源代码形式,没有改动
src/store/modules/user.js
const actions = {
// user login
login({ commit }, userInfo) {
//从用户信息中解析出用户名和密码
const { username, password } = userInfo
return new Promise((resolve, reject) => {
//传递用户名和密码参数
login({ username: username.trim(), password: password }).then(response => {
//从response解析出后端返回的数据
const { token } = response
//将返回的token信息保存到store,作为全局变量使用
commit('SET_TOKEN', token)
//设置token,将token信息保存到cookie中
setToken(token)
resolve()
}).catch(error => {
reject(error)
})
})
},
3.用户退出登录前后端联调
src/api/user.js
/**
* 退出登录
* @returns
*/
export async function logout() {
return await http.post("/api/sysUser/loginOut",param)
}
这一步由于是我在前端自己测试,所以按照源代码形式,没有改动
src/utils/auth.js
/**
* 清空sessionStorage
*/
export function clearStorage(){
return sessionStorage.clear();
}
src/layout/components/Navbar.vue
//先引入相应脚本
import{logout} from "@/api/user"
import {gettoken,removeToken,clearStorage} from "@/utils/auth"
//修改页面下拉菜单
<el-dropdown-menu slot="dropdown">
<router-link to="/profile/index">
<el-dropdown-item>首页</el-dropdown-item>
</router-link>
<router-link to="/">
<el-dropdown-item>个人中心</el-dropdown-item>
</router-link>
<!-- <a target="_blank" href="https://github.com/PanJiaChen/vue-element-admin/">
<el-dropdown-item>Github</el-dropdown-item>
</a>
<a target="_blank" href="https://panjiachen.github.io/vue-element-admin-site/#/"> -->
<!-- <el-dropdown-item>Docs</el-dropdown-item>
</a> -->
<el-dropdown-item divided @click.native="logout">
<span style="display:block;">退出系统</span>
</el-dropdown-item>
</el-dropdown-menu>
src/layout/components/Nabar.vue
logout() {
this.$confirm('确定要退出系统吗?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(async () => {
await this.$store.dispatch('user/logout')
this.$router.push(`/login?redirect=${this.$route.fullPath}`)
// // type: 'success',
// // message: '删除成功!'
// // 以上代码替换成下面代码,这段代码,到时候连后端之后需要改!!
// // 发送退出请求
// //先封装请求参数
// let params={token:getToken()}
// let res = logout(params);
// logout(params);
// //判断是否成功
// if(res.success){
// //清空token信息
// removeToken();
// clearStorage();
// //跳转到登陆页面
// window.location.href="/login/"
}
由于只是前端自己调试,所以没有改动这部分代码,到时候前后端联调需要改动。
4.封装信息确认提示框
src/utils,创建myconfirm.js
import {MessageBox, Message} from 'element-ui'
//删除弹窗
export default function myconfirm(text){
return new Promise((resolve,reject)=>{
MessageBox.confirm(text,'系统提示',{
confirmButtonText:'确定',
cancelButtionText:'取消',
type:'warning'
}).then(()=>{
resolve(true)
}).catch(()=>{
reject(false)
})
})
}
main.js,挂载信息确认提示框
/导入封装信息确定提示框组件脚本
import myconfirm from '@/utils/myconfirm'
vue.prototype.$myconfirm=myconfirm;
src/layout/conponents/Nabar.vue,使用封装改造之前的代码
async logout() {
let confirm =await this.$myconfirm("确定退出系统吗?")
//是否确认
if (confirm){
await this.$store.dispatch('user/logout')
this.$router.push(`/login?redirect=${this.$route.fullPath}`)
}
}
8.16-8.17学习汇报与总结
1.问题解决
(1)问题:
解决:①淘宝镜像源下载,②切换浏览器(这里我是换成了谷歌浏览器)
(2)问题:npm i报错
解决:命令行输入这个git config --global url."https://".insteadOf ssh://git@;再下载依赖npm i
(3)问题:首页页面登陆不上
解决:这里是要前后端联动,但是我只是操作了前端。为了光前端进行测试,所以我不改动代理,也就是继续使用mock
2.8.16-8.17学习体会与感受
这两天进度稍慢,主要的是在配置环境上出了各种奇奇怪怪的问题,花了好多时间,还有改各种报错。改报错的时候最大的体会是,不知道代码干嘛的,所以改的就会很懵,之前改错是去搜,然后一个个试。好家伙,直接搜不到,只能靠自己。不过是改错花的最长(尊的好长),也是最大的收获就是,原视频要使用后端接口测试,但是我这里暂时没有,于是就发现了问题所在(得亏有印象mock),就不能按照视频的方法改代码,于是自己慢慢重新再写了一遍,改一次,就运行看看效果,最后终于找到了那些地方能改,那些地方不能改,成功登录了!
总结:目前这个阶段,还是慢慢来,每改一段代码,就要运行看效果看报错,否则,就会不知道错在哪里(没错,我这个改了好久也改不出来重新打了一遍的大怨种)。
3.8.18-8.19学习计划
把握好进度继续完成项目搭建