【问题标题】:ListView: setItemChecked only works with standard ArrayAdapter - does NOT work when using customized ArrayAdapter?ListView:setItemChecked 仅适用于标准 ArrayAdapter - 在使用自定义 ArrayAdapter 时不起作用?
【发布时间】:2012-01-12 06:26:12
【问题描述】:

这真的很奇怪。

当我为 ListView 使用标准 ArrayAdapter 时,调用 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;
    }

}

【问题讨论】:

    标签: android listview android-arrayadapter


    【解决方案1】:

    您的行布局需要为Checkable 才能使setItemChecked() 工作,在这种情况下,Android 将在用户单击行时管理在您的Checkable 上调用setChecked()。您无需设置自己的OnCheckedChangeListener

    更多信息,请参见:

    【讨论】:

    • 难以置信 - 为什么这必须如此复杂和难以发现?我也做 iPhone 编程,而到目前为止我真的更喜欢 Android,这是一个真正的挫折。非常感谢!
    • 我现在详细查看了示例代码,并决定不遵循此解决方案。相反,我在我的 ArrayAdapter 中构建了一个 CheckBoxes 数组。有了这个,我就可以轻松地做我需要的那种东西。再次非常感谢您为我指明方向!
    • 更新了 marvinlabs 教程的链接blog.marvinlabs.com/2010/10/29/…
    • 完全正确。我已经在 Android 应用上工作了几年,这是我第一次使用 CheckedTextView。
    • @user387184 是的,android 文档很浪费时间。另外,有些事情是以最复杂的方式实现的。这东西还没有成熟。
    【解决方案2】:

    当然,你必须为你的 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>
    

    【讨论】:

      【解决方案3】:

      Checkable 是我现在不想采用的学习曲线。您可以在活动的OnItemClickListener 中手动设置和取消设置复选框。在ArrayAdapter&lt;MyObject&gt;MyObject 列表中维护一个布尔型isChecked 瞬态变量。

      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本身;在您的自定义数组适配器的getView 实现中,在CheckBoxOnClickListener 中使用相同的addItem(item)/removeItem(item) 逻辑。

      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() 是这里的关键。
      只要您的数组适配器可以假设相同的活动,这似乎就可以解决问题,并且您的数组适配器可以假设 res/layout/ 中的相同列表项 xml 定义。

      可检查列表项 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>
      

      如果您最初需要检查某些 CheckBox,只需在膨胀之前在 MyObject 中设置您的 isChecked 成员。

      【讨论】:

        【解决方案4】:

        没有一个响应对我有用。我的问题是,只有从 OnCreate/OnStart/OnPause 对 ExpandableListView.setItemChecked 的初始调用不起作用,就好像视图尚未完全初始化一样。为了解决这个问题,我必须做到以下几点:

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

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 2019-04-13
          • 1970-01-01
          • 1970-01-01
          • 2023-03-14
          • 1970-01-01
          • 1970-01-01
          相关资源
          最近更新 更多