Странное поведение ListView с CheckBox - PullRequest
6 голосов
/ 01 декабря 2011

У меня есть CheckBox над ListView, в котором есть элементы Checkbox. Поэтому, когда я проверяю флажок, я применяю notifyDataSetChanged(), чтобы проверить все флажки, все элементы списка. Поэтому в то время я получаю логи для метода getView два раза, это означает, что getView вызывается дважды, когда я вызываю notifyDataSetChanged() только один раз. Итак, кто-нибудь может сказать мне, почему я получаю логи дважды? И почему getView () вызывается дважды?

Основной класс -

checkAll = (CheckBox) findViewById(R.id.my_check_box);
        checkAll.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {

            @Override
            public void onCheckedChanged(CompoundButton checkbox, boolean arg1) {
                adapter.notifyDataSetChanged();
            }
        });

Код для класса адаптера -

    public class myAdapter extends ArrayAdapter<Model> {

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

    public myAdapter(Activity context, List<Model> list, CheckBox checkAll) {
        super(context, R.layout.row, list);
        this.context = context;
        this.list = list;
        this.checkAll = checkAll;

    }

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

    @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        View view = null;
        if (convertView == null) {
            LayoutInflater inflator = context.getLayoutInflater();
            view = inflator.inflate(R.layout.row, 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();
                            element.setSelected(buttonView.isChecked());
                        }
                    });
            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());


        if(checkAll.isChecked() == true){
            System.out.println(position+" checked th position");
        }
        else if(checkAll.isChecked() == false){
            System.out.println(position+" not checked th position");
        }

        holder.checkbox.setChecked(list.get(position).isSelected());    
        return view;
    }
}

Выход LogCat Когда я могу adapter.notifyDataSetChanged(); на флажках CheckAll Проверить прослушиватель изменений.

11-30 20:31:33.751: INFO/System.out(1300): 0 checked th position
11-30 20:31:33.761: INFO/System.out(1300): 1 checked th position
11-30 20:31:33.770: INFO/System.out(1300): 2 checked th position
11-30 20:31:33.780: INFO/System.out(1300): 3 checked th position
11-30 20:31:33.810: INFO/System.out(1300): 4 checked th position
11-30 20:31:33.810: INFO/System.out(1300): 0 checked th position
11-30 20:31:33.831: INFO/System.out(1300): 1 checked th position
11-30 20:31:33.831: INFO/System.out(1300): 2 checked th position
11-30 20:31:33.851: INFO/System.out(1300): 3 checked th position
11-30 20:31:33.860: INFO/System.out(1300): 4 checked th position

Вы можете видеть, что у меня вывод Logcat, я получаю один и тот же вывод два раза. Так может кто-нибудь сказать, почему это происходит?

Ответы [ 3 ]

5 голосов
/ 01 декабря 2011

Попробуйте заставить ListView высоту fill_parent.

Lisiview попытаться немного отрегулировать в соответствии с предоставленным пространством, это то, что я узнал, и похоже, что это является причиной.

Посмотрите, поможет ли это.

3 голосов
/ 01 декабря 2011

Это не совсем ответ на ваш вопрос - просто мне немного больно, когда я вижу это:

if(checkAll.isChecked() == true){
    System.out.println(position+" checked th position");
}
else if(checkAll.isChecked() == false){
    System.out.println(position+" not checked th position");
}

Для if-statement требуется выражение boolean, и, поскольку метод isChecked() возвращает boolean, нет необходимости сравнивать его с true/false. Кроме того, поскольку boolean может быть только true или false, нет необходимости проверять оба. Короче говоря, вы можете свести вышеизложенное к:

if( checkAll.isChecked() )
  System.out.println(position+" checked th position");
else
  System.out.println(position+" not checked th position");

Что гораздо красивее: -)

Не воспринимайте это как критику ваших навыков программирования - просто рассматривайте это как возможность обучения.

0 голосов
/ 01 декабря 2011

это потому что viewHolder внутри checkchangelistener будет меняться вместе с прокруткой списка.поэтому никакое сопоставление между этим флажком и viewHolder не может быть достигнуто с помощью этого способа.

создание класса sepatera checkchangelistener реализует checnChecnglistener.Передайте checkchangelistener через конструктор.

EG.снаружи getView

class MyCheckChangeLsitenet implements CompoundButton.OnCheckedChangeListener

ViewHolder viewHolder
 MyCheckChangeLsitenet(ViewHolder viewHolder)
 {
this.viewHolder =  viewHolder ;
}
                        @Override
                        public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                            Model element = (Model) viewHolder.checkbox.getTag();
                            element.setSelected(buttonView.isChecked());
                        }
                    });

в GetView

viewHolder.checkbox .setOnCheckedChangeListener (new MycheckChangeListenet (viewHolder))

...