打造RecyclerView萬能適配器

騷年咒吐,如果你還在用ListView趕緊換RecyclerView吧,用了之后你會(huì)發(fā)現(xiàn)從此你再也不想用ListView了娜谊。下面我說說兩者的優(yōu)缺點(diǎn)确买,listView簡單易用,google工程師已經(jīng)把很多功能封裝好了纱皆,比如點(diǎn)擊事件湾趾,分割線芭商,頭部,尾部搀缠。缺點(diǎn):性能差蓉坎,可拓展性不好。RecyclerView用法稍微復(fù)雜胡嘿。但是功能強(qiáng)大蛉艾,可拓展性好。依靠LayoutManager(通俗叫法布局管理器)可以實(shí)現(xiàn)多種效果衷敌,比如listview勿侯,graidview,以及瀑布流效果缴罗。并且可以設(shè)置動(dòng)畫助琐,點(diǎn)擊效果那叫一個(gè)狂拽酷炫啊。Google工程師呢也是留下了很多東西讓開發(fā)者自己去發(fā)揮面氓。條目點(diǎn)擊我們得自己寫兵钮,最惡心的是分割線和grid形式的間距設(shè)置有點(diǎn)惡心。還有一點(diǎn)我發(fā)現(xiàn)對(duì)于api22以上(開發(fā)時(shí)的編譯版本)和api22以下是有差異的舌界。比如我要設(shè)置recyclerview高度為wrap_content掘譬,但是視圖并沒有包裹而是占了一個(gè)頁面。解決方式是寫一個(gè)可包裹的L愛有天Mannager呻拌。對(duì)于api22以上不需要考慮葱轩,可以正常設(shè)置弥激。這點(diǎn)算一個(gè)版本bug吧缘圈。那么無論listview還是recyclerview最重要的就是適配器了。終于講到正題了嚼松。那么下面我們來聊聊Adapter
適配器其實(shí)就是把數(shù)據(jù)適配成view猾普。通俗點(diǎn)說就像是一個(gè)模具袜炕,你把料(數(shù)據(jù))放進(jìn)來,我們倒個(gè)模出來初家。平時(shí)開發(fā)中這個(gè)列表需求是很多的偎窘,那么意味著我們要用很多次這種控件。每次用我們又得寫適配器笤成。是不是很煩评架,一大堆相同代碼。寫得手疼炕泳,于是想著要把這些相同的代碼是不是可以做個(gè)封裝,我們只需要把布局上祈,把數(shù)據(jù)這些不同業(yè)務(wù)提供給外面實(shí)現(xiàn)培遵≌丬剑基于這個(gè)想法,有了今天我們要講的這個(gè)主題籽腕。

業(yè)務(wù)需求1.普通常見列表嗡呼。不包含頭尾。要求只給一個(gè)布局皇耗,一些數(shù)據(jù)可以實(shí)現(xiàn)需求南窗。

分析:數(shù)據(jù)寫死肯定是不行的,這里采用泛型設(shè)計(jì)郎楼。數(shù)據(jù)設(shè)置我們可以寫一個(gè)set方法設(shè)置万伤。布局得抽象出去實(shí)現(xiàn),還有數(shù)據(jù)和view之間的綁定也得抽象出去呜袁。holder也需要根據(jù)布局來重新寫敌买。

適配器代碼如下:

public abstract class BaseRecyclerViewAdapter<T> extends RecyclerView.Adapter<RecyclerView.ViewHolder> {
    protected Context mContext;
    protected final LayoutInflater inflater;
    protected List<T> mDatas;
    protected int mLayoutId;
    private OnItemClickListener onItemClickListener;

    public BaseRecyclerViewAdapter(Context mContext, int layoutId, List<T> datas) {
        this.mContext = mContext;
        this.mLayoutId = layoutId;
        this.mDatas = datas;
        inflater = LayoutInflater.from(mContext);
    }

    /*數(shù)據(jù)操作*/
    public void addData(T bean) {
        mDatas.add(bean);
        notifyDataSetChanged();
    }

    public void addDatas(List<T> datas) {
        mDatas.addAll(datas);
        notifyDataSetChanged();
    }

    public void remove(int index) {
        if (index < 0 && index > mDatas.size()) {
            throw new IndexOutOfBoundsException("index not right");
        }
        else mDatas.remove(index);
        notifyDataSetChanged();
    }

    public void removeAll() {
        mDatas.clear();
        notifyDataSetChanged();
    }

    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = inflater.inflate(mLayoutId, parent, false);
        return onCreatHolder(view);
    }


    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        final int pos = getRealPosition(holder);
        final T bean = mDatas.get(pos);
        onBindHolder(holder, pos, bean);
        holder.itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                if (onItemClickListener != null) {
                    onItemClickListener.onItemClick(pos, bean);
                }

            }
        });
    }

    private int getRealPosition(RecyclerView.ViewHolder holder) {
        return holder.getLayoutPosition();
    }

    @Override
    public int getItemCount() {
        return mDatas == null ? 0 : mDatas.size();
    }

    /*寫一個(gè)接口回調(diào)點(diǎn)擊事件*/
    public interface OnItemClickListener<T> {
        void onItemClick(int position, T bean);
    }

    public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
        this.onItemClickListener = onItemClickListener;
    }


    /**
     * @param view
     * @return 抽象出去
     */
    public abstract RecyclerView.ViewHolder onCreatHolder(View view);

    protected abstract void onBindHolder(RecyclerView.ViewHolder holder, int pos, T bean);
}

2.使用過程中發(fā)現(xiàn)ViewHolder以及查找id并不好用,把上面的適配器升級(jí)一下阶界,我們來一個(gè)BaseViewHolder把一些數(shù)據(jù)裝配操作也封裝起來虹钮。BaseViewHolder封裝了一些常用控件的數(shù)據(jù)操作,事件監(jiān)聽等膘融≤搅唬基于這些想法,我們?cè)僮錾?jí)改造氧映。

public abstract class BaseRecyclerAdapter<T> extends RecyclerView.Adapter<BaseViewHolder> {


    protected Context mContext;
    protected List<T> mDatas;
    private int totalList;

    private int itemLayoutId;
    private OnItemClickListener onItemClickListener;
    private OnItemLongClickListener onItemLongClickListener;
    private final LayoutInflater mInflater;
    private static final int TYPE_HEAD = 0;
    private static final int TYPE_ITEM = 1;
    private static final int TYPE_FOOT = 2;
    private boolean isScrolling = true;  //false表示滑動(dòng)宅倒,true表示不滑動(dòng)
    private View headerView, footerView;  //頭和尾的view


    public BaseRecyclerAdapter(Context ctx, int itemLayoutId, List<T> list) {
        this.mContext = ctx;
        this.itemLayoutId = itemLayoutId;
        mDatas = (list != null) ? list : new ArrayList<T>();
        mInflater = LayoutInflater.from(ctx);
    }
    /*數(shù)據(jù)操作*/
    public void addData(T bean) {
        mDatas.add(bean);
        notifyDataSetChanged();
    }

    public void addDatas(List<T> datas) {
        mDatas.addAll(datas);
        notifyDataSetChanged();
    }

    public void remove(int index) {
        if (index < 0 && index > mDatas.size()) {
            throw new IndexOutOfBoundsException("index not right");
        }
        else mDatas.remove(index);
        notifyDataSetChanged();
    }

    public void removeAll() {
        mDatas.clear();
        notifyDataSetChanged();
    }

    public void setFooterView(View view) {
        this.footerView = view;
    }

    public void setHeaderView(View view) {
        this.headerView = view;
    }

    @Override
    public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        if (viewType == TYPE_HEAD && headerView != null) {
            return new BaseViewHolder(mContext, headerView);
        }
        if (viewType == TYPE_FOOT && footerView != null) {
            return new BaseViewHolder(mContext, footerView);
        }

        View view = mInflater.inflate(itemLayoutId, parent, false);
        return new BaseViewHolder(mContext, view);
    }

    @Override
    public void onBindViewHolder(BaseViewHolder holder, int position) {
        if (getItemViewType(position) != TYPE_ITEM) {
            //如果不是正常的Item,就不去綁定數(shù)據(jù)
            return;
        }
        final int p = getRealPosition(position);
        convert(holder, p, mDatas.get(p));
        if (onItemClickListener != null) {
            holder.itemView.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    onItemClickListener.onItemClick(p, mDatas.get(p));
                }
            });
        }
        if (onItemLongClickListener != null) {
            holder.itemView.setOnLongClickListener(new View.OnLongClickListener() {
                @Override
                public boolean onLongClick(View view) {
                    onItemLongClickListener.onLongItemClick(p, mDatas.get(p));
                    return true;
                }
            });
        }


    }


    private int getRealPosition(int position) {
        return headerView == null ? position : position - 1;
    }

    @Override
    public int getItemCount() {
        if (mDatas == null) {
            return 0;
        }
        if (headerView != null && footerView != null) {
            //頭尾都不為空
            totalList = mDatas.size() + 2;
        } else if (headerView == null && footerView == null) {
            //頭尾都為空
            totalList = mDatas.size();
        } else {
            //頭尾有一個(gè)不為空
            totalList = mDatas.size() + 1;
        }
        return totalList;
    }

    @Override
    public int getItemViewType(int position) {
        if (position == 0 && headerView != null) {
            return TYPE_HEAD;
        } else if (position + 1 == getItemCount() && footerView != null) {
            return TYPE_FOOT;
        } else {
            return TYPE_ITEM;
        }
    }


    /*
    * 需要根據(jù)實(shí)際情況設(shè)置的部分抽象出去
    * */
    protected abstract void convert(BaseViewHolder holder, int p, T t);

    /*接口回調(diào)點(diǎn)擊和長按事件*/
    public interface OnItemClickListener<T> {
        void onItemClick(int position, T bean);
    }

    public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
        this.onItemClickListener = onItemClickListener;
    }

    public interface OnItemLongClickListener<T> {
        void onLongItemClick(int position, T bean);
    }

    public void setOnItemLongClickListener(OnItemLongClickListener onLongClickListener) {
        this.onItemLongClickListener = onLongClickListener;
    }
}

BaseViewHolder的代碼如下

public class BaseViewHolder extends RecyclerView.ViewHolder {

    /*
    * 數(shù)據(jù)量不大屯耸,最好在千級(jí)以內(nèi)
    * key必須為int類型拐迁,這中情況下的HashMap可以用SparseArray代替:比如
    * HashMap<Integer, Object> map = new HashMap<>();
    * 用SparseArray代替:
    * SparseArray<Object> array = new SparseArray<>();
    * */
    private SparseArray<View> mViews; //集合類,layout里包含的View,以view的id作為key疗绣,value是view對(duì)象
    private Context mContext;

    public BaseViewHolder(Context mContext, View itemView) {
        super(itemView);
        this.mContext = mContext;
        mViews = new SparseArray<>();
    }

    public <T extends View> T getView(int viewId) {
        View view = mViews.get(viewId);
        if (view == null) {
            view = itemView.findViewById(viewId);
            mViews.put(viewId, view);
        }
        return (T) view;
    }

    //  封裝一些常用的控件  根據(jù)對(duì)應(yīng)的id獲取控件
    public TextView getTextView(int viewId) {
        return (TextView) getView(viewId);
    }

    public Button getButton(int viewId) {
        return (Button) getView(viewId);
    }

    public ImageView getImageView(int viewId) {
        return (ImageView) getView(viewId);
    }

    public ImageButton getImageButton(int viewId) {
        return (ImageButton) getView(viewId);
    }

    public EditText getEditText(int viewId) {
        return (EditText) getView(viewId);
    }

    public BaseViewHolder setText(int viewId, String value) {
        TextView textView = getView(viewId);
        textView.setText(value);
        return this;
    }

    public BaseViewHolder setBitmapImage(int viewId, int imageId) {
        ImageView iv = getView(viewId);
        iv.setImageResource(imageId);
        return this;
    }

    public BaseViewHolder setImagebyUrl(int viewId, String url) {
        ImageView iv = getView(viewId);
        ShowImageUtils.showImageView(mContext, url, iv);
        return this;
    }

    public BaseViewHolder setCircleImagebyUrl(int viewId, String url) {
        ImageView iv = getView(viewId);
        ShowImageUtils.showImageViewToCircle(mContext, R.mipmap.ic_launcher, url, iv);
        return this;
    }

    public BaseViewHolder setBackground(int viewId, int resId) {
        View view = getView(viewId);
        view.setBackgroundResource(resId);
        return this;
    }

    public BaseViewHolder setOnClickListener(int viewId, View.OnClickListener listener) {
        View view = getView(viewId);
        view.setOnClickListener(listener);
        return this;
    }

}

3.使用中發(fā)現(xiàn)對(duì)于GridLayout线召,瀑布流這種會(huì)導(dǎo)致頭部和尾部變成其中的item。這顯然不是我們想要的多矮。那么就對(duì)著兩種情況特殊處理缓淹。
怎么做呢?在adapter中重寫onAttachedToRecyclerView
但是這種做法無法解決瀑布流的問題塔逃,對(duì)于StaggeredGridLayoutManager我們需要重寫onViewAttachedToWindow

 /**
     * @param recyclerView 處理GridLayoutManager
     */
    @Override
    public void onAttachedToRecyclerView(RecyclerView recyclerView) {
        super.onAttachedToRecyclerView(recyclerView);
        RecyclerView.LayoutManager manager = recyclerView.getLayoutManager();
        if (manager instanceof GridLayoutManager) {
            final GridLayoutManager gridLayoutManager = (GridLayoutManager) manager;
            //setSpanSizeLookup的getSpanSize可以控制列數(shù)
            gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
                @Override
                public int getSpanSize(int position) {
                    return (getItemViewType(position) == TYPE_HEAD || getItemViewType(position) == TYPE_FOOT) ? gridLayoutManager.getSpanCount() : 1;
                }
            });
        }/*else if (manager instanceof StaggeredGridLayoutManager) {  StaggeredGridLayoutManager并沒有setSpanSizeLookup這個(gè)方法讯壶,所以此路不通
            StaggeredGridLayoutManager staggeredManager = (StaggeredGridLayoutManager) manager;

        }*/
    }

    /**
     * @param holder
     * 處理StaggerdGridLayoutManager
     */
    @Override
    public void onViewAttachedToWindow(BaseViewHolder holder) {
        super.onViewAttachedToWindow(holder);
        ViewGroup.LayoutParams lp = holder.itemView.getLayoutParams();
        if (lp != null && lp instanceof StaggeredGridLayoutManager.LayoutParams) {
            StaggeredGridLayoutManager.LayoutParams p = (StaggeredGridLayoutManager.LayoutParams) lp;
            if (headerView != null && holder.getLayoutPosition() == 0) {
                p.setFullSpan(true);
            }

            if (footerView != null && holder.getLayoutPosition() == getItemCount() - 1) {
                p.setFullSpan(true);
            }
        }

    }

至此,我們的目的就達(dá)到了湾盗,使用起來超簡單可以繼承BaseRecyclerAdapter伏蚊,也可以直接用它的匿名內(nèi)部類。這里就不贅述了格粪。使用本類可以減少你Adapter百分之70的代碼躏吊,趕緊拿去擼吧氛改!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市比伏,隨后出現(xiàn)的幾起案子胜卤,更是在濱河造成了極大的恐慌,老刑警劉巖赁项,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件葛躏,死亡現(xiàn)場離奇詭異,居然都是意外死亡悠菜,警方通過查閱死者的電腦和手機(jī)舰攒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來李剖,“玉大人芒率,你說我怎么就攤上這事「菟常” “怎么了偶芍?”我有些...
    開封第一講書人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長德玫。 經(jīng)常有香客問我匪蟀,道長,這世上最難降的妖魔是什么宰僧? 我笑而不...
    開封第一講書人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任材彪,我火速辦了婚禮,結(jié)果婚禮上琴儿,老公的妹妹穿的比我還像新娘段化。我一直安慰自己,他們只是感情好造成,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開白布显熏。 她就那樣靜靜地躺著,像睡著了一般晒屎。 火紅的嫁衣襯著肌膚如雪喘蟆。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,255評(píng)論 1 308
  • 那天鼓鲁,我揣著相機(jī)與錄音蕴轨,去河邊找鬼。 笑死骇吭,一個(gè)胖子當(dāng)著我的面吹牛橙弱,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼膘螟,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼成福!你這毒婦竟也來了碾局?” 一聲冷哼從身側(cè)響起荆残,我...
    開封第一講書人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎净当,沒想到半個(gè)月后内斯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡像啼,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年俘闯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片忽冻。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡真朗,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出僧诚,到底是詐尸還是另有隱情遮婶,我是刑警寧澤,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布湖笨,位于F島的核電站旗扑,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏慈省。R本人自食惡果不足惜臀防,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望边败。 院中可真熱鬧袱衷,春花似錦、人聲如沸笑窜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽怖侦。三九已至篡悟,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間匾寝,已是汗流浹背搬葬。 一陣腳步聲響...
    開封第一講書人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留艳悔,地道東北人急凰。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親抡锈。 傳聞我的和親對(duì)象是個(gè)殘疾皇子疾忍,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

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