`

【Android】ListView与Button的共存问题解决

阅读更多

这两天在捣鼓ListView widget,为了在ListView中加入Button这类的有 “点击” 事件的widget,请教了不少高手,感谢LandMark对我的认真讲解,下面把解决过程描述一下。
 
ListView 和 其它能触发点击事件的widget无法一起正常工作的原因是加入其它widget后,ListView的itemclick事件将无法触发,被其它widget的click事件屏蔽。
 
  • 首先,说明一下,ListView中每一行包括以下三项:
 
   一个ImageView, 一个TextView,一个ImageButton,依次排开。
 
以下是layout的内容,分为两部分:
  • res/layout/main.xml

 

< ? 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>

因为继承了ListActivity,所以ListView 的id设置为"@id/android:list"是必须的

  • 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类的实现。

public class lvWithButtonExt extends ListActivity { 
    @Override 
    protected void 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 = new ArrayList < HashMap< String , Object > > ( ) ; 
        for ( int i= 0; i< 10; i+ + ) 
        { 
            HashMap < String , Object > map = new HashMap < 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对应的子项  
            new String [ ] { "ItemImage" , "ItemWinName" , "ItemCloseWin" } , 
            //ImageItem的XML文件里面的一个ImageView,两个TextView ID  
            new int [ ] { R. id . ItemImage, R. id . ItemWinName, R. id . ItemCloseWin} 
        ) ; 
        
        vncListView. setAdapter( listItemAdapter) ; 
    } 

    @Override 
    protected void onListItemClick( ListView l, View v, int position , long id ) { 
        // TODO Auto-generated method stub 
        super . onListItemClick( l, v, position , id ) ; 
        l. getItemAtPosition( position ) ; 
    } 
}

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

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

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

public class lvButtonAdapter extends BaseAdapter { 
    private class buttonViewHolder { 
        ImageView appIcon; 
        TextView appName; 
        ImageButton buttonClose; 
    } 
    
    private ArrayList < HashMap < String , Object > > mAppList; 
    private LayoutInflater mInflater; 
    private Context mContext; 
    private String [ ] keyString; 
    private int [ ] valueViewID; 
    private buttonViewHolder holder; 
    
    public lvButtonAdapter( Context c, ArrayList < HashMap < String , Object > > appList, intresource, 
            String [ ] from , int [ ] to) { 
        mAppList = appList; 
        mContext = c; 
        mInflater = ( LayoutInflater) mContext. getSystemService( Context .LAYOUT_INFLATER_SERVICE) ; 
        keyString = new String [ from . length ] ; 
        valueViewID = new int [ to. length ] ; 
        System . arraycopy ( from , 0, keyString, 0, from . length ) ; 
        System . arraycopy ( to, 0, valueViewID, 0, to. length ) ; 
    } 
    
    @Override 
    public int getCount ( ) { 
        return mAppList. size ( ) ; 
    } 

    @Override 
    public Object getItem ( int position ) { 
        return mAppList. get ( position ) ; 
    } 

    @Override 
    public long getItemId( int position ) { 
        return position ; 
    } 

    public void removeItem ( int position ) { 
        mAppList. remove ( position ) ; 
        this . notifyDataSetChanged( ) ; 
    } 
    
    @Override 
    public View getView ( int position , 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 { 
        private int position ; 

        lvButtonListener( int pos) { 
            position = pos; 
        } 
        
        @Override 
        public void 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的内容

分享到:
评论
3 楼 求知者long 2014-06-30  
受益了!
2 楼 892848153 2013-08-15  
你这个listView有headView么?有的话,headView是第0项。
1 楼 沉哖夕逝傷 2013-08-15  
你好,我遇到一个感觉比较奇葩的问题:就是item中的button监听的时候得到的position和该item在listview中显示的position不一致。也就是 holder. buttonClose. setOnClickListener( new lvButtonListener( position ) ) ; 中的position和listview.setOnItemClickListener();中得到 的position不一样。导致删除数据的时候没有删除自定项。有时候甚至会有数组下标越界的情况。不知你有没有遇到

相关推荐

    android中ListView和Button共存的问题

    使用BaseAdapter自定义适配器,采用了回收converView和ViewHolder优化代码,并且解决了在ListView中使用Button时,ListView的itemClick不能被触发的问题。

    android中listView的Button监听

    最近在做android项目时 发现要在listView用到Button等各种点击事件 经过各种百度 发现有两个简单的方法实现 我把demo放上去 为了方便自己以后用到 也方便学习listview实现button监听的同学们 注意 我用最新sdk...

    Android ListView添加Button及其事件

    以上就是关于“Android ListView添加Button及其事件”的详细知识,希望对你理解ListView的自定义和事件处理有所帮助。在实际开发中,你可以根据需求扩展这个例子,例如添加更多的交互元素,或者处理更复杂的业务逻辑...

    Android listview和button点击事件冲突问题解决方法

    然而,这样处理后,虽然解决了Button获取焦点的问题,但新的问题是,点击ListView项时,Button仍会出现按下的视觉效果。这是因为即使Button没有焦点,它的`setPressed()`方法仍会被调用。 为了解决这个问题,我们...

    android listview 中button 事件处理

    本文用两种方法BaseAdapter 和SimpleAdapter分别实现三种情况。listview中有button事件的时候,针对不同的需求,响应不同的事件。只实现响应listview的点击事件。只响应listview中的button事件。两个事件都需要。

    ListView和Button的结合使用

    本文将详细介绍如何在Android应用中将ListView与Button结合使用,以及如何处理它们的点击事件,实现不同的操作。 首先,我们需要创建一个ListView。在布局文件(如activity_list_view.xml)中,添加ListView元素,...

    Android listview和viewpager解决冲突 滑动冲突

    Android listview viewpager 滑动 跳动 冲突解决 ListView中嵌套ViewPage有或者滑动手势冲突解决 blog 地址 : http://blog.csdn.net/aaawqqq/article/details/43824631

    解决ListView 和 ScrollView 共存冲突的问题

    总之,解决ListView和ScrollView共存问题的关键在于理解它们的滚动机制,并根据具体需求选择合适的方法。通常,尽量避免直接嵌套,而是利用RecyclerView的灵活性和NestedScrollView的嵌套滚动特性。对于复杂的布局...

    Android ListView添加两个Button焦点获取

    Android ListView添加两个Button焦点获取 listview 放弃焦点 让Button获取焦点 很简单的一个demo 点击相应 我没弄 也很简单 焦点都给你获取了 点击都不会做 我也没办法了 百度了这么久 自己写了个 希望能帮到大家

    Android_ScrollView与ListView_GridView共存冲突解决方案

    ### Android ScrollView与ListView/GridView共存冲突解决方案 #### 背景介绍 在Android开发过程中,经常需要将多个UI组件组合在一起以实现丰富的界面效果。其中,`ScrollView`是一种常用的容器,可以用来包裹一系列...

    Android完美解决listview中多个edittext显示混乱问题

    综上所述,解决Android ListView中多个EditText显示混乱的问题需要理解ListView的复用机制,并针对性地调整Adapter、EditText以及ListView的行为。结合提供的源码和示例,开发者可以更直观地学习和应用这些解决方案...

    android ViewPager 和 listview的共存问题

    在Android开发中,ViewPager和ListView是...总之,虽然ViewPager和ListView的共存带来了一些挑战,但通过合理的布局设计、手势处理和性能优化,我们可以有效地解决这些问题,实现功能丰富且用户体验良好的Android应用。

    listView上增加button点击事件

    在Android开发中,ListView是一种常用的组件,用于展示大量的列表数据。然而,在实际应用中,我们经常需要在ListView的每一项(item)中添加按钮(button),以便用户进行交互操作。标题"listView上增加button点击...

    android的listview嵌套listview,列表嵌套列表 android studio版本

    当需求涉及列表中的每个条目本身也需要展示一个子列表时,我们就会遇到ListView的嵌套问题。本教程将详细讲解如何在Android Studio环境下实现ListView的嵌套,以满足这种复杂布局的需求。 首先,我们需要了解基本的...

    ListView的Button点击更改Button本身背景

    同时,为了防止状态丢失,我们还需要在`getView()`中处理View的复用问题,确保每次显示的数据与数据集中的对应项匹配。 代码示例: ```java public class MyAdapter extends ArrayAdapter&lt;String&gt; { private ...

    Android ListView边框圆角美化

    在Android开发中,ListView是常用的一种控件,用于展示大量数据列表。为了提升用户体验和界面设计的美观性,我们常常需要对ListView进行定制化,包括设置边框和实现圆角效果。本文将深入探讨如何在Android中实现...

    Android listview自动换行

    然而,当ListView的子项(item)宽度超过屏幕宽度时,如何实现自动换行就成了一个需要解决的问题。在本篇中,我们将深入探讨如何在Android中为ListView实现自动换行的功能。 首先,我们需要理解ListView的工作原理...

    Android提高篇之listView点击button翻页功能实现源码

    Android提高篇之listView点击button翻页功能实现源码,最近的开发需要在手机上实现列表分页功能,可以设置每页显示的记录数,第一页和最后一页翻页按钮自动置灰。代码中包括RelativeLayout对视图(View)和按钮位置...

Global site tag (gtag.js) - Google Analytics