ListView与Button的共存问题解决

在项目当中我们 会经常遇到ListView与button共存的问题:

ListView 和 其它能触发点击事件的widget无法一起正常工作的原因是加入其它widget后,ListView的itemclick事件将无法触发,被其它widget的click事件屏蔽。

下面把解决过程描述一下:


以下是layout的内容,分为两部分:

<?xml version="1.0"encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent" android:layout_height="fill_parent"
android:padding="10dip" android:orientation="vertical">

<ListView android:id="@id/android:list" android:layout_width="fill_parent"
android:layout_height="fill_parent"/>
</LinearLayout>

res/layout/lvitem.xml

注意:

<RelativeLayout>中

android:descendantFocusability="blocksDescendants"

<ImageButton>中

android:focusable="false"

这两项的设置很关键,如果不设置,将导致ListView的ItemClick事件将无法触发,该事件被ImageButton的click事件屏蔽了。

<?xml version="1.0"encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:padding="5dip"
android:descendantFocusability="blocksDescendants">

<ImageView 
android:id="@+id/ItemImage"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:padding="5dip"
/>


<!-- 
把按钮背景设置为透明: android:background="#00000000"
把按钮背景设置为半透明: android:background="#e0000000"
-->
<ImageButton 
android:id="@+id/ItemCloseWin"
android:layout_alignParentRight="true"
android:layout_alignTop="@+id/ItemWinName"
android:layout_alignBottom="@+id/ItemWinName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#e0000000"
android:gravity="left|center_vertical"
android:focusable="false"
android:src="@android:drawable/ic_menu_close_clear_cancel"
/>
<TextView 
android:id="@+id/ItemWinName"
android:layout_toRightOf="@+id/ItemImage"
android:layout_toLeftOf="@+id/ItemCloseWin"
android:layout_alignTop="@+id/ItemImage"
android:layout_alignBottom="@+id/ItemImage"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:gravity="left|center_vertical"
android:textSize="20dip"
android:text="title"
/>
</RelativeLayout>
  • 接下来,我们看看继承ListActivity的实现

lvWithButtonExt中,为了能处理ImageButton的click事件,我继承了BaseAdapter类,并重新实现了getView()接口,在其中加入了Button的clicklistener,详见lvButtonAdapter类的实现。

publicclass lvWithButtonExt extends ListActivity {
@Override
protectedvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main);

// 关联Layout中的ListView
ListView vncListView =(ListView)findViewById(android.R.id.list);

// 生成动态数组,加入数据 
ArrayList<HashMap<String,Object>> remoteWindowItem =newArrayList<HashMap<String,Object>>();
for(int i=0;i<10;i++)
{
HashMap<String,Object>map=newHashMap<String,Object>();
map.put("ItemImage", R.drawable.firefox);//图像资源的ID 
map.put("ItemWinName","Window ID "+i);
map.put("ItemCloseWin", android.R.drawable.ic_menu_close_clear_cancel);
remoteWindowItem.add(map);
}

// 生成适配器的Item和动态数组对应的元素 
lvButtonAdapter listItemAdapter =new lvButtonAdapter(
this,
remoteWindowItem,//数据源 
R.layout.lvitem,//ListItem的XML实现 

//动态数组与ImageItem对应的子项 
newString[]{"ItemImage","ItemWinName","ItemCloseWin"},
//ImageItem的XML文件里面的一个ImageView,两个TextView ID 
newint[]{R.id.ItemImage,R.id.ItemWinName,R.id.ItemCloseWin}
);

vncListView.setAdapter(listItemAdapter);
}

@Override
protectedvoid onListItemClick(ListView l,View v,intposition,longid){
// TODO Auto-generated method stub
super.onListItemClick(l, v,position,id);
l.getItemAtPosition(position);
}
}

  • 接下来,我们看看lvButtonAdapter的实现

为了响应按钮的点击事件,首先要记录按钮的位置,然后为按钮设置clicklistener。

在重新实现的getView()接口中,我使用了lvButtonListener监听类,在构造函数中,记录行号,以便在OnClick接口中能准确的定位按钮所在的位置,进而对相应的行进行处理。

publicclass lvButtonAdapter extends BaseAdapter {
privateclass buttonViewHolder {
ImageView appIcon;
TextView appName;
ImageButton buttonClose;
}

privateArrayList<HashMap<String,Object>> mAppList;
private LayoutInflater mInflater;
privateContext mContext;
privateString[] keyString;
privateint[] valueViewID;
private buttonViewHolder holder;

public lvButtonAdapter(Context c,ArrayList<HashMap<String,Object>> appList,int resource,
String[]from,int[] to){
mAppList = appList;
mContext = c;
mInflater =(LayoutInflater)mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
keyString =newString[from.length];
valueViewID =newint[to.length];
System.arraycopy(from, 0, keyString, 0,from.length);
System.arraycopy(to, 0, valueViewID, 0, to.length);
}

@Override
publicintgetCount(){
return mAppList.size();
}

@Override
publicObjectgetItem(intposition){
return mAppList.get(position);
}

@Override
publiclong getItemId(intposition){
returnposition;
}

publicvoidremoveItem(intposition){
mAppList.remove(position);
this.notifyDataSetChanged();
}

@Override
publicViewgetView(intposition,View convertView, ViewGroup parent){
if(convertView !=null){
holder =(buttonViewHolder) convertView.getTag();
}else{
convertView = mInflater.inflate(R.layout.lvitem,null);
holder =new buttonViewHolder();
holder.appIcon =(ImageView)convertView.findViewById(valueViewID[0]);
holder.appName =(TextView)convertView.findViewById(valueViewID[1]);
holder.buttonClose =(ImageButton)convertView.findViewById(valueViewID[2]);
convertView.setTag(holder);
}

HashMap<String,Object> appInfo = mAppList.get(position);
if(appInfo !=null){
String aname =(String) appInfo.get(keyString[1]);
int mid =(Integer)appInfo.get(keyString[0]);
int bid =(Integer)appInfo.get(keyString[2]);
holder.appName.setText(aname);
holder.appIcon.setImageDrawable(holder.appIcon.getResources().getDrawable(mid));
holder.buttonClose.setImageDrawable(holder.buttonClose.getResources().getDrawable(bid));
holder.buttonClose.setOnClickListener(new lvButtonListener(position));
} 
return convertView;
}

class lvButtonListener implements OnClickListener {
privateintposition;

lvButtonListener(int pos){
position= pos;
}

@Override
publicvoid onClick(View v){
int vid=v.getId();
if(vid == holder.buttonClose.getId())
removeItem(position);
}
}
}

备注1; 对于Android开发来说处理一些界面需要和Adapter适配器打交道,虽然Android自带了一些比如ArrayAdapter但是大多数情况下无法满足我们需要,所以就要从BaseAdapter派生一个类满足我们特殊的需要。

首先我们可能重写getView(),通过LayoutInflater的inflate方法映射一个自己定义的Layout布局xml加载或从xxxView中创建。这些大家可能滚瓜烂熟了但是仍然很多Android开发者对于BaseAdapter中notifyDataSetChanged()方法不是很理解,notifyDataSetChanged方法通过一个外部的方法控制如果适配器的内容改变时需要强制调用getView来刷新每个Item的内容。



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
WinForms中的ListViewItem按钮是一种在ListView中显示按钮的控件。ListView控件是WinForms中用于展示和编辑大量项目的强大控件。它允许我们以表格的形式显示数据,并且可以自定义每个项的外观和行为。 在ListView中显示按钮,可以为每个ListViewItem添加一个按钮来执行自定义操作。为ListViewItem添加按钮的一种常见方法是使用SubItems属性。每个ListViewItem的SubItems属性可以添加多个子项,而每个子项可以包含一个按钮。通过为每个按钮设置Click事件处理程序,我们可以为按钮定义自定义操作。 例如,我们可以创建一个包含按钮的ListViewItem列表,当用户单击按钮时,会弹出一个消息框显示该按钮所在的ListViewItem的文本。以下是一个示例代码: ```csharp ListView listView = new ListView(); // 创建按钮列 listView.Columns.Add("操作", 100); // 创建ListViewItem ListViewItem item1 = new ListViewItem("项1"); item1.SubItems.Add(new ListViewItem.ListViewSubItem(item1, "按钮1")); item1.SubItems[1].Tag = item1; // 为了在按钮的Click事件处理程序中获取ListViewItem // 添加按钮到ListViewItem Button button1 = new Button(); button1.Text = "点击我"; button1.Click += (sender, e) => { ListViewItem clickedItem = (ListViewItem)((Button)sender).Tag; MessageBox.Show(clickedItem.Text); }; listView.Controls.Add(button1); // 将按钮添加到ListView控件中 listView.Items.Add(item1); // 将ListViewItem添加到ListView控件的项目列表中 ``` 这样,当用户点击按钮时,将会弹出一个消息框,显示按钮所在的ListViewItem的文本。这就是如何使用WinForms的ListViewListViewItem按钮来实现自定义操作的简单示例。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值