博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
vue 在移动端体验上的优化解决方案
阅读量:6649 次
发布时间:2019-06-25

本文共 7671 字,大约阅读时间需要 25 分钟。

去年年底自己搭了一个vue在移动端的开发框架,感觉体验不是很好。上个星期又要做移动端的项目了。所以我花了两天时间对之前的那个开发框架做了以下优化

  • 自定义vuex-plugins-loading
  • 路由切换动画 + keep alive 动态管理缓存组件
  • better-scroll与vue的最佳实践(better-scroll的vue化)
  • 自定义指令(vue-finger:包括点击,长按,双击,拖拽移动,多点触控,滑动,旋转,缩放手势)
  • 移动端适配方案
  • 如何分情况处理页面置顶
  • 路由懒加载

自定义 vuex-plugins-loading

如果每个页面在数据加载完成前,展示loading。你首先想到的是每个页面设置状态,show和hide状态。但是这样冗余代码太多了,而且自己写的都烦。我之前的react的项目中用到了dva,其中有dva-loading库,之前就有研究过,所以我就用他的思想,自己写一个vuex-loading。

实现思路:vuex中注册一个管理loading的module,通过绑定异步的action,将每个action的loading存在vuex中,这样我在每个页面只需要在vuex的store中拿相对应的action loading就能达到此目的

## 核心代码    store.subscribeAction({      before: action => {        if (shouldEffect(action, includes, excludes)) {          store.commit({ type: namespace + '/SHOW', payload: action.type })        }      },      after: action => {        if (shouldEffect(action, includes, excludes)) {          store.commit({ type: namespace + '/HIDE', payload: action.type })        }      }    })  }}复制代码

使用之前大家可以先了解一下

想安装此插件,,记得给个star哟
注意: 使用上述代码,vuex必需为3.1.0版本。因为subscribeAction在3.1.0才新增的

补充

我觉得还是写一下为什么推荐大家使用vuex-plugins-loading这种处理数据流的思路。

我也是从前年接触dva.js的时候,感觉到这种处理数据流的思路会让你的整个项目更清晰,更易迭代。别人接手你的整个框架理解起来比较轻松。

现在我就讲一下我这边是怎么根据dva.js的思想封装vuex的。其实很简单

## store 文件目录下的 index.jsimport Vue from 'vue'import Vuex from 'vuex'import home from './modules/home'import createLoadingPlugin from 'vuex-plugins-loading'Vue.use(Vuex)export default new Vuex.Store({  modules: {    home,  },  plugins: [createLoadingPlugin()]})复制代码
# home.jsimport { loadDataApi } from '../../service/api'const state = {  listData: [],}const mutations = {  getData (state, payload) {    state.listData = state.listData.concat(payload.res.data.data)    state.page = payload.res.data.page    state.pageNumber = Math.ceil(payload.res.data.total / payload.res.data.pageSize)  },  refreshData (state, payload) {    state.listData = payload.res.data.data  },}const getters = {}const actions = {  loadMore ({ commit, state }, data) {    return new Promise((resolve, reject) => {      setTimeout(() => {        const { page, type } = data        loadDataApi({ page }).then(res => {          if (res.code === 200) {            if (type === 'loadMore') {              commit({                type: 'getData',                res: res              })            } else {              commit({                type: 'refreshData',                res: res              })            }            resolve()          } else {            reject(res.error)            Toast(res.error)          }        })      }, 1000)    })  }}复制代码
## Home.vuecomputed: {    // Getting Vuex State from store/modules/home    ...mapState({      listData: state => state.home.listData,      loading: state => state['@@loading'].effects['home/loadMore']    }),  },  methods: {    ...mapActions('home', ['initData', 'plusPage', 'initPage']),    // onLoad 加载数据    onLoad () {      this.requestData('loadMore')    },    requestData (type) {      setTimeout(() => {        this.$store.dispatch('home/loadMore', {        }).then(() => {        })      }, 1000)    },    onRefresh () {      this.initPage().then(() => {        this.requestData('refresh')      })    },复制代码

1.先说vuex-plugins-loading内层是怎么实现的?

1.plugins: [createLoadingPlugin()],

2.loading: state => state['@@loading'].effects['home/loadMore']

使用1此插件的时候会绑定你使用过的每个action,在我使用2的时候会将绑定action的loading拿到。

这里面是有两个过程的。store.subscribeAction 中的before和after。在我调用action之后我的loading一直是true,这就是在before里面的操作。当我的action里面包裹的是promise,则走完resolve()或者reject()后整个action才算完成。之后就会走after。

before

after

我知道大家会问为什么把大量逻辑放在vuex里面处理。其实我反而觉得这样会让你的整个项目清晰化,可维护性高。易于迭代。

.vue文件只是将vuex里面的数据进行页面渲染。

store里面将数据存取,进行过滤。

serve层用来封装你的api

这样一目了然

路由切换动画 + keep alive 动态管理缓存组件

之前采用的是全局设置路由切换动画,但是体验效果不是很好,特别是返回列表页,页面会引起回弹,页面切换时会有暂时的空白。

未改造前的,也是参考别人的做法

## app.vue 
computed: { // 数据存放在vuex里面 ...mapState({ data: state => { return state.global.data } }) }, methods: { // 设置Keep_alive路由 setKeep_alive (to) { if (to.meta.keepAlive) { this.$store.dispatch({ type: 'global/setData', payload: to.name }) } } }, watch: { '$route' (to, from) { // 此时从from页面跳转到to页面 this.setKeep_alive(to) const routeDeep = ['/', '/list', '/detail', '/reservation', '/addCars'] const toDepth = routeDeep.indexOf(to.path) const fromDepth = routeDeep.indexOf(from.path) if (!from.name) { this.transitionName = 'fold' return } this.transitionName = toDepth > fromDepth ? 'fold-left' : 'fold-right' } },复制代码
## router.jsscrollBehavior (to, from, savedPosition) {    // keep-alive 返回缓存页面后记录浏览位置    if (savedPosition && to.meta.keepAlive) {      return savedPosition    }    // 异步滚动操作    return new Promise((resolve) => {      setTimeout(() => {        resolve({ x: 0, y: 1 })      }, 0)    })  },复制代码

两个问题

  1. 列表页滑动到一定位置后跳转到详情页,返回列表页页面回弹
    原因:原生滚动条的位置是不变的,使用scrollBehavior,根据上述代码可知滚动条会有一个闪烁的过程先置顶,然后滚动到上次保留的位置。
  2. 页面切换时会有暂时的空白,过渡不正常。

改造后

## app.vue
computed: { // 数据存放在vuex里面 ...mapState({ data: state => { return state.global.data } }) }, methods: { // 设置Keep_alive路由 setKeep_alive (to) { if (to.meta.keepAlive) { this.$store.dispatch({ type: 'global/setData', payload: to.name }) } } }, watch: { '$route' (to, from) { // 此时从from页面跳转到to页面 this.setKeep_alive(to) } },复制代码
list.vue
复制代码

1.采用better-scroll后,第一个问题可以直接解决。而且不用设置scrollBehavior,不懂可以去看

2.给页面CSS添加设置“position:absolute;”,此时页面脱离文档流,不占空间,这样就不会把下一页挤下去,完成平滑过渡。使用better-scroll给页面CSS添加设置“position:fixed;”。

如果页面布局里面有用到flex布局,一定要给flex组件加一个position为absolute或者fixed的div。

上述代码中已有keep alive 动态管理缓存路由的思路。

better-scroll与vue的最佳实践

之前在一篇文章上看到BetterScroll可能是目前最好用的移动端滚动插件,所以这次就想试试,滴滴开源的cube-ui组件库里面大多数用到的滑动组件都是基于better-scroll,体验了一下感觉还挺好。为什么没有用cube了?因为个人感觉主题颜色有点丑。所以自己就打算基于better-scroll封装一个vue版本的scroll组件。不说那么多了,立马上图:

想用better-scroll还有另外一个原因,我想自定义上下拉的动画。

想看demo及源码。记得给个star哟

自定义指令 vue-finger

包括点击,长按,双击,拖拽移动,多点触控,滑动,旋转,缩放手势

这一块我这边是基于,在这些指令里面你可以做很多在移动端手势方面想做的事情。后续我会继续迭代这些指令,制定出体验接近原生的组件,大家要关注我的哟

移动端适配方案

## rem.jsconst baseSize = 32// 设置 rem 函数function setRem () {  // 当前页面宽度相对于 750 宽的缩放比例,可根据自己需要修改。  const scale = document.documentElement.clientWidth / 750  // 设置页面根节点字体大小  document.documentElement.style.fontSize = (baseSize * Math.min(scale, 2)) + 'px'}// 初始化setRem()// 改变窗口大小时重新设置 remwindow.onresize = function () {  setRem()}## main.jsimport './rem'复制代码

还有最后还有一步。对于经常写样式的同学,px转rem是不是感觉很烦。 我这边处理的方式是,在项目根目录新建一个postcss.config.js文件。这样你只需按照设计稿的样式,正常写px就好。运行项目时会自动帮你转成rem。

module.exports = {  plugins: {    'autoprefixer': {      browsers: ['Android >= 4.0', 'iOS >= 7']    },    'postcss-pxtorem': {      rootValue: 16,      propList: ['*']    }  }}复制代码

如何分情况处理页面置顶

上文有讲到vue-router里面这个方法。

## router.jsscrollBehavior (to, from, savedPosition) {    // keep-alive 返回缓存页面后记录浏览位置    if (savedPosition && to.meta.keepAlive) {      return savedPosition    }    // 异步滚动操作    return new Promise((resolve) => {      setTimeout(() => {        resolve({ x: 0, y: 1 })      }, 0)    })  },复制代码

但是感觉添加页面转场动画后。页面会有回弹。所以我就放弃它了。不添加动画的可以考虑。

我这边用到了better-scroll后就不用担心这个问题。看完better-scroll文档介绍,你就会发现better-scroll就是为移动端运用而生的。

路由懒加载

当打包构建应用时,JavaScript 包会变得非常大,影响页面加载。如果我们能把不同路由对应的组件分割成不同的代码块,然后当路由被访问的时候才加载对应组件,这样就更加高效了。 这是路由懒加载就很重要了。看过大家应该都会用了,这里我就不介绍了。

// 路由懒加载const _import_ = file => () => import('./views/' + file + '.vue')routes: [    {      path: '/',      name: 'home',      component: _import_('Home/Home'),      meta: {        title: '首页',        keepAlive: true      }    },]复制代码

终于写完了,以上这些就是我在移动端体验优化的实战。希望能帮到大家。如果往后有什么好的优化方案我会继续更新。谢谢大家的观看。觉得好的点个赞哟?

转载于:https://juejin.im/post/5cdd2457f265da034e7eb2f9

你可能感兴趣的文章
用户体验要素——产品设计的优秀方法论
查看>>
Hyper-V 2016 系列教程27 Hyper-v平台串口外设解决方案介绍
查看>>
Python3.2官方文档翻译--作用域和命名空间
查看>>
I.MX6 2014 u-boot 测试修改
查看>>
【java规则引擎】drools6.5.0版本api简介
查看>>
中国人工智能学会通讯——沿着Marr的道路前进——视觉计算的前世今生
查看>>
curd库
查看>>
从IP数据包到端口发送究竟经历了什么?
查看>>
前端小白入门区块链系列04
查看>>
ShineButton-一个动效的按钮
查看>>
Java并发编程-锁及并发容器
查看>>
Function 与 Classes 组件的区别在哪?
查看>>
[杭州/新加坡] imToken 招聘前端工程师
查看>>
Flutter视图基础简介--Widget、Element、RenderObject
查看>>
SpringBoot中json转换问题(客户端交互)
查看>>
isKindOfClass和isMemberOfClass的区别
查看>>
JS检测对象中是否存在某个属性
查看>>
简析社群运营的玩法
查看>>
模拟服务器返回数据|掘金技术征文
查看>>
用百度云的时序数据库给京价保加了个价格走势图功能
查看>>