diff --git a/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieAnimation.kt b/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieAnimation.kt index 886747c3a..837cbb866 100644 --- a/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieAnimation.kt +++ b/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieAnimation.kt @@ -54,6 +54,12 @@ import kotlin.math.roundToInt * features so it defaults to off. The only way to know if your animation will work * well with merge paths or not is to try it. If your animation has merge paths and * doesn't render correctly, please file an issue. + * @param renderMode Allows you to specify whether you want Lottie to use hardware or software rendering. + * Defaults to AUTOMATIC. Refer to [LottieAnimationView.setRenderMode] for more info. + * @param maintainOriginalImageBounds When true, dynamically set bitmaps will be drawn with the exact bounds of the original animation, + * regardless of the bitmap size. + * When false, dynamically set bitmaps will be drawn at the top left of the original image but with its own bounds. + * Defaults to false. * @param dynamicProperties Allows you to change the properties of an animation dynamically. To use them, use * [rememberLottieDynamicProperties]. Refer to its docs for more info. * @param alignment Define where the animation should be placed within this composable if it has a different @@ -69,6 +75,7 @@ fun LottieAnimation( applyOpacityToLayers: Boolean = false, enableMergePaths: Boolean = false, renderMode: RenderMode = RenderMode.AUTOMATIC, + maintainOriginalImageBounds: Boolean = false, dynamicProperties: LottieDynamicProperties? = null, alignment: Alignment = Alignment.Center, contentScale: ContentScale = ContentScale.Fit, @@ -106,6 +113,7 @@ fun LottieAnimation( drawable.isApplyingOpacityToLayersEnabled = applyOpacityToLayers drawable.enableMergePathsForKitKatAndAbove(enableMergePaths) drawable.useSoftwareRendering(useSoftwareRendering) + drawable.maintainOriginalImageBounds = maintainOriginalImageBounds drawable.progress = progress drawable.setBounds(0, 0, composition.bounds.width(), composition.bounds.height()) drawable.draw(canvas.nativeCanvas, matrix) @@ -133,6 +141,7 @@ fun LottieAnimation( applyOpacityToLayers: Boolean = false, enableMergePaths: Boolean = false, renderMode: RenderMode = RenderMode.AUTOMATIC, + maintainOriginalImageBounds: Boolean = false, dynamicProperties: LottieDynamicProperties? = null, alignment: Alignment = Alignment.Center, contentScale: ContentScale = ContentScale.Fit, @@ -153,6 +162,7 @@ fun LottieAnimation( applyOpacityToLayers, enableMergePaths, renderMode, + maintainOriginalImageBounds, dynamicProperties, alignment, contentScale, diff --git a/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieDynamicProperties.kt b/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieDynamicProperties.kt index e050a5a68..ea1331c84 100644 --- a/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieDynamicProperties.kt +++ b/lottie-compose/src/main/java/com/airbnb/lottie/compose/LottieDynamicProperties.kt @@ -19,15 +19,14 @@ import com.airbnb.lottie.value.ScaleXY * This takes a vararg of individual dynamic properties which should be created with [rememberLottieDynamicProperty]. * * @see rememberLottieDynamicProperty - * @see LottieDrawable.setRescaleBitmaps + * @see LottieDrawable.setMaintainOriginalImageBounds */ @Composable fun rememberLottieDynamicProperties( - rescaleBitmaps: Boolean = false, vararg properties: LottieDynamicProperty<*>, ): LottieDynamicProperties { - return remember(rescaleBitmaps, properties) { - LottieDynamicProperties(rescaleBitmaps, properties.toList()) + return remember(properties) { + LottieDynamicProperties(properties.toList()) } } @@ -93,7 +92,6 @@ class LottieDynamicProperty internal constructor( * @see rememberLottieDynamicProperties */ class LottieDynamicProperties internal constructor( - private val rescaleBitmaps: Boolean, private val intProperties: List>, private val pointFProperties: List>, private val floatProperties: List>, @@ -105,8 +103,7 @@ class LottieDynamicProperties internal constructor( private val bitmapProperties: List>, ) { @Suppress("UNCHECKED_CAST") - constructor(rescaleBitmaps: Boolean, properties: List>) : this( - rescaleBitmaps, + constructor(properties: List>) : this( properties.filter { it.property is Int } as List>, properties.filter { it.property is PointF } as List>, properties.filter { it.property is Float } as List>, @@ -118,7 +115,6 @@ class LottieDynamicProperties internal constructor( ) internal fun addTo(drawable: LottieDrawable) { - drawable.rescaleBitmaps = rescaleBitmaps intProperties.forEach { p -> drawable.addValueCallback(p.keyPath, p.property, p.callback.toValueCallback()) } diff --git a/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java b/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java index 538ac06a2..e4e3556b4 100644 --- a/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java +++ b/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java @@ -900,7 +900,7 @@ public String getImageAssetsFolder() { * Defaults to false. */ public void setRescaleBitmaps(boolean rescaleBitmaps) { - lottieDrawable.setRescaleBitmaps(rescaleBitmaps); + lottieDrawable.setMaintainOriginalImageBounds(rescaleBitmaps); } /** @@ -910,7 +910,7 @@ public void setRescaleBitmaps(boolean rescaleBitmaps) { * Defaults to false. */ public boolean getRescaleBitmaps() { - return lottieDrawable.getRescaleBitmaps(); + return lottieDrawable.getMaintainOriginalImageBounds(); } /** diff --git a/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java b/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java index eef6f23ec..1521163e0 100644 --- a/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java +++ b/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java @@ -99,7 +99,7 @@ public void onAnimationUpdate(ValueAnimator animation) { @Nullable TextDelegate textDelegate; private boolean enableMergePaths; - private boolean rescaleBitmaps = true; + private boolean maintainOriginalImageBounds = false; @Nullable private CompositionLayer compositionLayer; private int alpha = 255; @@ -221,23 +221,23 @@ public String getImageAssetsFolder() { } /** - * When true, dynamically set bitmaps will be drawn at the size of the original bitmap. - * When false, dynamically set bitmaps will be drawn at 0,0 at the original bitmap but at whatever size the dynamic bitmap is. + * When true, dynamically set bitmaps will be drawn with the exact bounds of the original animation, regardless of the bitmap size. + * When false, dynamically set bitmaps will be drawn at the top left of the original image but with its own bounds. * - * Defaults to true. + * Defaults to false. */ - public void setRescaleBitmaps(boolean rescaleBitmaps) { - this.rescaleBitmaps = rescaleBitmaps; + public void setMaintainOriginalImageBounds(boolean maintainOriginalImageBounds) { + this.maintainOriginalImageBounds = maintainOriginalImageBounds; } /** - * When true, dynamically set bitmaps will be drawn at the size of the original bitmap. - * When false, dynamically set bitmaps will be drawn at 0,0 at the original bitmap but at whatever size the dynamic bitmap is. + * When true, dynamically set bitmaps will be drawn with the exact bounds of the original animation, regardless of the bitmap size. + * When false, dynamically set bitmaps will be drawn at the top left of the original image but with its own bounds. * - * Defaults to true. + * Defaults to false. */ - public boolean getRescaleBitmaps() { - return rescaleBitmaps; + public boolean getMaintainOriginalImageBounds() { + return maintainOriginalImageBounds; } /** diff --git a/lottie/src/main/java/com/airbnb/lottie/model/layer/ImageLayer.java b/lottie/src/main/java/com/airbnb/lottie/model/layer/ImageLayer.java index e68ebf0b3..11ade548e 100644 --- a/lottie/src/main/java/com/airbnb/lottie/model/layer/ImageLayer.java +++ b/lottie/src/main/java/com/airbnb/lottie/model/layer/ImageLayer.java @@ -48,7 +48,7 @@ public class ImageLayer extends BaseLayer { canvas.save(); canvas.concat(parentMatrix); src.set(0, 0, bitmap.getWidth(), bitmap.getHeight()); - if (lottieDrawable.getRescaleBitmaps()) { + if (lottieDrawable.getMaintainOriginalImageBounds()) { dst.set(0, 0, (int) (lottieImageAsset.getWidth() * density), (int) (lottieImageAsset.getHeight() * density)); } else { dst.set(0, 0, (int) (bitmap.getWidth() * density), (int) (bitmap.getHeight() * density)); diff --git a/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/tests/ComposeDynamicPropertiesTestCase.kt b/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/tests/ComposeDynamicPropertiesTestCase.kt index 6356baf9d..e11af6d00 100644 --- a/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/tests/ComposeDynamicPropertiesTestCase.kt +++ b/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/tests/ComposeDynamicPropertiesTestCase.kt @@ -1,10 +1,13 @@ package com.airbnb.lottie.snapshots.tests +import android.graphics.Bitmap +import android.graphics.BitmapFactory import android.graphics.Color import androidx.compose.ui.platform.ComposeView import com.airbnb.lottie.LottieCompositionFactory import com.airbnb.lottie.LottieProperty import com.airbnb.lottie.compose.LottieAnimation +import com.airbnb.lottie.compose.LottieDynamicProperty import com.airbnb.lottie.compose.rememberLottieDynamicProperties import com.airbnb.lottie.compose.rememberLottieDynamicProperty import com.airbnb.lottie.snapshots.SnapshotTestCase @@ -14,8 +17,8 @@ import com.airbnb.lottie.snapshots.snapshotComposition class ComposeDynamicPropertiesTestCase : SnapshotTestCase { override suspend fun SnapshotTestCaseContext.run() { - val composition = LottieCompositionFactory.fromAssetSync(context, "Tests/DynamicGradient.json").value!! snapshotComposable("Compose Dynamic Gradient") { + val composition = LottieCompositionFactory.fromAssetSync(context, "Tests/DynamicGradient.json").value!! val dynamicProperties = rememberLottieDynamicProperties( rememberLottieDynamicProperty( LottieProperty.GRADIENT_COLOR, @@ -34,5 +37,46 @@ class ComposeDynamicPropertiesTestCase : SnapshotTestCase { ) LottieAnimation(composition, 0f, dynamicProperties = dynamicProperties) } + + val heartComposition = LottieCompositionFactory.fromAssetSync(context, "Tests/Heart.json").value!! + snapshotComposable("Compose Dynamic Image", "Default") { + LottieAnimation(heartComposition, 0f) + } + snapshotComposable("Compose Dynamic Image", "Default - Maintain Original Bounds") { + LottieAnimation(heartComposition, 0f, maintainOriginalImageBounds = true) + } + snapshotComposable("Compose Dynamic Image", "Smaller") { + val bitmap = getBitmapFromAssets("Images/Heart-80.png") + val dynamicProperties = rememberLottieDynamicProperties( + rememberLottieDynamicProperty(LottieProperty.IMAGE, bitmap, "Heart"), + ) + LottieAnimation(heartComposition, 0f, dynamicProperties = dynamicProperties) + } + snapshotComposable("Compose Dynamic Image", "Smaller - Maintain Original Bounds") { + val bitmap = getBitmapFromAssets("Images/Heart-80.png") + val dynamicProperties = rememberLottieDynamicProperties( + rememberLottieDynamicProperty(LottieProperty.IMAGE, bitmap, "Heart"), + ) + LottieAnimation(heartComposition, 0f, dynamicProperties = dynamicProperties, maintainOriginalImageBounds = true) + } + snapshotComposable("Compose Dynamic Image", "Larger") { + val bitmap = getBitmapFromAssets("Images/Heart-1200.png") + val dynamicProperties = rememberLottieDynamicProperties( + rememberLottieDynamicProperty(LottieProperty.IMAGE, bitmap, "Heart"), + ) + LottieAnimation(heartComposition, 0f, dynamicProperties = dynamicProperties) + } + snapshotComposable("Compose Dynamic Image", "Larger - Maintain Original Bounds") { + val bitmap = getBitmapFromAssets("Images/Heart-1200.png") + val dynamicProperties = rememberLottieDynamicProperties( + rememberLottieDynamicProperty(LottieProperty.IMAGE, bitmap, "Heart"), + ) + LottieAnimation(heartComposition, 0f, dynamicProperties = dynamicProperties, maintainOriginalImageBounds = true) + } + } + + private fun SnapshotTestCaseContext.getBitmapFromAssets(name: String): Bitmap { + @Suppress("BlockingMethodInNonBlockingContext") + return BitmapFactory.decodeStream(context.assets.open(name), null, BitmapFactory.Options())!! } } \ No newline at end of file diff --git a/snapshot-tests/src/main/assets/Images/Heart-1200.png b/snapshot-tests/src/main/assets/Images/Heart-1200.png new file mode 100644 index 000000000..27f52bf76 Binary files /dev/null and b/snapshot-tests/src/main/assets/Images/Heart-1200.png differ diff --git a/snapshot-tests/src/main/assets/Images/Heart-80.png b/snapshot-tests/src/main/assets/Images/Heart-80.png new file mode 100644 index 000000000..8f3eae4ab Binary files /dev/null and b/snapshot-tests/src/main/assets/Images/Heart-80.png differ diff --git a/snapshot-tests/src/main/assets/Tests/Heart.json b/snapshot-tests/src/main/assets/Tests/Heart.json new file mode 100644 index 000000000..43edfe5b6 --- /dev/null +++ b/snapshot-tests/src/main/assets/Tests/Heart.json @@ -0,0 +1 @@ +{"v":"5.8.1","fr":60,"ip":0,"op":32400,"w":400,"h":400,"nm":"Comp 1","ddd":0,"assets":[{"id":"image_0","w":400,"h":400,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"Heart","refId":"image_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[200,200,0],"ix":2,"l":2},"a":{"a":0,"k":[200,200,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":32400,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":1,"nm":"Blue Solid 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[200,200,0],"ix":2,"l":2},"a":{"a":0,"k":[600,600,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"sw":1200,"sh":1200,"sc":"#0600ff","ip":0,"op":32400,"st":0,"bm":0}],"markers":[]} \ No newline at end of file