修复在ViewPager2中使用PhotoView出现的滑动异常,实践出真知

当图片缩放处于上边缘或下边缘的时候,直接调用了parent.requestDisallowInterceptTouchEvent(false);让父View有机会走拦截逻辑,而父View,这里是ViewPager2也很不客气的拦截了事件,导致触发外部ViewPager2滑动。

这个应该算是PhotoView的bug,外部用ViewPager的话也一样会有这个问题。

解决方法

由于PhotoView的代码不多,我直接复制到项目里,修改了onGestureListener#onDrag里的代码:

@Override
public void onDrag(float dx, float dy) {

if (mScaleDragDetector.isScaling()) {
return; // Do not drag if we are already scaling
}
if (mOnViewDragListener != null) {
mOnViewDragListener.onDrag(dx, dy);
}

mSuppMatrix.postTranslate(dx, dy);
checkAndDisplayMatrix();

/*

  • Here we decide whether to let the ImageView’s parent to start taking
  • over the touch event.
  • First we check whether this function is enabled. We never want the
  • parent to take over if we’re scaling. We then check the edge we’re
  • on, and the direction of the scroll (i.e. if we’re pulling against
  • the edge, aka ‘overscrolling’, let the parent take over).
    */
    ViewParent parent = mImageView.getParent();
    //拖动的时候
    if (mAllowParentInterceptOnEdge && !mScaleDragDetector.isScaling() && !noScaleAndDrag) {
    float absDx = Math.abs(dx);
    float absDy = Math.abs(dy);
    if (mHorizontalScrollEdge == HORIZONTAL_EDGE_BOTH && absDx > absDy
    || (mHorizontalScrollEdge == HORIZONTAL_EDGE_LEFT && dx >= 1f && absDx > absDy)
    || (mHorizontalScrollEdge == HORIZONTAL_EDGE_RIGHT && dx <= -1f && absDx > absDy)
    || (mVerticalScrollEdge == VERTICAL_EDGE_TOP && dy >= 1f && absDy > absDx)
    || (mVerticalScrollEdge == VERTICAL_EDGE_BOTTOM && dy <= -1f && absDy > absDx
    || (mVerticalScrollEdge == VERTICAL_EDGE_BOTH) && absDy > absDx)
    ) {
    if (parent != null) {
    parent.requestDisallowInterceptTouchEvent(false);
    }
    }
    } else {
    if (parent != null) {
    parent.requestDisallowInterceptTouchEvent(true);
    }
    }
    }
    简单说明下,就是在请求父View继续拦截的条件里,增加了滑动方向的距离>另一方向上的距离的判定条件。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

img
img

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V:vip204888 备注Android获取(资料价值较高,非无偿)
img

文末

很多人在刚接触这个行业的时候或者是在遇到瓶颈期的时候,总会遇到一些问题,比如学了一段时间感觉没有方向感,不知道该从那里入手去学习,对此我整理了一些资料,需要的可以免费分享给大家

这里笔者分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司2021年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

【视频教程】

天道酬勤,只要你想,大厂offer并不是遥不可及!希望本篇文章能为你带来帮助,如果有问题,请在评论区留言。

链图片转存中…(img-y4NsgVTs-1711603375929)]

天道酬勤,只要你想,大厂offer并不是遥不可及!希望本篇文章能为你带来帮助,如果有问题,请在评论区留言。

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

### 回答1: ViewPager2 可以使用 androidx 库ViewPager2,在 dependencies 添加 ```Groovy implementation 'androidx.viewpager2:viewpager2:1.0.0' ``` 然后在布局文件添加 ViewPager2 控件,例如: ```xml <androidx.viewpager2.widget.ViewPager2 android:id="@+id/view_pager" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 接着,在代码创建 FragmentAdapter,例如: ```Kotlin class MyFragmentAdapter(fragmentManager: FragmentManager, lifecycle: Lifecycle) : FragmentStateAdapter(fragmentManager, lifecycle) { override fun getItemCount(): Int = 3 // Fragment 数量 override fun createFragment(position: Int): Fragment { return when (position) { 0 -> FragmentOne() 1 -> FragmentTwo() 2 -> FragmentThree() else -> throw IndexOutOfBoundsException() } } } ``` 其,getItemCount 返回 Fragment 数量,createFragment 返回指定位置的 Fragment 对象,例子返回了三个 FragmentOne、FragmentTwo、FragmentThree。 最后,在 Activity 或 Fragment 设置 ViewPager2 和 FragmentAdapter,例如: ```Kotlin val viewPager: ViewPager2 = findViewById(R.id.view_pager) val adapter = MyFragmentAdapter(supportFragmentManager, lifecycle) viewPager.adapter = adapter ``` 这样就实现了在 Fragment 之间滑动的功能。 ### 回答2: 要使用ViewPager2在Fragment之间滑动,首先需要在布局文件定义一个ViewPager2控件,并添加相应的Fragment。 在Activity,需要先获取ViewPager2控件的实例,并创建一个适配器来管理Fragment的切换。适配器需要继承FragmentStateAdapter,并重写getItemCount()方法和createFragment()方法。 getItemCount()方法返回Fragment的数量。 createFragment()方法根据position返回相应位置的Fragment。 然后,将适配器设置给ViewPager2控件。 接下来,就可以通过手势滑动或者使用setCurrentItem()方法来实现Fragment之间的切换。如果想要禁用滑动,可以使用setUserInputEnabled()方法来设置。 以下是一个简单的示例代码: 在布局文件: ``` <androidx.viewpager2.widget.ViewPager2 android:id="@+id/viewPager" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 在Activity: ```java ViewPager2 viewPager = findViewById(R.id.viewPager); FragmentStateAdapter adapter = new MyAdapter(this); viewPager.setAdapter(adapter); // 实现禁用滑动的功能 // viewPager.setUserInputEnabled(false); // 实现手动切换到指定Fragment的功能 // viewPager.setCurrentItem(2, false); ``` 适配器类: ```java private class MyAdapter extends FragmentStateAdapter { public MyAdapter(FragmentActivity fa) { super(fa); } @Override public int getItemCount() { return 3; // 返回Fragment的数量 } @NonNull @Override public Fragment createFragment(int position) { switch (position) { case 0: return new Fragment1(); case 1: return new Fragment2(); case 2: return new Fragment3(); default: return null; } } } ``` 这样,便可以在ViewPager2实现Fragment之间的滑动切换了。 ### 回答3: ViewPager2 是 Android Support Library 的一个组件,用于在多个 fragment 之间进行滑动切换。要在 fragment 之间使用 ViewPager2 进行滑动切换,首先需要进行以下几个步骤: 1. 添加依赖:在项目的 build.gradle 文件添加以下依赖: ``` implementation 'androidx.viewpager2:viewpager2:1.0.0' ``` 2. 创建 Fragment:创建需要滑动切换的多个 fragment,并在它们的布局文件添加相应的内容。 3. 创建 ViewPager2:在主 Activity 或者容器 fragment 的布局文件添加 ViewPager2 组件: ```xml <androidx.viewpager2.widget.ViewPager2 android:id="@+id/viewPager" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 4. 创建适配器:创建一个继承自 FragmentStateAdapter 的适配器类,用于管理 fragment 的创建和绑定。在适配器实现以下两个方法: ```java @Override public int getItemCount() { return fragmentList.size(); } @Override public Fragment createFragment(int position) { return fragmentList.get(position); } ``` 5. 绑定适配器:在主 Activity 或者容器 fragment 的代码,实例化 ViewPager2,并设置适配器: ```java ViewPager2 viewPager = findViewById(R.id.viewPager); viewPager.setAdapter(new MyAdapter(getSupportFragmentManager(), getLifecycle())); ``` 6. 处理切换事件:可以添加切换事件监听器,监听页面的切换操作,例如: ```java viewPager.registerOnPageChangeCallback(new ViewPager2.OnPageChangeCallback() { @Override public void onPageSelected(int position) { // 处理页面切换事件 } }); ``` 通过以上步骤,就可以在 fragment 之间使用 ViewPager2 进行滑动切换了。根据实际需求,还可以自定义每个 fragment 的布局和显示内容,以及指示器等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值