替换RecyclerView

时间:2016-06-28 12:55:01

标签: android android-recyclerview

我正在开发一个显示网站图片的应用。图像都具有不同的高度,它们显示的视图(视图)也是如此。

应用过滤器时,通过直接从适配器清除它们来删除先前的数据集(因此,当前可见的数据集)。通过

通知适配器
final int removedSize = items.size();
items.clear();
notifyItemRangeRemoved(0, removedSize);

这会使旧项目在屏幕上显示动画。该动画完成后,我使用

添加新数据集
RecyclerView attachedView;

attachedView.getItemAnimator().isRunning(new RecyclerView.ItemAnimator.ItemAnimatorFinishedListener() { // Add a listener to listen when the new items should be added
            @Override
            public void onAnimationsFinished() {
                attachedView.postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        addPhotosList(objects);
                        notifyItemRangeChanged(0, objects.size());                            
                        loading = false;
                        attachedView.requestLayout();
                    }
                },700);
            }
        });

这会动画添加的项目。 奇怪的是,我有时会在RecyclerView内部出现空白,就好像RecyclerView在绘制所有项目时不会将ViewHolder的新维度考虑在内。更重要的是,有些物品实际上是相互叠加的,只要他们愿意,它们就会闪现在前景中。放置在正确位置的物品顶部的物品是应放置在间隙内的物品。他们不会回到差距,但在手动调查项目集时,我注意到他们应该在那里。

RecyclerView是SwipeRefreshLayout视图的子代,但我不认为这应该是一个问题。

我为RecyclerView设置的一些参数:

recents.setLayoutManager(layoutManager);
recents.setHasFixedSize(false);
recents.setItemViewCacheSize(100);

布局文件内

android:animationCache="false"
android:scrollingCache="false"

我做错了什么?

我已经添加了一个0.7秒的小延迟,以确保所有动画都已完成,但这似乎没有多大帮助。

ItemAnimator:

@Override
    public boolean animateAdd(RecyclerView.ViewHolder viewHolder) {
        if (viewHolder.getItemViewType() != -8) { // ImageAdapter.ViewType.EMPTY = -8;
            if (viewHolder.getLayoutPosition() > lastAddAnimatedItem) {
                lastAddAnimatedItem++;
                runEnterAnimation((ImageListView) viewHolder, viewHolder.getLayoutPosition());
                return false;
            }
        }
        lastAddAnimatedItem = -6;
        dispatchAddFinished(viewHolder);
        return false;
    }

    @Override
    public boolean animateRemove(RecyclerView.ViewHolder viewHolder) {
        if (viewHolder.getItemViewType() != -8) { // ImageAdapter.ViewType.EMPTY = -8;
            if (viewHolder.getLayoutPosition() > lastRemoveAnimatedItem) {
                lastRemoveAnimatedItem++;
                runExitAnimation((ImageListView) viewHolder, viewHolder.getLayoutPosition());
                return false;
            }
        }

        lastRemoveAnimatedItem = -6;
        dispatchRemoveFinished(viewHolder);
        return false;
    }

    private void runEnterAnimation(final ImageListView holder, int position) {
        final int screenHeight = Utils.getScreenHeight(holder.itemView.getContext());
        holder.itemView.setTranslationY(screenHeight + holder.itemView.getHeight());
        holder.itemView.animate()
                .translationY(0)
                .setStartDelay(150 + (20 * (position)))
                .setInterpolator(new DecelerateInterpolator(3.f))
                .setDuration(700)
                .setListener(new AnimatorListenerAdapter() {
                    @Override
                    public void onAnimationEnd(Animator animation) {
                        dispatchAddFinished(holder);
                    }
                })
                .start();
    }

    private void runExitAnimation(final ImageListView holder, int position) {
        final int screenHeight = Utils.getScreenHeight(holder.itemView.getContext());
        holder.itemView.setTranslationY(0);
        holder.itemView.animate()
                .translationY(-screenHeight - holder.itemView.getHeight())
                .setStartDelay(Math.abs(20 * (position)))
                .setInterpolator(new DecelerateInterpolator(3.f))
                .setDuration(700)
                .setListener(new AnimatorListenerAdapter() {
                    @Override
                    public void onAnimationEnd(Animator animation) {
                        dispatchRemoveFinished(holder);
                    }
                })
                .start();
    }

编辑:

屏幕上的最后一个可见项后面的第一个项目显示在错误的位置。

1 个答案:

答案 0 :(得分:1)

我一直在与这个问题作斗争一个多星期,但我找到了解决方案:

将runEnterAnimation更改为:

private void runEnterAnimation(final ImageListView holder, int position) {
        final int screenHeight = Utils.getScreenHeight(holder.itemView.getContext());
        holder.itemView.setTranslationY(screenHeight);
        holder.itemView.animate()
                .translationY(0)
                .setStartDelay(150 + (20 * (position)))
                .setInterpolator(new DecelerateInterpolator(3.f))
                .setDuration(700)
                .setListener(new AnimatorListenerAdapter() {
                    @Override
                    public void onAnimationEnd(Animator animation) {
                        dispatchAddFinished(holder);
                        holder.itemView.setTranslationY(0f);
                    }
                    @Override
                    public void onAnimationCancel(Animator animation){
                        holder.itemView.setTranslationY(0f);
                    }
                })
                .start();
    }

注意onAnimationEnd和onAnimationCancel回调中的新行,这可以确保项目的转换始终设置为动画动画所需的转换。

将runExitAnimation替换为:

private void runExitAnimation(final ImageListView holder, int position) {
        final int screenHeight = Utils.getScreenHeight(holder.itemView.getContext());
        holder.itemView.setTranslationY(0);
        holder.itemView.animate()
                .translationY(-screenHeight)
                .setStartDelay(Math.abs(20 * (position)))
                .setInterpolator(new DecelerateInterpolator(3.f))
                .setDuration(700)
                .setListener(new AnimatorListenerAdapter() {
                    @Override
                    public void onAnimationEnd(Animator animation) {
                        dispatchRemoveFinished(holder);
                        holder.itemView.setTranslationY(-screenHeight);
                    }
                    @Override
                    public void onAnimationCancel(Animator animation){
                        holder.itemView.setTranslationY(-screenHeight);
                    }
                })
                .start();
    }

将endAnimation替换为:

@Override
    public void endAnimation(RecyclerView.ViewHolder item) {
        super.endAnimation(item);
        item.itemView.setTranslationY(0f);
    }

这将确保持有者始终返回其原始位置。

在适配器中,将以下行添加到onBindViewHolder方法:

holder.itemView.setTranslationY(0f);

通过上面的一行,我得到了另一个数据集的第一项被困在背景中的奇怪错误,但是如果你添加一个与RecyclerView背景颜色相同的背景到你的项目那个bug也是固定的。

有时候把事情写下去是非常有用的,因为他们要看到解决方案。就像这次它指出了解决方案。