自定义RecyclerView实现垂直滑动的ViewPager

2024-06-15 10:58

本文主要是介绍自定义RecyclerView实现垂直滑动的ViewPager,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

android原生的ViewPager默认水平方向滑动翻页的。突然,项目里有个签到需求要做成上下滑动翻页。我的第一反应是应用ViewPager,可是它只适用于水平滑动的情景,可不可以继承ViewPager重写它的onTouchEvent方法,把检测水平滑动的手势改成垂直滑动手势呢?答案是肯定的,这种做法可以实现上下翻页的功能,但是快速滑动时会导致页面错乱,而且fragment需要依赖activity作为主体,耦合度比较高,控制逻辑比较复杂。后来,我下定决心换另外一种方式实现,把目光转移到RecyclerView。经过一番改造,自定义的RecycleViewPager控件也实现同样效果,与activity耦合度有所降低,页面几乎没有发生错乱情况,而且加上翻页时切入切出的动画。效果图如下:



RecyclerViewPager里面有几个比较重要方法:initAttrs()、onRestoreInstanceState()、dispatchTouchEvent()与onTouchEvent()。首先看下initAttrs方法,它是用来设置滑动因子、触发偏移等属性参数,在构造方法里面被调用。具体如下:

    private void initAttrs(Context context, AttributeSet attrs, int defStyle) {final TypedArray a = context.obtainStyledAttributes(attrs, com.frank.pager.R.styleable.RecyclerViewPager, defStyle, 0);mFlingFactor = a.getFloat(com.frank.pager.R.styleable.RecyclerViewPager_rvp_flingFactor, 0.15f);mTriggerOffset = a.getFloat(com.frank.pager.R.styleable.RecyclerViewPager_rvp_triggerOffset, 0.25f);mSinglePageFling = a.getBoolean(com.frank.pager.R.styleable.RecyclerViewPager_rvp_singlePageFling, mSinglePageFling);a.recycle();}

      其中,上述的自定义属性需要在attrs文件里声明:

    <declare-styleable name="RecyclerViewPager"><attr name="rvp_triggerOffset" format="float" /><attr name="rvp_flingFactor" format="float" /><attr name="rvp_singlePageFling" format="boolean" /></declare-styleable>

      重写RecyclerView父类的onRestoreInstanceState方法,恢复上次记录的状态。

    protected void onRestoreInstanceState(Parcelable state) {try {//通过反射获得布局状态的属性Field fLayoutState = state.getClass().getDeclaredField("mLayoutState");//设置可以访问fLayoutState.setAccessible(true);Object layoutState = fLayoutState.get(state);//同样地,通过反射获得停止偏移与停止位置的属性Field fAnchorOffset = layoutState.getClass().getDeclaredField("mAnchorOffset");Field fAnchorPosition = layoutState.getClass().getDeclaredField("mAnchorPosition");fAnchorPosition.setAccessible(true);fAnchorOffset.setAccessible(true);if (fAnchorOffset.getInt(layoutState) > 0) {//如果偏移值大于0,设置停止位置为上一页fAnchorPosition.set(layoutState, fAnchorPosition.getInt(layoutState) - 1);} else if (fAnchorOffset.getInt(layoutState) < 0) {//如果偏移值小于0,设置停止位置为下一页fAnchorPosition.set(layoutState, fAnchorPosition.getInt(layoutState) + 1);}fAnchorOffset.setInt(layoutState, 0);} catch (Throwable e) {e.printStackTrace();}super.onRestoreInstanceState(state);}

重写父类的onDispatchTouchEvent方法,根据是否允许水平移动,返回X轴孩子的中心点或者是Y轴孩子的中心点。

    public boolean dispatchTouchEvent(MotionEvent ev) {//如果处于按下状态,并且LayoutManager不为空if (ev.getAction() == MotionEvent.ACTION_DOWN && getLayoutManager() != null) {//如果允许水平移动,返回X坐标轴的孩子中心点,否则返回Y坐标轴的孩子中心点mPositionOnTouchDown = getLayoutManager().canScrollHorizontally()? ViewUtils.getCenterXChildPosition(this): ViewUtils.getCenterYChildPosition(this);}return super.dispatchTouchEvent(ev);}

另外,重写父类的onTouchEvent方法,记录移动过程的最大值与最小值。

    public boolean onTouchEvent(MotionEvent e) {//记录移动过程的最大值与最小值if (e.getAction() == MotionEvent.ACTION_MOVE) {if (mCurView != null) {mMaxLeftWhenDragging = Math.max(mCurView.getLeft(), mMaxLeftWhenDragging);mMaxTopWhenDragging = Math.max(mCurView.getTop(), mMaxTopWhenDragging);mMinLeftWhenDragging = Math.min(mCurView.getLeft(), mMinLeftWhenDragging);mMinTopWhenDragging = Math.min(mCurView.getTop(), mMinTopWhenDragging);}}return super.onTouchEvent(e);}

接下来看下对应的Adapter,主要重写父类RecyclerView.Adapter的几个方法实现注册内容监听器、注销内容监听器、绑定窗体、解除绑定、设置布局宽度与高度。里面还有一个内部类:实现真正的业务逻辑处理的适配器。主要是在onBindViewHolder来处理UI界面的展示,其中负责更新页面的方法如下:

    //更新未签提示private void updateWaitSignTip(SimpleViewHolder mHolder){currentPosition = mHolder.getAdapterPosition();totalSize = getItemCount();if(totalSize > 1){if(currentPosition == 0){//在第一页,只显示向上滑动mHolder.rl_sign_up.setVisibility(View.VISIBLE);mHolder.tv_sign_up.setVisibility(View.VISIBLE);mHolder.rl_sign_down.setVisibility(View.GONE);mHolder.tv_sign_up.setText("请向上滑动,您有" + (totalSize-1) + "个未签");}else if(currentPosition == (totalSize - 1)){//在最后一页,只显示向下滑动mHolder.rl_sign_up.setVisibility(View.GONE);mHolder.tv_sign_up.setVisibility(View.GONE);mHolder.rl_sign_down.setVisibility(View.VISIBLE);mHolder.tv_sign_down.setText("请向下滑动,您有" + (totalSize-1) + "个未签");}else {//向上向下同时显示mHolder.rl_sign_up.setVisibility(View.VISIBLE);mHolder.tv_sign_up.setVisibility(View.VISIBLE);mHolder.rl_sign_down.setVisibility(View.VISIBLE);mHolder.tv_sign_up.setText("请向上滑动,您有" + (totalSize-currentPosition-1) + "个未签");mHolder.tv_sign_down.setText("请向下滑动,您有" + (currentPosition) + "个未签");}}}

在activity里,初始化自定义控件:

     LinearLayoutManager layout = new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false);mRecyclerView.setTriggerOffset(0.15f);//设置触发偏移量mRecyclerView.setFlingFactor(0.25f);//设置滑动因子mRecyclerView.setHasFixedSize(true);//设置自适应屏幕尺寸mRecyclerView.setLayoutManager(layout);mRecyclerView.setAdapter(new RecyclerViewPagerAdapter.LayoutAdapter(this, 5));


最后,为了实现上下翻页切入切出的动画效果,需要设置RecyclerView的滚动监听器,重写onScrolled方法。

     public void onScrolled(RecyclerView recyclerView, int i, int i2) {//获得RecyclerView的item数量int childCount = mRecyclerView.getChildCount();//获得第一个item的宽度int width = mRecyclerView.getChildAt(0).getWidth();//计算padding大小int padding = (mRecyclerView.getWidth() - width) / 2;//遍历所有itemfor (int j = 0; j < childCount; j++) {View v = recyclerView.getChildAt(j);float rate = 0;if (v.getTop() <= padding) {//如果top比padding小,说明该页面处于移出状态if (v.getTop() >= padding - v.getHeight()) {//根据padding、top与height计算缩放比例rate = (padding - v.getTop()) * 1f / v.getHeight();} else {rate = 1;}//设置XY轴的逐渐缩小v.setScaleX(1 - rate * 0.1f);v.setScaleY(1 - rate * 0.1f);} else {//如果top比padding大,说明该页面处于移ru状态if (v.getTop() <= recyclerView.getHeight() - padding) {//根据padding、top与height计算缩放比例rate = (recyclerView.getHeight() - padding - v.getTop()) * 1f / v.getHeight();}//设置XY轴的逐渐放大v.setScaleX(0.9f + rate * 0.1f);v.setScaleY(0.9f + rate * 0.1f);}}}

好了,直到这里,已经介绍完利用自定义RecyclerView实现上下翻页以及切入切出的动画效果。如有问题,希望大家指出,共同学习同步进步。点击下载demo: http://download.csdn.net/detail/u011686167/9660954

 


这篇关于自定义RecyclerView实现垂直滑动的ViewPager的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1063263

相关文章

MySQL索引的优化之LIKE模糊查询功能实现

《MySQL索引的优化之LIKE模糊查询功能实现》:本文主要介绍MySQL索引的优化之LIKE模糊查询功能实现,本文通过示例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一、前缀匹配优化二、后缀匹配优化三、中间匹配优化四、覆盖索引优化五、减少查询范围六、避免通配符开头七、使用外部搜索引擎八、分

Python实现特殊字符判断并去掉非字母和数字的特殊字符

《Python实现特殊字符判断并去掉非字母和数字的特殊字符》在Python中,可以通过多种方法来判断字符串中是否包含非字母、数字的特殊字符,并将这些特殊字符去掉,本文为大家整理了一些常用的,希望对大家... 目录1. 使用正则表达式判断字符串中是否包含特殊字符去掉字符串中的特殊字符2. 使用 str.isa

Spring Boot 集成 Quartz并使用Cron 表达式实现定时任务

《SpringBoot集成Quartz并使用Cron表达式实现定时任务》本篇文章介绍了如何在SpringBoot中集成Quartz进行定时任务调度,并通过Cron表达式控制任务... 目录前言1. 添加 Quartz 依赖2. 创建 Quartz 任务3. 配置 Quartz 任务调度4. 启动 Sprin

Android实现悬浮按钮功能

《Android实现悬浮按钮功能》在很多场景中,我们希望在应用或系统任意界面上都能看到一个小的“悬浮按钮”(FloatingButton),用来快速启动工具、展示未读信息或快捷操作,所以本文给大家介绍... 目录一、项目概述二、相关技术知识三、实现思路四、整合代码4.1 Java 代码(MainActivi

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停

基于Python实现读取嵌套压缩包下文件的方法

《基于Python实现读取嵌套压缩包下文件的方法》工作中遇到的问题,需要用Python实现嵌套压缩包下文件读取,本文给大家介绍了详细的解决方法,并有相关的代码示例供大家参考,需要的朋友可以参考下... 目录思路完整代码代码优化思路打开外层zip压缩包并遍历文件:使用with zipfile.ZipFil

Python实现word文档内容智能提取以及合成

《Python实现word文档内容智能提取以及合成》这篇文章主要为大家详细介绍了如何使用Python实现从10个左右的docx文档中抽取内容,再调整语言风格后生成新的文档,感兴趣的小伙伴可以了解一下... 目录核心思路技术路径实现步骤阶段一:准备工作阶段二:内容提取 (python 脚本)阶段三:语言风格调

C#实现将Excel表格转换为图片(JPG/ PNG)

《C#实现将Excel表格转换为图片(JPG/PNG)》Excel表格可能会因为不同设备或字体缺失等问题,导致格式错乱或数据显示异常,转换为图片后,能确保数据的排版等保持一致,下面我们看看如何使用C... 目录通过C# 转换Excel工作表到图片通过C# 转换指定单元格区域到图片知识扩展C# 将 Excel

基于Java实现回调监听工具类

《基于Java实现回调监听工具类》这篇文章主要为大家详细介绍了如何基于Java实现一个回调监听工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录监听接口类 Listenable实际用法打印结果首先,会用到 函数式接口 Consumer, 通过这个可以解耦回调方法,下面先写一个

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析