it-roy-ru.com

ListView: setItemChecked работает только со стандартным ArrayAdapter - НЕ работает при использовании настроенного ArrayAdapter?

Это действительно странно.

Когда я использую стандартный ArrayAdapter для вызова ListView, setItemChecked работает нормально

Но при использовании специального ArrayAdapter это не так.

В чем будет причина? Это ошибка? Или я что-то упустил?

public class Test_Activity extends Activity {

    /** Called when the activity is first created. */
    private List<Model> list;
    private ListView lView;

    public void onCreate(Bundle icicle) {
    super.onCreate(icicle);
    // Create an array of Strings, that will be put to our ListActivity
    setContentView(R.layout.main);
    lView = (ListView) findViewById(R.id.ListView01);
    lView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);

    list = getModel();

    //with this adapter setItemChecked works OK
    lView.setAdapter(new ArrayAdapter<Model>(this,
        Android.R.layout.simple_list_item_multiple_choice, list));


  //**************************
    //PROBLEM: with this adapter it does not check any items on the screen
    // ArrayAdapter<Model> adapter = new Test_Class1(this, list);
    // lView.setAdapter(adapter);



    }

    private List<Model> getModel() {
       List<Model> list = new ArrayList<Model>();
       list.add(get("0"));
       list.add(get("1"));
       list.add(get("2"));
       list.get(1).setSelected(true);
       Model m = list.get(1);
       list.add(get("3"));
       list.add(get("4"));
       list.add(get("5"));
       list.add(get("6"));
       list.add(get("7"));
       // Initially select one of the items
       return list;
    }

    private Model get(String s) {
       return new Model(s);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
       MenuInflater inflater = getMenuInflater();
       inflater.inflate(R.menu.results_screen_option_menu, menu);
       return true;
    }

    /**
     * @category OptionsMenu
     */
    @Override
 public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
    case R.id.select_all: {

        int size = lView.getAdapter().getCount();

        for (int i = 0; i <= size; i++) {

            //************************** PROBLEM
        lView.setItemChecked(i, true); // selects the item only for standard ArrayAdapter

    Log.i("xxx", "looping " + i);
        }
    }
        return true;
    case R.id.select_none:
        return true;
    }
    return false;
    }
}

// ------------------------------------------------ ------------

public class Test_Class1 extends ArrayAdapter<Model> {

    private final List<Model> list;
    private final Activity context;

 public Test_Class1(Activity context, List<Model> list) {
    super(context, R.layout.rowbuttonlayout2, list);
    this.context = context;
    this.list = list;
    }

  static class ViewHolder {
    protected TextView text;
    protected CheckBox checkbox;
    }

    @Override
 public View getView(int position, View convertView, ViewGroup parent) {
    View view = null;
    Log.i("xxx", "-> getView " + position);
    if (convertView == null) {
        LayoutInflater inflator = context.getLayoutInflater();
        view = inflator.inflate(R.layout.rowbuttonlayout, null);
        final ViewHolder viewHolder = new ViewHolder();
        viewHolder.text = (TextView) view.findViewById(R.id.label);
        viewHolder.checkbox = (CheckBox) view.findViewById(R.id.check);
        viewHolder.checkbox
            .setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {

            @Override
            public void onCheckedChanged(CompoundButton buttonView,
                boolean isChecked) {
                Model element = (Model) viewHolder.checkbox
                    .getTag();
                Log.i("xxx", "-> onCheckedChanged");
                element.setSelected(buttonView.isChecked());
                Log.i("xxx", "<- onCheckedChanged");

            }
            });
        view.setTag(viewHolder);
        viewHolder.checkbox.setTag(list.get(position));
    } else {
        view = convertView;
        ((ViewHolder) view.getTag()).checkbox.setTag(list.get(position));
    }
    ViewHolder holder = (ViewHolder) view.getTag();
    holder.text.setText(list.get(position).getName());
    Log.i("xxx", "holder.checkbox.setChecked: " + position);

    holder.checkbox.setChecked(list.get(position).isSelected());
    Log.i("xxx", "<-  getView " + position);

    return view;
    }

}
37
user387184

Ваш макет строки должен быть Checkable, чтобы setItemChecked() работал, и в этом случае Android будет управлять вызовом setChecked() для Checkable, когда пользователь нажимает на строку. Вам не нужно настраивать свою собственную OnCheckedChangeListener.

Для получения дополнительной информации см .:

47
CommonsWare

Конечно, вы должны установить режим выбора для вашего ListView, например:

list.setChoiceMode(ListView.CHOICE_MODE_SINGLE);

Но если вы также используете пользовательский макет для своего элемента списка (R.layout.drawing_list_item), то вы должны убедиться, что ваш макет реализует интерфейс Checkable:

public class CheckableRelativeLayout extends RelativeLayout implements Checkable {
    private boolean isChecked = false;

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

    public boolean isChecked() {
        return isChecked;
    }

    public void setChecked(boolean isChecked) {
        this.isChecked = isChecked;
        changeColor(isChecked);
    }

    public void toggle() {
        this.isChecked = !this.isChecked;
        changeColor(this.isChecked);
    }

    private void changeColor(boolean isChecked){
        if (isChecked) {
            setBackgroundColor(getResources().getColor(Android.R.color.holo_blue_light));
        } else {
            setBackgroundColor(getResources().getColor(Android.R.color.transparent));
        }
    }
}

Тогда ваш проверяемый макет будет определен как следующий пример:

<?xml version="1.0" encoding="utf-8"?>
<it.moondroid.banking.CheckableRelativeLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:id="@+id/drawer_item_layout"
    Android:layout_width="match_parent"
    Android:layout_height="wrap_content"
    Android:padding="10dp" >

    <ImageView
        Android:id="@+id/agenzie_item_icon"
        Android:layout_width="30dp"
        Android:layout_height="30dp"
        Android:layout_alignParentLeft="true"
        Android:layout_centerVertical="true"
        Android:background="@Android:color/darker_gray"
        Android:focusable="false" />

    <TextView
        Android:id="@+id/agenzie_item_name"
        Android:layout_width="wrap_content"
        Android:layout_height="wrap_content"
        Android:layout_centerVertical="true"
        Android:layout_toRightOf="@+id/agenzie_item_icon"
        Android:focusable="false"
        Android:paddingLeft="10dp"
        Android:text="item"
        Android:textAppearance="?android:attr/textAppearanceSmall"
        Android:textColor="#FFFFFF" />

</it.moondroid.banking.CheckableRelativeLayout>
10
moondroid

Checkable - это кривая обучения, которую я предпочитаю не брать прямо сейчас. Вы можете установить и снять флажок вручную в разделе «Действия» OnItemClickListener. Сохраните логическую переменную isChecked в списке MyObject в ArrayAdapter<MyObject>

public void onItemClick(AdapterView<?> av, View v, int position, long arg3) {
    final ListView listView = (ListView) findViewById(Android.R.id.list);
    MyObject item = (MyObject) listView.getItemAtPosition(position);
    item.isChecked = !item.isChecked;
    if(item.isChecked)
        addItem(item);
    else
        removeItem(item);
}  

Для учета пользователей, щелкающих само имя CheckBox; используйте ту же логику addItem (item)/removeItem (item) в реализации getView вашего адаптера массива, в CheckBoxOnClickListener

public View getView(int position, View convertView, ViewGroup viewGroup) {
        CheckBox cb = null;
        if (convertView == null) {
            if(inflater == null) //cache to avoid reconstructing for each view
            inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            convertView =inflater.inflate(R.layout.list_item_text_right_checkmark, null);

            CheckBox cb = (CheckBox) convertView.findViewById(R.id.check);
            cb.setChecked((list.get(position).isChecked));
            cb.setTag(list.get(position);
            public void onClick(View v) {
                if(v instanceof CheckBox) {
                    CheckBox cb = (CheckBox) v;
                    if(cb.isChecked()) //verify boolean logic here!!!
                        activityRef.get().addItem(cb.getTag()); //WeakReference to activity
                    else
                        activityRef.get().removeItem(cb.getTag());

                }
            });
    } else cb = (CheckBox) convertView.findViewById(R.id.check);
    cb.setChecked((list.get(position).isChecked));
...
}

Вызов CheckBox :: setChecked () является ключевым моментом здесь.
Похоже, это поможет, если ваш адаптер массива может выполнять те же действия, а адаптер массива может принимать то же самое определение xml элемента списка в res/layout /. 

проверяемый элемент списка XML:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:layout_width="match_parent"
    Android:layout_height="match_parent"
    Android:orientation="vertical" >

  <TextView
      Android:id="@+id/text"
      Android:layout_width="wrap_content"
      Android:layout_height="wrap_content"
      Android:layout_toLeftOf="@+id/check"
      Android:gravity="center_vertical"
      Android:paddingLeft="8dp"
      Android:textAppearance="?android:attr/textAppearanceMedium" />

  <CheckBox
    Android:id="@+id/check"
    Android:layout_width="wrap_content"
    Android:layout_height="wrap_content"
    Android:layout_alignParentRight="true"
    Android:focusable="false"/>
</RelativeLayout>

Если вам нужно проверить некоторые из ваших CheckBoxs изначально, просто установите членов isChecked в MyObject до раздувания. 

1
stephen

Ни один из ответов не работал для меня. Моя проблема заключалась в том, что только первоначальный вызов ExpandableListView.setItemChecked из OnCreate/OnStart/OnPause не работал, как будто представление еще не было полностью инициализировано. Чтобы решить эту проблему, мне пришлось сделать следующее:

_expandableListView.Post(() =>
{
    _expandableListView.SetItemChecked(fragmentPosition, true);
});
0
Alex