帶視差滾動頭部的RecyclerView

talk is cheap show me the pictures!

Linearlayoutmanager
Linearlayoutmanager

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

sample

demo目錄

enter description here
enter description here

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類的方法并重寫其中的必要方法onCreateonBaind

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的值 LinearLayoutManagerGridLayoutManager可用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下載地址

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容