RecyclerView Animations Part 2 – Behind The Scenes

1829次阅读  |  发布于5年以前

RecyclerView动画 第二篇-幕后

这是系列文章的第二篇,请先阅读第一篇

在第一篇文章中,我主要介绍了在RecyleyView中如何预测动画的运行。实际上有很多简单实现(对于LayoutManager)。这里有一些你需要知道的关键点。

当LayoutManager的children被移除的时候RecyclerView仍然保留他们的引用。这是如何工作的?LayoutManager和RecyclerView的约定是无效的么

是的,有点违反了和LayoutManager的约定,但是:

RecyclerView保持了View做为一个ViewGroup的child,但是对于LayoutManager又隐藏起来了。每次 LayoutManager调用了访问children的方法时,RecyclerView都把隐藏的View考虑在内。让我们看下第一篇中的一个例子,‘C’正在被移出Adapter。

Predictive Animation

当‘C’淡出时,如果LayoutManager调用了getChildCount(),RecyclerView会返回6虽说LayoutManager有7个children。如果LayoutManager调用了getChildAt(int),RecyclerView会跳过‘C’(或其他隐藏的children)。如果LayoutManager调用了addView(View, position),RecyclerView会在调用ViewGroup#addView前也会跳过。

当动画结束时,RecyclerView会移除并回收View。

更多信息你可以查看ChildHelper 这个内部类。

在preLayout过程中RecyclerView如何处理item的位置,是因为他们与Adapter不匹配么?

一个特殊的通知事件被添加到Adapter是可行的。当Adapter分派了notify**事件时,RecyclerView纪录它们并发送布局请求。任何下一个布局显示前获得的事件都会被一起应用。

当系统调用了onLayout时,RecyclerView做了如下动作:

  1. 纪录更新事件,例如move事件被增加到更新事件列表的末尾。移动move事件到列表末尾是一个简单步骤,所以我就不在这里展开了。如果你感兴趣你可以看OpReorderer类。

  2. 依次处理事件和更新当前的ViewHolders’的位置。如果一个ViewHolder被移除,他会被标记为removed。执行时,RecyclerView判断这个adapter的改变是否会被发送到LayoutManager的preLayout之前或之后的步骤。流程如下:

  1. 当Adapter更新被处理后,RecyclerView保存了当前View的位置和大小用于之后的动画使用。

  2. preLayoutRecyclerView调用LayoutManager#onLayoutChildren。就像我在第一篇文章中提到的,LayoutManager运行自己的布局规则。它所做的这些都是了布局那些被deletedchanged(LayoutParams#isItemRemovedLayoutParams#isItemChanged)的item。在这里需要提醒的是,被删除或被改变的item仍然会被Adapter API提供给LayoutManager。这样,LayoutManager就会简单的认为是其他View(添加、测量、位置等).

  3. preLayout结束后,RecyclerView再次纪录这些View的位置并告诉LayoutManager继续更新Adapter

  4. RecyclerView再次调用LayoutManager的onLayout(postLayout).这时,所有item的位置匹配Adapter现在的内容。LayoutManager再次用自己的规则显示布局。

  5. post layout结束后,RecyclerView再次检测这些View的位置,判断哪些item是被添加的,被删除的,被改变的和被移动的。它‘隐藏’了被删除的View,item没有被LayoutManager加入,加入到了RecyclerView里(因为它们应该开始动画了)

  6. 需要动画的Items被ItemAnimator开始运行它的动画效果。当动画执行完毕,ItemAnimator会调用RecyclerView里的回调方法,如果不再用了,View会被RecyclerView移除和回收。

如果item位置使用了LayoutManager保留的一些内部数据结构会发生什么?

一切工作...。感谢RecyclerView重写了Adapter的更新,当其中某一个adapter数据更新回调方法被调用时,所有的LayoutManager都可以自己处理更新。只要RecyclerView确保恰当的调用时机和顺序。

在布局时的任何时候,如果LayoutManager需要访问adapter额外的数据(一些自定义的API)。可以调用Recycler#convertPreLayoutPositionToPostLayout获得item在adapter中的位置。例如,GridLayoutManager用这个API去获得item的范围大小。

如果调用 notifyDataSetChanged 会发生什么?如何预测动画的运行?

什么也不会发生,这就是为什么notifyDataSetChanged应该最后调用。当adapter里的notifyDataSetChanged被调用时,RecyclerView不知道哪个的item被移动了所以他不能正确的假装getViewForPosition调用的样子。只是简单像LayoutTransition一样运行动画。

-- 我希望这两篇文章可以帮助你理解RecyclerView里的动画是如何工作的和为什么这样工作。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8