【前端——项目实践-权限管理系统】:环境搭建、登录退出

发布于:2023-01-13 ⋅ 阅读:(637) ⋅ 点赞:(0)

一、项目环境搭建

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)&&params[key]!==null&&params[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)&&params[key]!==null&&params[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学习计划

   把握好进度继续完成项目搭建

本文含有隐藏内容,请 开通VIP 后查看