ViewPager 滑动监听

当我们使用ViewPager+Fragment的时候经常会出现滑动的时候一卡一卡的,多半原因是在滑动的过程中同时操作Fragment中的逻辑,所以只需要正确用好 OnPageChangeListener();

@Override
public void onPageSelected(int position) {
    //position只有当页面真正切换的时候,才会调用这个方法,如果页面index一直没有变,则不会调用改方法
    Log.e(TAG,"onPageSelected"+position);
    //我们通常会在这里处理一些逻辑....
    doSomeThing(position);
}

如上图,大部分人都会在onPageSelected()方法中处理相应的逻辑,这样做最大的好处就是,在滑动到下一页前,可以提前预处理好相应逻辑。但是如果这部分逻辑过于复杂,就会造成卡顿(因为滑动和处理逻辑同时进行)。我们可以采取以下方式:

@Override
public void onPageScrollStateChanged(int state) {
    // press:state=1; up:state=2; idle:state=0;
    Log.e(TAG,"onPageScrollStateChanged"+state);
    if(state==0){
        doSomeThing();
    }
}

下面我们就来讲下这几个回调函数的执行顺序和其中的参数意义。

1:onPageScrollStateChanged(int state)

        从名字可以看出这个方法主要用来监测viewpager的滑动状态:当我们手指按下时 state=1,当我们手指抬起时 state=2,当viewpager处于空闲状态时 state=0;所以我们完全可以在state=0时 去加载或者处理我们的事情,因为这时候滑动已经结束。

2:onPageScrolled(int position , float positionOffset ,int positionOffsetPixels);

        这个方法主要是在viewpager滑动过程中执行,并且会被调用很多次。需要注意的是,position的值使用等于屏幕最左侧暴露出来的view的position,即使该view已经快看不见了。 第二个参数positionOffset:从左向右滑动时该值从1---0 变化,从右向左滑动时该值从0---1变化。第三个参数positionOffsetPixels:这个参数是viewpager滑动偏移量的具体值,比如屏幕宽度值是1080px而viewpager的宽度等于屏幕宽度,则这个值就会在0-1080之间变化。

3:onPageSelected(int position)

        这个方法会在viewpager快要成功切换的时候调用。怎么理解这句话呢,当我们滑动距离很小的时候,viewpager会自动回弹到当前页,也就是不进行页面切换,此时该方法不会被调用,因为页面并没有被成功切换;当我们滑动距离很大的时候,viewpager会自动帮我们滑动到下一页,此时手指抬起,viewpager判断即将成功切换到下一页,那么该方法就会被调用,并且position的值等于即将切换过去的页面的下标。

三个方法的执行顺序, onPageScrollStateChanged>onPageScrolled>onPageSelected :下面看下我手指从按下到抬起几个方法的执行情况。

06-13 11:03:38.245 24907-24907/com.example.win10.mytest E/====: onPageScrollStateChanged1
06-13 11:03:38.245 24907-24907/com.example.win10.mytest E/---->: onPageScrolled58
06-13 11:03:38.262 24907-24907/com.example.win10.mytest E/---->: onPageScrolled148
06-13 11:03:38.280 24907-24907/com.example.win10.mytest E/---->: onPageScrolled312
06-13 11:03:38.296 24907-24907/com.example.win10.mytest E/---->: onPageScrolled477
06-13 11:03:38.307 24907-24907/com.example.win10.mytest E/---->: onPageScrolled519
06-13 11:03:38.310 24907-24907/com.example.win10.mytest E/====: onPageScrollStateChanged2
06-13 11:03:38.310 24907-24907/com.example.win10.mytest E/****: onPageSelected1
06-13 11:03:38.314 24907-24907/com.example.win10.mytest E/---->: onPageScrolled579
06-13 11:03:38.331 24907-24907/com.example.win10.mytest E/---->: onPageScrolled778
06-13 11:03:38.349 24907-24907/com.example.win10.mytest E/---->: onPageScrolled915
06-13 11:03:38.366 24907-24907/com.example.win10.mytest E/---->: onPageScrolled989
06-13 11:03:38.381 24907-24907/com.example.win10.mytest E/---->: onPageScrolled1034
06-13 11:03:38.399 24907-24907/com.example.win10.mytest E/---->: onPageScrolled1061
06-13 11:03:38.415 24907-24907/com.example.win10.mytest E/---->: onPageScrolled1073
06-13 11:03:38.432 24907-24907/com.example.win10.mytest E/---->: onPageScrolled1078
06-13 11:03:38.449 24907-24907/com.example.win10.mytest E/---->: onPageScrolled0
06-13 11:03:38.532 24907-24907/com.example.win10.mytest E/====: onPageScrollStateChanged0

总结:如果我们想要在页面即将滑动成功的时候就做一些事情,那么完全可以在onPageSelected()方法中执行,但可能会造成卡顿感;如果想要在滑动结束后再执行相应的方法,可以列用onPageScrollStateChanged方法,判断当state==0的时候,做相应操作。

  • 7
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 可以通过以下两种方法禁止ViewPager滑动: 1. 自定义ViewPager类,重写onTouchEvent方法,不处理触摸事件即可。 2. 使用ViewPager的setOnTouchListener方法,监听触摸事件,当触摸事件发生时,返回true,即可禁止ViewPager滑动。 具体实现方法可以参考以下代码: 1. 自定义ViewPager类: ``` public class NoScrollViewPager extends ViewPager { public NoScrollViewPager(Context context) { super(context); } public NoScrollViewPager(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean onTouchEvent(MotionEvent ev) { return false; } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { return false; } } ``` 2. 使用setOnTouchListener方法: ``` viewPager.setOnTouchListener(new OnTouchListener() { @Override public boolean onTouch(View v, MotionEvent event) { return true; } }); ``` ### 回答2: Android中的ViewPager是一个非常流行的View,它提供了一种灵活的方式来切换不同的视图。默认情况下,ViewPager可以通过手势水平滑动来切换视图。然而,在某些情况下,您可能需要禁止用户手动滑动,而只是依靠代码来处理切换视图。 一种方法是使用ViewPager的setEnabled()方法来禁用ViewPager的手势滑动。将setEnabled()方法的参数设置为false即可。但是这种方法还可以让ViewPager仍然可以通过代码来进行切换视图,这可能并不是您想要的。 另一种方法是,您可以通过自定义ViewPager来禁止手势滑动,但仍保持使用代码来实现视图切换。 自定义ViewPager主要涉及两个方面。 第一方面涉及ViewPager的手势控制,默认情况下ViewPager具有手势控制功能,我们需要禁用这些手势控制。我们可以通过重写ViewPager的onInterceptTouchEvent和onTouchEvent方法来实现。在这两个方法中,我们可以判断用户的操作类型,如果是手势滑动,我们就不需要将事件传递给父View或子View,以此来禁用ViewPager的手势滑动。 第二个方面涉及ViewPager的动画效果。为了保证视图的切换效果,ViewPager默认添加了一些动画效果。而我们禁用了手势滑动后,这些动画效果可能会变得不协调或不适合我们的需求。因此,我们需要重写ViewPager的onPageScrolled方法来取消动画效果。 通过以上两个操作,我们就可以实现自定义ViewPager,并禁用手势滑动。但是需要注意的是,这样可能会影响用户的交互体验,因此在开发中需要根据具体情况慎重考虑是否要禁用手势滑动。 ### 回答3: 在 Android 中,如果想要禁止 ViewPager 的滑动,可以通过以下三种方法实现: 1. 自定义 ViewPager 可以自定义一个继承自 ViewPager 的类,重写它的 onTouchEvent() 方法和 onInterceptTouchEvent() 方法,在这两个方法中判断是否禁止了滑动,如果禁止了,则不执行父类的 onTouchEvent() 和onInterceptTouchEvent() 方法。示例代码如下: ``` public class NonSwipeableViewPager extends ViewPager { private boolean isSwipeEnabled = false; public NonSwipeableViewPager(@NonNull Context context) { super(context); } public NonSwipeableViewPager(@NonNull Context context, @Nullable AttributeSet attrs) { super(context, attrs); } @Override public boolean onTouchEvent(MotionEvent ev) { return isSwipeEnabled && super.onTouchEvent(ev); } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { return isSwipeEnabled && super.onInterceptTouchEvent(ev); } public void setSwipeEnabled(boolean enabled) { isSwipeEnabled = enabled; } } ``` 2. 给 ViewPager 设置监听器 可以在 Activity 或 Fragment 中给 ViewPager 设置 onPageChangeListener,然后在 onPageScrollStateChanged() 方法中判断是否禁止了滑动,如果禁止了,则将其返回到初始位置。示例如下: ``` viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() { @Override public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) { } @Override public void onPageSelected(int position) { } @Override public void onPageScrollStateChanged(int state) { if (state == ViewPager.SCROLL_STATE_IDLE) { int currentItem = viewPager.getCurrentItem(); if (currentItem == 0) { viewPager.setCurrentItem(0, false); } else if (currentItem == viewPager.getAdapter().getCount() - 1) { viewPager.setCurrentItem(viewPager.getAdapter().getCount() - 1, false); } } } }); ``` 3. 通过 layout 参数禁止滑动 在 XML 中设置 ViewPager 的 `android:layout_width` 属性为 0dp,可以禁止它的滑动。示例代码如下: ``` <ViewPager android:id="@+id/viewPager" android:layout_width="0dp" android:layout_height="wrap_content" /> ``` 以上是三种禁止 ViewPager 滑动的方法,开发者可以根据实际需求选用合适的方法来实现自己需要的效果。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值