vue中封装better-scroll

better-scroll的用法

//结构
<div class="wrapper">
  <ul class="content">
    <li>...</li>
    <li>...</li>
    ...
  </ul>
</div>
//js

import BScroll from 'better-scroll';
let wrap=document.querySelector('#wrapper');
let scroll =new BScroll(wrap,{})

原理
  • wrapper中只有第一个元素content可以滚动
  • 在初始化的时候,会计算父元素和子元素的高度和宽度,来决定是否可以纵向和横向滚动。因此,我们在初始化它的时候,必须确保父元素和子元素的内容已经正确渲染了。如果子元素或者父元素 DOM 结构发生改变的时候,必须重新调用 scroll.refresh() 方法重新计算来确保滚动效果的正常。

vue中使用better-scroll

<template>
  <div class="wrapper" ref='wrapper'>
    <ul class="content">
      <li>...</li>
      <li>...</li>
      ...
    </ul>
  </div>
</template>
<script>
import BScroll from 'better-scroll'
export default {
  mounted(){
    this.$nextTick(() => {
        this.scroll = new Bscroll(this.$refs.wrapper, {})
      })
  }
}
</script>  

实现下拉刷新,上拉加载更多的需求

<template>
   <div ref='wrapper'>
        <ul class="content">
            <li v-for="item in data">{{item}}</li>
        </ul>
        <div class="loading-wrapper"></div>
    </div>
</template>
<script>
new Vue({
        el:'#app',
        template:'#wrapper',
        data:{
            data:[1,2,3,4,5,6,7,8,9,0,2,1,3,4,5,6,7,8],
            dropDown:false
        },
        created() {
            this.loadData();
        },
        methods: {
            loadData() {
                //getData().then((res) => {
                    this.data = this.data.concat(this.data)
                    this.$nextTick(() => {
                        if (!this.scroll) {
                            this.scroll = new BScroll(this.$refs.wrapper, {})
                            this.scroll.on('scroll', (pos) => {
                                //如果下拉超过50px 就显示下拉刷新的文字
                                if(pos.y>50){
                                    this.dropDown = true
                                }else{
                                    this.dropDown = false
                                }
                            })
                            this.scroll.on('touchEnd', (pos) => {
                                // 下拉动作
                                if(pos.y > 50){
                                    console.log("下拉刷新成功")
                                    this.dropDown = false
                                }
                                //上拉加载 总高度>下拉的高度+10 触发加载更多
                                if(this.scroll.maxScrollY>pos.y+10){
                                    console.log("加载更多")
                                    //使用refresh 方法 来更新scroll  解决无法滚动的问题
                                    this.loadData()
                                    this.scroll.refresh()
                                }
                            })
                        } else {
                            this.scroll.refresh()
                        }
                    })
                //})
            }
        }
    })
</script>  

封装better-scroll组件

上边的代码实现了基本的功能,但是每次使用的时候都需要写入这么多的代码,所以我将他封装成一个组件scroll

<template>
    <div ref="wrapper">
        <slot></slot><!--分发内容-->
    </div>
</template>
<script>
    import BScroll from 'better-scroll'
    export default {
        props: {
            /**
             * 1 滚动的时候会派发scroll事件,会截流。
             * 2 滚动的时候实时派发scroll事件,不会截流。
             * 3 除了实时派发scroll事件,在swipe的情况下仍然能实时派发scroll事件
             */
            probeType: {
                type: Number,
                default: 1
            },
            /**
             * 点击列表是否派发click事件
             */
            click: {
                type: Boolean,
                default: true
            },
            /**
             * 是否开启横向滚动
             */
            scrollX: {
                type: Boolean,
                default: false
            },
            /**
             * 是否派发滚动事件
             */
            listenScroll: {
                type: Boolean,
                default: false
            },
            /**
             * 列表的数据
             */
            data: {
                type: Array,
                default: null
            },
            /**
             * 是否派发滚动到底部的事件,用于上拉加载
             */
            pullup: {
                type: Boolean,
                default: false
            },
            /**
             * 是否派发顶部下拉的事件,用于下拉刷新
             */
            pulldown: {
                type: Boolean,
                default: false
            },
            /**
             * 是否派发列表滚动开始的事件
             */
            beforeScroll: {
                type: Boolean,
                default: false
            },
            /**
             * 当数据更新后,刷新scroll的延时。
             */
            refreshDelay: {
                type: Number,
                default: 20
            }
        },
        mounted() {
        // 保证在DOM渲染完毕后初始化better-scroll
            setTimeout(() => {
                this._initScroll()
            }, 20)
        },

        methods: {
            _initScroll() {
                if (!this.$refs.wrapper) {
                    return
                }
                // better-scroll的初始化
                this.scroll = new BScroll(this.$refs.wrapper, {
                    probeType: this.probeType,
                    click: this.click,
                    scrollX: this.scrollX
                })
                //父组件可以通过@setScroll方法拿到scroll实例,进行配置
                this.$emit('setScroll',this.scroll);
                // 是否派发滚动事件
                if (this.listenScroll) {
                    this.scroll.on('scroll', (pos) => {
                        this.$emit('scroll', pos)
                    })
                }

                // 是否派发滚动到底部事件,用于上拉加载
                if (this.pullup) {
                    this.scroll.on('scrollEnd', () => {
                    // 滚动到底部
                        if (this.scroll.y <= (this.scroll.maxScrollY + 50)) {
                            this.$emit('scrollToEnd')
                        }
                    })
                }

                // 是否派发顶部下拉事件,用于下拉刷新
                if (this.pulldown) {
                    this.scroll.on('scrollEnd', (pos) => {
                        // 下拉动作
                        if (pos.y > 50) {
                            this.$emit('pulldown')
                        }
                     })
                }
                // 是否派发列表滚动开始的事件
                if (this.beforeScroll) {
                    this.scroll.on('beforeScrollStart', () => {
                        this.$emit('beforeScroll')
                    })
                }
            },
            //父组件可以通过this.@refs拿到组件 并调用组件上的这些方法
            disable() {
                // 代理better-scroll的disable方法
                this.scroll && this.scroll.disable()
            },
            enable() {
                // 代理better-scroll的enable方法
                this.scroll && this.scroll.enable()
            },
            refresh() {
                // 代理better-scroll的refresh方法
                this.scroll && this.scroll.refresh()
            },
            scrollTo() {
                // 代理better-scroll的scrollTo方法
                this.scroll && this.scroll.scrollTo.apply(this.scroll, arguments)
            },
            scrollToElement() {
                // 代理better-scroll的scrollToElement方法
                this.scroll && this.scroll.scrollToElement.apply(this.scroll, arguments)
            }
        },
        watch: {
            // 监听数据的变化,延时refreshDelay时间后调用refresh方法重新计算,保证滚动效果正常
            data() {
                setTimeout(() => {
                    this.refresh()
                }, this.refreshDelay)
            }
        }
    }
</script>

使用这个组件

<template>
    <scroll class="wrapper"
            :data="data"
            :pulldown="pulldown"
            @setScroll="setScroll"
            @pulldown="loadData"
            ref="scro">
        <ul class="content">
            <li v-for="item in data">{{item}}</li>
        </ul>
        <div class="loading-wrapper"></div>
    </scroll>
</template>
<script>
    import BScroll from 'better-scroll'
    export default {
        data() {
            return {
                data: [],
                pulldown: true
            }
        },
        created() {
            this.loadData()
        },
        mounted(){
            this.$nextTick(()=>{
                this.$refs.scro.scrollToElement('dom');
            })
        },
        methods: {
            loadData() {
                requestData().then((res) => {
                    this.data = res.data.concat(this.data)
                })
            },
            setScroll(scroll){
                this.scroll = scroll;
                console.log("scroll创建成功");
            },
        }

    }
</script>

better-scroll作为指令封装

未完待续

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352

推荐阅读更多精彩内容