搭建vue2项目

76 阅读5分钟

一、基于VueClig创建项目架子

新建一个文件夹

打开它的控制台

输入vue create 项目名

image.png

image.png

image.png

image.png

image.png

image.png

image.png

image.png

image.png

二、清理项目目录

三、实现vant组件库的按需引入

进入vant官网:

快速上手 Doc | 组件 中文文档 documentation | v2.13.2 v2.0 v2.x | Vant UI (for vue 2.0) | Vue UI Component for mobile phone | Vant js

image.png

image.png

image.png

image.png

进行vant组件全局注册:

在src文件夹下创建utils文件夹专门用于存放项目中用到的工具方法

src/utils/vant-ui.js

eg;

image.png

再将该文件引入到入口文件main.js中

image.png

四、postcss插件实现vw适配

npm install postcss-px-to-viewport --save-dev

image.png

五、新建styles文件夹用于存放项目通用样式

src/styles/common.less文件专门用于存放项目通用样式

然后再把样式文件引入到入口文件main.js中

六、二次封装axios,并完成在响应拦截器中统一处理错误信息,在请求拦截器中统一给请求添加token信息,在请求拦截器中实现打开发送请求之前出现的loading效果,在响应拦截器中实现关闭先前打开的loading效果。

utils/request.js

// 进行二次封装axios
// 二次封装axios是在原有的axios基础上进行的,所以需要先引入原有的axios
import axios from 'axios'
import { Toast } from 'vant'
import store from '@/store/index'
// 创建二次封装后的axios实例,需要用到其本身的create方法
const instance = axios.create({
  // 修改基地址
  baseURL: 'http://smart-shop.itheima.net/index.php?s=/api',
  timeout: 5000,
  headers: { platform: 'H5' }
})
// 配置请求拦截器  请求拦截器的作用:在发送请求之前,处理一些业务
instance.interceptors.request.use(function (config) {
  // 在发送请求之前做些什么

  //在请求拦截器中打开loading,这样每次请求之前都会显示loading效果
  Toast.loading({
    message: '请求中...',
    forbidClick: true,
    loadingType: 'spinner',
    duration: 0
  })

  //统一携带token请求信息,由于大多数接口在接受请求之前,都需要检查token信息,因此可以给发送的请求统一配置token信息,在请求拦截器中进行统一配置。
  const token = store.getters.token
  if (token) {
    config.headers['Access-Token'] = token
    config.headers.platform = 'H5'
  }
  return config
}, function (error) {
  // 对请求错误做些什么
  return Promise.reject(error)
})

// 配置响应拦截器  响应拦截器的作用:在获取到请求的数据之后,处理一些业务
instance.interceptors.response.use(function (response) {
  // 2xx 范围内的状态码都会触发该函数。
  // 对响应数据做点什么

  // 因为在发送请求的过程中,可能会出现很多错误。所以在项目开发中,决定在响应拦截器中统一处理在发送请求过程中出现的问题。
  // 首先发送请求,调用封装好的接口函数
  // 获取数据成功时执行的代码

  // 由于每次发送请求,都有可能或出现错误,因此需要统一处理错误信息
  // 获取到的数据先通过响应拦截器处理,在到达浏览器


  // 因此在响应拦截器中统一处理错误信息
  const res = response.data
  console.log(res)
  if (res.status !== 200) {
    // 当res.status!==200时,则说明返回的数据出现了错误
    // 当返回的数据出现错误时,需要先弹出错误信息,在抛出错误
    Toast(res.message)
    return Promise.reject(res.message)
  } else {
    //关闭loading效果
    Toast.clear()
  }
  return res
}, function (error) {
  // 超出 2xx 范围的状态码都会触发该函数。
  // 对响应错误做点什么
  // 获取数据失败时执行的代码
  return Promise.reject(error)
})
// 导出配置好的axios实例
export default instance

七、封装接口方法

src/api文件夹专门用于存放封装好的接口方法

eg:

image.png

引入二次封装好的axios实例request

八、vue2中实现发送请求

export default {
  name:'',
  data(){
    return {
      数据名1''
      数据名2''
      ...
     }
   },
   methods:{
     方法名一(){}
    ...
   }
methods:{
   async getPicCode () {
      const res = await getPicCode()
      this.picUrl = res.data.base64 // 存储地址
      this.picKey = res.data.key
      this.$toast('获取图形验证码成功')
    },
async created(){
   //在vue2中一般是在created声明周期函数中调用封装好的函数来调用接口,发送请求,获取数据
   
   this.getPicCode()
 }

九、实现项目中的登录功能

<template>
  <div>
    <van-nav-bar title="会员登录" left-arrow @click-left="$router.go(-1)" />
    <div class="container">
      <div class="title">
        <h3>手机号登录</h3>
        <p>未注册的手机号登录后将自动注册</p>
      </div>

      <div class="form">
        <div class="form-item">
          <input class="inp" v-model="mobile" maxlength="11" placeholder="请输入手机号码" type="text">
        </div>
        <div class="form-item">
          <input v-model="picCode" class="inp" maxlength="5" placeholder="请输入图形验证码" type="text">
          <img v-if="picUrl" :src="picUrl" alt="" @click="getPicCode()">
        </div>
        <div class="form-item">
          <input class="inp"  v-model="msgCode" placeholder="请输入短信验证码" type="text">
          <button @click="getCode()">{{ second===totalSecond?'获取短信验证码':second+'秒后重新发送' }}</button>
        </div>
      </div>

      <div class="login-btn" @click="login()">登录</div>
    </div>
  </div>
</template>

<script>
// 在登录页组件中,需要向服务器发送请求。因此需要引入二次封装后的axios实例

import { getPicCode, getMsgCode, codeLogin } from '@/api/login'

export default {
  name: 'LoginIndex',
  data () {
    return {
      picCode: '', // 这里是用户输入的图形验证码
      picKey: '', // 这里是获取到的图片的唯一标识
      picUrl: '', // 这里是用户获取到的图片的路径
      totalSecond: 60, // 总倒计时
      second: 60, // 当前倒计时
      timer: null, // 定时器
      mobile: '',
      msgCode: ''
    }
  },
  methods: {
    async getPicCode () {
      const res = await getPicCode()
      this.picUrl = res.data.base64 // 存储地址
      this.picKey = res.data.key
      this.$toast('获取图形验证码成功')
    },
    async getCode () {
      if (!this.validFn()) {
        return
      }
      if (!this.timer && this.second === this.totalSecond) {
        await getMsgCode(this.picCode, this.picKey, this.mobile)
        this.$toast('发送成功,请注意查收')
        this.timer = setInterval(() => {
          this.second--
          if (this.second <= 0) {
            this.timer = null
            this.second = this.totalSecond
            clearInterval(this.timer)
          }
        }, 1000)
      }
    },
    
    //验证表单中输入的内容是否合格
    validFn () {
      const reg1 = /^1[3-9][0-9]{9}$/
      if (!reg1.test(this.mobile)) {
        this.$toast('请输入正确的手机号')
        return false
      }
      const reg2 = /^\w{4}$/
      if (!reg2.test(this.picCode)) {
        this.$toast('请输入正确的图形验证码')
        return false
      }
      return true
    },
    async login () {
      if (!this.validFn()) {
        return false
      }
      const reg3 = /^[0-9]{6}$/
      if (!reg3.test(this.msgCode)) {
        this.$toast('请输入正确的手机验证码')
        return false
      }
      const res = await codeLogin(this.mobile, this.msgCode)
      this.$store.commit('user/setUserInfo', res.data)
      this.$toast('登陆成功')
      // 先进行判断是否有参数地址传来
      // 若有,则说明是从其他页面拦截而来,登陆成功后,需要回跳到原页面
      // 否则,就直接跳转回首页
      // 判断有无回跳地址
      const url = this.$route.query.backUrl || '/'
      this.$router.replace(url)
    }
  },
  async created () {
    this.getPicCode()
  },
  destroyed () {
    clearInterval(this.timer)
  }
}
</script>

<style lang="less" scoped>
.container {
  padding: 49px 29px;

  .title {
    margin-bottom: 20px;
    h3 {
      font-size: 26px;
      font-weight: normal;
    }
    p {
      line-height: 40px;
      font-size: 14px;
      color: #b8b8b8;
    }
  }

  .form-item {
    border-bottom: 1px solid #f3f1f2;
    padding: 8px;
    margin-bottom: 14px;
    display: flex;
    align-items: center;
    .inp {
      display: block;
      border: none;
      outline: none;
      height: 32px;
      font-size: 14px;
      flex: 1;
    }
    img {
      width: 94px;
      height: 31px;
    }
    button {
      height: 31px;
      border: none;
      font-size: 13px;
      color: #cea26a;
      background-color: transparent;
      padding-right: 9px;
    }
  }

  .login-btn {
    width: 100%;
    height: 42px;
    margin-top: 39px;
    background: linear-gradient(90deg,#ecb53c,#ff9211);
    color: #fff;
    border-radius: 39px;
    box-shadow: 0 10px 20px 0 rgba(0,0,0,.1);
    letter-spacing: 2px;
    display: flex;
    justify-content: center;
    align-items: center;
  }
}
</style>

十、实现全局前置导航守卫

如果用户当前处于未登录状态,去访问网站中的一些页面时,是无法访问的,此时需要做出拦截处理,此时将强制跳转到“登录页面”。

在router/index.js中进行路由拦截处理

image.png

十一、封装本地存储方法,封装成工具方法存放在utils文件夹下

utils/storage.js

eg:

image.png