14.状态管理器

 

Vuex 是什么? | Vuex

Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式

它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。

状态传值

  • 父组件给子组件传值

  • 子组件给父组件传值

  • 非父子组件(兄弟组件)传值

  • 子组件直接使用父组件的数据和方法

  • 父组件直接使用子组件的数据和方法

  • 跨组件传值

  • 状态管理器

  • 多个视图依赖于同一状态。

  • 来自不同视图的行为需要变更同一状态。

 

14.0 构建首页的头部组件

<template>
  <div class="box">
    <header class="header">
      <ul>
        <li>北京</li>
        <li>搜索框</li>
        <li v-if="true">登录</li>
        <li v-else>我的</li>
      </ul>
    </header>
    <div class="content" ref="content">
    ....
    </div>  
  </div>
</template>
<script>
.....
</script>
​
<style lang='stylus' scoped>
// scoped 代表该样式只在当前组件是有效的,不会影响其他组件的样式
....
.header
  ul
    height 100%
    display flex
    li
      flex 1
</style>
​

14.1 第一版登录状态

异步操作在组件,组件中直接提交mutations

定义状态管理器

// src/store/index.js
import Vue from 'vue'
import Vuex from 'vuex'
​
Vue.use(Vuex)
​
export default new Vuex.Store({
  state: {
    // ? 为什么要从本地存储设置 状态管理器的初始值
    // 因为每次刷新页面都会使 状态管理器 重置
    loginState: localStorage.getItem('loginState') === 'true'
  },
  mutations: {
    changeLoginState (state, payload) { // state 所有的状态  payload 参数
      state.loginState = payload
    }
  },
  actions: {
  },
  modules: {
  }
})
​

登录时保存状态

// 修改状态管理器中的代码 --- 代码不要看图,只看图中的代码位置,实际代码替换如下
this.$store.commit('changeLoginState', true)

底部组件获取登录状态

<template>
  <!-- div.container>div.box>header.header+div.content -->
  <div class="container">
    <router-view></router-view>
    <footer class="footer" v-if="!$route.meta.hidden">
      <ul>
        <router-link to="/home" tag="li">
          <span class="iconfont icon-shouye"></span>
          <p>首页</p>
        </router-link>
        <router-link to="/kind" tag="li">
          <span class="iconfont icon-fenlei"></span>
          <p>分类</p>
        </router-link>
        <router-link to="/cart" tag="li">
          <span class="iconfont icon-gouwuche"></span>
          <p>购物车</p>
        </router-link>
        <router-link v-if="loginState" to="/user" tag="li">
          <span class="iconfont icon-My"></span>
          <p>我的</p>
        </router-link>
        <router-link v-else to="/login" tag="li">
          <span class="iconfont icon-My"></span>
          <p>未登录</p>
        </router-link>
      </ul>
    </footer>
    <div class="tip">请将屏幕竖向浏览</div>
  </div>
</template>
<script>
export default {
  // 使用计算属性获取状态管理器中的状态
  computed: {
    loginState () {
      return this.$store.state.loginState
    }
  }
}
</script>
​
<style lang="stylus">
....
</style>
​
// src/views/home/index.vue
<template>
  <div class="box">
    <header class="header">
      <ul>
        <li>北京</li>
        <li>搜索框</li>
        <router-link tag="li" to="/user" v-if="loginState">我的</router-link>
        <router-link tag="li" to="/login" v-else>登录</router-link>
      </ul>
    </header>
    <div class="content" ref="content">
    ....
    </div>  
  </div>
</template>
<script>
export default {
     computed: {
    loginState () {
      return this.$store.state.loginState
    }
  },
  ....
}
</script>
​
<style lang='stylus' scoped>
// scoped 代表该样式只在当前组件是有效的,不会影响其他组件的样式
....
.header
  ul
    height 100%
    display flex
    li
      flex 1
</style>
​
​

退出登录

<template>
  <div class="box">
    <header class="header">user header</header>
    <div class="content">
​
      <button v-if="loginState" @click="logout">退出</button>
      <button v-else @click="login"> 登录</button>
    </div>
  </div>
</template>
<script>
export default {
  computed: {
    loginState () {
      return this.$store.state.loginState
    }
  },
  methods: {
    logout () {
      localStorage.removeItem('token')
      localStorage.removeItem('userid')
      localStorage.removeItem('loginState')
      this.$store.commit('changeLoginState', false) // 二次渲染的关键
      this.$router.push('/login')
    },
    login () {
      this.$router.push('/login')
    }
  }
}
</script>
​

总结:登录时,异步的登录操作放到了 登录组件中

14.2 第二版登录状态

登录时,异步的登录操作放到了 状态管理器中,通过组件触发状态管理器执行 异步操作

登录的代码拷贝到了 store/index.js中的actions内

import Vue from 'vue'
import Vuex from 'vuex'
import router from '../router'
import { Dialog, Toast } from 'vant'
import { login } from './../api/user'
Vue.use(Vuex)
​
export default new Vuex.Store({
  state: {
    // ? 为什么要从本地存储设置 状态管理器的初始值
    // 因为每次刷新页面都会使 状态管理器 重置
    loginState: localStorage.getItem('loginState') === 'true'
  },
  mutations: {
    changeLoginState (state, payload) { // state 所有的状态  payload 参数
      state.loginState = payload
    }
  },
  actions: {
    // context 上下文对象 ---  store
    // params 参数
    loginAction (context, parmas) {
      login({
        loginname: parmas.loginname,
        password: parmas.password
      }).then(res => {
        if (res.data.code === '10010') {
          // 账户不存在,提醒用户是否要立即注册
          Dialog.confirm({
            message: '该用户还未注册,是否立即注册',
            confirmButtonText: '立即注册',
            confirmButtonColor: '#ff6666',
            cancelButtonText: '取消',
            cancelButtonColor: '#999'
          })
            .then(() => {
              router.push('/register/step1')
            })
            .catch(() => {
              // on cancel
            })
        } else if (res.data.code === '10011') {
          // 提醒用户密码错误, 视情况而定是否需要清空密码输入框 this.password = ''
          Toast('密码错误')
        } else {
          // 登录成功
          Toast('登录成功')
          localStorage.setItem('userid', res.data.data.userid) // 知道是谁
          localStorage.setItem('token', res.data.data.token) // 后端验证用户的登录状态
          localStorage.setItem('loginState', true) // 前端自检登录状态
​
          // 使用状态管理器修改状态
          context.commit('changeLoginState', true) // 修改状态管理器中的loginState的值为true
          // 返回上一页
          router.back()
        }
      })
    }
  },
  modules: {
  }
})
​
<template>
  <div class="box">
    <header class="header">
      <van-nav-bar
        title="嗨购登录"
        left-arrow
        @click-left="$router.back()"
      ></van-nav-bar>
    </header>
    <div class="content">
      <div class="form" v-if="type === '1'">
        <van-field v-model="loginname" placeholder="用户名/手机号/邮箱" clearable/>
        <van-field v-model="password" type="password" placeholder="请输入密码" clearable/>
        <p class="passwordTip" v-if="passwordTipFlag">输入至少6位,包含至少一个大写字母,1个小写字母,1个数字</p>
        <div class="my-button">
          <van-button color="#ff6666" :disabled="adminameFlag"  block round @click="adminameLoginFn">登录</van-button>
        </div>
      </div>
      <div class="form" v-if="type === '0'">
        <van-field v-model="tel" type="tel" placeholder="手机号码" clearable/>
        <van-field
          v-model="telcode"
          center
          clearable
          placeholder="请输入短信验证码"
        >
          <template #button>
            <van-button size="small" >发送短信验证码</van-button>
          </template>
        </van-field>
        <div class="my-button">
          <van-button color="#ff6666"  block round>登录</van-button>
        </div>
      </div>
      <!-- 登录方式以及 注册提示 -->
      <ul class="more">
        <li @click="changeType">
          <span v-if="type === '1'">短信验证码登录</span>
          <span v-else>账号密码登录</span>
        </li>
        <router-link to="/register/step1" tag="li">
          手机快速注册
        </router-link>
      </ul>
      <div class="my-divider">
        <van-divider>其他登录方式</van-divider>
      </div>
      <div class="my-login-type">
        <van-row type="flex" justify="center">
          <van-col span="6">
            <van-image :src="qq" width="48" height="48"/>
            <p>QQ</p>
          </van-col>
          <van-col span="6">
            <van-image :src="wx" width="48" height="48"/>
            <p>微信</p>
          </van-col>
          <van-col span="6">
            <van-image :src="apple" width="48" height="48"/>
            <p>苹果</p>
          </van-col>
        </van-row>
      </div>
    </div>
  </div>
</template>
<script>
import Vue from 'vue'
import { NavBar, Field, Button, Divider, Col, Row, Image as VanImage } from 'vant'
​
Vue.use(NavBar)
Vue.use(Field)
Vue.use(Button)
Vue.use(Divider)
Vue.use(Col)
Vue.use(Row)
Vue.use(VanImage)
export default {
  data () {
    return {
      qq: '',
      wx: '',
      apple: '',
      tel: '',
      password: '',
      loginname: '',
      telcode: '',
      type: '1' // 1表示账户名密码 0表示手机验证码
    }
  },
  computed: {
    adminameFlag () {
      if (this.loginname !== '' && /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    },
    passwordTipFlag () {
      if (this.password === '' || /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    }
  },
  methods: {
    changeType () {
      this.type = this.type === '1' ? '0' : '1'
    },
    adminameLoginFn () {
      console.log('1111111')
      // 异步操作
      this.$store.dispatch('loginAction', {
        loginname: this.loginname,
        password: this.password
      })
    }
  }
}
</script>
​
<style lang="stylus">
.container .box .content
  padding 30px 15px
  background-color #ffffff
​
.form
  .my-button
    margin-top 30px
.more
  margin-top 20px
  display flex
  li
    flex 1
    &:nth-child(1)
      text-align left
    &:nth-child(2)
      text-align right
.my-divider
  margin-top 80px
.passwordTip
  color #f66
  font-size 12px
</style>
​

异步操作在组件,组件中通过

this.$store.commit()改变数据

异步操作在状态管理器,组件中通过

this.$store.dispatch() 触发,通过其内部的 context.commit() 改变数据

组件中通过this.$store.state获取状态,可以配合计算属性使用

14.3 使用辅助函数mapState获取状态

之前的案例是 通过 计算属性 计算得到 状态,通过 this.$store.state获取得到

mapState的是一个对象或者数组

// App.vue

<template>
  <!-- div.container>div.box>header.header+div.content -->
  <div class="container">
    <!-- <div class="box">
      <header class="header">header</header>
      <div class="content">content</div>
    </div> -->
    <router-view></router-view>
    <footer class="footer" v-if="!$route.meta.hidden">
      <ul>
        <router-link to="/home" tag="li">
          <span class="iconfont icon-shouye"></span>
          <p>首页</p>
        </router-link>
        <router-link to="/kind" tag="li">
          <span class="iconfont icon-fenlei"></span>
          <p>分类</p>
        </router-link>
        <router-link to="/cart" tag="li">
          <span class="iconfont icon-gouwuche"></span>
          <p>购物车</p>
        </router-link>
        <router-link v-if="loginState" to="/user" tag="li">
          <span class="iconfont icon-My"></span>
          <p>我的</p>
        </router-link>
        <router-link v-else to="/login" tag="li">
          <span class="iconfont icon-My"></span>
          <p>未登录</p>
        </router-link>
      </ul>
    </footer>
    <div class="tip">请将屏幕竖向浏览</div>
  </div>
</template>
<script>
import { mapState } from 'vuex'
export default {
  // 使用计算属性获取状态管理器中的状态
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.loginState
    })
  }
}
</script>
​
<style lang="stylus">
...
</style>
​

// src/views/home/index.vue

<template>
  <div class="box">
    ....
  </div>
</template>
<script>
...
import { mapState } from 'vuex'
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.loginState
    })
  },
  ....
}
</script>
​
<style lang='stylus' scoped>
/...
</style>
​

// src/views/user/index.vue

<template>
  <div class="box">
    <header class="header">user header</header>
    <div class="content">
​
      <button v-if="loginState" @click="logout">退出</button>
      <button v-else @click="login"> 登录</button>
    </div>
  </div>
</template>
<script>
import { mapState } from 'vuex'
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.loginState
    })
  },
  methods: {
    logout () {
      localStorage.removeItem('token')
      localStorage.removeItem('userid')
      localStorage.removeItem('loginState')
      this.$store.commit('changeLoginState', false) // 二次渲染的关键
      this.$router.push('/login')
    },
    login () {
      this.$router.push('/login')
    }
  }
}
</script>
​

14.4 使用辅助函数mapMutations 改变状态

这个辅助函数使用场景: 组件中去改变 mutation

// views/user/index.vue

<template>
  <div class="box">
    <header class="header">user header</header>
    <div class="content">
​
      <button v-if="loginState" @click="logout">退出</button>
      <button v-else @click="login"> 登录</button>
    </div>
  </div>
</template>
<script>
import { mapState, mapMutations } from 'vuex'
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.loginState
    })
  },
  methods: {
    ...mapMutations({
      changeLoginState: 'changeLoginState' // key随意 - 事件,value即为mutation中的值,一般情况下二者相同
    }),
    logout () {
      localStorage.removeItem('token')
      localStorage.removeItem('userid')
      localStorage.removeItem('loginState')
      // this.$store.commit('changeLoginState', false) // 二次渲染的关键
      this.changeLoginState(false) // 事件中通过 this.key事件(参数)
      this.$router.push('/login')
    },
    login () {
      this.$router.push('/login')
    }
  }
}
</script>
​

14.5 使用辅助函数 mapActions 触发 acitons

组件触发 action,aciton提交mutation

// views/login/index.vue

<template>
  <div class="box">
    <header class="header">
      <van-nav-bar
        title="嗨购登录"
        left-arrow
        @click-left="$router.back()"
      ></van-nav-bar>
    </header>
    <div class="content">
      <div class="form" v-if="type === '1'">
        <van-field v-model="loginname" placeholder="用户名/手机号/邮箱" clearable/>
        <van-field v-model="password" type="password" placeholder="请输入密码" clearable/>
        <p class="passwordTip" v-if="passwordTipFlag">输入至少6位,包含至少一个大写字母,1个小写字母,1个数字</p>
        <div class="my-button">
          <van-button color="#ff6666" :disabled="adminameFlag"  block round @click="adminameLoginFn">登录</van-button>
        </div>
      </div>
      <div class="form" v-if="type === '0'">
        <van-field v-model="tel" type="tel" placeholder="手机号码" clearable/>
        <van-field
          v-model="telcode"
          center
          clearable
          placeholder="请输入短信验证码"
        >
          <template #button>
            <van-button size="small" >发送短信验证码</van-button>
          </template>
        </van-field>
        <div class="my-button">
          <van-button color="#ff6666"  block round>登录</van-button>
        </div>
      </div>
      <!-- 登录方式以及 注册提示 -->
      <ul class="more">
        <li @click="changeType">
          <span v-if="type === '1'">短信验证码登录</span>
          <span v-else>账号密码登录</span>
        </li>
        <router-link to="/register/step1" tag="li">
          手机快速注册
        </router-link>
      </ul>
      <div class="my-divider">
        <van-divider>其他登录方式</van-divider>
      </div>
      <div class="my-login-type">
        <van-row type="flex" justify="center">
          <van-col span="6">
            <van-image :src="qq" width="48" height="48"/>
            <p>QQ</p>
          </van-col>
          <van-col span="6">
            <van-image :src="wx" width="48" height="48"/>
            <p>微信</p>
          </van-col>
          <van-col span="6">
            <van-image :src="apple" width="48" height="48"/>
            <p>苹果</p>
          </van-col>
        </van-row>
      </div>
    </div>
  </div>
</template>
<script>
import Vue from 'vue'
import { NavBar, Field, Button, Divider, Col, Row, Image as VanImage } from 'vant'
import { mapActions } from 'vuex'
Vue.use(NavBar)
Vue.use(Field)
Vue.use(Button)
Vue.use(Divider)
Vue.use(Col)
Vue.use(Row)
Vue.use(VanImage)
export default {
  data () {
    return {
      qq: '',
      wx: '',
      apple: '',
      tel: '',
      password: '',
      loginname: '',
      telcode: '',
      type: '1' // 1表示账户名密码 0表示手机验证码
    }
  },
  computed: {
    adminameFlag () {
      if (this.loginname !== '' && /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    },
    passwordTipFlag () {
      if (this.password === '' || /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    }
  },
  methods: {
    ...mapActions({
      loginAction: 'loginAction'
    }),
    changeType () {
      this.type = this.type === '1' ? '0' : '1'
    },
    adminameLoginFn () {
      console.log('1111111')
      // 异步操作
      // this.$store.dispatch('loginAction', {
      //   loginname: this.loginname,
      //   password: this.password
      // })
      this.loginAction({
        loginname: this.loginname,
        password: this.password
      })
    }
  }
}
</script>
​
<style lang="stylus">
.container .box .content
  padding 30px 15px
  background-color #ffffff
​
.form
  .my-button
    margin-top 30px
.more
  margin-top 20px
  display flex
  li
    flex 1
    &:nth-child(1)
      text-align left
    &:nth-child(2)
      text-align right
.my-divider
  margin-top 80px
.passwordTip
  color #f66
  font-size 12px
</style>
​

14.6 getters以及ma pGetters

// src/store/index.js

import Vue from 'vue'
import Vuex from 'vuex'
import router from '../router'
import { Dialog, Toast } from 'vant'
import { login } from './../api/user'
Vue.use(Vuex)
​
export default new Vuex.Store({
  state: {
    list: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 1, 2, 3, 4, 5],
    // ? 为什么要从本地存储设置 状态管理器的初始值
    // 因为每次刷新页面都会使 状态管理器 重置
    loginState: localStorage.getItem('loginState') === 'true'
  },
  getters: { // 可以看作是state的计算属性,类似于computed
    len: state => state.list.length
  },
  mutations: {
    changeLoginState (state, payload) { // state 所有的状态  payload 参数
      state.loginState = payload
    }
  },
  actions: {
    // context 上下文对象 ---  store
    // params 参数
    loginAction (context, parmas) {
      login({
        loginname: parmas.loginname,
        password: parmas.password
      }).then(res => {
        if (res.data.code === '10010') {
          // 账户不存在,提醒用户是否要立即注册
          Dialog.confirm({
            message: '该用户还未注册,是否立即注册',
            confirmButtonText: '立即注册',
            confirmButtonColor: '#ff6666',
            cancelButtonText: '取消',
            cancelButtonColor: '#999'
          })
            .then(() => {
              router.push('/register/step1')
            })
            .catch(() => {
              // on cancel
            })
        } else if (res.data.code === '10011') {
          // 提醒用户密码错误, 视情况而定是否需要清空密码输入框 this.password = ''
          Toast('密码错误')
        } else {
          // 登录成功
          Toast('登录成功')
          localStorage.setItem('userid', res.data.data.userid) // 知道是谁
          localStorage.setItem('token', res.data.data.token) // 后端验证用户的登录状态
          localStorage.setItem('loginState', true) // 前端自检登录状态
​
          // 使用状态管理器修改状态
          context.commit('changeLoginState', true) // 修改状态管理器中的loginState的值为true
          // 返回上一页
          router.back()
        }
      })
    }
  },
  modules: {
  }
})
​

// src/views/user/index.vue

<template>
  <div class="box">
    <header class="header">user header</header>
    <div class="content">
​
      <button v-if="loginState" @click="logout">退出</button>
      <button v-else @click="login"> 登录</button>
      {{ listLen }}
    </div>
  </div>
</template>
<script>
import { mapState, mapMutations, mapGetters } from 'vuex'
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.loginState
    }),
    // listLen () {
    //   return this.$store.getters.len
    // }
    ...mapGetters({
      listLen: 'len'
    })
  },
  methods: {
    ...mapMutations({
      changeLoginState: 'changeLoginState' // key随意 - 事件,value即为mutation中的值,一般情况下二者相同
    }),
    logout () {
      localStorage.removeItem('token')
      localStorage.removeItem('userid')
      localStorage.removeItem('loginState')
      // this.$store.commit('changeLoginState', false) // 二次渲染的关键
      this.changeLoginState(false) // 事件中通过 this.key事件(参数)
      this.$router.push('/login')
    },
    login () {
      this.$router.push('/login')
    }
  }
}
</script>
​

15 分模块的状态管理器

便于团队的开发协作

vue的实例中 可以 使用 computed 计算属性

vuex 中可以 使用 getters 计算属性

15.1 构建用户的相关的模块

// store/modules/user.js
import router from '../../router'
import { Dialog, Toast } from 'vant'
import { login } from './../../api/user'
export default {
  namespaced: true, // 命名空间,作用很大,用来标识是哪一个模块
  state: {
    list: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 1, 2, 3, 4, 5],
    // ? 为什么要从本地存储设置 状态管理器的初始值
    // 因为每次刷新页面都会使 状态管理器 重置
    loginState: localStorage.getItem('loginState') === 'true'
  },
​
  mutations: {
    changeLoginState (state, payload) { // state 所有的状态  payload 参数
      state.loginState = payload
    }
  },
  actions: {
    // context 上下文对象 ---  store
    // params 参数
    loginAction (context, parmas) {
      login({
        loginname: parmas.loginname,
        password: parmas.password
      }).then(res => {
        if (res.data.code === '10010') {
          // 账户不存在,提醒用户是否要立即注册
          Dialog.confirm({
            message: '该用户还未注册,是否立即注册',
            confirmButtonText: '立即注册',
            confirmButtonColor: '#ff6666',
            cancelButtonText: '取消',
            cancelButtonColor: '#999'
          })
            .then(() => {
              router.push('/register/step1')
            })
            .catch(() => {
              // on cancel
            })
        } else if (res.data.code === '10011') {
          // 提醒用户密码错误, 视情况而定是否需要清空密码输入框 this.password = ''
          Toast('密码错误')
        } else {
          // 登录成功
          Toast('登录成功')
          localStorage.setItem('userid', res.data.data.userid) // 知道是谁
          localStorage.setItem('token', res.data.data.token) // 后端验证用户的登录状态
          localStorage.setItem('loginState', true) // 前端自检登录状态
​
          // 使用状态管理器修改状态
          context.commit('changeLoginState', true) // 修改状态管理器中的loginState的值为true
          // 返回上一页
          router.back()
        }
      })
    }
  }
}
​
​
​
// store/index.js
import Vue from 'vue'
import Vuex from 'vuex'
import user from './modules/user'
Vue.use(Vuex)
​
export default new Vuex.Store({
  getters: { // 可以看作是state的计算属性,类似于computed
    len: state => state.user.list.length
  },
  modules: { // 整合模块
    user
  }
})
​

Src/App.vue

<template>
  <!-- div.container>div.box>header.header+div.content -->
  <div class="container">
    <!-- <div class="box">
      <header class="header">header</header>
      <div class="content">content</div>
    </div> -->
    <router-view></router-view>
    <footer class="footer" v-if="!$route.meta.hidden">
      <ul>
        <router-link to="/home" tag="li">
          <span class="iconfont icon-shouye"></span>
          <p>首页</p>
        </router-link>
        <router-link to="/kind" tag="li">
          <span class="iconfont icon-fenlei"></span>
          <p>分类</p>
        </router-link>
        <router-link to="/cart" tag="li">
          <span class="iconfont icon-gouwuche"></span>
          <p>购物车</p>
        </router-link>
        <router-link v-if="loginState" to="/user" tag="li">
          <span class="iconfont icon-My"></span>
          <p>我的</p>
        </router-link>
        <router-link v-else to="/login" tag="li">
          <span class="iconfont icon-My"></span>
          <p>未登录</p>
        </router-link>
      </ul>
    </footer>
    <div class="tip">请将屏幕竖向浏览</div>
  </div>
</template>
<script>
import { mapState } from 'vuex'
export default {
  // 使用计算属性获取状态管理器中的状态
  computed: {
    // loginState () {
    //   return this.$store.state.user.loginState
    // }
    ...mapState({
      loginState: state => state.user.loginState
    })
  }
}
</script>
​
<style lang="stylus">
...
</style>
​

// src/views/home/index.vue

<template>
  <div class="box">
   ....
</template>
<script>
import Vue from 'vue'
import { Swipe, SwipeItem, Grid, GridItem, Image as VanImage, CountDown, Icon, List, PullRefresh } from 'vant'
import { getBannerList, getSeckillList, getProList } from '@/api/home' // @ 代表的就是src的目录
import ProList from '@/components/ProList'
import { mapState } from 'vuex'
Vue.use(Swipe)
Vue.use(SwipeItem)
Vue.use(Grid)
Vue.use(GridItem)
Vue.use(VanImage)
Vue.use(CountDown)
Vue.use(Icon)
Vue.use(List)
Vue.use(PullRefresh)
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.loginState
    // }
    ...mapState({
      loginState: state => state.user.loginState
    })
  },
  ....
}
</script>
​
<style lang='stylus' scoped>
....
</style>
​

15.2 状态管理器中使用模块

15.3 登录时改变状态

Src/views/login/index.vue

<template>
  <div class="box">
    <header class="header">
      <van-nav-bar
        title="嗨购登录"
        left-arrow
        @click-left="$router.back()"
      ></van-nav-bar>
    </header>
    <div class="content">
      <div class="form" v-if="type === '1'">
        <van-field v-model="loginname" placeholder="用户名/手机号/邮箱" clearable/>
        <van-field v-model="password" type="password" placeholder="请输入密码" clearable/>
        <p class="passwordTip" v-if="passwordTipFlag">输入至少6位,包含至少一个大写字母,1个小写字母,1个数字</p>
        <div class="my-button">
          <van-button color="#ff6666" :disabled="adminameFlag"  block round @click="adminameLoginFn">登录</van-button>
        </div>
      </div>
      <div class="form" v-if="type === '0'">
        <van-field v-model="tel" type="tel" placeholder="手机号码" clearable/>
        <van-field
          v-model="telcode"
          center
          clearable
          placeholder="请输入短信验证码"
        >
          <template #button>
            <van-button size="small" >发送短信验证码</van-button>
          </template>
        </van-field>
        <div class="my-button">
          <van-button color="#ff6666"  block round>登录</van-button>
        </div>
      </div>
      <!-- 登录方式以及 注册提示 -->
      <ul class="more">
        <li @click="changeType">
          <span v-if="type === '1'">短信验证码登录</span>
          <span v-else>账号密码登录</span>
        </li>
        <router-link to="/register/step1" tag="li">
          手机快速注册
        </router-link>
      </ul>
      <div class="my-divider">
        <van-divider>其他登录方式</van-divider>
      </div>
      <div class="my-login-type">
        <van-row type="flex" justify="center">
          <van-col span="6">
            <van-image :src="qq" width="48" height="48"/>
            <p>QQ</p>
          </van-col>
          <van-col span="6">
            <van-image :src="wx" width="48" height="48"/>
            <p>微信</p>
          </van-col>
          <van-col span="6">
            <van-image :src="apple" width="48" height="48"/>
            <p>苹果</p>
          </van-col>
        </van-row>
      </div>
    </div>
  </div>
</template>
<script>
import Vue from 'vue'
import { NavBar, Field, Button, Divider, Col, Row, Image as VanImage } from 'vant'
import { mapActions } from 'vuex'
Vue.use(NavBar)
Vue.use(Field)
Vue.use(Button)
Vue.use(Divider)
Vue.use(Col)
Vue.use(Row)
Vue.use(VanImage)
export default {
  data () {
    return {
      qq: '',
      wx: '',
      apple: '',
      tel: '',
      password: '',
      loginname: '',
      telcode: '',
      type: '1' // 1表示账户名密码 0表示手机验证码
    }
  },
  computed: {
    adminameFlag () {
      if (this.loginname !== '' && /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    },
    passwordTipFlag () {
      if (this.password === '' || /^\S*(?=\S{6,})(?=\S*\d)(?=\S*[A-Z])(?=\S*[a-z])\S*$/.test(this.password)) {
        return false
      } else {
        return true
      }
    }
  },
  methods: {
    ...mapActions({
      loginAction: 'user/loginAction'
    }),
    changeType () {
      this.type = this.type === '1' ? '0' : '1'
    },
    adminameLoginFn () {
      console.log('1111111')
      // 异步操作
      // this.$store.dispatch('user/loginAction', {
      //   loginname: this.loginname,
      //   password: this.password
      // })
      this.loginAction({
        loginname: this.loginname,
        password: this.password
      })
    }
  }
}
</script>
​
<style lang="stylus">
.container .box .content
  padding 30px 15px
  background-color #ffffff
​
.form
  .my-button
    margin-top 30px
.more
  margin-top 20px
  display flex
  li
    flex 1
    &:nth-child(1)
      text-align left
    &:nth-child(2)
      text-align right
.my-divider
  margin-top 80px
.passwordTip
  color #f66
  font-size 12px
</style>
​

15.4 退出时改变状态

s r c/views/user/idnex.vue

<template>
  <div class="box">
    <header class="header">user header</header>
    <div class="content">
​
      <button v-if="loginState" @click="logout">退出</button>
      <button v-else @click="login"> 登录</button>
      {{ listLen }}
    </div>
  </div>
</template>
<script>
import { mapState, mapMutations, mapGetters } from 'vuex'
export default {
  computed: {
    // loginState () {
    //   return this.$store.state.user.loginState
    // }
    ...mapState({
      loginState: state => state.user.loginState
    }),
    // listLen () {
    //   return this.$store.getters.len
    // }
    ...mapGetters({
      listLen: 'len'
    })
  },
  methods: {
    ...mapMutations({
      changeLoginState: 'user/changeLoginState' // key随意 - 事件,value即为mutation中的值,一般情况下二者相同
    }),
    logout () {
      localStorage.removeItem('token')
      localStorage.removeItem('userid')
      localStorage.removeItem('loginState')
      // this.$store.commit('user/changeLoginState', false) // 二次渲染的关键
      this.changeLoginState(false) // 事件中通过 this.key事件(参数)
      this.$router.push('/login')
    },
    login () {
      this.$router.push('/login')
    }
  }
}
</script>
​

16.使用状态管理器管理购物车的数据 - 作业

16.1 添加模块

s r c/store/cart.js

import { getCartList } from './../../api/cart'
export default {
  namespaced: true,
  state: {
    cartList: []
  },
  mutations: {
    changeCartList (state, data) {
      state.cartList = data
    }
  },
  actions: {
    getCartListAction (context, params) {
      return new Promise(resolve => {
        // 后续的操作交给组件
        getCartList(params).then(res => {
          resolve(res)
        })
      })
    }
  }
}
​

16.2 注册模块

// src/store/index.js

import Vue from 'vue'
import Vuex from 'vuex'
import user from './modules/user'
import cart from './modules/cart'
Vue.use(Vuex)
​
export default new Vuex.Store({
  state: {},
  mutations: {},
  actions: {},
  getters: { // 可以看作状状态管理器的计算属性
    showNum (state) { // 控制底部选项卡中数量显示还是不显示
      // state 这个参数其实就是 所有的状态
      return state.user.isLogin
    },
   totalNum (state) { // 判断很关键
      return state.cart.cartList && state.cart.cartList.reduce((sum, item) => {
        return item.flag ? sum + item.num : sum + 0
      }, 0)
    },
    totalPrice (state) {
      return state.cart.cartList && state.cart.cartList.reduce((sum, item) => {
        return item.flag ? sum + item.originprice * item.num : sum + 0
      }, 0) * 100
    }
  },
  modules: {
    user,
    cart
  }
})
​

16.3 组件使用状态

s r c/views/cart/index.vue

<template>
  <div class="box">
    <header class="header">
      <van-nav-bar
        :title="'购物车-' + totalNum"
        left-arrow
        @click-left="$router.back()"
      />
    </header>
    <div class="content">
      <div v-if="empty">
        <van-empty description="购物车空空如也">
          <router-link to="/kind">
            <van-button round type="danger" class="bottom-button">
              立即选购
            </van-button>
          </router-link>
        </van-empty>
      </div>
      <div v-else>
        <van-row v-for="item of cartList" :key="item.cartid">
          <van-col span="3" class="checkboxList">
            <van-checkbox @change="updateFlag(item.cartid, item.flag)" v-model="item.flag"></van-checkbox>
          </van-col>
          <van-col span="21">
            <van-swipe-cell>
              <van-card
                :price="item.originprice"
                :title="item.proname"
                :thumb="item.img1"
              >
                <template #num>
                  <van-stepper @change="updateNum(item.cartid, item.num)" v-model="item.num" max="5" theme="round" button-size="22" />
                </template>
              </van-card>
              <template #right>
                <van-button @click="removeItem(item.cartid)" square text="删除" type="danger" class="delete-button" />
              </template>
            </van-swipe-cell>
          </van-col>
        </van-row>
        <van-submit-bar :disabled="flag" :price="totalPrice" button-text="提交订单" @submit="onSubmit">
          <van-checkbox @click="updateAllFlag" v-model="checked">全选</van-checkbox>
        </van-submit-bar>
      </div>
    </div>
  </div>
</template>
<script>
import Vue from 'vue'
import { mapState, mapActions, mapMutations, mapGetters } from 'vuex'
import { NavBar, Stepper, Empty, Button, Col, SubmitBar, Row, Card, Checkbox, SwipeCell } from 'vant'
import { remove, updateCartNum, selectall, selectone } from './../../api/cart'
Vue.use(NavBar)
Vue.use(Empty)
Vue.use(Button)
Vue.use(Col)
Vue.use(Row)
Vue.use(Checkbox)
Vue.use(Card)
Vue.use(SwipeCell)
Vue.use(Stepper)
Vue.use(SubmitBar)
export default {
  data () {
    return {
      // cartList: [], // 8.删除购物车的初始化状态
      empty: true,
      checked: false
    }
  },
  computed: {
    // 1.获取状态管理器中的登录状态
    ...mapState({
      isLogin: state => state.user.isLogin,
      // 9.获取购物车中的状态
      cartList: state => state.cart.cartList
    }),
    // 10.获取vuex中的totalNum - 不添加命名空间
    ...mapGetters({
      // key 代表当前组件中需要的字段,value代表的是vuex中的 getters中设置的计算属性
      totalNum: 'totalNum',
      totalPrice: 'totalPrice'
    }),
    // totalNum () {
    //   return this.cartList.reduce((sum, item) => {
    //     return item.flag ? sum + item.num : sum + 0
    //   }, 0)
    // },
    // totalPrice () {
    //   return this.cartList.reduce((sum, item) => {
    //     return item.flag ? sum + item.originprice * item.num : sum + 0
    //   }, 0) * 100
    // },
    flag () { // 提交订单按钮可点不可点
      return this.totalNum <= 0
    }
  },
  methods: {
    // 3.生成获取购物车数据的方法
    ...mapActions({
      getCartListAction: 'cart/getCartListAction'
    }),
    // 5.生成修改购物车数据的方法
    ...mapMutations({
      changeCartList: 'cart/changeCartList'
    }),
    onSubmit () {
    },
    updateFlag (cartid, flag) {
      console.log(cartid, flag)
      selectone({
        cartid, flag
      }).then(res => {
        this.getCartListData()
      })
    },
    updateAllFlag () {
      selectall({
        userid: localStorage.getItem('userid'),
        type: this.checked
      }).then(res => {
        this.getCartListData()
      })
    },
    getCartListData () {
      // 4.调用vuex中的actions
      this.getCartListAction({ userid: localStorage.getItem('userid') }).then(res => {
        // 没有数据
        if (res.data.code === '10020') {
          this.empty = true
          // 6.修改vuex中的状态
          this.changeCartList([])
        } else {
          this.empty = false
          // 7.修改vuex中的状态
          this.changeCartList(res.data.data)
          // 判断全选是不是被选中
          // every 所有的条件都满足返回为true,只要有一个不满足返回为false
          this.checked = res.data.data.every(item => item.flag)
        }
      })
    },
    updateNum (cartid, num) {
      console.log(cartid, num)
      updateCartNum({
        cartid,
        num
      }).then(res => {
        this.getCartListData() // 重新获取最新的数据
      })
    },
    removeItem (cartid) {
      remove({
        cartid
      }).then(res => {
        this.getCartListData() // 重新获取最新的数据
      })
    }
  },
  mounted () {
    // 2.依据状态管理器中的登录状态进一步操作
    if (this.isLogin) {
      this.getCartListData()
    } else {
      this.$router.push('/login')
    }
  }
}
</script>
<style lang="stylus" scoped>
.checkboxList
  display flex
  height 1.04rem
  justify-content center
  align-items center
  background-color #fafafa
.delete-button {
  height: 100%;
}
</style>
​

16.4 添加底部的数量标签

s r c/app.vue

<template>
  <div class="container">
    <!-- <div class="box">
      <header class="header">header</header>
      <div class="content">content</div>
    </div> -->
    <router-view></router-view>
    <footer class="footer" v-if="!$route.meta.hidden">
      <ul>
        <!-- router-link 默认会渲染为 a 标签
        使用tag属性设置 转换的标签 -->
        <router-link to="/home" tag="li">
          <span class="iconfont icon-shouye"></span>
          <p>首页</p>
        </router-link>
        <router-link to="/kind" tag="li">
          <span class="iconfont icon-leimupinleifenleileibie"></span>
          <p>分类</p>
        </router-link>
        <router-link to="/cart" tag="li">
          <span class="iconfont icon-shopping-cart"></span>
          <p>购物车{{ isLogin ? totalNum : ''}}</p>
        </router-link>
        <router-link v-if="isLogin" to="/user" tag="li">
          <span class="iconfont icon-wode"></span>
          <p>我的</p>
        </router-link>
        <router-link v-else to="/login" tag="li">
          <span class="iconfont icon-wode"></span>
          <p>未登录</p>
        </router-link>
      </ul>
    </footer>
    <div class="tip">请将屏幕竖向浏览</div>
  </div>
</template>
<script>
import { mapState, mapGetters, mapActions, mapMutations } from 'vuex'
// 组件中获取状态管理器的状态可以使用计算属性
export default {
  methods: { // 2.页面的刷新,导致数据的重置,获取最新的数据
    ...mapActions({
      getCartListAction: 'cart/getCartListAction'
    }),
    ...mapMutations({
      changeCartList: 'cart/changeCartList'
    })
  },
  mounted () { // 3.获取实时的购物车中的数据
    this.getCartListAction({ userid: localStorage.getItem('userid') }).then(res => {
      if (res.data.code === '10020') {
        this.changeCartList([])
      } else {
        this.changeCartList(res.data.data)
      }
    })
  },
  computed: {
    // isLogin () {
    //   return this.$store.state.user.isLogin
    // }
    // ... 扩展运算符其实代表的就是合并对象
    // computed 是一个对象,还可以写其他的自定义的计算属性
    ...mapState({ // 使用mapState辅助函数获取状态管理器中的状态,结合 扩展运算符完成
      isLogin: state => state.user.isLogin
​
    }),
    // 1.获取购物车中的数量
    ...mapGetters({
      totalNum: 'totalNum'
    })
  }
}
</script>
​
<style lang="stylus">
。。。。
</style>
​

16.5 详情页面底部的数量

<template>
  <div class="box">
    <header class="header">
      <!-- 使用导航栏组件 -->
      <van-nav-bar
        :title="proname"
        left-arrow
        @click-left="$router.back()"
      >
        <template #right>
          <!-- 头部右侧的下拉选择菜单 -->
          <van-popover
            v-model="showPopover"
            theme="dark"
            trigger="click"
            :actions="actions"
            placement="bottom-end"
            @select = "moreEvent"
          >
            <template #reference>
              <van-icon name="more-o" size="18"/>
            </template>
          </van-popover>
​
        </template>
      </van-nav-bar>
    </header>
    <div class="content">
      <van-swipe class="detail-swipe" indicator-color="white">
        <van-swipe-item @click="previewImage(index)" v-for="(item, index) of banners" :key="index">
          <van-image
            fit="cover"
            :src="item"
          />
        </van-swipe-item>
      </van-swipe>
      <!-- 播放视频的图标 -->
      <van-icon v-if="flag" @click="openOverLay" name="play-circle-o" class="playBtn" size="36" color="#f66"/>
      <!-- 详情 -->
      <van-tag type="danger">{{ brand }}</van-tag> {{ category }}
      <h3>{{ proname }}</h3>
      <div>{{ originprice }}</div>
​
      <van-goods-action>
        <van-goods-action-icon icon="chat-o" text="客服" color="#ee0a24" />
        <!-- 2.展示数量 -->
        <van-goods-action-icon icon="cart-o" text="购物车" :badge=" isLogin && totalNum > 0 ? totalNum : ''" />
        <van-goods-action-icon icon="star" text="已收藏" color="#ff5000" />
        <van-goods-action-button type="warning" @click="addProToCart" text="加入购物车" />
        <van-goods-action-button type="danger" text="立即购买" />
      </van-goods-action>
      <!-- 分享面板 -->
      <van-share-sheet
        v-model="showShare"
        title="立即分享给好友"
        :options="options"
      />
    </div>
    <!-- 视频播放的遮罩层 -->
    <van-overlay :show="show" @click="closeOverLay" class="vdoBox">
      <video ref="vdo" width="100%" src="https://vod.300hu.com/4c1f7a6atransbjngwcloud1oss/1b147065379705072794210305/v.f20.mp4?dockingId=f782a297-b570-4701-bcea-9ae0c0749efe&storageSource=3" controls>
      </video>
    </van-overlay>
​
  </div>
</template>
<script>
import Vue from 'vue'
import { mapState, mapGetters, mapMutations, mapActions } from 'vuex'
import { getProDetail } from './../../api/detail'
import { addCart } from './../../api/cart'
import { Toast, ImagePreview, Overlay, ShareSheet, Popover, NavBar, Swipe, SwipeItem, Tag, GoodsAction, GoodsActionIcon, GoodsActionButton } from 'vant'
Vue.use(Toast)
Vue.use(ImagePreview)
Vue.use(Overlay)
Vue.use(ShareSheet)
Vue.use(Popover)
Vue.use(NavBar)
Vue.use(Swipe)
Vue.use(SwipeItem)
Vue.use(Tag)
Vue.use(GoodsAction)
Vue.use(GoodsActionIcon)
Vue.use(GoodsActionButton)
export default {
  computed: {
    //
    ...mapState({
      isLogin: state => state.user.isLogin
    }),
    // 1.获取vuex中的totalNum数量
    ...mapGetters({
      totalNum: 'totalNum'
    })
  },
  data () {
    return {
      proid: '',
      proname: '',
      originprice: 0,
      discount: 0,
      banners: [],
      sales: 0,
      category: '',
      brand: '',
      showPopover: false, // 控制右侧更多菜单的显示和隐藏
      actions: [{ text: '首页' }, { text: '分类' }, { text: '购物车' }, { text: '我的' }, { text: '分享' }],
      showShare: false, // 控制分享面板的显示和隐藏
      options: [ // 分享面板
        { name: '微信', icon: 'wechat' },
        { name: '微博', icon: 'weibo' },
        { name: '复制链接', icon: 'link' },
        { name: '分享海报', icon: 'poster' },
        { name: '二维码', icon: 'qrcode' }
      ],
      show: false, // 控制视频播放遮罩层的显示和隐藏
      flag: true // 控制播放按钮的显示和隐藏
    }
  },
  mounted () {
    this.proid = this.$route.params.proid
    getProDetail(this.proid).then(res => {
      console.log(res)
      // 当前接口的banenrs数据有问题,需要前端自行处理 ---- 不意味着以后别的项目也需要
      this.banners = res.data.data.banners[0].split(',')
      this.proname = res.data.data.proname
      this.originprice = res.data.data.originprice
      this.discount = res.data.data.discount
      this.sales = res.data.data.sales
      this.category = res.data.data.category
      this.brand = res.data.data.brand
    })
  },
  methods: {
    // 3.为了加入购物车之后更新数据
    ...mapActions({
      getCartListAction: 'cart/getCartListAction'
    }),
    ...mapMutations({
      changeCartList: 'cart/changeCartList'
    }),
    addProToCart () {
      // 前端校验登录  本地存储的都是字符串
      if (localStorage.getItem('isLogin') === 'true') { // 前端已经表示登录
        // 调用加入购物车接口
        addCart({ // 一定要把参数传递完整,否则会有小发现
          userid: localStorage.getItem('userid'),
          proid: this.proid,
          num: 1
        }).then(res => {
          Toast('加入购物车成功')
          // 4.更新购物车的数据
          this.getCartListAction({ userid: localStorage.getItem('userid') }).then(res => {
            if (res.data.code === '10020') {
              this.changeCartList([])
            } else {
              this.changeCartList(res.data.data)
            }
          })
        })
      } else {
        this.$router.push('/login')
      }
    },
    previewImage (index) {
      ImagePreview({ // 预览图片
        images: this.banners,
        startPosition: index
      })
    },
    openOverLay () { // 打开遮罩层,播放视频,隐藏按钮
      this.show = true
      this.flag = false
      this.$refs.vdo.play()
    },
    closeOverLay () { // 关闭遮罩层,暂停视频,显示按钮
      this.$refs.vdo.pause()
      this.flag = true
      this.show = false
    },
    moreEvent (action, index) { // 右上角更多菜单
      console.log(index, action)
      switch (index) {
        case 0:
          this.$router.push('/home')
          break
        case 1:
          this.$router.push('/kind')
          break
        case 2:
          this.$router.push('/cart')
          break
        case 3:
          this.$router.push('/user')
          break
        case 4:
          this.showShare = true // 控制分享面板的显示
          break
        default:
          break
      }
    }
  }
}
</script>
​
<style lang="stylus">
.detail-swipe
  height 2.6rem
.van-popover[data-popper-placement=bottom-end] .van-popover__arrow {
    right: 0px;
}
.playBtn
  position fixed
  top 2.4rem
  left 50%
  transform translateX(-50%)
  z-index 1000
.vdoBox
  display flex
  justify-content center
  align-items center
</style>
​

如果删掉本地存储的数据,当用户在首页后者详情页面重新刷新页面时,自动跳转到了 登录页面, ------ qq 微信

但是我们希望 首页,分类,详情,活动,即使用户不登录也可以查看 ---- 修改axios的配置

// http://www.axios-js.com/zh-cn/docs/
import axios from 'axios'
import router from './../router'
import store from './../store' // **************************重中之重****************************************
// 开发环境 yarn serve
// 生产环境 yarn build
// development  production
const isDev = process.env.NODE_ENV === 'development'
​
// 创建axios实例
// http://www.axios-js.com/zh-cn/docs/#axios-create-config
const request = axios.create({
  // baseUrl 实际请求的地址是 baseURL + 请求地址
  // http://121.89.205.189/api/banner/list  ===》 baseURL + '/banner/list'
  // baseURL: 'http://121.89.205.189/api',
  // 项目上线时无需修改baseURL地址 ---- 需要提前知道线上接口的地址
  // baseURL: isDev ? 'http://121.89.205.189/api' : 'http://121.89.205.189/api',
  // 如果使用了 代理 解决跨域问题
  baseURL: isDev ? '' : 'http://121.89.205.189/api',
  timeout: 6000 // 网络超时时间
})
​
// axios 的拦截器
// http://www.axios-js.com/zh-cn/docs/#%E6%8B%A6%E6%88%AA%E5%99%A8
// 请求拦截器 ---- 所有的数据在请求之前都会执行的代码 --- 显示loading的动画效果/给接口添加token
request.interceptors.request.use((config) => {
  // 在请求之前做些什么
  // 给所有的请求都传递token信息
  config.headers.common.token = localStorage.getItem('token') || ''
  return config
}, (error) => {
  return Promise.reject(error)
})
// 响应拦截器 ---- 在拿到接口的数据之前都会执行的代码 --- 隐藏loading的动画效果/验证token
request.interceptors.response.use((response) => {
  // 在响应时做些什么
  // token 没有传递  token 传递了只不过是错误的  token 传递了 失效了
  if (response.data.code === '10119') {
    console.log(111111111)
    // 引入路由器,跳转到登录页面 ****************************重中之重****************************************
    if (store.state.user.isLogin) {
      router.push('/login')
    }
  }
  return response
}, (error) => {
  return Promise.reject(error)
})
​
// 暴露自定义的axios
export default request
​

16.6 总结

  • 选择是否需要分模块(不管项目大小都可分 - 参照16点,但是一般小项目部分模块 - 参照 15点)

  • 分模块步骤

    • 1.先写模块(state, mutations, actions, namespaced)

    namespaced 是关键,使用 mutations 和 actions 时需要 指明模块的名称

    • 2.注册模块 (state, getters, mutations, actions, modules)

    modules是关键,它的key就是模块的名称

    getters 是计算属性

    state 可以是 所有组件都需要的模块的状态

    • 3.组件中使用辅助函数执行业务(mapState, mapGetters,mapMutations, mapActions)

Logo

前往低代码交流专区

更多推荐