Android RecyclerView的Item點(diǎn)擊事件實(shí)現(xiàn)整理
自從開始使用RecyclerView代替ListView,會(huì)發(fā)現(xiàn)有很多地方需要學(xué)習(xí)。前一段時(shí)間的學(xué)習(xí)記錄有:
RecyclerView的滾動(dòng)事件研究 - DevWiki
RecyclerView的ViewHolder和Adapter的封裝優(yōu)化 - DevWiki
RecyclerView問題記錄 - DevWiki
實(shí)現(xiàn) RecyclerView的Item的點(diǎn)擊事件有三種方式:
- 在創(chuàng)建 ItemView時(shí)添加點(diǎn)擊監(jiān)聽
- 當(dāng) ItemView attach RecyclerView時(shí)實(shí)現(xiàn)
- 通過RecyclerView已有的方法addOnItemTouchListener()實(shí)現(xiàn)
1.在創(chuàng)建ItemView時(shí)添加點(diǎn)擊監(jiān)聽
思路是:因?yàn)閂iewHolder我們可以拿到每個(gè)Item的根布局,所以如果我們?yōu)楦季衷O(shè)置單獨(dú)的OnClick監(jiān)聽并將其開放給Adapter,那不就可以在組裝RecyclerView時(shí)就能夠設(shè)置ItemClickListener,只不過這個(gè)Listener不是設(shè)置到RecyclerView上而是設(shè)置到Adapter。具體實(shí)現(xiàn)代碼如下:
public class SampleAdapter extends RecyclerView.Adapter<SampleAdapter.SampleViewHolder> { private List<DataBean> mDatas; private OnItemClickListener mListener; // Item點(diǎn)擊事件 public DataBean getItem(int position) { return mDatas == null ? null : mDatas.get(position); } @Override public SampleViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent,false); return new SampleViewHolder(itemView); } @Override public void onBindViewHolder(SampleViewHolder holder, int position) { } @Override public int getItemCount() { return mDatas == null ? 0 : mDatas.size(); } class SampleViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener, View.OnLongClickListener { public SampleViewHolder(View itemView) { super(itemView); // TODO:初始化View ... itemView.setOnClickListener(this); itemView.setOnLongClickListener(this); } @Override public void onClick(View v) { if (mListener != null) { mListener.onItemClick(SampleAdapter.this, v, getLayoutPosition()); } } @Override public boolean onLongClick(View v) { if (mListener != null) { mListener.onItemLongClick(SampleAdapter.this, v, getLayoutPosition()); return true; } return false; } } }
2.當(dāng)ItemView attach RecyclerView時(shí)實(shí)現(xiàn)
該實(shí)現(xiàn)方法是在閱讀國(guó)外的一篇博客時(shí)發(fā)現(xiàn)的,原文鏈接如下:Getting your clicks on RecyclerView
實(shí)現(xiàn)的代碼如下:
public class ItemClickSupport { private static final int KEY = 0x99999999; private final RecyclerView mRecyclerView; private OnItemClickListener mOnItemClickListener; private OnItemLongClickListener mOnItemLongClickListener; private View.OnClickListener mOnClickListener = new View.OnClickListener() { @Override public void onClick(View v) { if (mOnItemClickListener != null) { RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v); mOnItemClickListener.onItemClicked(mRecyclerView, v, holder.getAdapterPosition()); } } }; private View.OnLongClickListener mOnLongClickListener = new View.OnLongClickListener() { @Override public boolean onLongClick(View v) { if (mOnItemLongClickListener != null) { RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v); return mOnItemLongClickListener.onItemLongClicked(mRecyclerView, v, holder.getAdapterPosition()); } return false; } }; private RecyclerView.OnChildAttachStateChangeListener mAttachListener = new RecyclerView.OnChildAttachStateChangeListener() { @Override public void onChildViewAttachedToWindow(View view) { if (mOnItemClickListener != null) { view.setOnClickListener(mOnClickListener); } if (mOnItemLongClickListener != null) { view.setOnLongClickListener(mOnLongClickListener); } } @Override public void onChildViewDetachedFromWindow(View view) { } }; /** * ItemClickSupport的私有構(gòu)造方法 */ private ItemClickSupport(RecyclerView recyclerView) { mRecyclerView = recyclerView; mRecyclerView.setTag(KEY, this); // 為RecyclerView設(shè)置OnChildAttachStateChangeListener事件監(jiān)聽 mRecyclerView.addOnChildAttachStateChangeListener(mAttachListener); } /** * 為RecyclerView設(shè)置ItemClickSupport */ public static ItemClickSupport addTo(RecyclerView view) { ItemClickSupport support = (ItemClickSupport) view.getTag(KEY); if (support == null) { support = new ItemClickSupport(view); } return support; } /** * 為RecyclerView移除ItemClickSupport */ public static ItemClickSupport removeFrom(RecyclerView view) { ItemClickSupport support = (ItemClickSupport) view.getTag(KEY); if (support != null) { support.detach(view); } return support; } /** * 為RecyclerView設(shè)置點(diǎn)擊事件監(jiān)聽 */ public ItemClickSupport setOnItemClickListener(OnItemClickListener listener) { mOnItemClickListener = listener; return this; } /** * 為RecyclerView設(shè)置長(zhǎng)按事件監(jiān)聽 */ public ItemClickSupport setOnItemLongClickListener(OnItemLongClickListener listener) { mOnItemLongClickListener = listener; return this; } /** * 為RecyclerView移除OnChildAttachStateChangeListener事件監(jiān)聽 */ private void detach(RecyclerView view) { view.removeOnChildAttachStateChangeListener(mAttachListener); view.setTag(KEY, null); } /** * RecyclerView的點(diǎn)擊事件監(jiān)聽接口 */ public interface OnItemClickListener { void onItemClicked(RecyclerView recyclerView, View itemView, int position); } /** * RecyclerView的長(zhǎng)按事件監(jiān)聽接口 */ public interface OnItemLongClickListener { boolean onItemLongClicked(RecyclerView recyclerView, View itemView, int position); } }
上面的代碼中給RecyclerView設(shè)置了OnChildAttachStateChangeListener事件監(jiān)聽,當(dāng)子View attach RecyclerView時(shí)設(shè)置事件監(jiān)聽。
private RecyclerView.OnChildAttachStateChangeListener mAttachListener = new RecyclerView.OnChildAttachStateChangeListener() { @Override public void onChildViewAttachedToWindow(View view) { if (mOnItemClickListener != null) { view.setOnClickListener(mOnClickListener); } if (mOnItemLongClickListener != null) { view.setOnLongClickListener(mOnLongClickListener); } } @Override public void onChildViewDetachedFromWindow(View view) {} };
使用時(shí)只需要調(diào)用addTo(RecycleView view)方法得到ItemClickSupport對(duì)象,然后調(diào)用setOnItemClickListener()方法和setOnItemLongClickListener()方法設(shè)置ItemView的點(diǎn)擊事件和長(zhǎng)按事件監(jiān)聽即可。
3.通過RecyclerView已有的方法addOnItemTouchListener()實(shí)現(xiàn)
3.1、查看源碼
查看RecyclerView源碼可以看到,RecyclerView預(yù)留了一個(gè)Item的觸摸事件方法:
/** * Add an {@link OnItemTouchListener} to intercept touch events before they are dispatched * to child views or this view's standard scrolling behavior. * * <p>Client code may use listeners to implement item manipulation behavior. Once a listener * returns true from * {@link OnItemTouchListener#onInterceptTouchEvent(RecyclerView, MotionEvent)} its * {@link OnItemTouchListener#onTouchEvent(RecyclerView, MotionEvent)} method will be called * for each incoming MotionEvent until the end of the gesture.</p> * * @param listener Listener to add * @see SimpleOnItemTouchListener */ public void addOnItemTouchListener(OnItemTouchListener listener) { mOnItemTouchListeners.add(listener); }
通過注釋我們可知,此方法是在滾動(dòng)事件之前調(diào)用,需要傳入一個(gè)OnItemTouchListener對(duì)象。OnItemTouchListener的代碼如下:
public static interface OnItemTouchListener { public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e); public void onTouchEvent(RecyclerView rv, MotionEvent e); public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept); }
此接口還提供了一個(gè)實(shí)現(xiàn)類,且官方推薦使用該實(shí)現(xiàn)類SimpleOnItemTouchListener:
/** * An implementation of {@link RecyclerView.OnItemTouchListener} that has empty method bodies and * default return values. * * You may prefer to extend this class if you don't need to override all methods. Another * benefit of using this class is future compatibility. As the interface may change, we'll * always provide a default implementation on this class so that your code won't break when * you update to a new version of the support library. */ public static class SimpleOnItemTouchListener implements RecyclerView.OnItemTouchListener { <span style="font-family:'Microsoft YaHei';"> </span> @Override public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) { return false; } @Override public void onTouchEvent(RecyclerView rv, MotionEvent e) { } @Override public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) { } }
在觸摸接口中,當(dāng)觸摸時(shí)會(huì)回調(diào)一個(gè)MotionEvent對(duì)象,通過使用GestureDetectorCompat來解析用戶的操作。
3.2、了解GestureDetector的工作原理
對(duì)于觸摸屏,其原生的消息無非按下、抬起、移動(dòng)這幾種,我們只需要簡(jiǎn)單重載onTouch或者設(shè)置觸摸偵聽器setOnTouchListener即可進(jìn)行處理。不過,為了提高我們的APP的用戶體驗(yàn),有時(shí)候我們需要識(shí)別用戶的手勢(shì),Android給我們提供的手勢(shì)識(shí)別工具GestureDetector就可以幫上大忙了。
GestureDetector的工作原理是,當(dāng)我們接收到用戶觸摸消息時(shí),將這個(gè)消息交給GestureDetector去加工,我們通過設(shè)置偵聽器獲得GestureDetector處理后的手勢(shì)。
GestureDetector提供了兩個(gè)偵聽器接口,OnGestureListener處理單擊類消息,OnDoubleTapListener處理雙擊類消息。
OnGestureListener的接口有這幾個(gè):
// 單擊,觸摸屏按下時(shí)立刻觸發(fā) abstract boolean onDown(MotionEvent e); // 抬起,手指離開觸摸屏?xí)r觸發(fā)(長(zhǎng)按、滾動(dòng)、滑動(dòng)時(shí),不會(huì)觸發(fā)這個(gè)手勢(shì)) abstract boolean onSingleTapUp(MotionEvent e); // 短按,觸摸屏按下后片刻后抬起,會(huì)觸發(fā)這個(gè)手勢(shì),如果迅速抬起則不會(huì) abstract void onShowPress(MotionEvent e); // 長(zhǎng)按,觸摸屏按下后既不抬起也不移動(dòng),過一段時(shí)間后觸發(fā) abstract void onLongPress(MotionEvent e); // 滾動(dòng),觸摸屏按下后移動(dòng) abstract boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY); // 滑動(dòng),觸摸屏按下后快速移動(dòng)并抬起,會(huì)先觸發(fā)滾動(dòng)手勢(shì),跟著觸發(fā)一個(gè)滑動(dòng)手勢(shì) abstract boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY); OnDoubleTapListener的接口有這幾個(gè):
// 雙擊,手指在觸摸屏上迅速點(diǎn)擊第二下時(shí)觸發(fā) abstract boolean onDoubleTap(MotionEvent e); // 雙擊的按下跟抬起各觸發(fā)一次 abstract boolean onDoubleTapEvent(MotionEvent e); // 單擊確認(rèn),即很快的按下并抬起,但并不連續(xù)點(diǎn)擊第二下 abstract boolean onSingleTapConfirmed(MotionEvent e);
有時(shí)候我們并不需要處理上面所有手勢(shì),方便起見,Android提供了另外一個(gè)類SimpleOnGestureListener實(shí)現(xiàn)了如上接口,我們只需要繼承SimpleOnGestureListener然后重載需要的手勢(shì)即可。
3.3、實(shí)現(xiàn)點(diǎn)擊事件監(jiān)聽
了解了GestureDetector的工作原理之后,便開始實(shí)現(xiàn)RecycleView的Item的點(diǎn)擊事件。首先寫一個(gè)SimpleRecycleViewItemClickListener類繼承SimpleOnItemTouchListener,構(gòu)造時(shí)傳入Item點(diǎn)擊回調(diào)OnItemClickListener,并覆寫父類的boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e)方法,具體代碼如下:
/** * RecyclerView的Item點(diǎn)擊事件監(jiān)聽 * * @author liyunlong * @date 2016/11/21 9:42 */ public class SimpleRecycleViewItemClickListener extends RecyclerView.SimpleOnItemTouchListener { private OnItemClickListener mListener; private GestureDetectorCompat mGestureDetector; public SimpleRecycleViewItemClickListener(OnItemClickListener listener) { this.mListener = listener; } @Override public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) { if (mGestureDetector == null) { initGestureDetector(rv); } if (mGestureDetector.onTouchEvent(e)) { // 把事件交給GestureDetector處理 return true; } else { return false; } } /** * 初始化GestureDetector */ private void initGestureDetector(final RecyclerView recyclerView) { mGestureDetector = new GestureDetectorCompat(recyclerView.getContext(), new GestureDetector.SimpleOnGestureListener() { // 這里選擇SimpleOnGestureListener實(shí)現(xiàn)類,可以根據(jù)需要選擇重寫的方法 /** * 單擊事件 */ @Override public boolean onSingleTapUp(MotionEvent e) { View childView = recyclerView.findChildViewUnder(e.getX(), e.getY()); if (childView != null && mListener != null) { mListener.onItemClick(childView, recyclerView.getChildLayoutPosition(childView)); return true; } return false; } /** * 長(zhǎng)按事件 */ @Override public void onLongPress(MotionEvent e) { View childView = recyclerView.findChildViewUnder(e.getX(), e.getY()); if (childView != null && mListener != null) { mListener.onItemLongClick(childView, recyclerView.getChildLayoutPosition(childView)); } } /** * 雙擊事件 */ @Override public boolean onDoubleTapEvent(MotionEvent e) { int action = e.getAction(); if (action == MotionEvent.ACTION_UP) { View childView = recyclerView.findChildViewUnder(e.getX(), e.getY()); if (childView != null && mListener != null) { mListener.onItemDoubleClick(childView, recyclerView.getChildLayoutPosition(childView)); return true; } } return false; } }); } /** * RecyclerView的Item點(diǎn)擊事件監(jiān)聽接口 * * @author liyunlong * @date 2016/11/21 9:43 */ public interface OnItemClickListener { /** * 當(dāng)ItemView的單擊事件觸發(fā)時(shí)調(diào)用 */ void onItemClick(View view, int position); /** * 當(dāng)ItemView的長(zhǎng)按事件觸發(fā)時(shí)調(diào)用 */ void onItemLongClick(View view, int position); /** * 當(dāng)ItemView的雙擊事件觸發(fā)時(shí)調(diào)用 */ void onItemDoubleClick(View view, int position); } /** * RecyclerView的Item點(diǎn)擊事件監(jiān)聽實(shí)現(xiàn) * * @author liyunlong * @date 2016/11/21 10:05 */ public class SimpleOnItemClickListener implements OnItemClickListener { @Override public void onItemClick(View view, int position) { } @Override public void onItemLongClick(View view, int position) { } @Override public void onItemDoubleClick(View view, int position) { } } }
在GestureDetectorCompat的手勢(shì)回調(diào)中我們覆寫:
- boolean onSingleTapUp(MotionEvent e):?jiǎn)螕羰录卣{(diào)
- void onLongPress(MotionEvent e):長(zhǎng)按事件回調(diào)
- boolean onDoubleTapEvent(MotionEvent e):雙擊事件回調(diào)
如果我們只需要監(jiān)聽單擊事件,而不需要監(jiān)聽長(zhǎng)按事件和雙擊事件,構(gòu)造SimpleRecycleViewItemClickListener時(shí)只需要傳入SimpleOnItemClickListener即可,如果需要處理其它的手勢(shì)監(jiān)聽,也可以覆寫對(duì)應(yīng)的手勢(shì)回調(diào)方法。
4.三種方法對(duì)比
以上三種方式分別是:
- 在創(chuàng)建ItemView時(shí)添加點(diǎn)擊監(jiān)聽
- 當(dāng)ItemView attach RecyclerView時(shí)實(shí)現(xiàn)
- 通過RecyclerView已有的方法addOnItemTouchListener()實(shí)現(xiàn)
從以上三種方式的實(shí)現(xiàn)過程可知:
三種均可實(shí)現(xiàn)ItemView的點(diǎn)擊事件和長(zhǎng)按事件的監(jiān)聽。
第一種和第二種方式可以很方便對(duì)ItemView中的子View進(jìn)行監(jiān)聽。
第三種方式可以很方便獲取用戶點(diǎn)擊的坐標(biāo)。
第二種方式和第三種方式可以寫在單獨(dú)的類中,相對(duì)于第一種寫在Adapter的方式可使代碼更獨(dú)立整潔。
綜上所述:
如果你只想監(jiān)聽I(yíng)temView的點(diǎn)擊事件或長(zhǎng)按事件,三種方式均可。
如果你想監(jiān)聽I(yíng)temView中每個(gè)子View的點(diǎn)擊事件,采用第一種或者第二種比較方便。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
詳解Android 手機(jī)衛(wèi)士設(shè)置向?qū)ы?yè)面
這篇文章主要介紹了詳解Android 手機(jī)衛(wèi)士設(shè)置向?qū)ы?yè)面的相關(guān)資料,需要的朋友可以參考下2016-04-04Android 自動(dòng)化測(cè)試經(jīng)驗(yàn)分享 深入U(xiǎn)iScrollable
UiScrollable是一個(gè)UiCollection(這東西還沒搞懂),我們可以使用它,在可滑動(dòng)的頁(yè)面(水平滑動(dòng)或上下滑動(dòng)都可以)上查找我們想要的控件(item)2013-05-05Android 側(cè)邊滑動(dòng)關(guān)閉Activity的示例代碼
這篇文章主要介紹了Android 側(cè)邊滑動(dòng)關(guān)閉Activity的方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-05-05Android中修改TabLayout底部導(dǎo)航條Indicator長(zhǎng)短的方法
Tablayout在我們?nèi)粘i_發(fā)中經(jīng)常會(huì)遇到,下面這篇文章主要給大家介紹了在Android中修改TabLayout底部導(dǎo)航條Indicator長(zhǎng)短的方法,文中給出了詳細(xì)的示例代碼供大家參考學(xué)習(xí),需要的朋友們下面來一起看看吧。2017-06-06Android設(shè)計(jì)模式之Builder模式詳解
這篇文章主要為大家詳細(xì)介紹了Android設(shè)計(jì)模式之Builder模式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08基于Android中dp和px之間進(jìn)行轉(zhuǎn)換的實(shí)現(xiàn)代碼
本篇文章是對(duì)在Android中dp和px之間進(jìn)行轉(zhuǎn)換的實(shí)現(xiàn)方法進(jìn)行了分析介紹,需要的朋友參考下2013-05-05Android實(shí)現(xiàn)監(jiān)聽音量的變化
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)監(jiān)聽音量的變化,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05