文章目录

  • 写在前面
  • 效果图
  • 原理分析
  • 核心代码
  • 源码地址

写在前面

本文基于 ViewPager2 实现的 Banner 效果,进而实现了仿淘宝、京东Banner滑动至最后一页时继续滑动来查看图文详情的效果。关于 ViewPager2 的原理及其封装,可以参见之前的两篇文章:
1、Android 深入理解ViewPager2原理及其实践(上篇)
2、Android 深入理解ViewPager2原理及其实践(下篇)

效果图

原理分析

  • Banner与右侧的查看更多View都是子View,被父View包裹,默认Banner的宽度是match_parent,而查看更多则是在屏幕的右侧,处于不可见状态;
  • Banner进行左右滑动时,当前的滑动事件是在Banner中消费的,即父View不会进行拦截。
  • Banner滑动到最右侧且要继续滑动时,此时父View会进行事件的拦截,从而事件由父View接管,并在父ViewonTouchEvent()中消费事件,此时就可以滑动父View中的内容了。怎么滑动呢?在MOVE事件时通过scrollTo()/scrollBy()滑动,而在UP/CANCEL事件时,需要通过ScrollerstartScroll()自动滑动到查看更多子View的左侧或右侧,从而完成一次事件的消费;
  • UP/CANCEL事件触发时,查看更多子View滑动的距离超过一半,认为需要触发查看更多操作了,当然这里的值都可以自行设置。

核心代码

  • TJBannerFragment.kt
/*** 仿淘宝京东宝贝详情Fragment*/
class TJBannerFragment : BaseFragment() {private val mModels: MutableList<Any> = mutableListOf()private val mContainer: VpLoadMoreView by id(R.id.vp2_load_more)override fun getLayoutId(): Int {return R.layout.fragment_tx_news_n}override fun onViewCreated(view: View, savedInstanceState: Bundle?) {initVerticalTxScroll()}private fun initVerticalTxScroll() {mModels.add(TxNewsModel(MConstant.IMG_4, "美轮美奂节目", "奥运五环缓缓升起"))mModels.add(TxNewsModel(MConstant.IMG_1, "精美商品", "9块9包邮"))mContainer.setData(mModels) {showToast("打开更多页面")}}
}
  • VpLoadMoreView.kt(父View)
class VpLoadMoreView @JvmOverloads constructor(context: Context,attrs: AttributeSet? = null,defStyle: Int = 0,
) : LinearLayout(context, attrs, defStyle) {private val mMVPager2: MVPager2 by id(R.id.mvp_pager2)private var mNeedIntercept: Boolean = false //是否需要拦截VP2事件private val mLoadMoreContainer: LinearLayout by id(R.id.load_more_container)private val mIvArrow: ImageView by id(R.id.iv_pull)private val mTvTips: TextView by id(R.id.tv_tips)private var mCurPos: Int = 0 //Banner当前滑动的位置private var mLastX = 0fprivate var mLastDownX = 0f //用于判断滑动方向private var mMenuWidth = 0 //加载更多View的宽度private var mShowMoreMenuWidth = 0 //加载更多发生变化时的宽度private var mLastStatus = false // 默认箭头样式private var mAction: (() -> Unit)? = nullprivate var mScroller: OverScrollerprivate var isTouchLeft = false //是否是向左滑动private var animRightStart = RotateAnimation(0f, -180f,Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f).apply {duration = 300fillAfter = true}private var animRightEnd = RotateAnimation(-180f, 0f,Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f).apply {duration = 300fillAfter = true}init {orientation = HORIZONTALView.inflate(context, R.layout.fragment_tx_news, this)mScroller = OverScroller(context)}/*** @param mModels 要加载的数据* @param action 回调Action*/fun setData(mModels: MutableList<Any>, action: () -> Unit) {this.mAction = actionmMVPager2.setModels(mModels).setLoop(false) //非循环模式.setIndicatorShow(false).setLoader(TxNewsLoader(mModels)).setPageTransformer(CompositePageTransformer().apply {addTransformer(MarginPageTransformer(15))}).setOrientation(ViewPager2.ORIENTATION_HORIZONTAL).setAutoPlay(false).setOnBannerClickListener(object : OnBannerClickListener {override fun onItemClick(position: Int) {showToast(mModels[position].toString())}}).registerOnPageChangeCallback(object : ViewPager2.OnPageChangeCallback() {override fun onPageScrollStateChanged(state: Int) {if (mCurPos == mModels.lastIndex && isTouchLeft && state == ViewPager2.SCROLL_STATE_DRAGGING) {//Banner在最后一页 & 手势往左滑动 & 当前是滑动状态mNeedIntercept = true //父View可以拦截mMVPager2.setUserInputEnabled(false) //VP2设置为不可滑动}}override fun onPageSelected(position: Int) {mCurPos = position}}).start()}override fun onLayout(changed: Boolean, l: Int, t: Int, r: Int, b: Int) {mMenuWidth = mLoadMoreContainer.measuredWidthmShowMoreMenuWidth = mMenuWidth / 3 * 2super.onLayout(changed, l, t, r, b)}override fun dispatchTouchEvent(ev: MotionEvent?): Boolean {when (ev?.action) {MotionEvent.ACTION_DOWN -> {mLastX = ev.xmLastDownX = ev.x}MotionEvent.ACTION_MOVE -> {isTouchLeft = mLastDownX - ev.x > 0 //判断滑动方向}}return super.dispatchTouchEvent(ev)}override fun onInterceptTouchEvent(ev: MotionEvent?): Boolean {var isIntercept = falsewhen (ev?.action) {MotionEvent.ACTION_MOVE -> isIntercept = mNeedIntercept //是否拦截Move事件}//log("ev?.action: ${ev?.action},isIntercept: $isIntercept")return isIntercept}@SuppressLint("ClickableViewAccessibility")override fun onTouchEvent(ev: MotionEvent?): Boolean {when (ev?.action) {MotionEvent.ACTION_MOVE -> {val mDeltaX = mLastX - ev.xif (mDeltaX > 0) {//向左滑动if (mDeltaX >= mMenuWidth || scrollX + mDeltaX >= mMenuWidth) {//右边缘检测scrollTo(mMenuWidth, 0)return super.onTouchEvent(ev)}} else if (mDeltaX < 0) {//向右滑动if (scrollX + mDeltaX <= 0) {//左边缘检测scrollTo(0, 0)return super.onTouchEvent(ev)}}showLoadMoreAnim(scrollX + mDeltaX)scrollBy(mDeltaX.toInt(), 0)mLastX = ev.x}MotionEvent.ACTION_UP, MotionEvent.ACTION_CANCEL -> {smoothCloseMenu()mNeedIntercept = falsemMVPager2.setUserInputEnabled(true)//执行回调val mDeltaX = mLastX - ev.xif (scrollX + mDeltaX >= mShowMoreMenuWidth) {mAction?.invoke()}}}return super.onTouchEvent(ev)}private fun smoothCloseMenu() {mScroller.forceFinished(true)/*** 左上为正,右下为负* startX:X轴开始位置* startY: Y轴结束位置* dx:X轴滑动距离* dy:Y轴滑动距离* duration:滑动时间*/mScroller.startScroll(scrollX, 0, -scrollX, 0, 300)invalidate()}override fun computeScroll() {if (mScroller.computeScrollOffset()) {showLoadMoreAnim(0f) //动画还原scrollTo(mScroller.currX, mScroller.currY)invalidate()}}private fun showLoadMoreAnim(dx: Float) {val showLoadMore = dx >= mShowMoreMenuWidthif (mLastStatus == showLoadMore) returnif (showLoadMore) {mIvArrow.startAnimation(animRightStart)mTvTips.text = "释放查看图文详情"mLastStatus = true} else {mIvArrow.startAnimation(animRightEnd)mTvTips.text = "滑动查看图文详情"mLastStatus = false}}
}

父View的注释很清晰,不用过多解释了,这里需要注意一点,已知在Banner的最后一页滑动时需要判断滑动方向:继续向左滑动,需要父View拦截滑动事件并自己进行消费;向右滑动时,父View不需要处理滑动事件,仍由Banner进行事件消费。

滑动方向需要起始位置(DOWN事件)的X坐标 - 滑动时的X坐标(MOVE事件) 的差值进行判断,那问题在哪里取起始位置的X坐标呢?在父ViewonInterceptTouchEvent()->DOWN事件里吗?这里是不行的,因为滑动方向是在MOVE事件里判断的,在父ViewonInterceptTouchEvent()->DOWN事件里拦截的话,后续事件不会往Banner里传递了。这里可以选择在父ViewdispatchTouchEvent()->DOWN事件里即可解决。

VpLoadMoreView对应的XML布局

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@color/white"android:orientation="horizontal"tools:parentTag="android.widget.LinearLayout"><!--ViewPager2--><org.ninetripods.lib_viewpager2.MVPager2android:id="@+id/mvp_pager2"android:layout_width="match_parent"android:layout_height="200dp"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintLeft_toLeftOf="parent"app:layout_constraintRight_toRightOf="parent"app:layout_constraintTop_toTopOf="parent" /><!--加载更多View--><LinearLayoutandroid:id="@+id/load_more_container"android:layout_width="100dp"android:layout_height="200dp"android:gravity="center_vertical"android:orientation="horizontal"><ImageViewandroid:id="@+id/iv_pull"android:layout_width="18dp"android:layout_height="18dp"android:layout_gravity="center_vertical"android:layout_marginStart="10dp"android:src="@drawable/icon_arrow_pull" /><TextViewandroid:id="@+id/tv_tips"android:layout_width="16dp"android:layout_height="match_parent"android:layout_marginStart="10dp"android:gravity="center_vertical"android:text="滑动查看图文详情"android:textColor="#333333"android:textSize="14sp"android:textStyle="bold" /></LinearLayout>
</merge>

这里的父View(VpLoadMoreView)LinearLayout,且必须是横向布局,XML的顶层布局使用的merge标签,这样既可以优化一层布局,又可以在父View中直接操作加载图文详情的子View

源码地址

完整代码地址参见:Android仿淘宝、京东Banner滑动至最后查看图文详情

Android仿淘宝、京东Banner滑动查看图文详情相关推荐

  1. Android仿淘宝京东商品规格参数颜色筛选

    Android 选择商品属性sku 最近项目中使用SKU属性查询,类似淘宝京东商品的选择,在网上查询了弄了几个源码看看,发现还是实现不了多属性选择问题,再原基础上改动相当费事,所以想干脆自己处理这个问 ...

  2. Android 仿淘宝京东等我的订单界面及任意列表拓展

    概述 目前像淘宝及展示列表等都有多个item展示的需求,可能大多数如果没做过,第一眼就是ListView去嵌套ListView,虽然这样是可以完成,但是这样做会导致手机过度绘制,为什么呢?因为当一个I ...

  3. Android 仿淘宝京东商品详情页阻力翻页效果

    原文链接:http://code.taobao.org/p/android-example/diff/46/trunk/%E5%95%86%E5%9F%8E%E8%AF%A6%E6%83%85/src ...

  4. Android 仿淘宝京东商品详情视频+图片与图片第一帧获取

    近日项目有个新需求就是把原本的商品详情只有图片展示,改为视频+图片方式展示. 此博客只提供记录,与思路具体根据自己需求实现.首先想到的是Google搜索下别人的实现方式来参考实现发现不怎么适合项目需求 ...

  5. Android仿淘宝详情页面viewPager滑动到最后一张图片跳转的功能

    需要做一个仿淘宝客户端ViewPager滑动到最后一页,再拖动的时候跳到详情的功能,刚开始我也迷糊了,通过查阅相关资料发现有好多种实现方法,下面小编给大家分享实例代码,感兴趣的朋友一起看看吧 需要做一 ...

  6. 仿淘宝京东商品规格属性选择的最简单实现

    仿淘宝京东商品规格属性选择的最简单实现 商城里面的规格选择,网上大部分是自定义控件实现的,显得很是麻烦,而我的实现方式是大家最常用的控件RecyclerView,特点是性能好,简单.废话不多说,先看实 ...

  7. 纯css仿淘宝京东导航菜单栏

    <html> <head> <meta http-equiv="Content-Type" content="text/html; char ...

  8. JavaScript仿淘宝京东放大镜效果(鼠标事件)------JavaScript学习之路10

    JavaScript仿淘宝京东放大镜效果 注意 一定计算好放大比例,本程序放大5倍,具体放大倍数,自定 效果 完整源码 <!DOCTYPE html> <html lang=&quo ...

  9. Android仿淘宝首页UI(附代源代码及示例图片)

    Android仿淘宝首页UI(附代源代码及示例图片) 可以收获 运行出来的效果 部分代码 源代码 可以收获 更改Layout中的文字和drawble中的图片即可生成适应于不同情景的APP,帮助开发者完 ...

最新文章

  1. day 2 基本类型和函数
  2. python中6 2是什么意思_python2.6中SyntaxError是什么错误?
  3. mac终端python不能显示中文_Matplotlib为Mac显示中文,ForMac
  4. python 画pr曲线
  5. hdu 3333 树状数组+离线处理
  6. iOS开发网络篇—网络编程基础
  7. python输出个数、给定一个n*n的矩阵m_简述Numpy
  8. server2008 mysql数据库病毒_sql server 2008 数据库可疑的解决步骤
  9. 华为马海旭:+智能,IoT行业云服务使能产业物联网
  10. SolidWorks的发展历史(1994~2007)
  11. Kylin多维分析引擎(四):Kylin Cude构建流程详解
  12. 学习编程,应该从哪里开始学习呢?
  13. C++程序员爱的表白,心形图示例
  14. 值传递和引用传递是什么?
  15. Centos磁盘管理
  16. 台达DVP-EH3系列PLC如何实现远程编程调试和程序上下载
  17. 计算机共享协议书,联合体资质共享协议书.doc
  18. 基于IOCP的局域网监控系统
  19. X3D: Expanding Architectures for Efficient Video Recognition 论文学习
  20. Cisco 3850交换机保存配置后重启配置文件丢失

热门文章

  1. 在微信开发者工具中,使用WeUI前端美化框架,微信小程序
  2. redis mysql 雪崩_Redis缓存雪崩问题
  3. Python实现多图合并成长图脚本
  4. 树莓派安装kali2020安装教程和坑点
  5. 基于安卓的校园跳蚤市场app
  6. android显示器,古董 or 真香? ThinkVision 28: 28'' 4K Android 显示器开箱
  7. PyCharm如何自定义调整字体大小的快捷键
  8. 灰狼优化算法(GWO)附代码
  9. oracle灾备同步_【oracle灾备方案系列】基于DDS的Oracle复制容灾方案(三)
  10. ROS学习----Publisher与Subscriber