Android 中使用ListView和CheckBox進行批量操作

jopen 11年前發布 | 43K 次閱讀 Android Android開發 移動開發

在使用ListView時,一般為了性能的提升,都會使用ViewHolder,也就是Item的View實現復用。

現在的問題是,當在ListView的Item中包含CheckBox,并且CheckBox的事件處理監聽器是holder.checkbox.setOnCheckedChangeListener()時,會出現第一項開始未選中,當第二項選中時第一項也跟著選中,這顯然不是我們想要的結果。

出現這個問題的原因是第一項和第二項用的是同一個Item,當第二項選中時,CheckBox的當前狀態為選中,這時setOnCheckedChangeListener里面會改變第一項關聯的實體對象的屬性(引用類型,變量A、B都引用同一個對象AA,當A把AA的某個屬性值修改了,B再次訪問時,AA對象的那個屬性的值為A引用改后的值),代碼如下:

holder.checkbox.setOnCheckedChangeListener(new OnCheckedChangeListener() {

            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                driver.setSelected(isChecked);
            }

        });

解決辦法:

1、在ListViewAdapter初始化時,將對象中有關CheckBox是否選中的屬性存儲起來。

  selectedMap = new HashMap<Integer, Boolean>(); 
        int size = mPersons.size();
        for (int i = 0; i < size; i++) { 
            selectedMap.put(i, mPersons.get(i).isSelected()); 
        }

2、去掉CheckBox的holder.checkbox.setOnCheckedChangeListener(){}事件監聽器

3、在Adapter里的 public View getView(final int position, View convertView, ViewGroup parent){}方法體里面,當前的CheckBox是否選中狀態,由之前初始化時保存的對象屬性值控制,代碼如下:

boolean selected = selectedMap.get(position);
holder.checkbox.setChecked(selected);

3、用戶點擊ListView的Item時,改變CheckBox的狀態,代碼如下:

 convertView.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                checkbox.toggle();
                selectedMap.put(position, checkbox.isChecked()); 
                driver.setSelected(checkbox.isChecked());
            }
        });

數據適配器ListViewAdapter的完整代碼:

package com.easipass.cloud.ccp.adapter;

import java.util.ArrayList;
import java.util.HashMap;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.CheckBox;
import android.widget.Filter;
import android.widget.Filterable;
import android.widget.ImageView;
import android.widget.TextView;

import com.easipass.R;
import com.easipass.cloud.ccp.entity.UserInfo;

/**
 * 用戶列表數據適配器
 * 
 * @author android_ls
 */
public final class UserListViewAdapter extends BaseAdapter implements Filterable {
    private LayoutInflater inflater;

    private MyFilter myFilter;

    private final Object mLock = new Object();

    private ArrayList<UserInfo> mPersons;

    private ArrayList<UserInfo> mCheckValues;

    public HashMap<Integer, Boolean> selectedMap;

    public UserListViewAdapter(Context context, ArrayList<UserInfo> cms) {
        inflater = LayoutInflater.from(context);
        mPersons = cms;

        selectedMap = new HashMap<Integer, Boolean>();
        int size = mPersons.size();
        for (int i = 0; i < size; i++) {
            selectedMap.put(i, mPersons.get(i).isSelected());
        }

    }

    @Override
    public int getCount() {
        return mPersons.size();
    }

    @Override
    public Object getItem(int arg0) {
        return mPersons.get(arg0);
    }

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

    @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        ViewHolder holder = null;
        if (convertView == null) {
            convertView = inflater.inflate(R.layout.ccp_carmanager_lv_item, null);
            holder = new ViewHolder();
            holder.text1 = (TextView) convertView.findViewById(R.id.tv_name);
            holder.text2 = (TextView) convertView.findViewById(R.id.tv_phnoe);
            holder.checkbox = (CheckBox) convertView.findViewById(R.id.checkbox);
            holder.imageView = (ImageView) convertView.findViewById(R.id.iv_icon);

            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        final UserInfo driver = mPersons.get(position);

        holder.text1.setText(driver.getName());

        holder.text2.setText(driver.getPhoneNumber());
        // TODO 測試
        holder.imageView.setBackgroundResource(Integer.valueOf(driver.getIconUrl()));
        holder.checkbox.setVisibility(View.VISIBLE);

        boolean selected = selectedMap.get(position);
        holder.checkbox.setChecked(selected);

        final CheckBox checkbox = holder.checkbox;
        convertView.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                checkbox.toggle();
                selectedMap.put(position, checkbox.isChecked());
                driver.setSelected(checkbox.isChecked());
            }
        });

        if (selected) {
            convertView.setClickable(false);
        }

        return convertView;
    }

    @Override
    public Filter getFilter() {
        if (myFilter == null) {
            myFilter = new MyFilter();
        }
        return myFilter;
    }

    class MyFilter extends Filter {

        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();
            if (mCheckValues == null) {
                synchronized (mLock) {
                    mCheckValues = new ArrayList<UserInfo>(mPersons);
                }
            }

            if (prefix == null || prefix.length() == 0) {
                synchronized (mLock) {
                    ArrayList<UserInfo> list = new ArrayList<UserInfo>(mCheckValues);
                    results.values = list;
                    results.count = list.size();
                }
            } else {
                String prefixString = prefix.toString().toLowerCase();
                final ArrayList<UserInfo> values = mCheckValues;
                final int count = values.size();

                final ArrayList<UserInfo> newValues = new ArrayList<UserInfo>(count);
                for (int i = 0; i < count; i++) {
                    final UserInfo value = (UserInfo) values.get(i);
                    if (value.getName().contains(prefixString)) {
                        newValues.add(value);
                    }
                }

                results.values = newValues;
                results.count = newValues.size();
            }

            return results;
        }

        @SuppressWarnings("unchecked")
        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            mPersons = (ArrayList<UserInfo>) results.values;
            if (results.count > 0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }
        }
    }

    static class ViewHolder {
        public TextView text1;

        public TextView text2;

        public ImageView imageView;

        public CheckBox checkbox;
    }

}

Activity中onCreate()里的寫法:

     mSearchToolbar = (SearchToolbar) this.findViewById(R.id.top_search_toolbar);
        mListView = (ListView) this.findViewById(R.id.listview);

        mDriverListAdapter = new UserListViewAdapter(this, driverList);
        mListView.setAdapter(mDriverListAdapter);

        mSearchToolbar.setFilter(mDriverListAdapter.getFilter());

SearchToolbar類的代碼:

package com.easipass.custom.view;
import android.content.Context;
import android.text.Editable;
import android.text.TextWatcher;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.AutoCompleteTextView;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.RelativeLayout;

import com.easipass.R;

/**
 * 功能描述:自定義搜索框組件
 * @author android_ls
 */
public class SearchToolbar extends FrameLayout {

    private RelativeLayout topSearchToolbar;

    /**
     * 頂部自動補全文本輸入框
     */
    private AutoCompleteTextView autoSearch;

    /**
     * 清除搜索結果按鈕
     */
    private ImageView btnClearSearch;

    public SearchToolbar(Context context) {
        super(context);
        setupViews();
    }

    public SearchToolbar(Context context, AttributeSet attrs) {
        super(context, attrs);
        setupViews();
    }

    private void setupViews() {
        final LayoutInflater mLayoutInflater = LayoutInflater.from(getContext());
        topSearchToolbar = (RelativeLayout) mLayoutInflater.inflate(R.layout.top_search_toolbar, null);
        addView(topSearchToolbar);

        btnClearSearch = (ImageView) topSearchToolbar.findViewById(R.id.iv_search_clear);
        autoSearch = (AutoCompleteTextView) topSearchToolbar.findViewById(R.id.auto_search);
    }

    public void setFilter(final  android.widget.Filter filter) {
        autoSearch.addTextChangedListener(new TextWatcher() {

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                String filterWord = autoSearch.getText().toString().trim();
                filter.filter(filterWord);
            }

            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
                // TODO Auto-generated method stub

            }

            @Override
            public void afterTextChanged(Editable s) {
                // TODO Auto-generated method stub

            }
        });

        btnClearSearch.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                autoSearch.setText(null);
            }
        });
    }

}

top_search_toolbar.xml文件:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/top_search_bar"
    android:layout_width="match_parent"
    android:layout_height="45dip"
    android:background="@drawable/search_bar_bg"
    android:visibility="visible" >
    <AutoCompleteTextView
        android:id="@+id/auto_search"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:layout_centerVertical="true"
        android:layout_marginLeft="5dip"
        android:layout_marginRight="5dip"
        android:background="@drawable/search_bar_edit_normal"
        android:completionThreshold="1"
        android:drawableLeft="@drawable/search_bar_icon_normal"
        android:dropDownHorizontalOffset="30dip"
        android:dropDownVerticalOffset="9dip"
        android:dropDownWidth="210dip"
        android:singleLine="true"
        android:textSize="15sp" />

    <ImageView
        android:id="@+id/iv_search_clear"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:layout_marginRight="12dip"
        android:layout_marginTop="-1dip"
        android:background="@drawable/btn_search_clear_selector" />

</RelativeLayout>


  來自:http://blog.csdn.net/android_ls/article/details/8644247

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!