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 ed4b10c5e..01a3c6a3b 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 @@ -70,6 +76,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, @@ -108,6 +115,7 @@ fun LottieAnimation( drawable.isApplyingOpacityToLayersEnabled = applyOpacityToLayers drawable.enableMergePathsForKitKatAndAbove(enableMergePaths) drawable.useSoftwareRendering(useSoftwareRendering) + drawable.maintainOriginalImageBounds = maintainOriginalImageBounds drawable.clipToCompositionBounds = clipToComposition drawable.progress = progress drawable.setBounds(0, 0, composition.bounds.width(), composition.bounds.height()) @@ -136,6 +144,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, @@ -157,6 +166,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 3bec2455d..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,6 +19,7 @@ import com.airbnb.lottie.value.ScaleXY * This takes a vararg of individual dynamic properties which should be created with [rememberLottieDynamicProperty]. * * @see rememberLottieDynamicProperty + * @see LottieDrawable.setMaintainOriginalImageBounds */ @Composable fun rememberLottieDynamicProperties( diff --git a/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java b/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java index bd8e56097..20d30cf8a 100644 --- a/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java +++ b/lottie/src/main/java/com/airbnb/lottie/LottieAnimationView.java @@ -843,6 +843,26 @@ public String getImageAssetsFolder() { return lottieDrawable.getImageAssetsFolder(); } + /** + * 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. + */ + public void setMaintainOriginalImageBounds(boolean maintainOriginalImageBounds) { + lottieDrawable.setMaintainOriginalImageBounds(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. + */ + public boolean getMaintainOriginalImageBounds() { + return lottieDrawable.getMaintainOriginalImageBounds(); + } + /** * Allows you to modify or clear a bitmap that was loaded for an image either automatically * through {@link #setImageAssetsFolder(String)} or with an {@link ImageAssetDelegate}. diff --git a/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java b/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java index d801f8009..10303c791 100644 --- a/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java +++ b/lottie/src/main/java/com/airbnb/lottie/LottieDrawable.java @@ -112,6 +112,7 @@ public void onAnimationUpdate(ValueAnimator animation) { @Nullable TextDelegate textDelegate; private boolean enableMergePaths; + private boolean maintainOriginalImageBounds = false; private boolean clipToCompositionBounds = true; @Nullable private CompositionLayer compositionLayer; @@ -254,6 +255,26 @@ public String getImageAssetsFolder() { return imageAssetsFolder; } + /** + * 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. + */ + public void setMaintainOriginalImageBounds(boolean maintainOriginalImageBounds) { + this.maintainOriginalImageBounds = 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. + */ + public boolean getMaintainOriginalImageBounds() { + return maintainOriginalImageBounds; + } + /** * Create a composition with {@link LottieCompositionFactory} * @@ -1115,7 +1136,11 @@ public Bitmap updateBitmap(String id, @Nullable Bitmap bitmap) { return ret; } + /** + * @deprecated use {@link #getBitmapForId(String)}. + */ @Nullable + @Deprecated public Bitmap getImageAsset(String id) { ImageAssetManager bm = getImageAssetManager(); if (bm != null) { @@ -1128,6 +1153,46 @@ public Bitmap getImageAsset(String id) { return null; } + /** + * Returns the bitmap that will be rendered for the given id in the Lottie animation file. + * The id is the asset reference id stored in the "id" property of each object in the "assets" array. + * + * The returned bitmap could be from: + * * Embedded in the animation file as a base64 string. + * * In the same directory as the animation file. + * * In the same zip file as the animation file. + * * Returned from an {@link ImageAssetDelegate}. + * or null if the image doesn't exist from any of those places. + */ + @Nullable + public Bitmap getBitmapForId(String id) { + ImageAssetManager assetManager = getImageAssetManager(); + if (assetManager != null) { + return assetManager.bitmapForId(id); + } + return null; + } + + /** + * Returns the {@link LottieImageAsset} that will be rendered for the given id in the Lottie animation file. + * The id is the asset reference id stored in the "id" property of each object in the "assets" array. + * + * The returned bitmap could be from: + * * Embedded in the animation file as a base64 string. + * * In the same directory as the animation file. + * * In the same zip file as the animation file. + * * Returned from an {@link ImageAssetDelegate}. + * or null if the image doesn't exist from any of those places. + */ + @Nullable + public LottieImageAsset getLottieImageAssetForId(String id) { + LottieComposition composition = this.composition; + if (composition == null) { + return null; + } + return composition.getImages().get(id); + } + private ImageAssetManager getImageAssetManager() { if (getCallback() == null) { // We can't get a bitmap since we can't get a Context from the callback. diff --git a/lottie/src/main/java/com/airbnb/lottie/LottieImageAsset.java b/lottie/src/main/java/com/airbnb/lottie/LottieImageAsset.java index f1acbe66c..39606060a 100644 --- a/lottie/src/main/java/com/airbnb/lottie/LottieImageAsset.java +++ b/lottie/src/main/java/com/airbnb/lottie/LottieImageAsset.java @@ -6,7 +6,7 @@ import androidx.annotation.RestrictTo; /** - * Data class describing an image asset exported by bodymovin. + * Data class describing an image asset embedded in a Lottie json file. */ public class LottieImageAsset { private final int width; @@ -36,6 +36,9 @@ public int getHeight() { return height; } + /** + * The reference id in the json file. + */ public String getId() { return id; } diff --git a/lottie/src/main/java/com/airbnb/lottie/manager/ImageAssetManager.java b/lottie/src/main/java/com/airbnb/lottie/manager/ImageAssetManager.java index d28061d8c..063435704 100644 --- a/lottie/src/main/java/com/airbnb/lottie/manager/ImageAssetManager.java +++ b/lottie/src/main/java/com/airbnb/lottie/manager/ImageAssetManager.java @@ -62,6 +62,10 @@ public void setDelegate(@Nullable ImageAssetDelegate assetDelegate) { return prevBitmap; } + @Nullable public LottieImageAsset getImageAssetById(String id) { + return imageAssets.get(id); + } + @Nullable public Bitmap bitmapForId(String id) { LottieImageAsset asset = imageAssets.get(id); if (asset == null) { 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 16534495f..6cd60b648 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 @@ -12,6 +12,7 @@ import androidx.annotation.Nullable; import com.airbnb.lottie.LottieDrawable; +import com.airbnb.lottie.LottieImageAsset; import com.airbnb.lottie.LottieProperty; import com.airbnb.lottie.animation.LPaint; import com.airbnb.lottie.animation.keyframe.BaseKeyframeAnimation; @@ -24,16 +25,18 @@ public class ImageLayer extends BaseLayer { private final Paint paint = new LPaint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG); private final Rect src = new Rect(); private final Rect dst = new Rect(); + @Nullable private final LottieImageAsset lottieImageAsset; @Nullable private BaseKeyframeAnimation colorFilterAnimation; @Nullable private BaseKeyframeAnimation imageAnimation; ImageLayer(LottieDrawable lottieDrawable, Layer layerModel) { super(lottieDrawable, layerModel); + lottieImageAsset = lottieDrawable.getLottieImageAssetForId(layerModel.getRefId()); } @Override public void drawLayer(@NonNull Canvas canvas, Matrix parentMatrix, int parentAlpha) { Bitmap bitmap = getBitmap(); - if (bitmap == null || bitmap.isRecycled()) { + if (bitmap == null || bitmap.isRecycled() || lottieImageAsset == null) { return; } float density = Utils.dpScale(); @@ -45,16 +48,21 @@ public class ImageLayer extends BaseLayer { canvas.save(); canvas.concat(parentMatrix); src.set(0, 0, bitmap.getWidth(), bitmap.getHeight()); - dst.set(0, 0, (int) (bitmap.getWidth() * density), (int) (bitmap.getHeight() * density)); + 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)); + } + canvas.drawBitmap(bitmap, src, dst, paint); canvas.restore(); } @Override public void getBounds(RectF outBounds, Matrix parentMatrix, boolean applyParents) { super.getBounds(outBounds, parentMatrix, applyParents); - Bitmap bitmap = getBitmap(); - if (bitmap != null) { - outBounds.set(0, 0, bitmap.getWidth() * Utils.dpScale(), bitmap.getHeight() * Utils.dpScale()); + if (lottieImageAsset != null) { + float scale = Utils.dpScale(); + outBounds.set(0, 0, lottieImageAsset.getWidth() * scale, lottieImageAsset.getHeight() * scale); boundsMatrix.mapRect(outBounds); } } @@ -63,11 +71,20 @@ public class ImageLayer extends BaseLayer { private Bitmap getBitmap() { if (imageAnimation != null) { Bitmap callbackBitmap = imageAnimation.getValue(); - if (callbackBitmap != null) + if (callbackBitmap != null) { return callbackBitmap; + } } String refId = layerModel.getRefId(); - return lottieDrawable.getImageAsset(refId); + Bitmap bitmapFromDrawable = lottieDrawable.getBitmapForId(refId); + if (bitmapFromDrawable != null) { + return bitmapFromDrawable; + } + LottieImageAsset asset = this.lottieImageAsset; + if (asset != null) { + return asset.getBitmap(); + } + return null; } @SuppressWarnings("SingleStatementInBlock") diff --git a/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/SnapshotTestCaseContext.kt b/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/SnapshotTestCaseContext.kt index b7e4e4be5..b05d0b0c9 100644 --- a/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/SnapshotTestCaseContext.kt +++ b/snapshot-tests/src/androidTest/java/com/airbnb/lottie/snapshots/SnapshotTestCaseContext.kt @@ -9,10 +9,13 @@ import android.graphics.PorterDuff import android.util.Log import android.view.View import android.view.ViewGroup +import android.view.ViewTreeObserver import android.widget.FrameLayout import android.widget.ImageView import android.widget.LinearLayout import androidx.compose.runtime.Composable +import androidx.compose.runtime.CompositionLocalProvider +import androidx.compose.runtime.compositionLocalOf import androidx.compose.ui.platform.ComposeView import androidx.core.view.doOnLayout import com.airbnb.lottie.FontAssetDelegate @@ -26,13 +29,12 @@ import com.airbnb.lottie.snapshots.utils.BitmapPool import com.airbnb.lottie.snapshots.utils.HappoSnapshotter import com.airbnb.lottie.snapshots.utils.ObjectPool import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.first import kotlinx.coroutines.suspendCancellableCoroutine -import kotlinx.coroutines.sync.Mutex import kotlinx.coroutines.withContext import kotlin.coroutines.resume -private val ActivityContentLock = Mutex() - /** * Set of properties that are available to all [SnapshotTestCase] runs. */ @@ -167,6 +169,12 @@ suspend fun SnapshotTestCaseContext.snapshotComposition( bitmapPool.release(bitmap) } +/** + * Use this to signal that the composition is not ready to be snapshot yet. + * This use useful if you are using things like `rememberLottieComposition` which parses a composition asynchronously. + */ +val LocalSnapshotReady = compositionLocalOf { MutableStateFlow(true) } + fun SnapshotTestCaseContext.loadCompositionFromAssetsSync(fileName: String): LottieComposition { return LottieCompositionFactory.fromAssetSync(context, fileName).value!! } @@ -180,53 +188,55 @@ suspend fun SnapshotTestCaseContext.snapshotComposable( log("Snapshotting $name") val composeView = ComposeView(context) composeView.layoutParams = LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT) - var bitmap = withContext(Dispatchers.Main) { - composeView.setContent { + val readyFlow = MutableStateFlow(null) + composeView.setContent { + CompositionLocalProvider(LocalSnapshotReady provides readyFlow) { content(RenderMode.SOFTWARE) } - suspendCancellableCoroutine { cont -> - composeView.doOnLayout { - log("Drawing $name") - val b = bitmapPool.acquire(composeView.width, composeView.height) - val canvas = Canvas(b) - composeView.draw(canvas) - cont.resume(b) - } - onActivity { activity -> - activity.binding.content.addView(composeView) - } - } + if (readyFlow.value == null) readyFlow.value = true } onActivity { activity -> - activity.binding.content.removeView(composeView) + activity.binding.content.addView(composeView) } + readyFlow.first { it == true } + composeView.awaitFrame() + log("Drawing $name - Software") + var bitmap = bitmapPool.acquire(composeView.width, composeView.height) + var canvas = Canvas(bitmap) + composeView.draw(canvas) snapshotter.record(bitmap, name, if (renderHardwareAndSoftware) "$variant - Software" else variant) bitmapPool.release(bitmap) if (renderHardwareAndSoftware) { - bitmap = withContext(Dispatchers.Main) { - composeView.setContent { + readyFlow.value = null + composeView.setContent { + CompositionLocalProvider(LocalSnapshotReady provides readyFlow) { content(RenderMode.HARDWARE) } - suspendCancellableCoroutine { cont -> - composeView.doOnLayout { - log("Drawing $name") - val b = bitmapPool.acquire(composeView.width, composeView.height) - val canvas = Canvas(b) - composeView.draw(canvas) - cont.resume(b) - } - onActivity { activity -> - activity.binding.content.addView(composeView) - } - } + if (readyFlow.value == null) readyFlow.value = true } - onActivity { activity -> - activity.binding.content.removeView(composeView) - } - snapshotter.record(bitmap, name, "$variant - Hardware") + readyFlow.first { it == true } + composeView.awaitFrame() + log("Drawing $name - Software") + bitmap = bitmapPool.acquire(composeView.width, composeView.height) + canvas = Canvas(bitmap) + composeView.draw(canvas) + snapshotter.record(bitmap, name, if (renderHardwareAndSoftware) "$variant - Hardware" else variant) bitmapPool.release(bitmap) } + + onActivity { activity -> + activity.binding.content.removeView(composeView) + } + LottieCompositionCache.getInstance().clear() +} + +private suspend fun View.awaitFrame() { + suspendCancellableCoroutine { cont -> + post { + cont.resume(Unit) + } + } } \ No newline at end of file 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..a4b9be7ca 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,21 +1,25 @@ 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 androidx.compose.runtime.getValue import com.airbnb.lottie.LottieCompositionFactory import com.airbnb.lottie.LottieProperty import com.airbnb.lottie.compose.LottieAnimation +import com.airbnb.lottie.compose.LottieCompositionSpec +import com.airbnb.lottie.compose.rememberLottieComposition import com.airbnb.lottie.compose.rememberLottieDynamicProperties import com.airbnb.lottie.compose.rememberLottieDynamicProperty +import com.airbnb.lottie.snapshots.LocalSnapshotReady import com.airbnb.lottie.snapshots.SnapshotTestCase import com.airbnb.lottie.snapshots.SnapshotTestCaseContext import com.airbnb.lottie.snapshots.snapshotComposable -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 +38,48 @@ class ComposeDynamicPropertiesTestCase : SnapshotTestCase { ) LottieAnimation(composition, 0f, dynamicProperties = dynamicProperties) } + + val heartComposition = LottieCompositionFactory.fromAssetSync(context, "Tests/Heart.json").value!! + snapshotComposable("Compose Dynamic Image", "Default") { + val composition by rememberLottieComposition(LottieCompositionSpec.Asset("Tests/Heart.json")) + LocalSnapshotReady.current.value = composition != null + LottieAnimation(composition, 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 diff --git a/snapshot-tests/src/main/java/com/airbnb/lottie/snapshots/FilmStripView.kt b/snapshot-tests/src/main/java/com/airbnb/lottie/snapshots/FilmStripView.kt index 310a43c60..5b4ae8b80 100644 --- a/snapshot-tests/src/main/java/com/airbnb/lottie/snapshots/FilmStripView.kt +++ b/snapshot-tests/src/main/java/com/airbnb/lottie/snapshots/FilmStripView.kt @@ -33,6 +33,8 @@ class FilmStripView @JvmOverloads constructor( fun setImageAssetDelegate(delegate: ImageAssetDelegate?) { animationViews.forEach { it.setImageAssetDelegate(delegate) } + // Enable bitmap rescaling for the first 4 views so both APIs get test coverage. + animationViews.forEachIndexed { i, av -> av.maintainOriginalImageBounds = i <= 4 } } fun setFontAssetDelegate(delegate: FontAssetDelegate?) {