2010-10-12 93 views
5

我有一个Android ListView其项目有一个复选框。复选框更改值两次

该复选框默认被选中。一旦未选中,它应该从列表中删除。

的问题是,onCheckedChanged是被解雇了两次:当我点击复选框取消选中它(与isCheckedfalse)后,我删除的项(含isCheckedtrue)。

这是我ArrayAdapter的相关代码:

public View getView(final int position, View convertView, ViewGroup parent) { 
    ViewHolder holder; 
    if (convertView == null) { 
     convertView = mInflater.inflate(R.layout.item, parent, false); 
     holder = new ViewHolder(); 
     holder.check = (CheckBox) convertView.findViewById(R.id.check); 
     convertView.setTag(holder); 
    } else { 
     holder = (ViewHolder) convertView.getTag(); 
    } 
    final Object item = this.getItem(position); 
    holder.check.setOnCheckedChangeListener(new CheckBox.OnCheckedChangeListener() { 
     @Override 
     public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 
      if (!isChecked) { 
       remove(item); // This somehow calls onCheckedChanged again 
      } 
     } 
    }); 
    return convertView; 
} 

我在做什么错?

+0

remove()方法做了什么,因为这不是'ListAdapter'上的方法? – CommonsWare 2010-10-12 11:48:03

+0

我正在使用ArrayAdapter。根据Android源代码,remove方法从数组中删除项目并通知数据已更改。 – hpique 2010-10-12 11:54:24

回答

0

我遇到同样的问题。

经过一番研究,我发现在RadioGroup中有一个类似于setCheckChangeListener的行为。

标记会弄乱事情,看起来像一个错误。

我的解决方法是将标记设置为null在侦听器的末尾它为我工作。

private OnCheckedChangeListener checkBoxitemClickListener = new OnCheckedChangeListener() { 

    public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { 


     if(isChecked) 
     {    
      //do smth 
     }else 
     { 
      //do smth 
     } 

     buttonView.setTag(null);       
    } 
}; 

如果您需要后更新您的ListView,不要忘记调用“notifyDataSetChanged”你的适配器上。

干杯

5

我遇到同样的问题,这似乎是一个Android的bug onCheckChanged方法的这两次执行。

我的解决方案:实施onClickListener,而不是onCheckedChangedListener

事情是这样的:

private final class CheckUpdateListener implements OnClickListener { 

    private Group parent; 
    private boolean isChecked; 

    private CheckUpdateListener(Group parent) { 

     this.parent = parent; 

    } 

    @Override 
    public void onClick(View box) { 
     this.isChecked = !parent.isChecked(); 

     parent.setChecked(isChecked); 

     notifyDataSetChanged(); 

    } 

} 
+0

是的,我认为听取onCheckedChange是矫枉过正,而不是在大多数情况下是正确的。但对我而言,基本问题是要获得正确的观点,即视图与所查看的数据对象不同。 – rwst 2014-02-01 10:12:31

2

我也有类似的问题,只是解决了它正确而无需避免使用意OnCheckedChangeListener

问题代码

holder.someCheckBox.setChecked(false); 
holder.someCheckBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 
    @Override 
    public void onCheckedChanged(CompoundButton compoundButton, boolean checked) { 
     ... 
    } 
}); 

注意我设置检查状态之前,我设置的听众,这是为了防止监听器发射的创作。

但是,只有在适配器被强制重新创建或通过使用notifyDataSetChanged后,监听器才开始启动正确检查事件两次。

解决方案

清除听者之前设置检查状态,即使是在创造在这里。

holder.someCheckBox.setOnCheckedChangeListener(null); 
holder.someCheckBox.setChecked(false); 
holder.someCheckBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { 
    @Override 
    public void onCheckedChanged(CompoundButton compoundButton, boolean checked) { 
     ... 
    } 
}); 

现在任何残余听众不会被设置的复选框的初始值时烧制。