Android ListView с шаблоном держателя при мерцании элемента обновления

Я создал listView с базовым адаптером, который реализует шаблон ViewHolder. Каждый элемент из списка имеет таймер обратного отсчета. Работает нормально, но если я хочу обновить список, countDownTimer мерцает между новым значением и старым значением.

Как решить эту проблему?

Базовый адаптер getView()

 @Override
public View getView(int position, View convertView, ViewGroup parent) {
    View rowView = convertView;
    if (rowView == null) {
        LayoutInflater inflater = LayoutInflater.from(context);
        rowView = inflater.inflate(R.layout.row_item, null);
        ViewHolder viewHolder = new ViewHolder();
        rowView.setTag(viewHolder);

        viewHolder.name = (TextView) rowView.findViewById(R.id.name);
        viewHolder.startingAt = (TextView) rowView.findViewById(R.id.starting_at);

    } else {
        rowView = convertView;

    }

    ViewHolder holder = (ViewHolder) rowView.getTag();
    Model current = list.get(position);
    holder.name.setText(current.name);

    return rowView;
}

CountDownTimer для каждого элемента

@Override
public void run() {
    int j = 0;
    for (int nTime : listOfTimes) {
        listOfTimes.set(j, nTime - 1);
        j++;
    }

    if (lv.getCount() <= 0) {
        return;
    }
    int nAmountVisible = lv.getLastVisiblePosition() - lv.getFirstVisiblePosition();
    for (int i = 0; i <= nAmountVisible; i++) {
        long lTimeFromList = listOfTimes.get(lv.getFirstVisiblePosition() + i);

        // transform timestamp to readable date
        long weeks = TimeUnit.SECONDS.toDays(lTimeFromList) / 7;
        long days = TimeUnit.SECONDS.toDays(lTimeFromList) - (7 * weeks);
        long hours = TimeUnit.SECONDS.toHours(lTimeFromList) - TimeUnit.DAYS.toHours(days) - TimeUnit.DAYS.toHours(7 * weeks);
        long minute = TimeUnit.SECONDS.toMinutes(lTimeFromList) - (TimeUnit.SECONDS.toHours(lTimeFromList) * 60);
        long second = TimeUnit.SECONDS.toSeconds(lTimeFromList) - (TimeUnit.SECONDS.toMinutes(lTimeFromList) * 60);

        View viewHolder = lv.getChildAt(i);
        AuctionAdapter.ViewHolder localHolder = (AuctionAdapter.ViewHolder) viewHolder.getTag();
        viewHolder.invalidate();

        localHolder.startingAt.setText(weeks + "w " + days + "d " + hours + "h " + minute + "m " + second + "s ");

    }
    mHandler.postDelayed(this, 1000);
}

и код из фрагмента, который заполняет адаптер и список времени для обратного отсчета

  baseAdapter = new baseAdapter(getActivity(),list);
        baseAdapter.notifyDataSetChanged();
        runningLv.setAdapter(baseAdapter);
        runningLv.invalidateViews();

        ArrayList<Integer> listOfTimes = new ArrayList<Integer>();
        listOfTimes.clear();
        for (Model m : list) {
            listOfTimes.add(Integer.parseInt(m.starting_at) - m.current_time);
        }
        CountDownTime countDownTimer = new CountDownTime(listOfTimes, runningLv);
        countDownTimer.run();
        baseAdapter.notifyDataSetChanged();

ОБНОВЛЕНИЕ: это onCreateView, я хочу повторно использовать представление

    @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = null;
    if (view != null) {
        ViewGroup parent = (ViewGroup) view.getParent();
        if (parent != null) {
            parent.removeView(view);
        }
    } else {
        view = inflater.inflate(R.layout.fragment_running, null);
        setupViewItems(view);

    }
    return view;
}

person ovluca    schedule 03.07.2014    source источник


Ответы (1)


Правильный способ использования держателя должен быть таким:

@Override
public View getView(int position, View convertView, ViewGroup parent)
{
    ViewHolder holder = null;
    if (convertView == null)
    {
        LayoutInflater inflater = LayoutInflater.from(context);
        convertView = inflater.inflate(R.layout.row_item, null);
        holder = new ViewHolder();

        holder.name = (TextView) convertView.findViewById(R.id.name);
        holder.startingAt = (TextView) convertView.findViewById(R.id.starting_at);

        convertView.setTag(viewHolder);
    }
    else
    {
        holder = (ViewHolder) convertView.getTag();
    }

    Model current = list.get(position);
    holder.name.setText(current.name);

    return convertView;
}

РЕДАКТИРОВАТЬ:

Вы не должны устанавливать значение TextView из работающего потока, а непосредственно из метода getView.

Добавьте эту строку прямо перед return convertView;:

holder.startingAt.setText(current.timer);

И в вашем классе Model добавьте новое поле:

public String timer = null;

Затем в вашем текущем потоке (таймер обратного отсчета):

@Override
public void run()
{
    for (int i = 0; i <= listOfTimes.size(); i++)
    {
        long lTimeFromList = listOfTimes.get(i);

        // transform timestamp to readable date
        long weeks = TimeUnit.SECONDS.toDays(lTimeFromList) / 7;
        long days = TimeUnit.SECONDS.toDays(lTimeFromList) - (7 * weeks);
        long hours = TimeUnit.SECONDS.toHours(lTimeFromList) - TimeUnit.DAYS.toHours(days) - TimeUnit.DAYS.toHours(7 * weeks);
        long minute = TimeUnit.SECONDS.toMinutes(lTimeFromList) - (TimeUnit.SECONDS.toHours(lTimeFromList) * 60);
        long second = TimeUnit.SECONDS.toSeconds(lTimeFromList) - (TimeUnit.SECONDS.toMinutes(lTimeFromList) * 60);

        listOfTimes.get(i).timer = weeks + "w " + days + "d " + hours + "h " + minute + "m " + second + "s ";
    }
    mHandler.postDelayed(this, 1000);
}
person Manitoba    schedule 03.07.2014
comment
спасибо, но моя проблема не решена ... я обновляю свой вопрос - person ovluca; 03.07.2014
comment
Почему вы хотите использовать onCreateView для повторного использования представлений? Механизмы Listview уже делают свою работу. - person Manitoba; 03.07.2014
comment
я думаю, что это лучший подход. чтобы повторно использовать вид, сколько вы можете. попробую без. - person ovluca; 03.07.2014
comment
Я закончил модификации, проблема осталась - person ovluca; 03.07.2014
comment
Я обновил свой ответ. Проблема, похоже, находится в вашем методе run. - person Manitoba; 03.07.2014
comment
спасибо за помощь .. но если я сделаю это textview, start_at больше не изменится в одну секунду - person ovluca; 03.07.2014
comment
1. Получить предмет. 2. Измените значение timer. 3. Верните его в список (set(position, item)) - person Manitoba; 03.07.2014