android viewpager 禁止滑动

在这里插入图片描述


前言

本文介绍了本人有一个相关的需求需要实现这一功能,在过程中发现自己之前没做过,然后记录下实现这一功能的过程及相关的小知识点。


一、viewpager 禁止滑动是什么,有现成方法吗?

viewpager 禁止滑动,就是在特殊的条件,比如说编辑的状态下,是不允许用户滑动的,但是看了下viewpager 的源码,也百度了一下,并没有现成的API可以直接禁用调viewpager 的滑动,毕竟人家这个组件就是要滑动的,没有这个API也很正常。所以我们需要自己实现。

二、使用setOnTouchListener

自己实现嘛,一般来说就是能偷懒则偷懒的,先定义了一个isInterceptTouch的布尔值变量,在编辑态下设置为true,拦截事件。示例代码如下:

        mBinding.viewPager.setOnTouchListener(object : View.OnTouchListener{
            override fun onTouch(v: View?, event: MotionEvent?): Boolean {
                Log.d(TAG, "onTouch: $isInterceptTouch ${event?.action}")
                return isInterceptTouch
            }

        })

调试过后发现还是能滑动,就是不流畅了,所以动了个小脑筋,在滑动的时候再把页面设置回来。划了一下没问题交给测试小伙伴了。代码现在是这个样子:

        mBinding.viewPager.setOnTouchListener(object : View.OnTouchListener{
            override fun onTouch(v: View?, event: MotionEvent?): Boolean {
                Log.d(TAG, "onTouch: $isInterceptTouch ${event?.action}")
                if (isInterceptTouch){
                    mBinding.viewPager.setCurrentItem(mPos,false)
                }
                return isInterceptTouch
            }

        })

三、使用自定义viewpager

然后把,就不出意外的出意外了,测试反馈说我一点一点的滑动,还是可以话,我试了下确实是这样,但是日志又打印的确实是true,表示确实把事件已经拦截了,我一下子脑子卡主了,想了好一会才反应过来。顺便带大家复习一下setOnTouchListener和onTouchEvent的区别

setOnTouchListener和onTouchEvent是Android中用于处理触摸事件的两种方法,它们有以下区别:

  1. 职责不同:onTouchEvent是View类中的一个方法,用于处理由View对象接收到的触摸事件。而setOnTouchListener则是View类的一个方法,它允许你为View对象设置一个OnTouchListener对象,这个对象可以响应View的触摸事件。
  2. 触发时机不同:当用户的触摸事件发生时,会最先由最内层的View对象(如按钮等)响应,如果该View对象没有消费该事件(即onTouch方法返回false),则会将事件传递给其父View对象,依此类推。如果最内层的View对象消费了该事件(即onTouch方法返回true),则该事件不会被传递给其父View对象。
  3. 处理方式不同:onTouchEvent是直接在View类中定义的方法,它的触发需要依赖传递过来的触摸事件信息。而setOnTouchListener则是为View对象设置了一个监听器,当用户的触摸事件发生时,会先由最内层的View对象响应,如果没有被消费,则会传递给其父View对象的onTouch方法;如果已经被消费,则不会传递给其父View对象的onTouch方法。

从复习结果上看当走到setOnTouchListener的时候,触摸事件其实已经被viewPager消费了,所以才会拦截不住,动歪脑筋设置setCurrentItem是没有用的。
因此,为了实现这一个需求,我不仅要自定义viewpager,使用onTouchEvent方法对触摸事件进行拦截,还要使用onInterceptTouchEvent防止其他子层级拿到消费事件。

再复习下onInterceptTouchEvent方法:这个方法主要存在于ViewGroup类中,用于拦截触摸事件。当返回值为false时,可以将点击事件传递到下层去;当返回值为true时,在这一层拦截点击事件,不能传递到下层去。

所以这个时候我的自定义viewpager就变成了这个样子

import android.content.Context
import android.util.AttributeSet
import android.util.Log
import android.view.MotionEvent
import androidx.viewpager.widget.ViewPager


class NoScrollViewPager : ViewPager {
    companion object {
        private const val TAG = "NoScrollViewPager"
    }

    private var isScrollAllowed = true

    constructor(context: Context?) : super(context!!) {}
    constructor(context: Context?, attrs: AttributeSet?) : super(
        context!!, attrs
    ) {
    }

    override fun onTouchEvent(event: MotionEvent): Boolean {
        Log.d(TAG, "onTouchEvent: $isScrollAllowed")
        return if (isScrollAllowed) {
            super.onTouchEvent(event)
        } else false
    }

    override fun onInterceptTouchEvent(event: MotionEvent): Boolean {
        Log.d(TAG, "onInterceptTouchEvent: $isScrollAllowed")
        return if (isScrollAllowed) {
            super.onInterceptTouchEvent(event)
        } else false
    }

    fun setScrollAllowed(scrollAllowed: Boolean) {
        isScrollAllowed = scrollAllowed
    }
}

然后通过setScrollAllowed这个方法来控制是否禁止滑动ViewPager,这次没问题了。


总结

本文介绍了如何禁止Android中的ViewPager组件的滑动操作,由于ViewPager本就是为了滑动而设计的,因此并没有现成的API可以直接实现禁止滑动。通过自定义ViewPager并重写onTouchEvent和onInterceptTouchEvent方法,可以实现滑动拦截并禁止ViewPager的滑动操作。同时也介绍了setOnTouchListener和onTouchEvent的区别。

  • 29
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答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 滑动的方法,开发者可以根据实际需求选用合适的方法来实现自己需要的效果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我怀里的猫

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值