talk is cheap show me the pictures!

另外兩種布局與LinearlayoutManager類似,頭部布局獨(dú)占一行,正常item跟平時使用的recyclerView一致
sample
demo目錄

values目錄下的attrs中添加資源
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="MyRecyclerView">
<attr name="layout" format="reference"/>
<attr name="parallaxMultiplier" format="float"/>
</declare-styleable>
</resources>
使用布局文件中引入控件
<com.pandaq.collapsingheaderrecyclerview.myrecyclerview.MyRecyclerView
android:id="@+id/myrecycler"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout="@layout/recycler_header"
app:parallaxMultiplier="0.8"/>
layout對應(yīng)的是頭部視圖的布局文件
parallaxMultiplier是滑動視差因子默認(rèn)值是1
使用時寫一個繼承自BaseRecyclerAdapter類的方法并重寫其中的必要方法onCreate
和onBaind
class RecyclerAdapter extends BaseRecyclerAdapter<String> {
private Context mContext;
RecyclerAdapter(Context context) {
mContext = context;
}
@Override
public RecyclerView.ViewHolder onCreate(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(mContext).inflate(R.layout.recycler_item, parent, false);
return new ViewHolder(view);
}
@Override
public void onBind(RecyclerView.ViewHolder viewHolder, int RealPosition, String data) {
ViewHolder holder = (ViewHolder) viewHolder;
holder.mText.setText(data);
}
private class ViewHolder extends BaseRecyclerAdapter.Holder {
@BindView(R.id.text)
TextView mText;
ViewHolder(View view) {
super(view);
ButterKnife.bind(this, view);
}
}
}
使用recyclerView的地方與普通recyclerview一樣,設(shè)置布局,為適配器添加數(shù)據(jù),設(shè)置適配器
mMyrecycler.setLayoutManager(new LinearLayoutManager(this));
adapter.addDatas(dataList);
mMyrecycler.setAdapter(adapter);
實(shí)現(xiàn)
頭部視圖的滑動視差效果是通過滑動的時候不斷設(shè)置更新頭部的底部margin
來實(shí)現(xiàn)的即MyrecyclerView
類中的onScrolled方法的這段代碼
//當(dāng)添加了頭部視圖且當(dāng)前頭部視圖可見的時候給頭部視圖添加滾動視差效果
if (headerView != null&&firstVisibleItemPosition == 0) {
if (distance <= headerView.getHeight()) {
distance = dy;
} else {
distance = headerView.getHeight();
}
LayoutParams layoutParams = (LayoutParams) headerView.getLayoutParams();
//重新賦值給底部邊距
scrolledMargin = -distance + scrolledMargin;
if (scrolledMargin > 0) {
scrolledMargin = 0;
}
layoutParams.setMargins(0, 0, 0, (int) (multiplier * scrolledMargin));
headerView.setLayoutParams(layoutParams);
}
其中的firstVisibleItemPosition
的值 LinearLayoutManager
和GridLayoutManager
可用findFirstVisibleItemPosition()
方法獲取。(PS:findLastVisibleItemPosition()
可以獲取最后一個item的position??梢杂眠@兩個方法來判斷是否滑動到底部或頂部,滑動是否進(jìn)行加載或刷新)
StaggeredFridLayoutManager
則只有findLastVisibleItemPositions(int []positions)
和findFirstVisibleItemPositions(int[]positions)
方法。這兩個方法會將每一頁顯示的第一個或者最后一個item的position放置到傳入的數(shù)組中。取出數(shù)組中的最大最小值則是整個recyclerView的最后一個或者第一個Item的位置
布局兼容
如果不對Adapter進(jìn)行任何處理像正常使用recyclerView一樣使用會發(fā)現(xiàn)除了LinearLayoutManager布局會正常顯示其他兩種方式頭部視圖都是占一個item的位置而不是一整行。對于這個問題我在這兒找到了答案
通過構(gòu)建一個繼承自RecyclerView.Adapter<RecyclerView.ViewHolder>
的抽象類BaseRecyclerAdapter<T>
重寫其中的onAttachedToRecyclerView(RecyclerView recyclerView)
方法和onViewAttachedToWindow(RecyclerView.ViewHolder holder)
方法來實(shí)現(xiàn)頭部視圖獨(dú)占一行。使用Adapter的時候直接繼承這個類就好了。
@Override
public void onAttachedToRecyclerView(RecyclerView recyclerView) {
super.onAttachedToRecyclerView(recyclerView);
RecyclerView.LayoutManager manager = recyclerView.getLayoutManager();
if (manager instanceof GridLayoutManager) {
final GridLayoutManager gridManager = ((GridLayoutManager) manager);
gridManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
@Override
public int getSpanSize(int position) {
return getItemViewType(position) == TYPE_HEADER
? gridManager.getSpanCount() : 1;
}
});
}
}
@Override
public void onViewAttachedToWindow(RecyclerView.ViewHolder holder) {
super.onViewAttachedToWindow(holder);
ViewGroup.LayoutParams lp = holder.itemView.getLayoutParams();
if (lp != null
&& lp instanceof StaggeredGridLayoutManager.LayoutParams
&& holder.getLayoutPosition() == 0) {
StaggeredGridLayoutManager.LayoutParams p = (StaggeredGridLayoutManager.LayoutParams) lp;
p.setFullSpan(true);
}
}
HeaderView
headerView使用也很簡單,MyRecyclerView中構(gòu)造了一個getHeaderView()
方法。該方法會返回一個View對象,獲取到這個對象之后想怎么處理就隨便你自己發(fā)揮了。我正在寫的這個項目中headerView就是一個頂部的ViewPager實(shí)現(xiàn)的輪播Bander,目前還沒有發(fā)現(xiàn)什么不良反應(yīng)。
最后貼上demo下載地址