VueDemo-14-16状态管理器
14.状态管理器Vuex 是什么? | VuexVuex 是一个专为 Vue.js 应用程序开发的状态管理模式。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。状态传值父组件给子组件传值子组件给父组件传值非父子组件(兄弟组件)传值子组件直接使用父组件的数据和方法父组件直接使用子组件的数据和方法跨组件传值状态管理器多个视图依赖于同一状态。来自不
14.状态管理器
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)
-
更多推荐
所有评论(0)