我正在为我的RecyclerView构建一个更改动画。我已覆盖canReuseUpdatedViewHolder()
以返回true
以保留之前的观看者。我在animateChange()
中启动动画,并在结束时立即致电dispatchAnimationFinished()
。参数oldHolder
和newHolder
是animateChange()
中的相同实例。
然后,一旦动画开始,就会为列表中的每个孩子调用RecyclerView的onBindViewHolder()
。令人惊讶的是,只有动画项目才会在onCreateViewHolder()
中生成一个新的ViewHolder,正如我所理解的那样,这不是正确的行为。对于每个其他孩子,旧的ViewHolder绑定在onBindViewHolder()
。
作为旁注,在动画结束之前,onBindViewHolder()
过早被调用。即使在动画结束后调用dispatchAnimationFinished(holder)
。
这是ItemAnimator子类。
public class CustomItemAnimator extends DefaultItemAnimator {
@Override
public boolean canReuseUpdatedViewHolder(@NonNull RecyclerView.ViewHolder viewHolder) {
return true;
}
@Override
public boolean canReuseUpdatedViewHolder(@NonNull RecyclerView.ViewHolder viewHolder, @NonNull List<Object> payloads) {
return true;
}
@Override
public boolean animateChange(@NonNull RecyclerView.ViewHolder oldHolder, @NonNull RecyclerView.ViewHolder newHolder, @NonNull ItemHolderInfo preLayoutInfo, @NonNull ItemHolderInfo postLayoutInfo) {
CustomHolder holder = (CustomHolder) newHolder;
CustomView customView = holder.customView;
Animator animator = customView.revealAnimation();
animator.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
dispatchAnimationFinished(holder);
}
});
animator.start();
return false;
}
}
这是自定义视图动画代码:
CustomView.java
public Animator revealAnimation() {
return circularRevealView(visibleView, hiddenView);
}
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
private Animator circularRevealView(View visibleView, View invisibleView) {
// get the center for the clipping circle
int cx = visibleView.getWidth() / 2;
int cy = visibleView.getHeight() / 2;
// get the final radius for the clipping circle
float finalRadius = (float) Math.hypot(cx, cy);
// create the animator for this view (the start radius is zero)
Animator anim =
ViewAnimationUtils.createCircularReveal(invisibleView, cx, cy, 0, finalRadius);
visibleView.setVisibility(INVISIBLE);
invisibleView.setVisibility(View.VISIBLE);
// return the animation for later use
return anim;
}
感谢。
答案 0 :(得分:1)
animateChange(...)在notifyDataSetChanged()之后调用,这意味着onBindViewHolder()在animateChange(...)之前。并且holder的状态已经改变。你可以这样做吗
animator.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationStart(Animator animation) {
visibleView.setVisibility(INVISIBLE);
invisibleView.setVisibility(View.VISIBLE);
}
@Override
public void onAnimationEnd(Animator animation) {
dispatchAnimationFinished(holder);
}
});