ListView:setItemChecked 仅适用于标准 ArrayAdapter - 在使用自定义 ArrayAdapter 时不起作用?
Posted
技术标签:
【中文标题】ListView:setItemChecked 仅适用于标准 ArrayAdapter - 在使用自定义 ArrayAdapter 时不起作用?【英文标题】:ListView: setItemChecked only works with standard ArrayAdapter - does NOT work when using customized 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;
【问题讨论】:
【参考方案1】:您的行布局需要为Checkable
才能使setItemChecked()
工作,在这种情况下,Android 将在用户单击行时管理在您的Checkable
上调用setChecked()
。您无需设置自己的OnCheckedChangeListener
。
更多信息,请参见:
ListView with CHOICE_MODE_MULTIPLE using CheckedText in a custom view Multiple choice list with custom view? http://www.marvinlabs.com/2010/10/custom-listview-ability-check-items/ http://tokudu.com/2010/android-checkable-linear-layout/ http://alvinalexander.com/java/jwarehouse/apps-for-android/RingsExtended/src/com/example/android/rings_extended/CheckableRelativeLayout.java.shtml【讨论】:
难以置信 - 为什么这必须如此复杂和难以发现?我也做 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_
android:layout_
android:padding="10dp" >
<ImageView
android:id="@+id/agenzie_item_icon"
android:layout_
android:layout_
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_
android:layout_
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<MyObject>
的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
实现中,在CheckBox
的OnClickListener
中使用相同的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_
android:layout_
android:orientation="vertical" >
<TextView
android:id="@+id/text"
android:layout_
android:layout_
android:layout_toLeftOf="@+id/check"
android:gravity="center_vertical"
android:paddingLeft="8dp"
android:textAppearance="?android:attr/textAppearanceMedium" />
<CheckBox
android:id="@+id/check"
android:layout_
android:layout_
android:layout_alignParentRight="true"
android:focusable="false"/>
</RelativeLayout>
如果您最初需要检查某些 CheckBox,只需在膨胀之前在 MyObject 中设置您的 isChecked 成员。
【讨论】:
【参考方案4】:没有一个响应对我有用。我的问题是,只有从 OnCreate/OnStart/OnPause 对 ExpandableListView.setItemChecked 的初始调用不起作用,就好像视图尚未完全初始化一样。为了解决这个问题,我必须做到以下几点:
_expandableListView.Post(() =>
_expandableListView.SetItemChecked(fragmentPosition, true);
);
【讨论】:
以上是关于ListView:setItemChecked 仅适用于标准 ArrayAdapter - 在使用自定义 ArrayAdapter 时不起作用?的主要内容,如果未能解决你的问题,请参考以下文章