最近碰到了個問題,在正在下載頁面舶担,點擊取消一個下載任務(wù)的時候坡疼,總是出現(xiàn)點擊事件的混亂,要么是點擊取消按鈕(一個imageview)沒反應(yīng)衣陶,要么是選中了整個item柄瑰。
通過分析發(fā)現(xiàn),暫停下載的話剪况,是不存在這個問題的教沾,因此排除了imageview點擊區(qū)域不夠大的原因,也排除了父控件和子控件的focus爭搶的問題译断。
最終解決問題是刷到了這個:
http://blog.csdn.net/guolin_blog/article/details/45586553
這篇文章讓我對listview的原理有了更加充分的認(rèn)識授翻,“ListView中的子View其實來來回回就那么幾個,移出屏幕的子View會很快被移入屏幕的數(shù)據(jù)重新利用起來”,也就是說堪唐,listview的子view是被不斷復(fù)用的巡语,相同的view,設(shè)上不同的數(shù)據(jù)淮菠,出來的就是不同的行男公。
基于此,我猜想合陵,當(dāng)我點擊的時候理澎,被點擊的view可能已經(jīng)被刷新了(不是當(dāng)時的那個view了)。仔細(xì)排查代碼發(fā)現(xiàn)曙寡,每次來一條下載進(jìn)度更新的消息糠爬,都使用NotifyDataSetChanged()刷新了整個列表。這樣也就解釋通了举庶,為什么暫停下載時點擊沒有問題执隧,而下載的時候列表點擊就有問題了。
解決問題的方式:
(1)列表局部更新:每當(dāng)收到一條進(jìn)度更新的消息的時候户侥,從消息中取出對應(yīng)的數(shù)據(jù)镀琉,來只更新其中的一行。參考其中的解決方案一 蕊唐,使用findViewWithTag屋摔,來找到要更新的view,這更新這一行的相關(guān)view就行替梨,避免刷新整個列表
(2)給view設(shè)tag钓试,tag能幫忙標(biāo)示一個view。在onclick事件中副瀑,通過getTag的方式弓熏,把view中的數(shù)據(jù)取出來。
在自定義Adapter的getview()函數(shù)里面糠睡,給需要和進(jìn)度關(guān)聯(lián)的幾個控件設(shè)上tag:
public View getView(int position, View convertView, ViewGroup parent) {
if (mListView == null) {
mListView = (ListView) parent;
}
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.downloading_item, null);
itemView = new ViewHolder();
itemView.mCtrl = (ImageView) convertView.findViewById(R.id.ctrl);
itemView.mProgressText = (TextView) convertView.findViewById(R.id.progress_text);
itemView.mProgressBar = (ProgressBar) convertView.findViewById(R.id.progress);
itemView.mDeleteView = convertView.findViewById(R.id.delete);
convertView.setTag(itemView);
} else {
itemView = (ViewHolder) convertView.getTag();
}
long cloudId = mSonglist.get(position).getCloudId();
String progress_text_tag = String.valueOf(cloudId).concat("_text");
String ctrl_tag = String.valueOf(cloudId).concat("_ctrl");
itemView.mProgressBar.setTag(cloudId);
itemView.mProgressText.setTag(progress_text_tag);
itemView.mCtrl.setTag(ctrl_tag);
// imageview的點擊事件
itemView.mDeleteView.setOnClickListener(mOnDeleteClickListener);
itemView.mDeleteView.setTag(cloudId);
//整個item的點擊事件
convertView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View view) {
//...
}
});
return convertView;
}
在收到某條進(jìn)度更新的消息后挽鞠,通過mListView的findViewWithTag,找到要更新的view狈孔,設(shè)置上相應(yīng)的狀態(tài)信息
(注意:不要直接調(diào)用notifyDataSetChanged()信认,這樣會刷新整個列表)
public void onEventMainThread(DownloadEvents.OnDownloadProgressChanged event) {
DownloadSong downloadSong = event.downloadSong;
long progress = downloadSong.getFileLengthDownloaded();
long total = downloadSong.getFileLength();
long cloudId = downloadSong.getCloudId();
String progress_text_tag = String.valueOf(cloudId).concat("_text");
String progress_ctrl_tag = String.valueOf(cloudId).concat("_ctrl");
ProgressBar progressBar = (ProgressBar)mListView.findViewWithTag(cloudId);
TextView progressText = (TextView)mListView.findViewWithTag(progress_text_tag);
ImageView ctrl = (ImageView)mListView.findViewWithTag(progress_ctrl_tag);
if (progressBar != null && progressText != null && ctrl != null) {
updateView(status, progress, total, progressBar, progressText, ctrl);
}
}
解決問題的過程中還參考學(xué)習(xí)了以下:
http://txlong-onz.iteye.com/blog/907186
listview 更新進(jìn)度條
http://www.myexception.cn/mobile/1880588.html
http://f303153041.iteye.com/blog/1838092
android:descendantFocusability用法簡析
開發(fā)中很常見的一個問題,項目中的listview不僅僅是簡單的文字均抽,常常需要自己定義listview嫁赏,自己的Adapter去繼承BaseAdapter,在adapter中按照需求進(jìn)行編寫到忽,問題就出現(xiàn)了橄教,可能會發(fā)生點擊每一個item的時候沒有反應(yīng)清寇,無法獲取的焦點。原因多半是由于在你自己定義的Item中存在諸如ImageButton护蝶,Button华烟,CheckBox等子控件(也可以說是Button或者Checkable的子類控件),此時這些子控件會將焦點獲取到持灰,所以常常當(dāng)點擊item時變化的是子控件盔夜,item本身的點擊沒有響應(yīng)。
這時候就可以使用descendantFocusability來解決啦堤魁,API描述如下:
android:descendantFocusability
Defines the relationship between the ViewGroup and its descendants when looking for a View to take focus.
Must be one of the following constant values.
該屬性是當(dāng)一個為view獲取焦點時喂链,定義viewGroup和其子控件兩者之間的關(guān)系。
屬性的值有三種:
beforeDescendants:viewgroup會優(yōu)先其子類控件而獲取到焦點
afterDescendants:viewgroup只有當(dāng)其子類控件不需要獲取焦點時才獲取焦點
blocksDescendants:viewgroup會覆蓋子類控件而直接獲得焦點
通常我們用到的是第三種妥泉,即在Item布局的根布局加上android:descendantFocusability=”blocksDescendants”的屬性就好了椭微,至此listview點擊的靈異事件告一段落。心得:遇到不會不懂的地方除了網(wǎng)上查詢資料之外盲链,也可以多多去嘗試每種屬性的作用蝇率,多閱讀官方文檔(我始終覺得還是讀原文的比翻譯的理解的會更好)。