Android中解決RecyclerView各種點(diǎn)擊事件的方法
完美解決RecyclerView點(diǎn)擊事件、長(zhǎng)按事件、子項(xiàng)點(diǎn)擊事件
自從Google推出了RecyclerView之后,便可以完全取代ListView,個(gè)人感覺唯一的美中不足是對(duì)于itemView的各種點(diǎn)擊事件不夠完美。觀點(diǎn)只代表個(gè)人看法。應(yīng)最近項(xiàng)目需求實(shí)現(xiàn)itemView的子項(xiàng)點(diǎn)擊事件,便寫篇博客記錄一下,若是能夠幫到你,我深感榮幸。接下來,便對(duì)RecyclerView進(jìn)行簡(jiǎn)單的封裝,使得它更方便實(shí)現(xiàn)各種點(diǎn)擊事件。
我們都知道,對(duì)與RecyclerView的使用,是創(chuàng)建一個(gè)adapter類,然后在adapter類中再創(chuàng)建一個(gè)ViewHolder的內(nèi)部類。我們要做的,正是對(duì)這兩個(gè)類進(jìn)行封裝,讓其實(shí)現(xiàn)itemView點(diǎn)擊事件、長(zhǎng)按事件、子項(xiàng)點(diǎn)擊事件。
首先,我的處理方式是,對(duì)于開發(fā)者來說,只需要對(duì)adapter進(jìn)行setxxx()方法的調(diào)用,例如設(shè)置itemView的點(diǎn)擊事件:adapter.setOnRecyclerViewItemClickListener(...);對(duì)該方法傳入自定義的接口即可。也就是說,我們需要自定義一個(gè)adapter類。那我們就先創(chuàng)建一個(gè)類,命名為BaseRecylerAdapter,此后,我們也應(yīng)當(dāng)創(chuàng)建一個(gè)BaseViewHolder類,接下來開始搞事情。
BaseRecylerAdapter類
public abstract class BaseRecyclerAdapter extends RecyclerView.Adapter<BaseViewHolder>
implements View.OnClickListener
,View.OnLongClickListener {
private OnRecyclerViewItemClickListener onRecyclerViewItemClickListener;
private OnRecyclerViewItemLongClickListener onRecyclerViewItemLongClickListener;
private OnSubViewClickListener onSubViewClickListener;
@Override
public void onBindViewHolder(BaseViewHolder holder, int position) {
holder.itemView.setTag(position);
holder.onBind(position);
if (onRecyclerViewItemClickListener != null) {
holder.itemView.setOnClickListener(this);
}
if (onRecyclerViewItemLongClickListener != null) {
holder.itemView.setOnClickListener(this);
}
if (onSubViewClickListener != null) {
holder.setSubViewClickListener(onSubViewClickListener,position);
}
}
public void setOnRecyclerViewItemClickListener(OnRecyclerViewItemClickListener onRecyclerViewItemClickListener) {
this.onRecyclerViewItemClickListener = onRecyclerViewItemClickListener;
}
public void setOnRecyclerViewItemLongClickListener(OnRecyclerViewItemLongClickListener onRecyclerViewItemLongClickListener) {
this.onRecyclerViewItemLongClickListener = onRecyclerViewItemLongClickListener;
}
public void setOnSubViewClickListener(OnSubViewClickListener listener){
this.onSubViewClickListener = listener;
}
@Override
public void onClick(View v) {
if (v.getTag() != null) {
int position = (int) v.getTag();
onRecyclerViewItemClickListener.onItemClick(position);
}
}
@Override
public boolean onLongClick(View v) {
if (v.getTag() != null){
int position = (int)v.getTag();
onRecyclerViewItemLongClickListener.onItemLongClick(position);
}
return true;
}
public interface OnRecyclerViewItemClickListener {
void onItemClick(int position);
}
public interface OnSubViewClickListener{
void onSubViewClick(View v, int position);
}
public interface OnRecyclerViewItemLongClickListener {
void onItemLongClick(int position);
}
}
可以看到我們?cè)陬愔袆?chuàng)建了三個(gè)接口類
public interface OnRecyclerViewItemClickListener {
void onItemClick(int position);
}
public interface OnSubViewClickListener{
void onSubViewClick(View v, int position);
}
public interface OnRecyclerViewItemLongClickListener {
void onItemLongClick(int position);
}
這三個(gè)接口便是用于點(diǎn)擊事件的回調(diào),看名字就能分別出各自的功能。itemView的點(diǎn)擊回調(diào)public interface OnRecyclerViewItemClickListener,itemView的長(zhǎng)按public interface OnRecyclerViewItemLongClickListener,子項(xiàng)View的點(diǎn)擊回調(diào)public interface OnSubViewClickListener。都是點(diǎn)擊事件的處理,沒有點(diǎn)擊發(fā)送怎么行呢,對(duì)吧!所以,這個(gè)類還實(shí)現(xiàn)了View.OnClickListener 和View.OnLongClickListener 這兩個(gè)接口,本別實(shí)現(xiàn)itemView的點(diǎn)擊事件和長(zhǎng)按事件。
可以看到,BaseRecyclerAdapter繼承自RecyclerView.Adapter<BaseViewHolder>,此時(shí)我們只需要實(shí)現(xiàn)onBindViewHolder 這個(gè)方法即可。來分析這個(gè)方法。
@Override
public void onBindViewHolder(BaseViewHolder holder, int position) {
holder.itemView.setTag(position);
holder.onBind(position);
if (onRecyclerViewItemClickListener != null) {
holder.itemView.setOnClickListener(this);
}
if (onRecyclerViewItemLongClickListener != null) {
holder.itemView.setOnClickListener(this);
}
if (onSubViewClickListener != null) {
holder.setSubViewClickListener(onSubViewClickListener,position);
}
}
可以看出 這個(gè)方法里都是操作我們自定義的BaseViewHolder類。接下來就是三個(gè)空判斷,也就是說,我們?nèi)羰菦]有設(shè)置相應(yīng)的點(diǎn)擊事件,就不會(huì)初始化對(duì)應(yīng)的點(diǎn)擊事件,這樣的處理方式還是很常見的。處理這個(gè)點(diǎn)擊事件最麻煩的就是position的問題,因此我們使用的技巧是,對(duì)View對(duì)象設(shè)置tag的方式。查看源碼便知道,View有個(gè)方法 setTag(Object obj); 我們就可以將對(duì)應(yīng)的position賦值給這個(gè)tag,我們使用View的getTag() 方法就可以得到對(duì)應(yīng)點(diǎn)擊View的position了。在BaseRecylerAdapter類實(shí)現(xiàn)的點(diǎn)擊接口和長(zhǎng)按接口就可以知道這樣的操作,類容如下。
@Override
public void onClick(View v) {
if (v.getTag() != null) {
int position = (int) v.getTag();
onRecyclerViewItemClickListener.onItemClick(position);
}
}
@Override
public boolean onLongClick(View v) {
if (v.getTag() != null){
int position = (int)v.getTag();
onRecyclerViewItemLongClickListener.onItemLongClick(position);
}
return true;
}
BaseViewHolder類
public abstract class BaseViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener{
private BaseRecyclerAdapter.OnSubViewClickListener onSubViewClickListener;
public BaseViewHolder(View itemView) {
super(itemView);
findViewById(itemView);
}
/**
* 傳入子項(xiàng)點(diǎn)擊事件所需參數(shù)
* @param listener 自定義的接口
* @param tagPosition tag
*/
public void setSubViewClickListener(BaseRecyclerAdapter.OnSubViewClickListener listener, int tagPosition){
this.onSubViewClickListener = listener;
initSubViewClick(tagPosition);
}
/**
* 通過id匹配控件(開發(fā)者自行實(shí)現(xiàn))
* @param itemView 父布局
*/
abstract protected void findViewById(View itemView);
/**
* 用于裝載數(shù)據(jù)(開發(fā)者自行實(shí)現(xiàn))
* @param position 當(dāng)前位置
*/
abstract protected void onBind(int position);
/**
* 初始化子項(xiàng)的點(diǎn)擊事件(為子項(xiàng)設(shè)置tag)
* @param tagPosition tag
*/
protected void initSubViewClick(int tagPosition){
}
/**
* 實(shí)現(xiàn)子項(xiàng)點(diǎn)擊事件的轉(zhuǎn)發(fā)
* @param v
*/
@Override
public void onClick(View v) {
if (v.getTag() != null) {
int position = (int) v.getTag();
onSubViewClickListener.onSubViewClick(v,position);
}
}
}
這是個(gè)抽象類,也就是說,在使用的時(shí)候需要實(shí)現(xiàn)其中的抽象方法。為了邏輯清晰,我在這里寫了兩個(gè)抽象方法
/** * 通過id匹配控件(開發(fā)者自行實(shí)現(xiàn)) * @param itemView 父布局 */ abstract protected void findViewById(View itemView); /** * 用于裝載數(shù)據(jù)(開發(fā)者自行實(shí)現(xiàn)) * @param position 當(dāng)前位置 */ abstract protected void onBind(int position);
看注釋也就很清楚這兩個(gè)方法的作用是什么,這里就不多說了。
到此,我們已經(jīng)實(shí)現(xiàn)了itemView的點(diǎn)擊和長(zhǎng)按事件,接下來我們來實(shí)現(xiàn)對(duì)itemView子項(xiàng)的點(diǎn)擊事件。
在BaseViewHolder類中,也實(shí)現(xiàn)了一個(gè)View的點(diǎn)擊事件接口。子項(xiàng)的點(diǎn)擊方式和itemView的點(diǎn)擊事件是一樣的套路,使用tag。接下來我們來看個(gè)例子,就明白了。
public class RecyclerAdapterMyActivity extends BaseRecyclerAdapter{
private List<MyActivityBean> list;
public RecyclerAdapterMyActivity(List<MyActivityBean>list){
this.list = list;
}
@Override
public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_recycler_myactivity_activity,parent,false);
ViewHolder holder = new ViewHolder(view);
return holder;
}
@Override
public int getItemCount() {
return list.size();
}
public class ViewHolder extends BaseViewHolder {
private TextView tv_name,tv_title,tv_content;
private Button activityBtnChat;
private Button activityBtnCancel;
ViewHolder(View itemView) {
super(itemView);
}
@Override
protected void findViewById(View itemView) {
tv_name = itemView.findViewById(R.id.tv_my_activity_name);
tv_title = itemView.findViewById(R.id.tv_my_activity_title);
tv_content = itemView.findViewById(R.id.tv_my_activity_content);
activityBtnChat = itemView.findViewById(R.id.activity_btn_chat);
activityBtnCancel = itemView.findViewById(R.id.activity_btn_cancel);
}
@Override
protected void onBind(int position) {
MyActivityBean bean = list.get(position);
tv_name.setText(bean.getName());
tv_title.setText(bean.getTitle());
tv_content.setText(bean.getContent());
}
@Override
protected void initSubViewClick(int tagPosition) {
activityBtnChat.setTag(tagPosition);
activityBtnCancel.setTag(tagPosition);
activityBtnChat.setOnClickListener(this);
activityBtnCancel.setOnClickListener(this);
}
}
}
這段代碼是最近項(xiàng)目中的一小段代碼。其中,adapter類繼承BaseRecyclerAdapter,viewHolder類繼承BaseViewHolder。尤其要注意的是ViewHolder的構(gòu)造方法中一定要有super(itemView); 其余的方法都會(huì)按照正確的邏輯執(zhí)行。若要實(shí)現(xiàn)itemView的子項(xiàng)點(diǎn)擊事件,需要重寫父類的initSubViewClick(int tagPosition) ; 方法。其中參數(shù)tagPosition便是對(duì)應(yīng)的itemVIew處于RecyclerView中的位置。在這里是為兩個(gè)button添加點(diǎn)擊事件,先為其設(shè)置tag,再設(shè)置點(diǎn)擊事件,我們這里的setOnClickListener(this) ; 參數(shù)傳的是this,是因?yàn)?,我們?cè)俑割愔袑?shí)現(xiàn)了View的onClick(View v); 方法。
這樣,我們便完成了各類點(diǎn)擊事件。
使用方法也很簡(jiǎn)單,就是直接操作你的adapter就可以了,調(diào)用adapter.setXxxx(...) 即可方便地實(shí)現(xiàn)各種點(diǎn)擊事件。當(dāng)然,要是你地需求是Touchu事件,或子項(xiàng)地長(zhǎng)按事件等,都可以通過這樣類似地方式來實(shí)現(xiàn)。
最后
方法不止一種,這樣地操作方式,這只是我的一種思考。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android應(yīng)用關(guān)閉的情況以及識(shí)別方法詳解
對(duì)于現(xiàn)在的安卓手機(jī)而言,很多功能都是在逐步完善的,這篇文章主要給大家介紹了關(guān)于Android應(yīng)用關(guān)閉的情況以及識(shí)別的相關(guān)資料,文章通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06
Android webview加載https鏈接錯(cuò)誤或無響應(yīng)的解決
這篇文章主要介紹了Android webview加載https鏈接錯(cuò)誤或無響應(yīng)的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-03-03
Android開發(fā)技巧之永不關(guān)閉的Toast信息框(長(zhǎng)時(shí)間顯示而非系統(tǒng)關(guān)閉)
Toast信息提示框之所以在顯示一定時(shí)間后會(huì)自動(dòng)關(guān)閉,是因?yàn)樵谙到y(tǒng)中有一個(gè)Toast隊(duì)列;那么有些時(shí)候需要這個(gè)Toast信息提示框長(zhǎng)時(shí)間顯示,直到需要關(guān)閉它時(shí)通過代碼來控制,而不是讓系統(tǒng)自動(dòng)來關(guān)閉Toast信息提示框2013-01-01
Android使用OKHttp庫(kù)實(shí)現(xiàn)視頻文件的上傳到服務(wù)器功能
這篇文章主要介紹了Android使用OKHttp庫(kù)實(shí)現(xiàn)視頻文件的上傳到服務(wù)器功能,需要的朋友可以參考下2018-03-03
Android ViewPager中顯示圖片與播放視頻的填坑記錄
這篇文章主要給介紹了關(guān)于Android ViewPager中顯示圖片與播放視頻的一些填坑記錄,文中通過示例代碼介紹的非常詳細(xì),對(duì)各位Android開發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-05-05
Kotlin Channel處理多個(gè)數(shù)據(jù)組合的流
最近項(xiàng)目中對(duì) kotlin 的使用比較多。不得不說 kotlin 確實(shí)可以極大的提高 android 的開發(fā)效率,channel用于協(xié)程之間的通訊,使用send和receive往通道里寫入或者讀取數(shù)據(jù),2個(gè)方法為非阻塞掛起函數(shù),channel是熱流,不管有沒有訂閱者都會(huì)發(fā)送2022-11-11
android 6.0下webview的定位權(quán)限設(shè)置方法
今天小編就為大家分享一篇android 6.0下webview的定位權(quán)限設(shè)置方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-07-07
Android編程之線性布局LinearLayout實(shí)例簡(jiǎn)析
這篇文章主要介紹了Android編程之線性布局LinearLayout用法,結(jié)合實(shí)例形式簡(jiǎn)單分析了Android線性布局的使用技巧,需要的朋友可以參考下2016-01-01

