Android两条并排RecyclerView实时联动滑动增强
在附录1中,我初步实现了两条垂直摆放的RecyclerView的实时联动滚动,但是附录文章1的实现存在一定问题,比如当用户的手指同时在屏幕的两条RecyclerView区域内滑动,将发生异常。还有就是,比如在RecyclerView1正在滚动时候,用户又在RecyclerView2的区域内手指滑动,这样将触发循环嵌套滚动事件,引发错误,因此本文在附录1的基础上改进,通过拦截处理系统的触摸事件解决造成RecyclerView滚动引发的问题。
代码:
package zhangphil.test; import android.graphics.Color; import android.os.Bundle; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.support.v7.app.AppCompatActivity; import android.support.v7.widget.LinearLayoutManager; import android.support.v7.widget.RecyclerView; import android.support.v7.widget.StaggeredGridLayoutManager; import android.view.LayoutInflater; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; import java.util.ArrayList; public class RVActivity extends AppCompatActivity { private RecyclerView mRecyclerView1; private RecyclerView mRecyclerView2; private ArrayList<Integer> mItems; /** * 返回true,表示拦截事件。 * 返回false,表示不做任何处理,交给子View处理。 * * @param event * @return */ @Override public boolean dispatchTouchEvent(MotionEvent event) { /** * 如果用户的手指同时放在屏幕上滑动,不要触发滚动事件。 * */ if (event.getPointerCount() >= 2) { return true; } /** * 如果左侧的RecyclerView1在滚动中,但是此时用户又在RecyclerView2中触发滚动事件,则停止所有滚动,等待新一轮滚动。 * */ if (mRecyclerView1.getScrollState() != RecyclerView.SCROLL_STATE_IDLE) { if (touchEventInView(mRecyclerView2, event.getX(), event.getY())) { mRecyclerView1.stopScroll(); mRecyclerView2.stopScroll(); return true; } } /** * 如果右侧的RecyclerView2在滚动中,但是此时用户又在RecyclerView1中触发滚动事件,则停止所有滚动,等待新一轮滚动。 * */ if (mRecyclerView2.getScrollState() != RecyclerView.SCROLL_STATE_IDLE) { if (touchEventInView(mRecyclerView1, event.getX(), event.getY())) { mRecyclerView2.stopScroll(); mRecyclerView1.stopScroll(); return true; } } return super.dispatchTouchEvent(event); } private boolean touchEventInView(View view, float x, float y) { if (view == null) { return false; } int[] location = new int[2]; view.getLocationOnScreen(location); int left = location[0]; int top = location[1]; int right = left + view.getMeasuredWidth(); int bottom = top + view.getMeasuredHeight(); if (y >= top && y <= bottom && x >= left && x <= right) { return true; } return false; } @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.rv_activity); mItems = new ArrayList<>(); for (int i = 0; i < 10000; i++) { mItems.add(i); } initRecyclerView1(); initRecyclerView2(); mRecyclerView1.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { if (recyclerView.getScrollState() != RecyclerView.SCROLL_STATE_IDLE) { mRecyclerView2.scrollBy(dx, dy); } } }); mRecyclerView2.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { if (recyclerView.getScrollState() != RecyclerView.SCROLL_STATE_IDLE) { mRecyclerView1.scrollBy(dx, dy); } } }); } private void initRecyclerView1() { mRecyclerView1 = findViewById(R.id.recycler_view_1); LinearLayoutManager layoutManager = new LinearLayoutManager(this); layoutManager.setOrientation(LinearLayoutManager.VERTICAL); mRecyclerView1.setLayoutManager(layoutManager); RecyclerViewAdapter mAdapter = new RecyclerViewAdapter(1); mRecyclerView1.setAdapter(mAdapter); } private void initRecyclerView2() { mRecyclerView2 = findViewById(R.id.recycler_view_2); StaggeredGridLayoutManager layoutManager = new StaggeredGridLayoutManager(12, StaggeredGridLayoutManager.VERTICAL); layoutManager.setOrientation(LinearLayoutManager.VERTICAL); mRecyclerView2.setLayoutManager(layoutManager); RecyclerViewAdapter mAdapter = new RecyclerViewAdapter(2); mRecyclerView2.setAdapter(mAdapter); } private class RecyclerViewAdapter extends RecyclerView.Adapter<MyVH> { private int id; public RecyclerViewAdapter(int id) { this.id = id; } @NonNull @Override public MyVH onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { View view = LayoutInflater.from(getApplicationContext()).inflate(R.layout.rv_item, parent, false); return new MyVH(view); } @Override public void onBindViewHolder(@NonNull MyVH holder, int position) { holder.text1.setText("RecyclerView:" + id); holder.text2.setText(mItems.get(position) + ""); switch (id) { case 1: holder.text1.setBackgroundColor(Color.RED); break; case 2: holder.text1.setBackgroundColor(Color.BLUE); break; } } @Override public int getItemCount() { return mItems.size(); } } private class MyVH extends RecyclerView.ViewHolder { public TextView text1; public TextView text2; public MyVH(View itemView) { super(itemView); text1 = itemView.findViewById(R.id.text1); text1.setTextColor(Color.WHITE); text2 = itemView.findViewById(R.id.text2); text2.setTextColor(Color.DKGRAY); } } }
相应布局文件:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal"> <android.support.v7.widget.RecyclerView android:id="@+id/recycler_view_1" android:layout_width="100dp" android:layout_height="match_parent" /> <View android:layout_width="1px" android:layout_height="match_parent" android:background="#e0e0e0" /> <HorizontalScrollView android:layout_width="match_parent" android:layout_height="match_parent"> <android.support.v7.widget.RecyclerView android:id="@+id/recycler_view_2" android:layout_width="1000dp" android:layout_height="match_parent" /> </HorizontalScrollView> </LinearLayout>
附录:
1,《Android若干条并排RecyclerView滑动实时联动》链接:https://blog.csdn.net/zhangphil/article/details/79925803
2,《 Android dispatchTouchEvent检测多点触摸事件是否落入在某一个View区域内》链接:https://blog.csdn.net/zhangphil/article/details/80067150
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Android dispatchTouchEvent检测多点触摸事件是否落入在某一个View区域内
Android dispatchTouchEvent检测多点触摸事件是否落入在某一个View区域内 检测发生在Android设备屏幕上到的点击事件是否发生在某一个View的区域内,重点方法是touchEventInView,该方法进行具体的检测,如果返回true,则表明该事件是发生在当前的参数view范围内,返回false则不是。 利用dispatchTouchEvent传出发生在全部区域上的点击事件,然后用touchEventInView计算判断。 dispatchTouchEvent返回true,则表明该事件消费掉了,不需要再次进行事件传递。换句话说,如果dispatchTouchEvent返回true,相当于对触摸事件进行了拦截处理。如果不想拦截不做任何处理,切忌不要简单粗暴的返回false,返回false导致意想不到的结果,而是要在dispatchTouchEvent里面利用super进行返回: return super.dispatchTouchEvent(event); 例: package zhangphil.test; import android.os.Bundle;...
- 下一篇
Android 沉浸式状态栏的多种样式
小菜最近正在处理客户端顶部沉浸式展示图片,借此整理了一下小菜自己研究测试的沉浸式状态栏。 沉浸式状态栏大家都很熟悉,即 APP 界面图片延伸到状态栏, 应用本身沉浸于状态栏,即顶部不会默认展示系统的黑条。因为小菜技术有限,理解不透彻,所以仅分享一下自己应用测试中可以呈现的几种样式。 基本样式 公共的步骤: 布局文件中添加使用 Toolbar 控件(纯色 Toolbar 背景色为颜色,图片 Toolbar 样式设置背景色为图片或添加一个 ImageView 控件),在文件根布局与 Toolbar 中添加 android:fitsSystemWindows="true",这个很重要,可以使背景图片延伸至状态栏,当然在 Java 文件中设置一样的效果; <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS6,CentOS7官方镜像安装Oracle11G
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- CentOS8编译安装MySQL8.0.19
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Windows10,CentOS7,CentOS8安装Nodejs环境
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7