Android笔记(四十):ViewPager2嵌套RecyclerView滑动冲突进一步办理 ...

打印 上一主题 下一主题

主题 985|帖子 985|积分 2955

配景


ViewPager2内嵌套横向滑动的RecyclerView,会有滑动冲突的情况,引入官方提供的NestedScrollableHost类可以办理冲突题目,但是有一些瑕疵,滑动横向RecyclerView到顶部,按住它不放手继续往左拖再往右拖,这时候会发现外层ViewPager2滑动了,而不是横向RecyclerView滑动,于是参考NestedScrollableHost进行逻辑美满
完备代码



  • 重要是增加判断外层ViewPager2是否可滚动来设置是否允许父View拦截变乱
  1. open class NestRecyclerView @JvmOverloads constructor(
  2.     context: Context,
  3.     attrs: AttributeSet? = null
  4. ): RecyclerView(context, attrs) {
  5.    
  6.     private var initialX = 0f
  7.     private var initialY = 0f
  8.     private val parentViewPager: ViewPager2?
  9.         get() {
  10.             var v: View? = parent as? View
  11.             while (v != null && v !is ViewPager2) {
  12.                 v = v.parent as? View
  13.             }
  14.             return v as? ViewPager2
  15.         }
  16.     private fun canViewScroll(target: View?, orientation: Int, delta: Float): Boolean {
  17.         val direction = -delta.sign.toInt()
  18.         return when (orientation) {
  19.             0 -> target?.canScrollHorizontally(direction) ?: false
  20.             1 -> target?.canScrollVertically(direction) ?: false
  21.             else -> throw IllegalArgumentException()
  22.         }
  23.     }
  24.     override fun onInterceptTouchEvent(event: MotionEvent?): Boolean {
  25.         val orientation = parentViewPager?.orientation ?: return super.onInterceptTouchEvent(event)
  26.         if (!canViewScroll(this, orientation, -1f) && !canViewScroll(this, orientation, 1f)) {
  27.             return super.onInterceptTouchEvent(event)
  28.         }
  29.         when (event?.action) {
  30.             MotionEvent.ACTION_DOWN -> {
  31.                 initialX = event.x
  32.                 initialY = event.y
  33.                 parent.requestDisallowInterceptTouchEvent(true)
  34.             }
  35.             MotionEvent.ACTION_MOVE -> {
  36.                 val dx = event.x - initialX
  37.                 val dy = event.y - initialY
  38.                 val isVpHorizontal = orientation == ViewPager2.ORIENTATION_HORIZONTAL
  39.                 if (isVpHorizontal == dy.absoluteValue > dx.absoluteValue) {
  40.                     parent.requestDisallowInterceptTouchEvent(false)
  41.                 } else {
  42.                     if (canViewScroll(this, orientation, if (isVpHorizontal) dx else dy)) {
  43.                         parent.requestDisallowInterceptTouchEvent(true)
  44.                     } else {
  45.                         if (canViewScroll(parentViewPager, orientation, if (isVpHorizontal) dx else dy)) {
  46.                             parent.requestDisallowInterceptTouchEvent(false)
  47.                         } else {
  48.                             parent.requestDisallowInterceptTouchEvent(true)
  49.                         }
  50.                     }
  51.                 }
  52.             }
  53.         }
  54.         return super.onInterceptTouchEvent(event)
  55.     }
  56. }
复制代码
向上滑动AppBarLayout不联动题目

假如布局CoordinatorLayout + AppBarLayout + ViewPager2内嵌套横向滑动的RecyclerView,这时拖拽横向滑动的RecyclerView向上移,AppBarLayout不会跟着向上移
原因分析



  • 拖拽横向滑动的RecyclerView向上移时,CoordinatorLayout.onNestedPreScroll内的lp.isNestedScrollAccepted(type)返回false,造成AppBarLayout没有实行scroll

  • lp.isNestedScrollAccepted(type)被赋值的地方,会根据AppBarLayout$Behavior.onStartNestedScroll返回的accepted进行赋值

  • AppBarLayout$Behavior.onStartNestedScroll内,会判断nestedScrollAxes的值不是2就返回false

  • RecyclerView也支持嵌套滑动。startNestedScroll是由NestedScrollingChildHelper实现的,它会将嵌套滑动上传,也就是NestedScrollingChild都会将嵌套滑动先交给NestedScrollingParent处置处罚。
  1. class RecyclerView
  2.   ...
  3.      public boolean onInterceptTouchEvent(MotionEvent e) {
  4.         if (mLayoutSuppressed) {
  5.             // When layout is suppressed,  RV does not intercept the motion event.
  6.             // A child view e.g. a button may still get the click.
  7.             return false;
  8.         }
  9.         // Clear the active onInterceptTouchListener.  None should be set at this time, and if one
  10.         // is, it's because some other code didn't follow the standard contract.
  11.         mInterceptingOnItemTouchListener = null;
  12.         if (findInterceptingOnItemTouchListener(e)) {
  13.             cancelScroll();
  14.             return true;
  15.         }
  16.         if (mLayout == null) {
  17.             return false;
  18.         }
  19.         final boolean canScrollHorizontally = mLayout.canScrollHorizontally();
  20.         final boolean canScrollVertically = mLayout.canScrollVertically();
  21.         if (mVelocityTracker == null) {
  22.             mVelocityTracker = VelocityTracker.obtain();
  23.         }
  24.         mVelocityTracker.addMovement(e);
  25.         final int action = e.getActionMasked();
  26.         final int actionIndex = e.getActionIndex();
  27.         switch (action) {
  28.             case MotionEvent.ACTION_DOWN:
  29.                 if (mIgnoreMotionEventTillDown) {
  30.                     mIgnoreMotionEventTillDown = false;
  31.                 }
  32.                 mScrollPointerId = e.getPointerId(0);
  33.                 mInitialTouchX = mLastTouchX = (int) (e.getX() + 0.5f);
  34.                 mInitialTouchY = mLastTouchY = (int) (e.getY() + 0.5f);
  35.                 if (mScrollState == SCROLL_STATE_SETTLING) {
  36.                     getParent().requestDisallowInterceptTouchEvent(true);
  37.                     setScrollState(SCROLL_STATE_DRAGGING);
  38.                     stopNestedScroll(TYPE_NON_TOUCH);
  39.                 }
  40.                 // Clear the nested offsets
  41.                 mNestedOffsets[0] = mNestedOffsets[1] = 0;
  42.                 int nestedScrollAxis = ViewCompat.SCROLL_AXIS_NONE;
  43.                 if (canScrollHorizontally) {
  44.                     nestedScrollAxis |= ViewCompat.SCROLL_AXIS_HORIZONTAL;
  45.                 }
  46.                 if (canScrollVertically) {
  47.                     nestedScrollAxis |= ViewCompat.SCROLL_AXIS_VERTICAL;
  48.                 }
  49.                 startNestedScroll(nestedScrollAxis, TYPE_TOUCH);
  50.                 break;
  51.                 ...
  52.         return mScrollState == SCROLL_STATE_DRAGGING;
  53.     }
复制代码
这里RecyclerView是横向的,所以nestedScrollAxis会被赋值为1,RecyclerView内调用startNestedScroll会向上层view传递,直到交给CoordinatorLayout处置处罚,而CoordinatorLayout在调用onStartNestedScroll的时候,AppBarLayout$Behavior.onStartNestedScroll又返回false了,造成CoordinatorLayout回调onNestedPreScroll(由RecyclerView在ACTION_MOVE时调用dispatchNestedPreScroll触发)时无法调用AppBarLayout的滚动。
办理方法

在CoordinatorLayout调用onStartNestedScroll的时候不处置处罚横向的情况,就不会导致lp.isNestedScrollAccepted(type)被赋值
  1. class NestedCoordinatorLayout @JvmOverloads constructor(
  2.     context: Context,
  3.     attrs: AttributeSet? = null
  4. ): CoordinatorLayout(context, attrs) {
  5.     override fun onStartNestedScroll(child: View, target: View, axes: Int, type: Int): Boolean {
  6.         return if (axes and ViewCompat.SCROLL_AXIS_HORIZONTAL != 0) {
  7.             false
  8.         } else super.onStartNestedScroll(child, target, axes, type)
  9.     }
  10. }
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

乌市泽哥

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表