1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.deskclock 18 19 import android.animation.Animator 20 import android.animation.AnimatorListenerAdapter 21 import android.animation.AnimatorSet 22 import android.animation.ObjectAnimator 23 import android.animation.PropertyValuesHolder 24 import android.view.View 25 import androidx.collection.ArrayMap 26 import androidx.recyclerview.widget.RecyclerView.State 27 import androidx.recyclerview.widget.RecyclerView.ViewHolder 28 import androidx.recyclerview.widget.RecyclerView.ItemAnimator 29 import androidx.recyclerview.widget.SimpleItemAnimator 30 31 class ItemAnimator : SimpleItemAnimator() { 32 private val mAddAnimatorsList: MutableList<Animator> = ArrayList() 33 private val mRemoveAnimatorsList: MutableList<Animator> = ArrayList() 34 private val mChangeAnimatorsList: MutableList<Animator> = ArrayList() 35 private val mMoveAnimatorsList: MutableList<Animator> = ArrayList() 36 37 private val mAnimators: MutableMap<ViewHolder, Animator> = ArrayMap() 38 animateRemovenull39 override fun animateRemove(holder: ViewHolder): Boolean { 40 endAnimation(holder) 41 42 val prevAlpha: Float = holder.itemView.getAlpha() 43 44 val removeAnimator: Animator? = ObjectAnimator.ofFloat(holder.itemView, View.ALPHA, 0f) 45 removeAnimator!!.duration = getRemoveDuration() 46 removeAnimator.addListener(object : AnimatorListenerAdapter() { 47 override fun onAnimationStart(animator: Animator) { 48 dispatchRemoveStarting(holder) 49 } 50 51 override fun onAnimationEnd(animator: Animator) { 52 animator.removeAllListeners() 53 mAnimators.remove(holder) 54 holder.itemView.setAlpha(prevAlpha) 55 dispatchRemoveFinished(holder) 56 } 57 }) 58 mRemoveAnimatorsList.add(removeAnimator) 59 mAnimators[holder] = removeAnimator 60 return true 61 } 62 animateAddnull63 override fun animateAdd(holder: ViewHolder): Boolean { 64 endAnimation(holder) 65 66 val prevAlpha: Float = holder.itemView.getAlpha() 67 holder.itemView.setAlpha(0f) 68 69 val addAnimator: Animator = ObjectAnimator.ofFloat(holder.itemView, View.ALPHA, 1f) 70 .setDuration(getAddDuration()) 71 addAnimator.addListener(object : AnimatorListenerAdapter() { 72 override fun onAnimationStart(animator: Animator) { 73 dispatchAddStarting(holder) 74 } 75 76 override fun onAnimationEnd(animator: Animator) { 77 animator.removeAllListeners() 78 mAnimators.remove(holder) 79 holder.itemView.setAlpha(prevAlpha) 80 dispatchAddFinished(holder) 81 } 82 }) 83 mAddAnimatorsList.add(addAnimator) 84 mAnimators[holder] = addAnimator 85 return true 86 } 87 animateMovenull88 override fun animateMove( 89 holder: ViewHolder, 90 fromX: Int, 91 fromY: Int, 92 toX: Int, 93 toY: Int 94 ): Boolean { 95 endAnimation(holder) 96 97 val deltaX = toX - fromX 98 val deltaY = toY - fromY 99 val moveDuration: Long = getMoveDuration() 100 101 if (deltaX == 0 && deltaY == 0) { 102 dispatchMoveFinished(holder) 103 return false 104 } 105 106 val view: View = holder.itemView 107 val prevTranslationX = view.translationX 108 val prevTranslationY = view.translationY 109 view.translationX = -deltaX.toFloat() 110 view.translationY = -deltaY.toFloat() 111 112 val moveAnimator: ObjectAnimator? 113 if (deltaX != 0 && deltaY != 0) { 114 val moveX = PropertyValuesHolder.ofFloat(View.TRANSLATION_X, 0f) 115 val moveY = PropertyValuesHolder.ofFloat(View.TRANSLATION_Y, 0f) 116 moveAnimator = ObjectAnimator.ofPropertyValuesHolder(holder.itemView, moveX, moveY) 117 } else if (deltaX != 0) { 118 val moveX = PropertyValuesHolder.ofFloat(View.TRANSLATION_X, 0f) 119 moveAnimator = ObjectAnimator.ofPropertyValuesHolder(holder.itemView, moveX) 120 } else { 121 val moveY = PropertyValuesHolder.ofFloat(View.TRANSLATION_Y, 0f) 122 moveAnimator = ObjectAnimator.ofPropertyValuesHolder(holder.itemView, moveY) 123 } 124 125 moveAnimator?.duration = moveDuration 126 moveAnimator.interpolator = AnimatorUtils.INTERPOLATOR_FAST_OUT_SLOW_IN 127 moveAnimator.addListener(object : AnimatorListenerAdapter() { 128 override fun onAnimationStart(animator: Animator) { 129 dispatchMoveStarting(holder) 130 } 131 132 override fun onAnimationEnd(animator: Animator) { 133 animator?.removeAllListeners() 134 mAnimators.remove(holder) 135 view.translationX = prevTranslationX 136 view.translationY = prevTranslationY 137 dispatchMoveFinished(holder) 138 } 139 }) 140 mMoveAnimatorsList.add(moveAnimator) 141 mAnimators[holder] = moveAnimator 142 143 return true 144 } 145 animateChangenull146 override fun animateChange( 147 oldHolder: ViewHolder, 148 newHolder: ViewHolder, 149 preInfo: ItemHolderInfo, 150 postInfo: ItemHolderInfo 151 ): Boolean { 152 endAnimation(oldHolder) 153 endAnimation(newHolder) 154 155 val changeDuration: Long = getChangeDuration() 156 val payloads = if (preInfo is PayloadItemHolderInfo) preInfo.payloads else null 157 158 if (oldHolder === newHolder) { 159 val animator = (newHolder as OnAnimateChangeListener) 160 .onAnimateChange(payloads, preInfo.left, preInfo.top, preInfo.right, 161 preInfo.bottom, changeDuration) 162 if (animator == null) { 163 dispatchChangeFinished(newHolder, false) 164 return false 165 } 166 animator.addListener(object : AnimatorListenerAdapter() { 167 override fun onAnimationStart(animator: Animator) { 168 dispatchChangeStarting(newHolder, false) 169 } 170 171 override fun onAnimationEnd(animator: Animator) { 172 animator.removeAllListeners() 173 mAnimators.remove(newHolder) 174 dispatchChangeFinished(newHolder, false) 175 } 176 }) 177 mChangeAnimatorsList.add(animator) 178 mAnimators[newHolder] = animator 179 return true 180 } else if (oldHolder !is OnAnimateChangeListener || 181 newHolder !is OnAnimateChangeListener) { 182 // Both holders must implement OnAnimateChangeListener in order to animate. 183 dispatchChangeFinished(oldHolder, true) 184 dispatchChangeFinished(newHolder, true) 185 return false 186 } 187 188 val oldChangeAnimator = (oldHolder as OnAnimateChangeListener) 189 .onAnimateChange(oldHolder, newHolder, changeDuration) 190 if (oldChangeAnimator != null) { 191 oldChangeAnimator.addListener(object : AnimatorListenerAdapter() { 192 override fun onAnimationStart(animator: Animator) { 193 dispatchChangeStarting(oldHolder, true) 194 } 195 196 override fun onAnimationEnd(animator: Animator) { 197 animator.removeAllListeners() 198 mAnimators.remove(oldHolder) 199 dispatchChangeFinished(oldHolder, true) 200 } 201 }) 202 mAnimators[oldHolder] = oldChangeAnimator 203 mChangeAnimatorsList.add(oldChangeAnimator) 204 } else { 205 dispatchChangeFinished(oldHolder, true) 206 } 207 208 val newChangeAnimator = (newHolder as OnAnimateChangeListener) 209 .onAnimateChange(oldHolder, newHolder, changeDuration) 210 if (newChangeAnimator != null) { 211 newChangeAnimator.addListener(object : AnimatorListenerAdapter() { 212 override fun onAnimationStart(animator: Animator) { 213 dispatchChangeStarting(newHolder, false) 214 } 215 216 override fun onAnimationEnd(animator: Animator) { 217 animator.removeAllListeners() 218 mAnimators.remove(newHolder) 219 dispatchChangeFinished(newHolder, false) 220 } 221 }) 222 mAnimators[newHolder] = newChangeAnimator 223 mChangeAnimatorsList.add(newChangeAnimator) 224 } else { 225 dispatchChangeFinished(newHolder, false) 226 } 227 228 return true 229 } 230 animateChangenull231 override fun animateChange( 232 oldHolder: ViewHolder, 233 newHolder: ViewHolder, 234 fromLeft: Int, 235 fromTop: Int, 236 toLeft: Int, 237 toTop: Int 238 ): Boolean { 239 /* Unused */ 240 throw IllegalStateException("This method should not be used") 241 } 242 runPendingAnimationsnull243 override fun runPendingAnimations() { 244 val removeAnimatorSet = AnimatorSet() 245 removeAnimatorSet.playTogether(mRemoveAnimatorsList) 246 mRemoveAnimatorsList.clear() 247 248 val addAnimatorSet = AnimatorSet() 249 addAnimatorSet.playTogether(mAddAnimatorsList) 250 mAddAnimatorsList.clear() 251 252 val changeAnimatorSet = AnimatorSet() 253 changeAnimatorSet.playTogether(mChangeAnimatorsList) 254 mChangeAnimatorsList.clear() 255 256 val moveAnimatorSet = AnimatorSet() 257 moveAnimatorSet.playTogether(mMoveAnimatorsList) 258 mMoveAnimatorsList.clear() 259 260 val pendingAnimatorSet = AnimatorSet() 261 pendingAnimatorSet.addListener(object : AnimatorListenerAdapter() { 262 override fun onAnimationEnd(animator: Animator) { 263 animator.removeAllListeners() 264 dispatchFinishedWhenDone() 265 } 266 }) 267 // Required order: removes, then changes & moves simultaneously, then additions. There are 268 // redundant edges because changes or moves may be empty, causing the removes to incorrectly 269 // play immediately. 270 pendingAnimatorSet.play(removeAnimatorSet).before(changeAnimatorSet) 271 pendingAnimatorSet.play(removeAnimatorSet).before(moveAnimatorSet) 272 pendingAnimatorSet.play(changeAnimatorSet).with(moveAnimatorSet) 273 pendingAnimatorSet.play(addAnimatorSet).after(changeAnimatorSet) 274 pendingAnimatorSet.play(addAnimatorSet).after(moveAnimatorSet) 275 pendingAnimatorSet.start() 276 } 277 endAnimationnull278 override fun endAnimation(holder: ViewHolder) { 279 val animator = mAnimators[holder] 280 281 mAnimators.remove(holder) 282 mAddAnimatorsList.remove(animator) 283 mRemoveAnimatorsList.remove(animator) 284 mChangeAnimatorsList.remove(animator) 285 mMoveAnimatorsList.remove(animator) 286 287 animator?.end() 288 dispatchFinishedWhenDone() 289 } 290 endAnimationsnull291 override fun endAnimations() { 292 val animatorList: MutableList<Animator?> = ArrayList(mAnimators.values) 293 for (animator in animatorList) { 294 animator?.end() 295 } 296 dispatchFinishedWhenDone() 297 } 298 isRunningnull299 override fun isRunning(): Boolean = mAnimators.isNotEmpty() 300 301 private fun dispatchFinishedWhenDone() { 302 if (!isRunning()) { 303 dispatchAnimationsFinished() 304 } 305 } 306 recordPreLayoutInformationnull307 override fun recordPreLayoutInformation( 308 state: State, 309 viewHolder: ViewHolder, 310 @AdapterChanges changeFlags: Int, 311 payloads: MutableList<Any> 312 ): ItemAnimator.ItemHolderInfo { 313 val itemHolderInfo: ItemHolderInfo = 314 super.recordPreLayoutInformation(state, viewHolder, changeFlags, payloads) 315 if (itemHolderInfo is PayloadItemHolderInfo) { 316 itemHolderInfo.payloads = payloads 317 } 318 return itemHolderInfo 319 } 320 obtainHolderInfonull321 override fun obtainHolderInfo(): ItemAnimator.ItemHolderInfo { 322 return PayloadItemHolderInfo() 323 } 324 canReuseUpdatedViewHoldernull325 override fun canReuseUpdatedViewHolder( 326 viewHolder: ViewHolder, 327 payloads: MutableList<Any?> 328 ): Boolean { 329 val defaultReusePolicy: Boolean = super.canReuseUpdatedViewHolder(viewHolder, payloads) 330 // Whenever we have a payload, this is an in-place animation. 331 return payloads.isNotEmpty() || defaultReusePolicy 332 } 333 334 private class PayloadItemHolderInfo : ItemHolderInfo() { 335 private val mPayloads: MutableList<Any> = ArrayList() 336 337 var payloads: MutableList<Any> 338 get() = mPayloads 339 set(payloads) { 340 mPayloads.clear() 341 mPayloads.addAll(payloads) 342 } 343 } 344 345 interface OnAnimateChangeListener { onAnimateChangenull346 fun onAnimateChange(oldHolder: ViewHolder, newHolder: ViewHolder, duration: Long): Animator? 347 348 fun onAnimateChange( 349 payloads: List<Any>?, 350 fromLeft: Int, 351 fromTop: Int, 352 fromRight: Int, 353 fromBottom: Int, 354 duration: Long 355 ): Animator? 356 } 357 }