完善Vue简单项目的部分功能(ElementUI之首页导航&左侧菜单)

1.Mockjs

1.1.什么是Mock.js

         前后端分离开发开发过程当中,经常会遇到以下几个尴尬的场景:

  • 老大,接口文档还没输出,我的好多活干不下去啊!

  • 后端小哥,接口写好了没,我要测试啊!

前后端分离之后,前端迫切需要一种机制,不再需要依赖后端接口开发,而今天的主角mockjs就可以做到这一点。

         Mock.js是一个模拟数据的生成器,用来帮助前端调试开发、进行前后端的原型分离以及用来提高自动化测试效率。

         众所周知Mock.js因为两个重要的特性风靡前端:

  • 数据类型丰富:支持生成随机的文本、数字、布尔值、日期、邮箱、链接、图片、颜色等。
  • 拦截Ajax请求:不需要修改既有代码,就可以拦截Ajax请求,返回模拟的响应数据。

更多内容,可查看Mockjs官方

1.2.安装与配置

1.2.1.安装mockjs

npm i mockjs -D

在这里插入图片描述
         当我们安装完成后,就会出现这个样子
在这里插入图片描述

1.2.2.引入mockjs

         为了只在开发环境使用mock,而打包到生产环境时自动不使用mock,我们可以在config目录中的dev.env.jsprod.env.js做一个配置,如下:

  • dev.env.js
module.exports = merge(prodEnv, {
  NODE_ENV: '"development"',
	MOCK: 'true'
})
  • prod.env.js
module.exports = {
  NODE_ENV: '"production"',
	MOCK: 'false'
}
  • main.js
//开发环境下才会引入mockjs
process.env.MOCK && require('@/mock') 

         导入写好的mock-login.js文件和mock中的index.js文件

1.3.mockjs使用

1.3.1.定义测试数据文件

         为每个*.vue定义单独的xxx-mock.js文件,并在其中添加自定义的json数据,还可以通过mockjs的模块生成随机数据信息,动态测试ajax请求效果。

         创建src/mock/json目录,定义登录测试数据文件login-mock.js:

  • login-mock.js
// const loginInfo = {
// 	code: -1,
// 	message: '密码错误'
// }

//使用mockjs的模板生成随机数据
const loginInfo = {
	'code|-1-0': 0,
	'message|3-10': 'msg'
}
export default loginInfo;

1.3.2.mock拦截ajax请求

         在src/mock目录下创建index.js,定义拦截路由配置:

  • index.js
import Mock from 'mockjs' //引入mockjs,npm已安装
import action from '@/api/action' //引入请求地址

//全局设置:设置所有ajax请求的超时时间,模拟网络传输耗时
Mock.setup({
	// timeout: 400  //延时400s请求到数据
	timeout: 200 - 400 //延时200-400s请求到数据
})

//引登陆的测试数据,并添加至mockjs
import loginInfo from '@/mock/json/login-mock.js'
let s1 = action.getFullPath('SYSTEM_USER_DOLOGIN')
Mock.mock(s1, "post", loginInfo)
// Mock.mock(s1, /post|get/i, loginInfo)

         配置完成之后,直接使用axios配置post或get请求,模拟测试效果。
         启动一下项目 npm run dev
         点击登录,出现下界面,能够拿到模拟数据
在这里插入图片描述

1.3.3.数据模板定义规则(额外知识,要用的时候有用)

         请在使用的Vue组件中,先导入Mock模块

import Mock from 'mockjs'
  • 生成随机id值

         使用Mock.Random 的方法在数据模板中称为『占位符』,书写格式为 @占位符(参数 [, 参数])

let uid1 = Mock.Random.guid();
//使用占位符的方式
let uid2 = Mock.mock("@guid");
console.log(uid1);
console.log(uid2);
  • string
let obj1 = Mock.mock({
  'star|1-5':"★",//生成指定范围长度的字符串
  info:'静态的字符串',//静态字符串
  name:'@cname'//使用规定占位符@cname,随机生成中国人的名字
})
console.log(obj1);
  • number 自增

+1 是指,在id的起始值每次调用+1

// +1 是指,在id的起始值每次调用+1
let temp = {
	"id|+1":1000001
}
console.log(Mock.mock(temp)); // 1000001
console.log(Mock.mock(temp)); // 1000002
console.log(Mock.mock(temp)); // 1000003
console.log(Mock.mock(temp)); // 1000004
console.log(Mock.mock(temp)); // 1000005
console.log(Mock.mock(temp)); // 1000006
  • number随机生成

小数生成 |min-max.dmin(小数最小位数)-dmax(小数最大位数)

let obj2 = Mock.mock({
    'age|1-100':0,//随机生成0-100整数
    'price|1-100.1-4':0,//随机生成整数部分1-100,小数位数1-4位
    'num|.2-4':0,//整数部分省略不写,默认是0(js的默认规则)
    'num2|100.3-3':0//整数部分保持100不变,小数位数恒定3位数,小数值随机生成
})
console.log(obj2);
  • boolean值
let flag = {
    'flag1|0':true, //模板规则:0数值是0时,表示布尔值恒定是false
    'flag2|1':false,//值位>=1时,boolean随机取值
    "flag3|1-2":true
}
console.log(Mock.mock(flag));
  • 城市随机取值
let city = Mock.mock({
    "city1|2":{//数字2表示随机取2个值
        addr1:'@city',//占位符@city表示随机生成城市
        addr2:'@city',//占位符@city表示随机生成城市
        addr3:'@city',//占位符@city表示随机生成城市
        addr4:'@city',//占位符@city表示随机生成城市
    },
    "city2|1-3":{//数字1-3表示随机取1-3个值
        addr1:'@city',//占位符@city表示随机生成城市
        addr2:'@city',//占位符@city表示随机生成城市
        addr3:'@city',//占位符@city表示随机生成城市
        addr4:'@city',//占位符@city表示随机生成城市
    }
})
console.log(city);
  • 枚举取值,从数组中随机取值
let gender = Mock.mock({
    'sex|1':['男','女','不知道'],//数字1表示随机取数组中的一个值
    'sex2|+1':['男','女','不知道']//+1表示循环一次取数组中的值
})
console.log(gender);
  • 正则表达式规则
let obj7 = Mock.mock({
    user:{
    	name:'@cname',
    },
    tel:/1[0-9]{10}/,
    //正则表达式可以使用插件
    email:/^[A-Za-z0-9\u4e00-\u9fa5]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/,
    //函数表达式
    info:function(){
        //注意:这里的this指向是数据模板对象,所以可以在函数表达式里使用this来获取已有值;
        return `我的手机号码是:${this.tel},我的邮箱是:${this.email},我的身份证号码是:${this.cardId}`
    },
    // (8)路径规则中: @表示启动关键字(注意:@前面不能有除了空格的任何其他字符)
    //          ==>   / 表示在模板数据中的层级关系,上下级
    message:"我的名字是: @/user/name ,我的手机号码是: @/tel,我的邮箱是: @/email,我的身份证号码是: @/cardId"
})
console.log(obj7);
  • 日期生成
let timer = Mock.mock({
    day1:'@date',// @date占位符表示随机生成mock默认格式的年月日
    day2:'@date("yyyy年MM月dd日")',//@date()可以传参,自定义时间格式,
    time1:'@time',//@time占位符可以随机生成mock格式的时分秒
    time2:'@time("HH时mm分ss秒SS毫秒")',//同样可以在@time()中传参数自定义格式
    daytime1:'@datetime("yyyy年MM月dd日 HH:mm:ss:SS")',//@daytime生成完整的时间
    nowTime:'@now'
})
console.log(timer);
  • 图片生成

生成的是带参数的图片地址,可以根据地址获取图片

let imgs = Mock.mock({
    img:"@image('200x100', '#894FC4', '#FFF', 'png', '!')",
    img1:"@image('200x100', '#aaaaa', '#FFF', 'png', '!')",
    img2:"@image('200x100', '#bbbbb', '#FFF', 'png', '!')",
    // mock占位符调用时,需要传递参数,参数也可以是占位符
    img3:"@image('200x100', '@color', '#FFF', 'png', '!')",
    //可以使用随机生成图片的网址,但是注意,统一网址的请求图片地址时,如果不加参数,那么浏览器默认地址一样的会使用浏览器的缓存,那么所请求的图片只有一种;一般加一个时间戳作为参数,保持参数的唯一,图片可以请求到不同的值
    img4:"https://api.ixiaowai.cn/mcapi/mcapi.php?_=@time('T')"
});
console.log(imgs);

         效果展示
                  打开emlementUI官网,复制两个模板过来
                  贴到自己代码上

在这里插入图片描述

          if (resp.data.code == 0) {
            this.$message({
              message: resp.data.message,
              type: 'success'
            });
          } else {
            this.$message.error(resp.data.message);
          }

         然后注意 要用es6中的箭头函数的写法,不然如果嵌套在函数中的话,这个this就不知道指向的是哪个this了,指向的就不是vue的实例了
在这里插入图片描述

2.注册登录界面的切换(js如何做组件路由)

         复制一份Login.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="gotoLogin()">已有账号?去登录</el-link>
      </el-row>
    </el-form>
  </div>
</template>

<script>
  // import axios from 'axios'
  // import qs from 'qs'
  export default {
    name: 'Login',
    data() {
      return {
        username: null,
        password: null
      }
    },
    methods: {
      // doSubmit() {
      //   let param = {
      //     username: this.username,
      //     password: this.password
      //   }
      //   let url = 'http://localhost:8081/ssm/user/userLogin'
      //   console.log(param)
      //   axios.get(url, {
      //     params: param
      //   }).then(function(resp) {
      //     console.log(resp)
      //   }).catch(function(resp) {

      //   })
      // },
      doSubmit() {
        let param = {
          username: this.username,
          password: this.password
        }
        let url = this.axios.urls.SYSTEM_USER_DOLOGIN
        console.log(param)
        this.axios.post(url, param).then(resp => {
          console.log(resp)
          if (resp.data.code == 0) {
            this.$message({
              message: resp.data.message,
              type: 'success'
            });
          } else {
            this.$message.error(resp.data.message);
          }
        }).catch(resp => {

        })
      },
      gotoLogin() {
        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>

构建注册页面。
         别忘了在router/index.js页面中导入Register.vue哦

在这里插入图片描述

         使用this.$router.push({})实现路由跳转的几种方式:

  • 参数为字符串,即路径名称
//  路径/home对应router目录下index.js中定义的path属性值
this.$router.push('/home');
  • 参数为对象
//  对应router目录下index.js中定义的path
this.$router.push({path:'/home'});
  • 参数为路由命名
//  对应router目录下index.js中定义的name
this.$router.push({name:'Home'});
  • 带参数
// 传递参数:params里面放置的是我们要传递过去的参数
this.$router.push({name:'Home',params:{user:'david'}});

// 查询参数:传递过去的内容会自动拼接变成/home?user=david
this.$router.push({path:'/home',query:{user:'david'}});

query语法与params语法的区别

  • **写法的不同:**query的语法用于path编写传参地址;params的语法用于name编写传参地址
  • **接收方式不同:**query语法使用this.$route.params.name;params语法使用this.$route.query.name
  • **刷新页面参数消失问题:**query在刷新页面时参数不会消失;params再刷新页面时参数会消失
  • **参数显示不同:**query传递参数会显示在地址栏上,相当于get请求;params传递参数不会显示在地址栏上,相当于post请求

3.总线组件通信

主页搭建参考目录资料\src\vue中的vue组件。

  • AppMain.vue:布局容器组件
  • LeftAside.vue:左侧菜单组件
  • TopNav.vue:首页导航组件

根据vue组件之间传递数据实现element-ui的NavMenu菜单折叠、展开效果。

  • 子组件往父组件传递数据(this.$emit):TopNav -> AppMain
  • 父组件往子组件传递数据(props):AppMain -> LeftAside
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在AppMain中
    //注册组件
    components:{TopNav,LeftNav},
    data(){
      return{
        asideClass:'main-aside'
      }
    },
    created(){
      this.$root.BUS.$on('docollapsed',v=>{
        //v指的是topNav传递过来的this.collapsed
        this.asideClass = v?'main-aside-collapsed':'main-aside';
      });
    }

在LeftNav中

<script>
	export default {
    data(){
      return{
        collapsed:false
      }
    },
    created(){
      this.$root.BUS.$on('docollapsed',v=>{
        //v指的是topNav传递过来的this.collapsed
        this.collapsed = v;
      });
    }
	}
</script>

在TopNav中

<script>
	export default {
    data(){
      return{
        collapsed:false,
        imgshow:require('@/assets/img/show.png'),
        imgsq:require('@/assets/img/sq.png')
      }
    },
    methods:{
      doToggle(){
        this.collapsed=!this.collapsed;
        //自定义收起事件
        this.$root.BUS.$emit('docollapsed',this.collapsed);
      }
    }

	}
</script>

在这里插入图片描述
在这里插入图片描述

4.退出功能

         在methods中写上

 exit(){
        this.$router.push({path:'/'})
      }

在这里插入图片描述

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值