From e6e251e496c51b9b327e52eab2ead795843ced27 Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Sat, 9 Feb 2019 03:03:19 +0100 Subject: [PATCH 1/6] added cancelAnimations method to ZoomApi removed mClearAnimation flag and replaced it with mActiveAnimators which holds a list of all active animators that will be cancelled if a new animation starts or the state changes for other reasons updated readme --- README.md | 2 + .../java/com/otaliastudios/zoom/ZoomApi.kt | 8 ++ .../java/com/otaliastudios/zoom/ZoomEngine.kt | 122 ++++++++++-------- 3 files changed, 76 insertions(+), 56 deletions(-) diff --git a/README.md b/README.md index f951de3..ab3ac84 100644 --- a/README.md +++ b/README.md @@ -243,6 +243,7 @@ will make more sense than the other - e. g., in a PDF viewer, you might want to |`zoomOut()`|Applies a small, animated zoom-out.|`-`| |`setZoomEnabled(boolean)`|If true, the content will be allowed to zoom in and out by user input.|`true`| + The `moveTo(float, float, float, boolean)` API will let you animate both zoom and [pan](#pan) at the same time. ### Pan @@ -267,6 +268,7 @@ In any case the current scale is not considered, so your system won't change if |`setAllowFlingInOverscroll(boolean)`|If true, fling gestures will be allowed even when detected while overscrolled. This might cause artifacts so it is disabled by default.|`false`| |`panTo(float, float, boolean)`|Pans to the given values, animating if needed.|`-`| |`panBy(float, float, boolean)`|Applies the given deltas to the current pan, animating if needed.|`-`| +|`cancelAnimations()`|Cancels all currently active code driven animations (including fling animations).|`-`| The `moveTo(float, float, float, boolean)` API will let you animate both [zoom](#zoom) and pan at the same time. diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt index 39c011e..8bb3a1c 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt @@ -336,6 +336,14 @@ interface ZoomApi { */ fun setAnimationDuration(duration: Long) + /** + * Cancels all currently active code driven animations (including fling animations) + * If no animation is currently active this is a no-op. + * + * @return true if anything was cancelled, false otherwise + */ + fun cancelAnimations(): Boolean + companion object { /** diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt index af4c99d..f95a0b4 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt @@ -85,7 +85,6 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener private var mInitialized = false private var mContentScaledRect = RectF() private var mContentRect = RectF() - private var mClearAnimation = false private var mAnimationDuration = DEFAULT_ANIMATION_DURATION // Gestures @@ -677,27 +676,31 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener // Returns true if we should go to that mode. @SuppressLint("SwitchIntDef") - private fun setState(@State state: Int): Boolean { - LOG.v("trySetState:", state.toStateName()) + private fun setState(@State newState: Int): Boolean { + LOG.v("trySetState:", newState.toStateName()) if (!mInitialized) return false - if (state == mState) return true - val oldMode = mState - - when (state) { - SCROLLING -> if (oldMode == PINCHING || oldMode == ANIMATING) return false - FLINGING -> if (oldMode == ANIMATING) return false - PINCHING -> if (oldMode == ANIMATING) return false + // we need to do some cleanup in case of ANIMATING so we can't return just yet + if (newState == mState && newState != ANIMATING) return true + val oldState = mState + + when (newState) { + SCROLLING -> if (oldState == PINCHING || oldState == ANIMATING) return false + FLINGING -> if (oldState == ANIMATING) return false + PINCHING -> if (oldState == ANIMATING) return false NONE -> dispatchOnIdle() } // Now that it succeeded, do some cleanup. - when (oldMode) { + when (oldState) { + ANIMATING -> { + mActiveAnimators.forEach { it.cancel() } + mActiveAnimators.clear() + } FLINGING -> mFlingScroller.forceFinished(true) - ANIMATING -> mClearAnimation = true } - LOG.i("setState:", state.toStateName()) - mState = state + LOG.i("setState:", newState.toStateName()) + mState = newState return true } @@ -1202,17 +1205,13 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener * Applies a small, animated zoom-in. * Shorthand for [zoomBy] with factor 1.3. */ - override fun zoomIn() { - zoomBy(1.3f, animate = true) - } + override fun zoomIn() = zoomBy(1.3f, animate = true) /** * Applies a small, animated zoom-out. * Shorthand for [zoomBy] with factor 0.7. */ - override fun zoomOut() { - zoomBy(0.7f, animate = true) - } + override fun zoomOut() = zoomBy(0.7f, animate = true) /** * Animates the actual matrix zoom to the given value. @@ -1277,24 +1276,46 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener //region Apply values + private val mActiveAnimators = mutableSetOf() private val mCancelAnimationListener = object : AnimatorListenerAdapter() { - override fun onAnimationEnd(animation: Animator?) { - setState(NONE) + override fun onAnimationEnd(animator: Animator) { + cleanup(animator) + } + + override fun onAnimationCancel(animator: Animator) { + cleanup(animator) } - override fun onAnimationCancel(animation: Animator?) { + private fun cleanup(animator: Animator) { + animator.removeListener(this) + mActiveAnimators.remove(animator) setState(NONE) } } /** * Prepares a [ValueAnimator] for the first run + * + * @return itself (for chaining) */ - private fun ValueAnimator.prepare() { + private fun ValueAnimator.prepare(): ValueAnimator { this.duration = mAnimationDuration this.addListener(mCancelAnimationListener) this.interpolator = ANIMATION_INTERPOLATOR + return this + } + + /** + * Starts a [ValueAnimator] with the given update function + * + * @return itself (for chaining) + */ + private fun ValueAnimator.start(onUpdate: (ValueAnimator) -> Unit): ValueAnimator { + this.addUpdateListener(onUpdate) + this.start() + mActiveAnimators.add(this) + return this } /** @@ -1306,21 +1327,12 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener */ private fun animateZoom(@Zoom zoom: Float, allowOverPinch: Boolean) { if (setState(ANIMATING)) { - mClearAnimation = false @Zoom val startZoom = this.zoom @Zoom val endZoom = checkZoomBounds(zoom, allowOverPinch) - val zoomAnimator = ValueAnimator.ofFloat(startZoom, endZoom) - zoomAnimator.prepare() - zoomAnimator.addUpdateListener { + ValueAnimator.ofFloat(startZoom, endZoom).prepare().start { LOG.v("animateZoom:", "animationStep:", it.animatedFraction) - if (mClearAnimation) { - it.cancel() - return@addUpdateListener - } applyZoom(it.animatedValue as Float, allowOverPinch) } - - zoomAnimator.start() } } @@ -1336,6 +1348,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener * @param zoomTargetX the x-axis zoom target * @param zoomTargetY the y-axis zoom target */ + @SuppressLint("ObjectAnimatorBinding") private fun animateZoomAndAbsolutePan(@Zoom zoom: Float, @AbsolutePan x: Float, @AbsolutePan y: Float, allowOverScroll: Boolean, @@ -1343,7 +1356,6 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener zoomTargetX: Float? = null, zoomTargetY: Float? = null) { if (setState(ANIMATING)) { - mClearAnimation = false @Zoom val startZoom = this.zoom @Zoom val endZoom = checkZoomBounds(zoom, allowOverScroll) val startPan = pan @@ -1351,8 +1363,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener LOG.i("animateZoomAndAbsolutePan:", "starting.", "startX:", startPan.x, "endX:", x, "startY:", startPan.y, "endY:", y) LOG.i("animateZoomAndAbsolutePan:", "starting.", "startZoom:", startZoom, "endZoom:", endZoom) - @SuppressLint("ObjectAnimatorBinding") - val animator = ObjectAnimator.ofPropertyValuesHolder(mContainer, + ObjectAnimator.ofPropertyValuesHolder(mContainer, PropertyValuesHolder.ofObject( "pan", TypeEvaluator { fraction: Float, startValue: AbsolutePoint, endValue: AbsolutePoint -> @@ -1362,20 +1373,13 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener PropertyValuesHolder.ofFloat( "zoom", startZoom, endZoom) - ) - animator.prepare() - animator.addUpdateListener { - if (mClearAnimation) { - it.cancel() - return@addUpdateListener - } + ).prepare().start { val newZoom = it.getAnimatedValue("zoom") as Float val currentPan = it.getAnimatedValue("pan") as AbsolutePoint applyZoomAndAbsolutePan(newZoom, currentPan.x, currentPan.y, allowOverScroll, allowOverPinch, zoomTargetX, zoomTargetY) } - animator.start() } } @@ -1390,25 +1394,16 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener private fun animateScaledPan(@ScaledPan deltaX: Float, @ScaledPan deltaY: Float, allowOverScroll: Boolean) { if (setState(ANIMATING)) { - mClearAnimation = false val startPan = scaledPan val endPan = startPan + ScaledPoint(deltaX, deltaY) - val panAnimator = ValueAnimator.ofObject(TypeEvaluator { fraction, startValue: ScaledPoint, endValue: ScaledPoint -> + ValueAnimator.ofObject(TypeEvaluator { fraction, startValue: ScaledPoint, endValue: ScaledPoint -> startValue + (endValue - startValue) * fraction - scaledPan - }, startPan, endPan) - panAnimator.prepare() - panAnimator.addUpdateListener { + }, startPan, endPan).prepare().start { LOG.v("animateScaledPan:", "animationStep:", it.animatedFraction) - if (mClearAnimation) { - it.cancel() - return@addUpdateListener - } val currentPan = it.animatedValue as ScaledPoint applyScaledPan(currentPan.x, currentPan.y, allowOverScroll) } - - panAnimator.start() } } @@ -1596,6 +1591,21 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener return true } + /** + * Cancels all currently active code driven animations (including fling animations) + * If no animation is currently active this is a no-op. + * + * @return true if anything was cancelled, false otherwise + */ + override fun cancelAnimations(): Boolean { + if (mState == FLINGING || mState == ANIMATING) { + setState(NONE) + return true + } + + return false + } + //endregion //region scrollbars helpers From e0c313b345bbf66a522275cd77efa5b0f6aa0627 Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Sat, 9 Feb 2019 03:21:50 +0100 Subject: [PATCH 2/6] only set state to none if all active animators are done --- library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt index f95a0b4..d3ea0c8 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt @@ -1290,7 +1290,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener private fun cleanup(animator: Animator) { animator.removeListener(this) mActiveAnimators.remove(animator) - setState(NONE) + if (mActiveAnimators.isEmpty()) setState(NONE) } } From c0ab743be67f525a3e6b864711db2a1f4d2b4eec Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Sat, 9 Feb 2019 03:57:59 +0100 Subject: [PATCH 3/6] cancelAnimations when using positioning or zoom API methods without animation --- library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt | 3 +++ 1 file changed, 3 insertions(+) diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt index d3ea0c8..9a9cb7c 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt @@ -1134,6 +1134,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener if (animate) { animateZoomAndAbsolutePan(zoom, x, y, allowOverScroll = false) } else { + cancelAnimations() applyZoomAndAbsolutePan(zoom, x, y, allowOverScroll = false) } } @@ -1169,6 +1170,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener if (animate) { animateZoomAndAbsolutePan(zoom, panX + dx, panY + dy, allowOverScroll = false) } else { + cancelAnimations() applyZoomAndAbsolutePan(zoom, panX + dx, panY + dy, allowOverScroll = false) } } @@ -1185,6 +1187,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener if (animate) { animateZoom(zoom, allowOverPinch = false) } else { + cancelAnimations() applyZoom(zoom, allowOverPinch = false) } } From de753b9c878db30cef9d6b2e823159f4ef8f9731 Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Sun, 10 Feb 2019 04:55:19 +0100 Subject: [PATCH 4/6] remove unnecessary valueanimator parameter --- library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt index 9a9cb7c..0509aa6 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt @@ -1366,7 +1366,7 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener LOG.i("animateZoomAndAbsolutePan:", "starting.", "startX:", startPan.x, "endX:", x, "startY:", startPan.y, "endY:", y) LOG.i("animateZoomAndAbsolutePan:", "starting.", "startZoom:", startZoom, "endZoom:", endZoom) - ObjectAnimator.ofPropertyValuesHolder(mContainer, + ObjectAnimator.ofPropertyValuesHolder( PropertyValuesHolder.ofObject( "pan", TypeEvaluator { fraction: Float, startValue: AbsolutePoint, endValue: AbsolutePoint -> From 2d76fec155698d44a6ce195212a581ac3cd9c5d6 Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Tue, 12 Feb 2019 01:25:35 +0100 Subject: [PATCH 5/6] review fix --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index ab3ac84..9043fe8 100644 --- a/README.md +++ b/README.md @@ -268,7 +268,7 @@ In any case the current scale is not considered, so your system won't change if |`setAllowFlingInOverscroll(boolean)`|If true, fling gestures will be allowed even when detected while overscrolled. This might cause artifacts so it is disabled by default.|`false`| |`panTo(float, float, boolean)`|Pans to the given values, animating if needed.|`-`| |`panBy(float, float, boolean)`|Applies the given deltas to the current pan, animating if needed.|`-`| -|`cancelAnimations()`|Cancels all currently active code driven animations (including fling animations).|`-`| +|`cancelAnimations()`|Cancels all currently active animations triggered by either API calls with `animate = true` or touch input flings.|`-`| The `moveTo(float, float, float, boolean)` API will let you animate both [zoom](#zoom) and pan at the same time. From 2e2b37c62786a417329dcd1a585b1ca2ad8ecced Mon Sep 17 00:00:00 2001 From: Markus Ressel Date: Tue, 12 Feb 2019 01:27:10 +0100 Subject: [PATCH 6/6] review fix --- library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt | 4 ++-- library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt index 8bb3a1c..bc6023e 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomApi.kt @@ -337,8 +337,8 @@ interface ZoomApi { fun setAnimationDuration(duration: Long) /** - * Cancels all currently active code driven animations (including fling animations) - * If no animation is currently active this is a no-op. + * Cancels all currently active animations triggered by either API calls with `animate = true` + * or touch input flings. If no animation is currently active this is a no-op. * * @return true if anything was cancelled, false otherwise */ diff --git a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt index 0509aa6..5f7b008 100644 --- a/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt +++ b/library/src/main/java/com/otaliastudios/zoom/ZoomEngine.kt @@ -1595,8 +1595,8 @@ internal constructor(context: Context) : ViewTreeObserver.OnGlobalLayoutListener } /** - * Cancels all currently active code driven animations (including fling animations) - * If no animation is currently active this is a no-op. + * Cancels all currently active animations triggered by either API calls with `animate = true` + * or touch input flings. If no animation is currently active this is a no-op. * * @return true if anything was cancelled, false otherwise */