100字范文,内容丰富有趣,生活中的好帮手!
100字范文 > vue.js移动端实现上拉加载下拉刷新

vue.js移动端实现上拉加载下拉刷新

时间:2022-01-01 21:47:39

相关推荐

vue.js移动端实现上拉加载下拉刷新

web前端|js教程

javascript,vue.js,下拉

web前端-js教程跟横向滚动一样,我们还是采用better-scroll这个库来实现。由于better已经更新了新的版本,之前是0.几的版本,更新了一下发现,现在已经是1.2.6这个版本了,新版本多了些

比较好用的api,所以我也重写了之前的代码,用新的api来实现上拉加载以及下拉刷新。

jsp网站后台源码,ubuntu中pdf软件,用爬虫模拟登录,php大型,常见seo策略lzw

首先把基本的样式写好,这里就略过了,然后引入better-scroll库

网络三级分销系统源码,手机ubuntu安装教程,如何重装tomcat服务器,图片爬虫同类软件,php7网站模板,谷歌SEO优点lzw

import BScroll from etter-scroll

其次,在mounted生命周期实例化scroll,可以获取完数据后再new,也可以先new后,获取完数据调用refresh。

视频app ios源码,ubuntu启动绑定硬件,开源网络爬虫程序,php设置全局,前端框架seolzw

实例时需要传入一个配置参数,由于参数比较多,具体的请参考文档,这里只讲2个重点的:

//是否开启下拉刷新,可传入true或者false,如果需要更多配置可以传入一个对象pullDownRefresh:{ threshold:80, stop:40}//是否开启上拉加载,同上,上拉无stop参数,这里需要注意是负数pullUpLoad:{ threshold:-80,}/** * * @param threshold 触发事件的阀值,即滑动多少距离触发 * @param stop 下拉刷新后回滚距离顶部的距离(为了给loading留出一点空间) */

以上的数字个人感觉比较合适,但是这里有一个问题,由于我采用的是淘宝flexible.js来适配,这就导致:在安卓下80这个距离是合适的,但是到了iphone6s下,由于被缩放了3陪,所以现在80在iphone6s下就是27左右了。

所以,对于不同缩放程度的屏幕,还需要乘以对应的缩放比。

淘宝flexible.js里面其实已经有这个获取屏幕缩放比方法,这里直接从里面拿:

//在util.js里面加一个方法export function getDeviceRatio(){ var isAndroid = window.navigator.appVersion.match(/android/gi); var isIPhone = window.navigator.appVersion.match(/iphone/gi); var devicePixelRatio = window.devicePixelRatio; var dpr; if (isIPhone) { // iOS下,对于2和3的屏,用2倍的方案,其余的用1倍方案 if (devicePixelRatio >= 3) { dpr = 3; } else if (devicePixelRatio >= 2){dpr = 2; } else {dpr = 1; } } else { // 其他设备下,仍旧使用1倍的方案 dpr = 1; } return dpr}

import{ DEVICE_RATIO} from ../base/js/api.js/*获取当前缩放比*/const DEVICE_RATIO=getDeviceRatio(); /*下拉配置*/const DOWN_CONFIG={ threshold:80*DEVICE_RATIO, stop:40*DEVICE_RATIO}/*上拉配置*/const UP_CONFIG={ threshold:-80*DEVICE_RATIO,}this.scroller = new BScroll(scrollWrap,{ click:true, probeType:3, pullDownRefresh:DOWN_CONFIG, pullUpLoad:UP_CONFIG});

实例化后,接下来就是监听上拉和下拉事件了。betterScroll新增了一些事件,主要的有:

/*下拉事件*/this.scroller.on(pullingDown,()=> {});/*上拉事件*/this.scroller.on(pullingUp,()=>{});

触发上拉或者下拉事件后,需要我们调用 this.scroller.finishPullDown() 或者 this.scroller.finishPullUp() 来通知better-scroll事件完成。

大致的流程是这样的:

this.scroller.on(pullingDown,()=> { });

通常操作完成后都需要我们手动触发refresh方法来重新计算可滚动的距离,因此可以写一个watch监听数据的变化,这样我们只需要改变数据,不用每次操作数据后都调用refresh方法。

watch:{ dataList(){ this.$nextTick(()=>{ this.scroller.refresh(); }) }},

如果你使用的版本还是旧的,那可以在on( scroll )事件的时候进行判断来实现功能

this.scroller.on("scroll",(pos)=>{ //获取整个滚动列表的高度 var height=getStyle(scroller,"height"); //获取滚动外层wrap的高度 var pageHeight=getStyle(scrollWrap,"height"); //触发事件需要的阀值 var distance=80*DEVICE_RATIO; //参数pos为当前位置 if(pos.y>distance){//console.log("下拉"); //do something}else if(pos.y-pageHeight<-height-distance){ //console.log("上拉"); //do something }

为了防止多次触发,需要加2个开关类的东西;

var onPullUp=true;var onPullDown=true;

每次触发事件时,將对应的开关设置为false, 等操作完成后,再重新设置为true,否则多次下拉或者上拉就会触发多次事件。通过设置开关可以保证每次只有一个事件在进行。

最后,来封装成一个组件

由于每个页面需要滚动的具体内容都是不一样的,所以用了一个插槽来分发。

组件需要的参数由父级传入,通过prop来接收并设置默认值

export default { props: { dataList:{ type: Array, default: [] }, probeType: { type: Number, default: 3 }, click: { type: Boolean, default: true },pullDownRefresh: { type: null, default: false }, pullUpLoad: { type: null, default: false }, }

组件挂载后,在事件触发时并不直接处理事件,而是向父级发送一个事件,父级通过在模板v-on接收事件并处理后续的逻辑

mounted() { this.scroll = new BScroll(this.$refs.wrapper, {probeType: this.probeType,click: this.click,pullDownRefresh: this.pullDownRefresh,pullUpLoad: this.pullUpLoad, }) this.scroll.on(pullingUp,()=> { if(this.continuePullUp){this.beforePullUp();this.$emit("onPullUp","当前状态:上拉加载"); } }); this.scroll.on(pullingDown,()=> { this.beforePullDown(); this.$emit("onPullDown","当前状态:下拉加载更多"); }); }

父组件在使用时,需要传入配置参数Props以及处理子组件发射的事件,并且用具体的内容并替换掉 slot 标签

{{v.title}}

导演:{{filterDirectors(v.directors)}}

年份:{{v.year}}{{v.stock}}

类别:{{v.genres.join(" / ")}}

评分:{{v.rating.average}}分

父组件可以通过this.$refs.xxx来获取到子组件,可以调用子组件里面的方法;

computed:{ scrollElement(){return this.$refs.scroll } }

完整的scroller组件内容如下

0" :loadingWord="beforePullUpWord">

import BScroll from etter-scroll import Loading from ./loading.vue import PullingWord from ./pulling-word const PullingUpWord="正在拼命加载中..."; const beforePullUpWord="上拉加载更多"; const finishPullUpWord="加载完成"; const PullingDownWord="加载中..."; export default {props: {dataList:{ type: Array, default: []},probeType: { type: Number, default: 3},click: { type: Boolean, default: true}, pullDownRefresh: { type: null, default: false},pullUpLoad: { type: null, default: false}, },data() { return {scroll:null, inPullUp:false, inPullDown:false, beforePullUpWord, PullingUpWord, PullingDownWord, continuePullUp:true }}, mounted() { setTimeout(()=>{ this.initScroll(); this.scroll.on(pullingUp,()=> { if(this.continuePullUp){ this.beforePullUp(); this.$emit("onPullUp","当前状态:上拉加载"); } }); this.scroll.on(pullingDown,()=> { this.beforePullDown(); this.$emit("onPullDown","当前状态:下拉加载更多"); }); },20) },methods: { initScroll() { if (!this.$refs.wrapper) { return } this.scroll = new BScroll(this.$refs.wrapper, { probeType: this.probeType, click: this.click,pullDownRefresh: this.pullDownRefresh, pullUpLoad: this.pullUpLoad, }) }, beforePullUp(){ this.PullingUpWord=PullingUpWord; this.inPullUp=true; }, beforePullDown(){ this.disable(); this.inPullDown=true; }, finish(type){ this["finish"+type](); this.enable(); this["in"+type]=false; }, disable() { this.scroll && this.scroll.disable() }, enable() { this.scroll && this.scroll.enable() }, refresh() { this.scroll && this.scroll.refresh() }, finishPullDown(){ this.scroll&&this.scroll.finishPullDown() }, finishPullUp(){ this.scroll&&this.scroll.finishPullUp() }, }, watch: { dataList() { this.$nextTick(()=>{ this.refresh(); }) }},components: { Loading, PullingWord} }

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。