zoukankan      html  css  js  c++  java
  • vue+vuex 回退定位到初始位置

    先放出两张图(没错,你还在9012,做为一名资深设计师我唯一的技能点就是留白),简单说明下问题
    未做回退定位(从落地页回退,每次都回到A位置)想死啊有木有,每次都需要手动重新定位来选择,你大哥看到你做个这肯定想扣死你:

    添加回退定位后(从落地页回退,定位到点击位置)哈,好用到爆 有木有~:


    按照WBD国际通用惯例(我编的),先对这个demo中用到的文件做一个索引,方便对整个回退功能有个宏观的视角,更容易理解整个流程是怎么走通的,做到心中有术。
    这个回退定位demo共涉及到5个文件,分别如下:
    BackToA.vue 组件A,也就是需要回退定位的组件,也就是开篇的车祸现场
    Alphabet.vue 组件B,跳转落地页,不重要
    index.js 在这个文件中进行,配置路由,添加meta属性(keepAlive),很重要
    App.vue 模板文件,引入组件(router-view),依据路由是否添加meta属性判断是否启用keep-alive进行组件缓存
    main.js 入口文件,引入vuex,在state定义_scrollTop来记录scrollTop值,和改变state值的actions方法来提交mutation更新状态
    demo目录结构如下图:

    看到这,有没有感觉很宏观。是不是心中有了术,对这个小demo产生了王的蔑视。少年请留步,话说,有术无道,止于术,再给我两分钟,让我把记忆结成冰,说“道”。
    道:首先对组件BackToA启用keep-alive缓存,当组件BackToA上下滚动时记录当前页的scrollTop值,并把这个值存储在vuex的state中。当从落地组件B回退到组件BackToA时,缓存生效,组件BackToA不再重新渲染。此时拿到vuex state中存储的scrollTop值,赋值给BackToA组件,实现定位到跳转前的位置。
    其实我是一名程序员,ok,下面开始我们喜闻乐见的编程环节。
    1.引入vuex存储scrollTop值
    因为vuex是整个流程的核心,先把这部分放到第一个梳理清楚方便后面的流程持续进行(我们都受过被打断的伤痛在我这决不允许)(打断的都是工作,没有其它的乱七八糟的什么东西不要乱联想)。
    打开入口文件main.js(一般都是这个文件),在这个文件中定义一个store常量、存储一个_scrollTop来存储组件上下滚动时页面的scrollTop值、引入vuex,并启用

    源码如下

    import Vuex from 'vuex'
    Vue.use(Vuex)
    
    const store = new Vuex.Store({
    state: {
    _scrollTop: 0 // 存储组件的scrollTop值
    },
    mutations: {
    updatePosition (state, payload) {
    state._scrollTop = payload.top // 更改scrollTop值
    }
    },
    actions: {
    updatePosition ({commit}, top) { 
    commit({type: 'updatePosition', top: top}) // 提交mutation 更改状态
    }
    }
    })

    2.创建组件BackToA

    ①引入vuex中的mapState和mapActions

    import { mapState, mapActions } from 'vuex'
    

    ②在computed计算属性中定义demoTop变量,并读取state中的_scrollTop值给其赋值

    computed: {
    ...mapState({
    demoTop: state => state._scrollTop, // 获取 state 中存储的top值
    }),
    },
    

    ③创建监听页面滚动方法,实时监听当前页面的scrollTop值,这个监听放在mounted钩子中触发的原因是:此时页面dom已经渲染完毕

    mounted: function () {
    console.log("==mounted===");
    document.querySelector('.hello').addEventListener('scroll', this.handleScroll, false);
    },
    

      

    methods:{
    handleScroll(){
    this.box = document.querySelector('.hello')
    }, 
    },
    

    ④实时获取当前页面scrollTop值,提交mutation更新状态

    methods:{
    handleScroll(){
    this.box = document.querySelector('.hello')
    this.updatePosition(this.box.scrollTop)
    }, 
    ...mapActions([
    'updatePosition'
    ]), 
    },
    

    ⑤缓存数据读取,更新scrollTop值,实现定位

    activated () {
    document.querySelector('.hello').scrollTop = this.demoTop // 更新缓存组件的scrollTop值
    },
    

    ⑥页面布局

    注:我们在这个例子中得到的scrollTop值,是属于class='hello'的,所以在组件中BackToA必须只有class='hello'的div支持 overflow:scroll才行

    <template>
    <div class="hello">
    <ul>
    <li v-for="(item,index) in 26" :key="index">
    <router-link :to="{ name: 'Alphabet', params: { letter: String.fromCharCode(64+item) }}">{{String.fromCharCode(64+item)}}</router-link> 
    </li> 
    </ul>
    </div>
    </template>
    <style scoped>
    .hello{
    height:600px;
    overflow:scroll;
    }
    .hello>ul{
    height:auto;
    overflow:hidden;
    background:#eee;
    }
    .hello>ul>li{
    height:60px;
    line-height:60px;
    100%;
    float:left;
    margin:0;
    border-bottom:1px dashed #ccc;
    }
    </style>
    

    3.创建落地组件Alphabet

    <template>
    <div class="hello">
    <div><button @click="goback">goback</button></div>
    <ul>
    <li>
    this is <strong>{{ $route.params.letter }}</strong>
    </li> 
    </ul>
    </div>
    </template>
    

    4.配置组件缓存

    ①在router/index.js文件中给/backto/backtoa路由添加meta缓存属性:meta: { keepAlive: true }

    {
    path: '/backto/backtoa',
    name: 'BackToA',
    component: BackToA,
    meta: { keepAlive: true }
    }	
    

    ②在App.vue模板文件中根据是否添加meta属性判断是否开启缓存机制:

    <template>
    <div id="app">
    <keep-alive>
    <router-view v-if="$route.meta.keepAlive"></router-view>
    </keep-alive>
    <router-view v-if="!$route.meta.keepAlive"></router-view>
    </div>
    </template>
    

    **本篇完再见**

  • 相关阅读:
    ios 将Log日志重定向输出到文件中保存
    IOS中GPS定位偏移纠正(适用于Google地图)
    java与IOS之间的RSA加解密
    [iPhone高级] 基于XMPP的IOS聊天客户端程序(IOS端一)
    [IOS] Storyboard全解析-第一部分
    NTP时间同步
    Prthon多线程和模块
    Python循环语句,对象
    Python起步学习
    Nginx实战之反向代理WebSocket的配置实例
  • 原文地址:https://www.cnblogs.com/ligulalei/p/10723841.html
Copyright © 2011-2022 走看看