完美解决Android在listview添加checkbox实现单选多选操作问题

http://blog.csdn.net/simtice/article/details/7068774


在Android某些开发需求当中,有时候需要在listveiw中加入checkbox实现单选,多选操作。表面上看上去只是改变checkbox那么简单,然而实际开发中,实现起来并不是那么得心应手。尤其当listview比较多(比如屏幕最多只能显示10个item,但总共有12个item,也就是说listview的item数大于屏幕能够显示的item数)滑动屏幕的时候,由于适配器中getview()会重复使用被移除屏幕的item,所以会造成checkbox选择状态不正常的现象。自己在开发中碰到这样的问题很是苦恼,查了下资料,发现网上很少没有针对这类批量操作并没有一个完整的例子。搜了很多篇帖子才完美的实现这一常用的操作。所以在这里把这个Demo贴出来,供大家参考,希望能对大家有所帮助。

主界面的布局main.xml 这个就不多说什么

  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="wrap_content"
  5. android:orientation="vertical">
  6. <LinearLayout
  7. android:orientation="vertical"
  8. android:layout_width="fill_parent"
  9. android:layout_height="wrap_content"
  10. >
  11. <TextView
  12. android:id="@+id/tv"
  13. android:layout_width="fill_parent"
  14. android:layout_height="50dip"
  15. android:textColor="#FCFCFC"
  16. android:textSize="11pt"
  17. android:gravity="center_vertical"
  18. android:layout_marginLeft="10dip"
  19. />
  20. <ListView
  21. android:id="@+id/lv"
  22. android:layout_width="fill_parent"
  23. android:layout_height="381dip"
  24. android:cacheColorHint="#00000000"
  25. ></ListView>
  26. </LinearLayout>
  27. <RelativeLayout
  28. android:layout_width="fill_parent"
  29. android:layout_height="53dip"
  30. android:orientation="horizontal"
  31. >
  32. <Button
  33. android:id="@+id/selectall"
  34. android:layout_width="80dip"
  35. android:layout_height="50dip"
  36. android:layout_marginLeft="20dip"
  37. android:text="全选"
  38. android:gravity="center"
  39. />
  40. <Button
  41. android:id="@+id/inverseselect"
  42. android:layout_width="80dip"
  43. android:layout_height="50dip"
  44. android:layout_marginLeft="118dip"
  45. android:text="反选"
  46. android:gravity="center"
  47. />
  48. <Button
  49. android:id="@+id/cancel"
  50. android:layout_width="80dip"
  51. android:layout_height="50dip"
  52. android:layout_marginLeft="213dip"
  53. android:text="取消已选"
  54. android:gravity="center"
  55. />
  56. </RelativeLayout>
  57. </LinearLayout>


ListView每个item的布局,listviewitem.xml:

这里需要注意的是,由于checkbox的点击事件优先级比listview的高,所以要添加android:focusable="false"属性,使得checkbox初始的时候没有获取焦点。

另外这里是点击ListView的item控制checkbox的状态改变,也就是让item接收clik事件,所以需要加上android:focusableInTouchMode="false"这一属性。

  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="55dip"
  5. android:orientation="horizontal"
  6. android:layout_marginTop="20dip"
  7. >
  8. <TextView
  9. android:id="@+id/item_tv"
  10. android:layout_width="267dip"
  11. android:layout_height="40dip"
  12. android:textSize="10pt"
  13. android:gravity="center_vertical"
  14. android:layout_marginLeft="10dip"
  15. />
  16. <CheckBox
  17. android:id="@+id/item_cb"
  18. android:layout_width="wrap_content"
  19. android:layout_height="wrap_content"
  20. android:focusable="false"
  21. android:focusableInTouchMode="false"
  22. android:clickable="false"
  23. android:layout_toRightOf="@id/item_tv"
  24. android:layout_alignParentTop="true"
  25. android:layout_marginRight="5dip"
  26. />
  27. </RelativeLayout>


ViewHolder类

  1. packagesimtice.test.listview.viewholder;
  2. importandroid.widget.CheckBox;
  3. importandroid.widget.TextView;
  4. publicclassViewHolder{
  5. publicTextViewtv=null;
  6. publicCheckBoxcb=null;
  7. }


为listview自定义适配器,该类为主Activity类MainActivity.java的内部类

  1. publicstaticclassMyAdapterextendsBaseAdapter{
  2. publicstaticHashMap<Integer,Boolean>isSelected;
  3. privateContextcontext=null;
  4. privateLayoutInflaterinflater=null;
  5. privateList<HashMap<String,Object>>list=null;
  6. privateStringkeyString[]=null;
  7. privateStringitemString=null;//记录每个item中textview的值
  8. privateintidValue[]=null;//id值
  9. publicMyAdapter(Contextcontext,List<HashMap<String,Object>>list,
  10. intresource,String[]from,int[]to){
  11. this.context=context;
  12. this.list=list;
  13. keyString=newString[from.length];
  14. idValue=newint[to.length];
  15. System.arraycopy(from,0,keyString,0,from.length);
  16. System.arraycopy(to,0,idValue,0,to.length);
  17. inflater=LayoutInflater.from(context);
  18. init();
  19. }
  20. //初始化设置所有checkbox都为未选择
  21. publicvoidinit(){
  22. isSelected=newHashMap<Integer,Boolean>();
  23. for(inti=0;i<list.size();i++){
  24. isSelected.put(i,false);
  25. }
  26. }
  27. @Override
  28. publicintgetCount(){
  29. returnlist.size();
  30. }
  31. @Override
  32. publicObjectgetItem(intarg0){
  33. returnlist.get(arg0);
  34. }
  35. @Override
  36. publiclonggetItemId(intarg0){
  37. return0;
  38. }
  39. @Override
  40. publicViewgetView(intposition,Viewview,ViewGrouparg2){
  41. ViewHolderholder=null;
  42. if(holder==null){
  43. holder=newViewHolder();
  44. if(view==null){
  45. view=inflater.inflate(R.layout.listviewitem,null);
  46. }
  47. holder.tv=(TextView)view.findViewById(R.id.item_tv);
  48. holder.cb=(CheckBox)view.findViewById(R.id.item_cb);
  49. view.setTag(holder);
  50. }else{
  51. holder=(ViewHolder)view.getTag();
  52. }
  53. HashMap<String,Object>map=list.get(position);
  54. if(map!=null){
  55. itemString=(String)map.get(keyString[0]);
  56. holder.tv.setText(itemString);
  57. }
  58. holder.cb.setChecked(isSelected.get(position));
  59. returnview;
  60. }
  61. }


最后,最重要的就是MainActivity.java中一些事件响应的处理

  1. publicclassMainActivityextendsActivity{
  2. TextViewtv=null;
  3. ListViewlv=null;
  4. Buttonbtn_selectAll=null;
  5. Buttonbtn_inverseSelect=null;
  6. Buttonbtn_calcel=null;
  7. Stringname[]={"G1","G2","G3","G4","G5","G6","G7","G8","G9",
  8. "G10","G11","G12","G13","G14"};
  9. ArrayList<String>listStr=null;
  10. privateList<HashMap<String,Object>>list=null;
  11. privateMyAdapteradapter;
  12. @Override
  13. publicvoidonCreate(BundlesavedInstanceState){
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.main);
  16. tv=(TextView)this.findViewById(R.id.tv);
  17. lv=(ListView)this.findViewById(R.id.lv);
  18. btn_selectAll=(Button)this.findViewById(R.id.selectall);
  19. btn_inverseSelect=(Button)this.findViewById(R.id.inverseselect);
  20. btn_calcel=(Button)this.findViewById(R.id.cancel);
  21. showCheckBoxListView();
  22. //全选
  23. btn_selectAll.setOnClickListener(newOnClickListener(){
  24. @Override
  25. publicvoidonClick(Viewarg0){
  26. listStr=newArrayList<String>();
  27. for(inti=0;i<list.size();i++){
  28. MyAdapter.isSelected.put(i,true);
  29. listStr.add(name[i]);
  30. }
  31. adapter.notifyDataSetChanged();//注意这一句必须加上,否则checkbox无法正常更新状态
  32. tv.setText("已选中"+listStr.size()+"项");
  33. }
  34. });
  35. //反选
  36. btn_inverseSelect.setOnClickListener(newOnClickListener(){
  37. @Override
  38. publicvoidonClick(Viewv){
  39. for(inti=0;i<list.size();i++){
  40. if(MyAdapter.isSelected.get(i)==false){
  41. MyAdapter.isSelected.put(i,true);
  42. listStr.add(name[i]);
  43. }
  44. else{
  45. MyAdapter.isSelected.put(i,false);
  46. listStr.remove(name[i]);
  47. }
  48. }
  49. adapter.notifyDataSetChanged();
  50. tv.setText("已选中"+listStr.size()+"项");
  51. }
  52. });
  53. //取消已选
  54. btn_calcel.setOnClickListener(newOnClickListener(){
  55. @Override
  56. publicvoidonClick(Viewv){
  57. for(inti=0;i<list.size();i++){
  58. if(MyAdapter.isSelected.get(i)==true){
  59. MyAdapter.isSelected.put(i,false);
  60. listStr.remove(name[i]);
  61. }
  62. }
  63. adapter.notifyDataSetChanged();
  64. tv.setText("已选中"+listStr.size()+"项");
  65. }
  66. });
  67. }
  68. //显示带有checkbox的listview
  69. publicvoidshowCheckBoxListView(){
  70. list=newArrayList<HashMap<String,Object>>();
  71. for(inti=0;i<name.length;i++){
  72. HashMap<String,Object>map=newHashMap<String,Object>();
  73. map.put("item_tv",name[i]);
  74. map.put("item_cb",false);
  75. list.add(map);
  76. adapter=newMyAdapter(this,list,R.layout.listviewitem,
  77. newString[]{"item_tv","item_cb"},newint[]{
  78. R.id.item_tv,R.id.item_cb});
  79. lv.setAdapter(adapter);
  80. listStr=newArrayList<String>();
  81. lv.setOnItemClickListener(newOnItemClickListener(){
  82. @Override
  83. publicvoidonItemClick(AdapterView<?>arg0,Viewview,
  84. intposition,longarg3){
  85. ViewHolderholder=(ViewHolder)view.getTag();
  86. holder.cb.toggle();//在每次获取点击的item时改变checkbox的状态
  87. MyAdapter.isSelected.put(position,holder.cb.isChecked());//同时修改map的值保存状态
  88. if(holder.cb.isChecked()==true){
  89. listStr.add(name[position]);
  90. }else{
  91. listStr.remove(name[position]);
  92. }
  93. tv.setText("已选中"+listStr.size()+"项");
  94. }
  95. });
  96. }
  97. }
  98. //为listview自定义适配器内部类
  99. publicstaticclassMyAdapterextendsBaseAdapter{
  100. ...
  101. }
  102. }

好了,来看运行结果

我选择了G2、G3、G11三项,现在屏幕滑动到底部,看以看到状态保存的很好,TextView显示已选中3项。全选、反选、取消已选功能正常,多选操作完美解决!


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值