前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >处理 WebView 与 ViewPager 滑动冲突

处理 WebView 与 ViewPager 滑动冲突

作者头像
技术小黑屋
发布2020-02-10 22:20:03
1.9K0
发布2020-02-10 22:20:03
举报
文章被收录于专栏:技术小黑屋技术小黑屋

问题场景 在项目的App中,有一个ViewPager,它内部包含了WebView,而内部的webview加载了一个可以滑动的网页。

当我们在网页滑动的时候,会直接切换到下一个viewpager的页面,而不是优先响应webview的滑动。

具体的效果如视频

期待的示例效果

解决思路

其实思路还是比较简单,大概如下

  • 优先响应webview内部滑动
  • 如果webview内部滑动完成,则响应外部的滑动

那么问题就来了,怎么判断webview内部滑动结束就是解决问题的关键了。

解决问题的关键就在于WebView.onOverScrolled方法

WebView.onOverScrolled
WebView.onOverScrolled

看了上面的文档,我们可能还是有一些疑惑,到底什么是overScroll。正所谓一图胜千言,看一下下图就知道了。

Webview overscrolled effect
Webview overscrolled effect

上面红框的内容就是overScroll的效果,其实就是划过了的意思(英语中over有过的意思)

了解了上面的信息,我们具体的实施办法也就有了。

  • 在WebView的onTouchEvent事件为ACTION_DOWN时,查找父视图是否是可以滑动的视图(如ViewPager),如果是,则通过requestDisallowInterceptTouchEvent(true)调用,请求父视图不要拦截touchEvent
  • 如果WebView不再响应内部滑动(即onOverScrolled中clampedX或者clampedY值为true),我们再起调用requestDisallowInterceptTouchEvent(false)请求父视图恢复拦截处理touchEvent.

核心代码

代码语言:javascript
复制
override fun onTouchEvent(event: MotionEvent): Boolean {

   if (event.action == MotionEvent.ACTION_DOWN) {

       val viewParent = findViewParentIfNeeds(this)

       viewParent?.requestDisallowInterceptTouchEvent(true)

   }

   return super.onTouchEvent(event)

}



override fun onOverScrolled(scrollX: Int, scrollY: Int, clampedX: Boolean, clampedY: Boolean) {

   dumpMessage("onOverScrolled scrollX=" + scrollX + ";scrollY=" + scrollY

           + ";clampedX=" + clampedX + ";clampedY=" + clampedY)

   if (clampedX) {

       val viewParent = findViewParentIfNeeds(this)

       viewParent?.requestDisallowInterceptTouchEvent(false)

   }

   super.onOverScrolled(scrollX, scrollY, clampedX, clampedY)

}



private fun findViewParentIfNeeds(tag: View): ViewParent? {

   val parent = tag.parent

   if (parent == null) {

       return parent

   }

   return if (parent is ViewPager ||

           parent is AbsListView ||

           parent is ScrollView ||

           parent is HorizontalScrollView ||

           parent is GridView) {

       parent

   } else {

       if (parent is View) {

           findViewParentIfNeeds(parent as View)

       } else {

           parent

       }

   }

}

利用上面的代码,我们就能完美的解决水平滑动的问题,对于垂直纵向的问题,大家可以参考本文方法做类似实现。

示例代码

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 解决思路
  • 示例代码
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档