最新文章专题视频专题问答1问答10问答100问答1000问答2000关键字专题1关键字专题50关键字专题500关键字专题1500TAG最新视频文章推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37视频文章20视频文章30视频文章40视频文章50视频文章60 视频文章70视频文章80视频文章90视频文章100视频文章120视频文章140 视频2关键字专题关键字专题tag2tag3文章专题文章专题2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章专题3
当前位置: 首页 - 科技 - 知识百科 - 正文

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

来源:动视网 责编:小采 时间:2020-11-27 19:54:18
文档

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

vue.js移动端实现上拉加载下拉刷新:这次给大家带来vue.js移动端实现上拉加载下拉刷新,vue.js移动端实现上拉加载下拉刷新的注意事项有哪些,下面就是实战案例,一起来看一下。 跟横向滚动一样,我们还是采用better-scroll这个库来实现。由于better已经更新了新的版本,之前是0.几的版本,更新
推荐度:
导读vue.js移动端实现上拉加载下拉刷新:这次给大家带来vue.js移动端实现上拉加载下拉刷新,vue.js移动端实现上拉加载下拉刷新的注意事项有哪些,下面就是实战案例,一起来看一下。 跟横向滚动一样,我们还是采用better-scroll这个库来实现。由于better已经更新了新的版本,之前是0.几的版本,更新


这次给大家带来vue.js移动端实现上拉加载下拉刷新,vue.js移动端实现上拉加载下拉刷新的注意事项有哪些,下面就是实战案例,一起来看一下。

跟横向滚动一样,我们还是采用better-scroll这个库来实现。由于better已经更新了新的版本,之前是0.几的版本,更新了一下发现,现在已经是1.2.6这个版本了,新版本多了些 比较好用的api,所以我也重写了之前的代码,用新的api来实现上拉加载以及下拉刷新。

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

import BScroll from 'better-scroll'

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

实例时需要传入一个配置参数,由于参数比较多,具体的请参考文档,这里只讲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',()=> {
 
 <!-- 1. 发送请求获取数据 -->
 
 <!-- 2. 获取成功后,通知事件完成 -->
 
 <!-- 3. 修改data数据,在nextTick调用refresh -->
});

通常操作完成后都需要我们手动触发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,否则多次下拉或者上拉就会触发多次事件。通过设置开关可以保证每次只有一个事件在进行。

最后,来封装成一个组件

 <template>
 <p ref="wrapper" class="list-wrapper"> 
 <p class="scroll-content"> 
 <slot></slot> 
 </p> 
 </p>
</template>

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

组件需要的参数由父级传入,通过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 标签

 <Scroller 
 id="scroll"
 ref="scroll" 
 :dataList="filmList"
 :pullDownRefresh="DOWN_CONFIG"
 :pullUpLoad="UP_CONFIG"
 @onPullUp="pullUpHandle"
 @onPullDown="pullDownHandle"
 >
 <ul>
 <router-link class="film-list" v-for="(v,i) in filmList" :key="v.id" tag="li" :to='{path:"/film-detail/"+v.id}'>
 <p class="film-listimg">
 <img v-lazy="v.images.small" alt="" /> 
 </p>
 <p class="film-listdetail">
 <p class="film-listdetailtitle">{{v.title}}</p>
 <p class="film-listdetaildirector">导演:{{filterDirectors(v.directors)}}</p>
 <p class="film-listdetailyear">年份:{{v.year}}<span>{{v.stock}}</span></p>
 <p class="film-listdetailtype">类别:{{v.genres.join(" / ")}}<span></span></p>
 <p class="film-listdetailrank">评分:<span>{{v.rating.average}}分</span></p>
 </p> 
 </router-link>
 </ul> 
 </Scroller>

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

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

完整的scroller组件内容如下

 <template>
 <p ref="wrapper" class="list-wrapper"> 
 <p class="scroll-content"> 
 <slot></slot>
 <p>
 <PullingWord v-show="!inPullUp&&dataList.length>0" :loadingWord="beforePullUpWord"></PullingWord>
 <Loading v-show="inPullUp" :loadingWord='PullingUpWord'></Loading>
 </p> 
 </p> 
 <transition name="pullDown">
 <Loading class="pullDown" v-show="inPullDown" :loadingWord='PullingDownWord'></Loading>
 </transition> 
 </p>
 </template>
 <script >
 import BScroll from 'better-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
 }
 }
 </script>

相信看了本文案例你已经掌握了方法,更多精彩请关注Gxl网其它相关文章!

推荐阅读:

文档

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

vue.js移动端实现上拉加载下拉刷新:这次给大家带来vue.js移动端实现上拉加载下拉刷新,vue.js移动端实现上拉加载下拉刷新的注意事项有哪些,下面就是实战案例,一起来看一下。 跟横向滚动一样,我们还是采用better-scroll这个库来实现。由于better已经更新了新的版本,之前是0.几的版本,更新
推荐度:
标签: VUE js 移动端
  • 热门焦点

最新推荐

猜你喜欢

热门推荐

专题
Top