`

RecyclerView:带header&footer&加载更多

阅读更多
接上一篇
由简到繁,循序渐进
实现了:
0,点击&长按事件;
1,header;
2,footer;
3,加载更多;
4,还想下拉刷新?“想的美,这本就不是RecyclerView该做的事情!”

import java.util.ArrayList;
import java.util.List;

import com.example.test.R;
import com.widget.view.HeaderFooterAdapter.ViewHolder;

import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

/**
 * 为了用RecyclerView创建一个带header的grid:

	1,定义一个具有两种view类型的adapter,一个为header一个为普通item。
	
	2,inflate一个header,把它传递给adapter。
	
	3,重写GridLayoutManager中的setSpanSizeLookup,在header所处的位置返回和span count(列数)相等的 span size。
 * @author Administrator
 *
 */
public class HeaderFooterAdapter<T> extends RecyclerView.Adapter<ViewHolder> {

	private static final int ITEM_VIEW_TYPE_HEADER = 0;
	private static final int ITEM_VIEW_TYPE_ITEM = 1;
	private static final int ITEM_VIEW_TYPE_FOOTER = 2;

	private final View header;
	private final View footer;
	private int headerCount=0;
	private int footerCount=0;
	private final List<T> beans;
	
	public HeaderFooterAdapter(View header, View footer,List<T> beans) {
		headerCount=header == null?0:1;
		footerCount=footer == null?0:1;
		this.header = header;
		this.footer = footer;
		this.beans = beans;
	}

	public boolean isHeader(int position) {
		if(headerCount==0){
			return false;
		}
		return position == 0;
	}
	public boolean isFooter(int position) {
		if(footerCount==0){
			return false;
		}
		return position == beans.size()+headerCount;// add 1 for footer
//		return position == getItemCount()-headerCount;// or Subtract 1 for footer
	}

	@Override
	public ViewHolder<T> onCreateViewHolder(ViewGroup parent, int viewType) {
		if (viewType == ITEM_VIEW_TYPE_HEADER) {
			return new ViewHolder<T>(header);
		}
		if (viewType == ITEM_VIEW_TYPE_FOOTER) {
			return new ViewHolder<T>(footer);
		}
		View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item, parent, false);
		return new ViewHolder<T>(view);
	}

	@Override
	public void onBindViewHolder(final ViewHolder holder, final int position) {
		if (isHeader(position)) {
			return;
		}
		if (isFooter(position)) {
			return;
		}
		final T bean = beans.get(position - headerCount); // Subtract 1 for header
		holder.textView.setText(bean.toString());
	}

	@Override
	public int getItemViewType(int position) {
		if(isHeader(position)){
			return ITEM_VIEW_TYPE_HEADER;
		}
		if(isFooter(position)){
			return ITEM_VIEW_TYPE_FOOTER;
		}
		return ITEM_VIEW_TYPE_ITEM;
	}

	/**
	 * 获得实际View的数量,包括header和footer
	 */
	@Override
	public int getItemCount() {
		return beans.size() + headerCount+footerCount;
	}

	class ViewHolder<T> extends RecyclerView.ViewHolder implements View.OnClickListener,View.OnLongClickListener{
		public TextView textView;

		public ViewHolder(View itemView) {
			super(itemView);
			if (onItemClickLitener != null){
				itemView.setOnClickListener(this);
            }
			if(onItemLongClickLitener!=null){
				itemView.setOnLongClickListener(this);
            }
			textView = (TextView) itemView.findViewById(R.id.item_0);
		}

		@Override
		public void onClick(View v) {
			// TODO Auto-generated method stub
			onItemClickLitener.onItemClick(v, getLayoutPosition());//getPosition()被废弃了
		}
		
		@Override
		public boolean onLongClick(View v) {
			// TODO Auto-generated method stub
			onItemLongClickLitener.onItemLongClick(v, getLayoutPosition());
			return true;
		}
		
	}
	
	private static OnItemClickLitener onItemClickLitener;
	
	public void setOnItemClickLitener(OnItemClickLitener onItemClickLitener) {
		this.onItemClickLitener = onItemClickLitener;
	}
	
	public interface OnItemClickLitener {
		/**
		 * 点击回调,注意position,指的是在Adapter中的位置,已经算上header
		 * @param view
		 * @param position
		 */
		void onItemClick(View view, int position);
	}

	private static OnItemLongClickLitener onItemLongClickLitener;
	
	public void setOnItemLongClickLitener(OnItemLongClickLitener onItemLongClickLitener) {
		this.onItemLongClickLitener = onItemLongClickLitener;
	}

	public interface OnItemLongClickLitener {
		/**
		 * 长按回调,注意position,指的是在Adapter中的位置,已经算上header
		 * @param view
		 * @param position
		 */
		void onItemLongClick(View view, int position);
	}
	
	/**
	 * 在position位置插入item
	 * @param position:指的是item在beans中的位置,而不是在Adapter中的位置;
	 * 所以在notifyItemInserted的时候需要算上header
	 * @param item
	 */
	public void addItemAt(int position,T item) {
		beans.add(position, item);
        notifyItemInserted(position+headerCount);
    }
	/**
	 * 在底部添加Items
	 * @param items
	 */
	public void addItemsAtLast(ArrayList<T> items) {
		int size=beans.size();
		beans.addAll(items);
		notifyItemRangeInserted(size+headerCount, items.size());
	}

	/**
	 * 移除position位置的Item
	 * @param position :指的是item在beans中的位置,而不是在Adapter中的位置;
	 * 所以在notifyItemInserted的时候需要算上header
	 */
	public void removeItemAt(int position) {
		beans.remove(position);
        notifyItemRemoved(position+headerCount);
    }
	
	/**
	 * 获得中间Item的数量,不包括header和footer
	 * @return
	 */
	public int getItemSize(){
		return beans.size();
	}
	
}


ArrayList<String> list=new ArrayList<String>();
    		for (int i = 0; i < 30; ++i) {
    			list.add(String.valueOf(i));
    		}
    		
        	View header = LayoutInflater.from(this).inflate(R.layout.header_view, recyclerView, false);
        	View footer = LayoutInflater.from(this).inflate(R.layout.footer_view, recyclerView, false);
			final HeaderFooterAdapter<String> adapter = new HeaderFooterAdapter<String>(header,footer, list);
			adapter.setOnItemClickLitener(new HeaderFooterAdapter.OnItemClickLitener() {
				
				@Override
				public void onItemClick(View view, int position) {
					// TODO Auto-generated method stub
					if(adapter.isHeader(position)){
						Toast.makeText(context, "header clicked", Toast.LENGTH_SHORT).show();
					}else if(adapter.isFooter(position)){
						Toast.makeText(context, "footer clicked", Toast.LENGTH_SHORT).show();
					}else{
						Toast.makeText(context, "header item "+position, Toast.LENGTH_SHORT).show();
					}
				}
			});
			
        	final GridLayoutManager manager = new GridLayoutManager(this, 2);
        	manager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
				@Override
				public int getSpanSize(int position) {
					if(adapter.isHeader(position)||adapter.isFooter(position)){
						return manager.getSpanCount();
					}
					return  1;
				}
			});
        	//如果可以确定每个item的高度是固定的,设置这个选项可以提高性能
        	recyclerView.setHasFixedSize(true);
        	recyclerView.setLayoutManager(manager);
			recyclerView.setAdapter(adapter);
			recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
				@Override 
				public void onScrolled(RecyclerView rv, int dx, int dy) {
	                boolean isBottom =
	                		manager.findLastCompletelyVisibleItemPosition()>= adapter.getItemCount()-1;
	                if (isBottom) {
	                	ArrayList<String> list=new ArrayList<String>();
	                	int size=adapter.getItemSize();
	            		for (int i = size; i < 30+size; ++i) {
	            			list.add(String.valueOf(i));
	            		}
	                	adapter.addItemsAtLast(list);
	                }
	            }
			});


这是一个如你熟悉ListView、GridView一样熟悉的RecyclerView类库,你可以用以前使用ListView / GridView的习惯来使用RecyclerView,这些可让你将项目原有的ListView / GridView / 瀑布流 迁移到RecyclerView时减少许多工作量.

你必须了解的RecyclerView的五大开源项目-解决上拉加载、下拉刷新和添加Header、Footer等问题
http://www.see-source.com/blog/300000035/1463.html

滑动到顶部/底部 可拉伸回弹的ListView与ScrollView
https://github.com/teisun/SunmiUI
分享到:
评论

相关推荐

    RecyclerView之添加header footer

    添加header和footer是RecyclerView中常见的需求,例如,我们可能希望在列表开头显示一些广告或分类信息,在列表结尾放置加载更多的按钮。 首先,我们需要创建一个自定义的RecyclerView.Adapter,它不仅包含原有的...

    Android-RecyclerView封装加载更多添加Header和Footer

    本教程将聚焦于如何对RecyclerView进行封装,实现加载更多、添加Header和Footer的功能。这在许多应用程序中都非常常见,例如社交应用、电商应用等,需要无限滚动加载新数据,并在顶部或底部显示额外的信息。 首先,...

    RecyclerView添加header ,添加footer同理

    注意,这只是一个基本实现,实际项目中可能需要处理更多细节,例如Header和Footer的点击事件、动态加载等。在处理这些问题时,可以结合使用接口回调、观察者模式或者其他设计模式来实现更灵活的交互。 总之,...

    RecyclerView_Header_Footer

    在默认情况下,RecyclerView只会显示Adapter中的数据项,但有时我们需要在列表的开头或末尾添加额外的视图,如广告、提示信息或者加载更多按钮,这就需要用到Header和Footer。 添加Header和Footer的方法主要有两种...

    RecyclerView添加Header和Footer.rar

    在RecyclerView中添加Header和Footer,通常是为了扩展视图的功能,比如在列表顶部显示广告、分类标签或底部添加加载更多按钮。这使得界面更加丰富和用户友好。 3. 实现步骤: - 创建Header和Footer的布局:首先,...

    recyclerView万能复用添加Header/FooterView及上拉加载下拉刷新源码

    对于上拉加载更多,我们可以在RecyclerView的滑动监听器中判断是否到达底部,如果达到,就触发加载更多数据的请求。通常需要添加一个状态字段来追踪加载状态,避免在加载过程中再次触发加载请求。 ```java ...

    整合多个开源项目,自认为相当简便使用的RecyclerView,支持下拉刷新,加载更多,添加Header和Footer。

    普通列表功能、为列表添加分割线,设置分割线高度(宽度)、颜色等。... 再不需要加载更多数据情况下,支持添加Footer。 未扩展下拉刷新功能,如需要,请使用Android官方提供的SwipeRefreshLayout 。

    Android-根据XRecyclerView源码修改支持自定义header和footer

    最后,充分测试在各种场景下的行为,确保header和footer的添加不影响其他功能的正常使用,如滚动、刷新和加载更多。使用Android Studio的布局预览和调试工具可以帮助查找和解决问题。 总结,通过对XRecyclerView...

    使用RecyclerView添加Header和Footer的方法

    在Android开发中,RecyclerView是一个非常重要的组件,它作为ListView的替代品,提供了更优秀的性能和更多的自定义选项。然而,RecyclerView并不直接支持ListView中的addHeaderView和addFooterView方法,这使得...

    Android RecyclerView下拉刷新和上拉加载更多

    今天终于有点时间,来写了一下: 为RecyclerView实现下拉刷新和上拉加载更多。今天会在前面的两篇文章的基础上: RecyclerView系列之(1):为RecyclerView添加Header和Footer RecyclerView系列之(2):为...

    自定义上拉刷新下拉加载更多

    在Android开发中,"自定义上拉刷新下拉加载更多"是一个常见的功能需求,尤其在列表滚动场景中,如RecyclerView、ListView和ScrollView等。这个功能可以让用户在浏览数据时便捷地获取新内容,提高用户体验。下面我们...

    支持下拉刷新,上拉加载,Header,Footer,复杂多种数据结构类型.zip

    4. **Footer(尾部)**:与Header相反,Footer位于视图的底部,常用于放置一些固定的操作按钮,如“加载更多”按钮,或者版权信息。在Android中,可以使用FooterView或者在RecyclerView中添加额外的item来实现,iOS...

    Android代码-下拉刷新和上拉加载更多的RecyclerView,具有下拉和刷新动画。

    瀑布流列表使用的是自定义Header和Footer的下拉刷新和上拉上拉加载,没有设置了下拉使放大的图片,使用默认的刷新动画。 用法: Gradle: dependencies { compile 'com.android.support:recyclerview-v7:23.1.0' ...

    Android开发-RecyclerView

    2. LoadMore功能:结合LinearLayoutManager实现无限滚动加载更多数据。 3. 延迟加载:根据屏幕可见范围加载数据,减少内存消耗。 五、高级用法 1. SnapHelper:帮助RecyclerView自动对齐到特定位置,如第一条、...

    Android-RecyclerView基础写法简单实现

    另外,还可以实现Item点击事件、动画效果以及下拉刷新和上拉加载更多功能。 对于更复杂的需求,例如自定义Item的视图类型,可以在`onCreateViewHolder()`中判断viewType,返回不同布局的ViewHolder。在`...

    上拉加载更多的RecyclerView

    源码AnimRefreshRecyclerViewDemo,下拉刷新和上拉加载更多的RecyclerView,具有下拉和刷新动画。 效果: 嗯...看起来有点卡,截图软件的问题: 使用说明: 上图中演示了三种不同的布局和下拉效果,三种布局和三种...

    下拉刷新,上拉加载更多.zip

    为了实现下拉刷新和上拉加载更多,开发者通常需要自定义ListView的Header和Footer,并监听滑动事件。 2. **SwipeRefreshLayout与ListView**:虽然SwipeRefreshLayout不是直接与ListView配合,但可以通过在...

Global site tag (gtag.js) - Google Analytics