android -------- RecyclerView的可(多个Item水平,垂直)滚动列表的实现
RecyclerView的可滚动的列表实现,视图相邻的视图在屏幕上部分或完全可见(水平3个item,第一个和第三个显示一部分,第二个完全显示)
效果如图:

基于RecyclerView完成的(自定义控件)
public class DiscreteRecyclerView extends RecyclerView {
public static final int NO_POSITION = DiscreteRecyclerLayoutManager.NO_POSITION;
private static final int DEFAULT_ORIENTATION = DSVOrientation.HORIZONTAL.ordinal();
private DiscreteRecyclerLayoutManager layoutManager;
private List<ScrollStateChangeListener> scrollStateChangeListeners;
private List<OnItemChangedListener> onItemChangedListeners;
private boolean isOverScrollEnabled;
public DiscreteRecyclerView(Context context) {
super(context);
init(null);
}
public DiscreteRecyclerView(Context context, AttributeSet attrs) {
super(context, attrs);
init(attrs);
}
public DiscreteRecyclerView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(attrs);
}
private void init(AttributeSet attrs) {
scrollStateChangeListeners = new ArrayList<>();
onItemChangedListeners = new ArrayList<>();
int orientation = DEFAULT_ORIENTATION;
if (attrs != null) {
TypedArray ta = getContext().obtainStyledAttributes(attrs, R.styleable.DiscreteRecyclerView);
orientation = ta.getInt(R.styleable.DiscreteRecyclerView_dsv_orientation, DEFAULT_ORIENTATION);
ta.recycle();
}
isOverScrollEnabled = getOverScrollMode() != OVER_SCROLL_NEVER;
layoutManager = new DiscreteRecyclerLayoutManager(
getContext(), new ScrollStateListener(),
DSVOrientation.values()[orientation]);
setLayoutManager(layoutManager);
}
@Override
public void setLayoutManager(LayoutManager layout) {
if (layout instanceof DiscreteRecyclerLayoutManager) {
super.setLayoutManager(layout);
} else {
throw new IllegalArgumentException(getContext().getString(R.string.dsv_ex_msg_dont_set_lm));
}
}
@Override
public boolean fling(int velocityX, int velocityY) {
boolean isFling = super.fling(velocityX, velocityY);
if (isFling) {
layoutManager.onFling(velocityX, velocityY);
} else {
layoutManager.returnToCurrentPosition();
}
return isFling;
}
@Nullable
public ViewHolder getViewHolder(int position) {
View view = layoutManager.findViewByPosition(position);
return view != null ? getChildViewHolder(view) : null;
}
/**
* @return adapter position of the current item or -1 if nothing is selected
*/
public int getCurrentItem() {
return layoutManager.getCurrentPosition();
}
public void setItemTransformer(DiscreteScrollItemTransformer transformer) {
layoutManager.setItemTransformer(transformer);
}
public void setItemTransitionTimeMillis(@IntRange(from = 10) int millis) {
layoutManager.setTimeForItemSettle(millis);
}
public void setSlideOnFling(boolean result){
layoutManager.setShouldSlideOnFling(result);
}
public void setSlideOnFlingThreshold(int threshold){
layoutManager.setSlideOnFlingThreshold(threshold);
}
public void setOrientation(DSVOrientation orientation) {
layoutManager.setOrientation(orientation);
}
public void setOffscreenItems(int items) {
layoutManager.setOffscreenItems(items);
}
public void setClampTransformProgressAfter(@IntRange(from = 1) int itemCount) {
if (itemCount <= 1) {
throw new IllegalArgumentException("must be >= 1");
}
layoutManager.setTransformClampItemCount(itemCount);
}
public void setOverScrollEnabled(boolean overScrollEnabled) {
isOverScrollEnabled = overScrollEnabled;
setOverScrollMode(OVER_SCROLL_NEVER);
}
public void addScrollStateChangeListener(@NonNull ScrollStateChangeListener<?> scrollStateChangeListener) {
scrollStateChangeListeners.add(scrollStateChangeListener);
}
public void addScrollListener(@NonNull ScrollListener<?> scrollListener) {
addScrollStateChangeListener(new ScrollListenerAdapter(scrollListener));
}
public void addOnItemChangedListener(@NonNull OnItemChangedListener<?> onItemChangedListener) {
onItemChangedListeners.add(onItemChangedListener);
}
public void removeScrollStateChangeListener(@NonNull ScrollStateChangeListener<?> scrollStateChangeListener) {
scrollStateChangeListeners.remove(scrollStateChangeListener);
}
public void removeScrollListener(@NonNull ScrollListener<?> scrollListener) {
removeScrollStateChangeListener(new ScrollListenerAdapter<>(scrollListener));
}
public void removeItemChangedListener(@NonNull OnItemChangedListener<?> onItemChangedListener) {
onItemChangedListeners.remove(onItemChangedListener);
}
private void notifyScrollStart(ViewHolder holder, int current) {
for (ScrollStateChangeListener listener : scrollStateChangeListeners) {
listener.onScrollStart(holder, current);
}
}
private void notifyScrollEnd(ViewHolder holder, int current) {
for (ScrollStateChangeListener listener : scrollStateChangeListeners) {
listener.onScrollEnd(holder, current);
}
}
private void notifyScroll(float position,
int currentIndex, int newIndex,
ViewHolder currentHolder, ViewHolder newHolder) {
for (ScrollStateChangeListener listener : scrollStateChangeListeners) {
listener.onScroll(position, currentIndex, newIndex,
currentHolder,
newHolder);
}
}
private void notifyCurrentItemChanged(ViewHolder holder, int current) {
for (OnItemChangedListener listener : onItemChangedListeners) {
listener.onCurrentItemChanged(holder, current);
}
}
private void notifyCurrentItemChanged() {
if (onItemChangedListeners.isEmpty()) {
return;
}
int current = layoutManager.getCurrentPosition();
ViewHolder currentHolder = getViewHolder(current);
notifyCurrentItemChanged(currentHolder, current);
}
private class ScrollStateListener implements DiscreteRecyclerLayoutManager.ScrollStateListener {
@Override
public void onIsBoundReachedFlagChange(boolean isBoundReached) {
if (isOverScrollEnabled) {
setOverScrollMode(isBoundReached ? OVER_SCROLL_ALWAYS : OVER_SCROLL_NEVER);
}
}
@Override
public void onScrollStart() {
if (scrollStateChangeListeners.isEmpty()) {
return;
}
int current = layoutManager.getCurrentPosition();
ViewHolder holder = getViewHolder(current);
if (holder != null) {
notifyScrollStart(holder, current);
}
}
@Override
public void onScrollEnd() {
if (onItemChangedListeners.isEmpty() && scrollStateChangeListeners.isEmpty()) {
return;
}
int current = layoutManager.getCurrentPosition();
ViewHolder holder = getViewHolder(current);
if (holder != null) {
notifyScrollEnd(holder, current);
notifyCurrentItemChanged(holder, current);
}
}
@Override
public void onScroll(float currentViewPosition) {
if (scrollStateChangeListeners.isEmpty()) {
return;
}
int currentIndex = getCurrentItem();
int newIndex = layoutManager.getNextPosition();
if (currentIndex != newIndex) {
notifyScroll(currentViewPosition,
currentIndex, newIndex,
getViewHolder(currentIndex),
getViewHolder(newIndex));
}
}
@Override
public void onCurrentViewFirstLayout() {
post(new Runnable() {
@Override
public void run() {
notifyCurrentItemChanged();
}
});
}
@Override
public void onDataSetChangeChangedPosition() {
notifyCurrentItemChanged();
}
}
public interface ScrollStateChangeListener<T extends ViewHolder> {
void onScrollStart(@NonNull T currentItemHolder, int adapterPosition);
void onScrollEnd(@NonNull T currentItemHolder, int adapterPosition);
void onScroll(float scrollPosition,
int currentPosition,
int newPosition,
@Nullable T currentHolder,
@Nullable T newCurrent);
}
public interface ScrollListener<T extends ViewHolder> {
void onScroll(float scrollPosition,
int currentPosition, int newPosition,
@Nullable T currentHolder,
@Nullable T newCurrent);
}
public interface OnItemChangedListener<T extends ViewHolder> {
/*
* This method will be also triggered when view appears on the screen for the first time.
* If data set is empty, viewHolder will be null and adapterPosition will be NO_POSITION
*/
void onCurrentItemChanged(@Nullable T viewHolder, int adapterPosition);
}
}
源码:https://github.com/DickyQie/android-discrete-recyclerview
android -------- RecyclerView的可(多个Item水平,垂直)滚动列表的实现的更多相关文章
- Android RecyclerView嵌套EditView实时更新Item数据
一.场景(例如:购物车) 1.当我们需要以列表样式管理某些数据时,可能需要列表项的某个字段可编辑 2.编辑Item上的某个字段后可能还要更新相关字段的值 二.可能遇到的问题 1.列表滑动导致输入框中的 ...
- Android开发 ListView(垂直滚动列表项视图)的简单使用
效果图: 使用方法: 1.在布局文件中加入ListView控件: <?xml version="1.0" encoding="utf-8"?> &l ...
- Android TextView多行垂直滚动
在Android应用中,有时候需要TextView可以垂直滚动,今天我就介绍一下怎么实现的.在布局里: <TextView android:id="@+id/tvCWJ" a ...
- Android RecyclerView(瀑布流)水平/垂直方向分割线
Android RecyclerView(瀑布流)水平/垂直方向分割线 Android RecyclerView不像过去的ListView那样随意的设置水平方向的分割线,如果要实现Recycle ...
- Android RecyclerView 实现支付宝首页效果
Android RecyclerView 实现支付宝首页效果 [TOC] 虽然我本人不喜欢支付宝的,但是这个网格本身其实还是不错的,项目更新中更改了一个布局为网格模式,类似支付宝.(估计是产品抄袭的= ...
- Android RecyclerView的基本使用
Android RecyclerView 在去年的Google I/O大会上就推出来了,以前经常使用的ListView 继承的是AbsListView,而RecyclerView则直接继承 ViewG ...
- Android RecyclerView体验(一)- 简介
在网上关于RecyclerView的基本使用方式已经有了比较详细介绍,而且其设计结构也类似于ListView,所以本文将不重点介绍如何使用,在文末的引用中都可以相关内容.这里主要是介绍Recycler ...
- 浅谈Android RecyclerView
Android RecyclerView 是Android5.0推出来的,导入support-v7包即可使用. 个人体验来说,RecyclerView绝对是一款功能强大的控件. 首先总结下Recycl ...
- (转载) Android RecyclerView 使用完全解析 体验艺术般的控件
Android RecyclerView 使用完全解析 体验艺术般的控件 标签: Recyclerviewpager瀑布流 2015-04-16 09:07 721474人阅读 评论(458) 收藏 ...
- Android RecyclerView与ListView比较
RecyclerView 概述 RecyclerView 集成自 ViewGroup .RecyclerView是Android-support-V7版本中新增的一个Widgets,官方对于它的介绍是 ...
随机推荐
- Ubuntu 16.04 安装 Apache, MySQL, PHP7.2
先更新系统 sudo apt update 安装PHP 通过修改PPA源的方式来安装 sudo apt-get install software-properties-common python-so ...
- Webpack 学习手记
官网:https://www.webpackjs.com/ 参考网址:https://www.cnblogs.com/cangqinglang/p/8964460.html 1.webpack简述:是 ...
- Aspose是一个很强大的控件,可以用来操作word,excel,ppt等文件
Aspose是一个很强大的控件,可以用来操作word,excel,ppt等文件,用这个控件来导入.导出数据非常方便.其中Aspose.Cells就是用来操作Excel的,功能有很多.我所用的是最基本的 ...
- it入门之:学会使用Git 分布式版本控制工具
环境:window 工具:git & TortoiseGit 下载安装以及配置环境变量 :略略略 创建Github账户:登录https://github.com/用自己的常用邮箱创建账户,用来 ...
- [NOIP赛前冲刺第一期]初赛基础知识归纳
关于计算机 1.CPU 中央处理器(CPU,Central Processing Unit)是一块超大规模的集成电路,是一台计算机的运算核心(Core)和控制核心(Control Unit).它的功能 ...
- python 常用turtle
python 常用turtle 常用命令1 import turtle turtle.bgcolor("black") 设置背景颜色 turtle.onscreenclick(x, ...
- ssh很慢的问题转子
根据网上的解决办法解决了,记录一下: 问题:ssh ***@192.*.*.* ,然后就一直卡在这个地方,很久以后才会出现让输入密码的提示 解决办法:1.关闭防火墙--chkconfig ipta ...
- 随手记一 2018/04/23 Ajax基础了解
1.什么是ajax? 主要目的是用来实现客户端和服务器之间的异步通信,实现页面的局部刷新 2.同步和异步! 同步:当多个线程同时向一个数据发送请求时,必须是A先执行完毕才可以给B,会出现阻塞的情况,但 ...
- PID算法(c 语言)(转)
PID算法(c 语言)(来自老外) #include <stdio.h> #include<math.h> //定义PID 的结构体 struct _pid { int pv; ...
- JZ2440学习笔记之第一个裸机程序(Keil-MDK)
CPU:S3C2440, ARM920T, Internal 4KB RAM, Support boot from NAND flash, 128MB for each bank. JZ2440:Me ...