From 9aaadcd5fa9611b70c9a365909060aca40f7891a Mon Sep 17 00:00:00 2001 From: Marco Saia Date: Thu, 12 Dec 2024 15:15:41 +0100 Subject: [PATCH 1/5] RUM-6195: Add support for Compose Checkbox --- .../android/internal/utils/ImageViewUtils.kt | 14 +- detekt_custom.yml | 9 + .../semantics/CheckboxSemanticsNodeMapper.kt | 205 ++++++++ .../semantics/RootSemanticsNodeMapper.kt | 3 +- .../internal/reflection/ComposeReflection.kt | 10 + .../compose/internal/utils/PathUtils.kt | 186 +++++++ .../compose/internal/utils/SemanticsUtils.kt | 70 +++ .../CheckboxSemanticsNodeMapperTest.kt | 475 ++++++++++++++++++ .../compose/internal/utils/PathUtilsTest.kt | 143 ++++++ .../api/apiSurface | 6 + .../api/dd-sdk-android-session-replay.api | 13 + .../internal/utils/DrawableUtils.kt | 11 +- .../recorder/wrappers/BitmapWrapper.kt | 27 +- .../recorder/wrappers/CanvasWrapper.kt | 17 +- .../DefaultImageWireframeHelperTest.kt | 25 +- .../internal/utils/DrawableUtilsTest.kt | 25 +- .../recorder/wrappers/CanvasWrapperTest.kt | 2 +- 17 files changed, 1198 insertions(+), 43 deletions(-) create mode 100644 features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt create mode 100644 features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt create mode 100644 features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt create mode 100644 features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt rename features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/{internal => }/recorder/wrappers/BitmapWrapper.kt (75%) rename features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/{internal => }/recorder/wrappers/CanvasWrapper.kt (80%) rename features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/{internal => }/recorder/wrappers/CanvasWrapperTest.kt (96%) diff --git a/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt b/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt index cf3f148758..73dbcaed52 100644 --- a/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt +++ b/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt @@ -19,7 +19,7 @@ import android.widget.ImageView object ImageViewUtils { /** * Resolves the absolute position on the screen of the given [View]. - * @param view the [View]. + * @param view: the [View]. * @return the [Rect] representing the absolute position of the view. */ fun resolveParentRectAbsPosition(view: View): Rect { @@ -42,9 +42,9 @@ object ImageViewUtils { /** * Calculates the clipping [Rect] of the given child [Rect] using its parent [Rect] and * the screen density. - * @param parentRect the parent [Rect]. - * @param childRect the child [Rect]. - * @param density the screen density. + * @param parentRect: the parent [Rect]. + * @param childRect: the child [Rect]. + * @param density: the screen density. * @return the clipping [Rect]. */ fun calculateClipping(parentRect: Rect, childRect: Rect, density: Float): Rect { @@ -78,9 +78,9 @@ object ImageViewUtils { /** * Resolves the [Drawable] content [Rect] using the given [ImageView] scale type. - * @param imageView the [ImageView]. - * @param drawable the [Drawable]. - * @param customScaleType optional custom [ImageView.ScaleType]. + * @param imageView: the [ImageView]. + * @param drawable: the [Drawable]. + * @param customScaleType: optional custom [ImageView.ScaleType]. * @return the resolved content [Rect]. */ fun resolveContentRectWithScaling( diff --git a/detekt_custom.yml b/detekt_custom.yml index f9c57759d3..5ad1efef3a 100644 --- a/detekt_custom.yml +++ b/detekt_custom.yml @@ -129,8 +129,10 @@ datadog: - "android.database.sqlite.SQLiteDatabase.setTransactionSuccessful():java.lang.IllegalStateException" - "android.graphics.Bitmap.compress(android.graphics.Bitmap.CompressFormat, kotlin.Int, java.io.OutputStream):java.lang.NullPointerException,java.lang.IllegalArgumentException" - "android.graphics.Bitmap.copy(android.graphics.Bitmap.Config, kotlin.Boolean):java.lang.IllegalArgumentException" + - "android.graphics.Bitmap.createBitmap(kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException" - "android.graphics.Bitmap.createBitmap(android.util.DisplayMetrics?, kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException" - "android.graphics.Bitmap.createScaledBitmap(android.graphics.Bitmap, kotlin.Int, kotlin.Int, kotlin.Boolean):java.lang.IllegalArgumentException" + - "android.graphics.Color.parseColor(kotlin.String?):java.lang.IllegalArgumentException" - "android.graphics.Canvas.constructor(android.graphics.Bitmap):java.lang.IllegalStateException" - "android.graphics.drawable.LayerDrawable.getDrawable(kotlin.Int):java.lang.IndexOutOfBoundsException" - "android.net.ConnectivityManager.registerDefaultNetworkCallback(android.net.ConnectivityManager.NetworkCallback):java.lang.IllegalArgumentException,java.lang.SecurityException" @@ -468,6 +470,7 @@ datadog: # endregion # region Android Graphics - "android.graphics.Bitmap.recycle()" + - "android.graphics.Canvas.drawColor(kotlin.Int)" - "android.graphics.Canvas.drawColor(kotlin.Int, android.graphics.PorterDuff.Mode)" - "android.graphics.Color.argb(kotlin.Int, kotlin.Int, kotlin.Int, kotlin.Int)" - "android.graphics.Color.blue(kotlin.Int)" @@ -484,6 +487,7 @@ datadog: - "android.graphics.drawable.Drawable.setTintList(android.content.res.ColorStateList?)" - "android.graphics.drawable.RippleDrawable.findIndexByLayerId(kotlin.Int)" - "android.graphics.drawable.DrawableContainer.DrawableContainerState.getChild(kotlin.Int)" + - "android.graphics.Paint.constructor()" - "android.graphics.Point.constructor()" - "android.graphics.Point.constructor(kotlin.Int, kotlin.Int)" - "android.graphics.Rect.centerX()" @@ -510,6 +514,11 @@ datadog: - "androidx.compose.runtime.tooling.CompositionGroup.stableId()" - "androidx.compose.ui.graphics.Color(kotlin.Long)" - "androidx.compose.ui.graphics.Color.toArgb()" + - "androidx.compose.ui.graphics.Matrix.constructor(kotlin.FloatArray)" + - "androidx.compose.ui.graphics.Matrix.scale(kotlin.Float, kotlin.Float, kotlin.Float)" + - "androidx.compose.ui.graphics.Matrix.translate(kotlin.Float, kotlin.Float, kotlin.Float)" + - "androidx.compose.ui.graphics.Path.getBounds()" + - "androidx.compose.ui.graphics.Path.transform(androidx.compose.ui.graphics.Matrix)" - "androidx.compose.ui.layout.LayoutCoordinates.positionInWindow()" - "androidx.compose.ui.layout.LayoutInfo.getModifierInfo()" - "androidx.compose.ui.unit.Density(kotlin.Float, kotlin.Float)" diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt new file mode 100644 index 0000000000..0d19c303c2 --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt @@ -0,0 +1,205 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright CHECKBOX_SIZE16-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.mappers.semantics + +import androidx.compose.ui.semantics.SemanticsNode +import androidx.compose.ui.semantics.SemanticsProperties +import androidx.compose.ui.semantics.getOrNull +import androidx.compose.ui.state.ToggleableState +import com.datadog.android.sessionreplay.ImagePrivacy +import com.datadog.android.sessionreplay.compose.internal.data.SemanticsWireframe +import com.datadog.android.sessionreplay.compose.internal.data.UiContext +import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils +import com.datadog.android.sessionreplay.compose.internal.utils.SemanticsUtils +import com.datadog.android.sessionreplay.model.MobileSegment +import com.datadog.android.sessionreplay.utils.AsyncJobStatusCallback +import com.datadog.android.sessionreplay.utils.ColorStringFormatter +import com.datadog.android.sessionreplay.utils.GlobalBounds + +internal class CheckboxSemanticsNodeMapper( + colorStringFormatter: ColorStringFormatter, + private val semanticsUtils: SemanticsUtils = SemanticsUtils(), + private val pathUtils: PathUtils = PathUtils() +) : AbstractSemanticsNodeMapper(colorStringFormatter, semanticsUtils) { + + override fun map( + semanticsNode: SemanticsNode, + parentContext: UiContext, + asyncJobStatusCallback: AsyncJobStatusCallback + ): SemanticsWireframe { + val globalBounds = resolveBounds(semanticsNode) + + val wireframes = if (isCheckboxChecked(semanticsNode)) { + createCheckedWireframes( + parentContext = parentContext, + asyncJobStatusCallback = asyncJobStatusCallback, + semanticsNode = semanticsNode, + globalBounds = globalBounds + ) + } else { + createUncheckedWireframes( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = DEFAULT_COLOR_WHITE + ) + } + + return SemanticsWireframe( + uiContext = null, + wireframes = wireframes + ) + } + + private fun createCheckedWireframes( + parentContext: UiContext, + asyncJobStatusCallback: AsyncJobStatusCallback, + semanticsNode: SemanticsNode, + globalBounds: GlobalBounds + ): List { + val rawFillColor = semanticsUtils.resolveCheckboxFillColor(semanticsNode) + val rawCheckmarkColor = semanticsUtils.resolveCheckmarkColor(semanticsNode) + + val fillColorRgba = rawFillColor?.let { + convertColor(it) + } ?: DEFAULT_COLOR_WHITE + val checkmarkColorRgba = rawCheckmarkColor?.let { + convertColor(it) + } ?: getFallbackCheckmarkColor(DEFAULT_COLOR_WHITE) + + val parsedFillColor = pathUtils.parseColorSafe(fillColorRgba) + val parsedCheckmarkColor = pathUtils.parseColorSafe(checkmarkColorRgba) + + if (parsedFillColor != null && parsedCheckmarkColor != null) { + val checkMarkBitmap = semanticsUtils + .resolveCheckPath(semanticsNode)?.let { checkPath -> + pathUtils.convertPathToBitmap( + checkPath = checkPath, + fillColor = parsedFillColor, + checkmarkColor = parsedCheckmarkColor + ) + } + + if (checkMarkBitmap != null) { + parentContext.imageWireframeHelper.createImageWireframeByBitmap( + id = resolveId(semanticsNode, 0), + globalBounds = globalBounds, + bitmap = checkMarkBitmap, + density = parentContext.density, + isContextualImage = false, + imagePrivacy = ImagePrivacy.MASK_NONE, + asyncJobStatusCallback = asyncJobStatusCallback, + clipping = null, + shapeStyle = MobileSegment.ShapeStyle( + backgroundColor = fillColorRgba, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ), + border = MobileSegment.ShapeBorder( + color = fillColorRgba, + width = BOX_BORDER_WIDTH_DP + ) + )?.let { imageWireframe -> + return listOf(imageWireframe) + } + } + } + + // if we failed to create a wireframe from the path + return createManualCheckedWireframe(semanticsNode, globalBounds, fillColorRgba) + } + + private fun createManualCheckedWireframe( + semanticsNode: SemanticsNode, + globalBounds: GlobalBounds, + backgroundColor: String + ): List { + val strokeColor = getFallbackCheckmarkColor(backgroundColor) + + val background: MobileSegment.Wireframe = createUncheckedWireframes( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = backgroundColor + )[0] + + val checkmarkWidth = globalBounds.width * CHECKMARK_SIZE_FACTOR + val checkmarkHeight = globalBounds.height * CHECKMARK_SIZE_FACTOR + val xPos = globalBounds.x + ((globalBounds.width / 2) - (checkmarkWidth / 2)) + val yPos = globalBounds.y + ((globalBounds.height / 2) - (checkmarkHeight / 2)) + val foreground: MobileSegment.Wireframe = MobileSegment.Wireframe.ShapeWireframe( + id = resolveId(semanticsNode, 1), + x = xPos.toLong(), + y = yPos.toLong(), + width = checkmarkWidth.toLong(), + height = checkmarkHeight.toLong(), + shapeStyle = MobileSegment.ShapeStyle( + backgroundColor = strokeColor, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ), + border = MobileSegment.ShapeBorder( + color = DEFAULT_COLOR_BLACK, + width = BOX_BORDER_WIDTH_DP + ) + ) + return listOf(background, foreground) + } + + private fun createUncheckedWireframes( + semanticsNode: SemanticsNode, + globalBounds: GlobalBounds, + backgroundColor: String + ): List { + val borderColor = + semanticsUtils.resolveBorderColor(semanticsNode) + ?.let { rawColor -> + convertColor(rawColor) + } ?: DEFAULT_COLOR_BLACK + + return listOf( + MobileSegment.Wireframe.ShapeWireframe( + id = resolveId(semanticsNode, 0), + x = globalBounds.x, + y = globalBounds.y, + width = globalBounds.width, + height = globalBounds.height, + shapeStyle = MobileSegment.ShapeStyle( + backgroundColor = backgroundColor, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ), + border = MobileSegment.ShapeBorder( + color = borderColor, + width = BOX_BORDER_WIDTH_DP + ) + ) + ) + } + + private fun isCheckboxChecked(semanticsNode: SemanticsNode): Boolean = + semanticsNode.config.getOrNull(SemanticsProperties.ToggleableState) == ToggleableState.On + + private fun getFallbackCheckmarkColor(backgroundColor: String?) = + if (backgroundColor == DEFAULT_COLOR_WHITE) { + DEFAULT_COLOR_BLACK + } else { + DEFAULT_COLOR_WHITE + } + + internal companion object { + internal const val DEFAULT_COLOR_BLACK = "#000000FF" + internal const val DEFAULT_COLOR_WHITE = "#FFFFFFFF" + + // when we create the checkmark manually, what % of the checkbox size should it be + internal const val CHECKMARK_SIZE_FACTOR = 0.5 + + // values from Compose Checkbox sourcecode + internal const val BOX_BORDER_WIDTH_DP = 2L + internal const val STROKE_WIDTH_DP = 2f + internal const val CHECKBOX_SIZE_DP = 20 + internal const val CHECKBOX_CORNER_RADIUS = 2f + } +} diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapper.kt index d1157f9dc7..8aa368013d 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapper.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapper.kt @@ -31,7 +31,8 @@ internal class RootSemanticsNodeMapper( Role.RadioButton to RadioButtonSemanticsNodeMapper(colorStringFormatter, semanticsUtils), Role.Tab to TabSemanticsNodeMapper(colorStringFormatter, semanticsUtils), Role.Button to ButtonSemanticsNodeMapper(colorStringFormatter, semanticsUtils), - Role.Image to ImageSemanticsNodeMapper(colorStringFormatter, semanticsUtils) + Role.Image to ImageSemanticsNodeMapper(colorStringFormatter, semanticsUtils), + Role.Checkbox to CheckboxSemanticsNodeMapper(colorStringFormatter, semanticsUtils) ), // Text doesn't have a role in semantics, so it should be a fallback mapper. private val textSemanticsNodeMapper: TextSemanticsNodeMapper = TextSemanticsNodeMapper( diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/reflection/ComposeReflection.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/reflection/ComposeReflection.kt index e0feb83111..652d1d36da 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/reflection/ComposeReflection.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/reflection/ComposeReflection.kt @@ -40,6 +40,16 @@ internal object ComposeReflection { val ColorField = BackgroundElementClass?.getDeclaredFieldSafe("color") val ShapeField = BackgroundElementClass?.getDeclaredFieldSafe("shape") + val CheckDrawingCacheClass = getClassSafe("androidx.compose.material.CheckDrawingCache") + val CheckboxKtClass = getClassSafe("androidx.compose.material.CheckboxKt\$CheckboxImpl\$1\$1") + val DrawBehindElementClass = getClassSafe("androidx.compose.ui.draw.DrawBehindElement") + val BorderColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$borderColor\$delegate") + val BoxColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$boxColor\$delegate") + val CheckCacheField = CheckboxKtClass?.getDeclaredFieldSafe("\$checkCache") + val CheckColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$checkColor\$delegate") + val CheckPathField = CheckDrawingCacheClass?.getDeclaredFieldSafe("checkPath") + val OnDrawField = DrawBehindElementClass?.getDeclaredFieldSafe("onDraw") + val PaddingElementClass = getClassSafe("androidx.compose.foundation.layout.PaddingElement") val StartField = PaddingElementClass?.getDeclaredFieldSafe("start") val EndField = PaddingElementClass?.getDeclaredFieldSafe("end") diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt new file mode 100644 index 0000000000..739a6996d4 --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt @@ -0,0 +1,186 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.utils + +import android.graphics.Bitmap +import android.graphics.Canvas +import android.graphics.Color +import android.graphics.Paint +import androidx.compose.ui.graphics.Matrix +import androidx.compose.ui.graphics.Path +import androidx.compose.ui.graphics.asAndroidPath +import com.datadog.android.api.InternalLogger +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_SIZE_DP +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.STROKE_WIDTH_DP +import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper +import java.util.Locale + +internal class PathUtils( + private val logger: InternalLogger = InternalLogger.UNBOUND, + private val canvasWrapper: CanvasWrapper = CanvasWrapper(logger), + private val bitmapWrapper: BitmapWrapper = BitmapWrapper() +) { + @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException + internal fun parseColorSafe(color: String): Int? { + return try { + Color.parseColor(color) + } catch (e: IllegalArgumentException) { + logger.log( + target = InternalLogger.Target.MAINTAINER, + level = InternalLogger.Level.WARN, + messageBuilder = { COLOR_PARSE_ERROR.format(Locale.US, color) }, + throwable = e + ) + null + } + } + + internal fun convertRgbaToArgb(rgbaString: String): String { + if (rgbaString.length < 2) return rgbaString + + // for takeLast: n > 0 + @Suppress("UnsafeThirdPartyFunctionCall") + val alphaValue = rgbaString.takeLast(2) + + // for substring: length is necessarily > 1 at this point + // for dropLast: n > 0 + @Suppress("UnsafeThirdPartyFunctionCall") + val rgbColor = rgbaString + .substring(1) + .dropLast(2) + return "#$alphaValue$rgbColor" + } + + @Suppress("UnsafeThirdPartyFunctionCall") // handling UnsupportedOperationException + internal fun asAndroidPathSafe(path: Path): android.graphics.Path? { + return try { + path.asAndroidPath() + } catch (e: UnsupportedOperationException) { + logger.log( + target = InternalLogger.Target.MAINTAINER, + level = InternalLogger.Level.WARN, + messageBuilder = { PATH_CONVERSION_ERROR }, + throwable = e + ) + null + } + } + + internal fun convertPathToBitmap( + checkPath: Path, + fillColor: Int, + checkmarkColor: Int + ): Bitmap? { + val scaledPath = scalePathToBitmapSize(checkPath) + val mutableBitmap = + bitmapWrapper.createBitmap(CHECKBOX_SIZE_DP, CHECKBOX_SIZE_DP, Bitmap.Config.ARGB_8888) + ?: return null + + return drawPathOntoBitmap( + mutableBitmap, + scaledPath, + fillColor, + checkmarkColor + ) + } + + private fun drawPathToBitmap(checkmarkColor: Int, path: Path, canvas: Canvas?) { + val paint = Paint().apply { + color = checkmarkColor + style = Paint.Style.STROKE + strokeWidth = STROKE_WIDTH_DP + isAntiAlias = true + } + + // Draw the Path onto the Canvas + asAndroidPathSafe(path)?.let { + drawPathSafe(canvas, it, paint) + } + } + + private fun scalePathToBitmapSize(path: Path): Path { + // path initial bounds + val originalBounds = path.getBounds() + + // calculate the scale factor + val scaleX = CHECKBOX_SIZE_DP / originalBounds.width + val scaleY = CHECKBOX_SIZE_DP / originalBounds.height + val scaleFactor = minOf(scaleX, scaleY) + + // current center + val currentCenterX = (originalBounds.left + originalBounds.right) / 2 + val currentCenterY = (originalBounds.top + originalBounds.bottom) / 2 + + // new center + val newCenterX = CHECKBOX_SIZE_DP / 2 + val newCenterY = CHECKBOX_SIZE_DP / 2 + + // center changes after scaling + val scaledCenterX = currentCenterX * scaleFactor + val scaledCenterY = currentCenterY * scaleFactor + + // translation needed to recenter + val translateX = newCenterX - scaledCenterX + val translateY = newCenterY - scaledCenterY + + // the order of operations is important + val matrix = Matrix() + matrix.translate(translateX, translateY) + matrix.scale(scaleFactor, scaleFactor) + path.transform(matrix) + + return path + } + + private fun drawPathOntoBitmap( + bitmap: Bitmap, + scaledPath: Path, + fillColor: Int, + checkmarkColor: Int + ): Bitmap? { + val canvas = canvasWrapper.createCanvas(bitmap) ?: return null + drawCanvasBackground(canvas, fillColor) + drawCanvasForeground(canvas, scaledPath, checkmarkColor) + return bitmap + } + + private fun drawCanvasBackground( + canvas: Canvas, + fillColor: Int + ) { + canvas.drawColor(fillColor) + } + + private fun drawCanvasForeground( + canvas: Canvas, + path: Path, + checkmarkColor: Int + ) { + drawPathToBitmap(checkmarkColor, path, canvas) + } + + @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException + private fun drawPathSafe(canvas: Canvas?, path: android.graphics.Path, paint: Paint) { + try { + canvas?.drawPath(path, paint) + } catch (e: IllegalArgumentException) { + logger.log( + target = InternalLogger.Target.MAINTAINER, + level = InternalLogger.Level.WARN, + messageBuilder = { PATH_DRAW_ERROR }, + throwable = e + ) + } + } + + internal companion object { + internal const val COLOR_PARSE_ERROR = "Failed to parse color: %s" + internal const val PATH_CONVERSION_ERROR = "Failed to convert Compose Path to Android Path" + internal const val PATH_DRAW_ERROR = "Failed to draw Path to Canvas" + } +} diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt index 239cd57dc3..509be5fa8d 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt @@ -8,10 +8,12 @@ package com.datadog.android.sessionreplay.compose.internal.utils import android.graphics.Bitmap import android.view.View +import androidx.compose.animation.core.AnimationState import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Size import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.Path import androidx.compose.ui.graphics.Shape import androidx.compose.ui.graphics.painter.BitmapPainter import androidx.compose.ui.graphics.vector.VectorPainter @@ -30,6 +32,8 @@ import com.datadog.android.sessionreplay.compose.TextInputSemanticsPropertyKey import com.datadog.android.sessionreplay.compose.TouchSemanticsPropertyKey import com.datadog.android.sessionreplay.compose.internal.data.BitmapInfo import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.TextLayoutInfo +import com.datadog.android.sessionreplay.compose.internal.reflection.ComposeReflection +import com.datadog.android.sessionreplay.compose.internal.reflection.getSafe import com.datadog.android.sessionreplay.utils.GlobalBounds @Suppress("TooManyFunctions") @@ -114,6 +118,31 @@ internal class SemanticsUtils(private val reflectionUtils: ReflectionUtils = Ref return backgroundModifier?.let { reflectionUtils.getShape(it) } } + internal fun resolveCheckPath(semanticsNode: SemanticsNode): Path? = + resolveOnDrawInstance(semanticsNode)?.let { onDraw -> + ComposeReflection.CheckCacheField?.getSafe(onDraw)?.let { checkCache -> + ComposeReflection.CheckPathField?.getSafe(checkCache) as? Path + } + } + + internal fun resolveCheckboxFillColor(semanticsNode: SemanticsNode): Long? = + resolveReflectedProperty( + semanticsNode, + CheckmarkFieldType.FILL_COLOR + ) + + internal fun resolveCheckmarkColor(semanticsNode: SemanticsNode): Long? = + resolveReflectedProperty( + semanticsNode, + CheckmarkFieldType.CHECKMARK_COLOR + ) + + internal fun resolveBorderColor(semanticsNode: SemanticsNode): Long? = + resolveReflectedProperty( + semanticsNode, + CheckmarkFieldType.BORDER_COLOR + ) + private fun shrinkInnerBounds( modifier: Modifier, currentBounds: GlobalBounds @@ -277,4 +306,45 @@ internal class SemanticsUtils(private val reflectionUtils: ReflectionUtils = Ref internal fun getInteropView(semanticsNode: SemanticsNode): View? { return reflectionUtils.getInteropView(semanticsNode) } + + private fun resolveOnDrawInstance(semanticsNode: SemanticsNode): Any? { + val drawBehindElement = + semanticsNode.layoutInfo.getModifierInfo().firstOrNull { modifierInfo -> + ComposeReflection.DrawBehindElementClass?.isInstance(modifierInfo.modifier) == true + }?.modifier + + return drawBehindElement?.let { + ComposeReflection.OnDrawField?.getSafe(drawBehindElement) + } + } + + private fun resolveReflectedProperty(semanticsNode: SemanticsNode, fieldType: CheckmarkFieldType): Long? { + val onDrawInstance = resolveOnDrawInstance(semanticsNode) + + val checkmarkColor: AnimationState<*, *>? = onDrawInstance?.let { + when (fieldType) { + CheckmarkFieldType.FILL_COLOR -> { + ComposeReflection.BoxColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + CheckmarkFieldType.CHECKMARK_COLOR -> { + ComposeReflection.CheckColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + CheckmarkFieldType.BORDER_COLOR -> { + ComposeReflection.BorderColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + } + } + + val result = (checkmarkColor?.value as? Color)?.value + + return result?.toLong() + } + + internal companion object { + internal enum class CheckmarkFieldType { + FILL_COLOR, + CHECKMARK_COLOR, + BORDER_COLOR + } + } } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt new file mode 100644 index 0000000000..a623bd56ba --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt @@ -0,0 +1,475 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.mappers.semantics + +import android.graphics.Bitmap +import androidx.compose.ui.graphics.Path +import androidx.compose.ui.semantics.SemanticsConfiguration +import androidx.compose.ui.semantics.SemanticsNode +import androidx.compose.ui.semantics.SemanticsProperties +import androidx.compose.ui.semantics.getOrNull +import androidx.compose.ui.state.ToggleableState +import com.datadog.android.sessionreplay.ImagePrivacy +import com.datadog.android.sessionreplay.compose.internal.data.UiContext +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.BOX_BORDER_WIDTH_DP +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_CORNER_RADIUS +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.DEFAULT_COLOR_BLACK +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.DEFAULT_COLOR_WHITE +import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils +import com.datadog.android.sessionreplay.compose.test.elmyr.SessionReplayComposeForgeConfigurator +import com.datadog.android.sessionreplay.model.MobileSegment +import com.datadog.android.sessionreplay.utils.AsyncJobStatusCallback +import com.datadog.android.sessionreplay.utils.ImageWireframeHelper +import fr.xgouchet.elmyr.Forge +import fr.xgouchet.elmyr.annotation.LongForgery +import fr.xgouchet.elmyr.annotation.StringForgery +import fr.xgouchet.elmyr.junit5.ForgeConfiguration +import fr.xgouchet.elmyr.junit5.ForgeExtension +import org.assertj.core.api.Assertions.assertThat +import org.junit.jupiter.api.BeforeEach +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.extension.ExtendWith +import org.junit.jupiter.api.extension.Extensions +import org.mockito.Mock +import org.mockito.Mockito.mock +import org.mockito.junit.jupiter.MockitoExtension +import org.mockito.junit.jupiter.MockitoSettings +import org.mockito.kotlin.any +import org.mockito.kotlin.anyOrNull +import org.mockito.kotlin.doReturn +import org.mockito.kotlin.eq +import org.mockito.kotlin.verify +import org.mockito.kotlin.whenever +import org.mockito.quality.Strictness + +@Extensions( + ExtendWith(MockitoExtension::class), + ExtendWith(ForgeExtension::class) +) +@MockitoSettings(strictness = Strictness.LENIENT) +@ForgeConfiguration(SessionReplayComposeForgeConfigurator::class) +internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest() { + private lateinit var testedMapper: CheckboxSemanticsNodeMapper + + @Mock + private lateinit var mockSemanticsNode: SemanticsNode + + @Mock + lateinit var mockConfig: SemanticsConfiguration + + @Mock + private lateinit var mockAsyncJobStatusCallback: AsyncJobStatusCallback + + @LongForgery(min = 0xffffffff) + var fakeBorderColor: Long = 0L + + @LongForgery(min = 0xffffffff) + var fakeFillColor: Long = 0L + + @LongForgery(min = 0xffffffff) + var fakeCheckmarkColor: Long = 0L + + @StringForgery(regex = "#[0-9A-F]{8}") + lateinit var fakeBorderColorHexString: String + + @StringForgery(regex = "#[0-9A-F]{8}") + lateinit var fakeFillColorHexString: String + + @StringForgery(regex = "#[0-9A-F]{8}") + lateinit var fakeCheckmarkColorHexString: String + + @Mock + private lateinit var mockUiContext: UiContext + + @Mock + private lateinit var mockPath: Path + + @Mock + private lateinit var mockBitmap: Bitmap + + @Mock + private lateinit var mockPathUtils: PathUtils + + @Mock + private lateinit var mockImageWireframeHelper: ImageWireframeHelper + + @BeforeEach + override fun `set up`(forge: Forge) { + super.`set up`(forge) + + whenever(mockUiContext.imageWireframeHelper) + .thenReturn(mockImageWireframeHelper) + + whenever(mockUiContext.density) + .thenReturn(fakeDensity) + + whenever(mockSemanticsUtils.resolveInnerBounds(mockSemanticsNode)) doReturn rectToBounds( + fakeBounds, + fakeDensity + ) + + mockSemanticsNodeWithBound { + whenever(mockSemanticsNode.layoutInfo).doReturn(mockLayoutInfo) + whenever(mockSemanticsNode.config).doReturn(mockConfig) + } + + whenever(mockPath.getBounds()) + .thenReturn(fakeBounds) + + whenever(mockSemanticsUtils.resolveCheckmarkColor(mockSemanticsNode)) + .thenReturn(fakeCheckmarkColor) + + whenever(mockSemanticsUtils.resolveBorderColor(mockSemanticsNode)) + .thenReturn(fakeBorderColor) + + mockColorStringFormatter(fakeBorderColor, fakeBorderColorHexString) + mockColorStringFormatter(fakeFillColor, fakeFillColorHexString) + mockColorStringFormatter(fakeCheckmarkColor, fakeCheckmarkColorHexString) + + testedMapper = CheckboxSemanticsNodeMapper( + colorStringFormatter = mockColorStringFormatter, + semanticsUtils = mockSemanticsUtils, + pathUtils = mockPathUtils + ) + } + + @Test + fun `M return the unchecked wireframe W map { unchecked, reflection resolution successful }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.Off) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val actualWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + + val expectedShapeBorder = MobileSegment.ShapeBorder( + color = fakeBorderColorHexString, + width = BOX_BORDER_WIDTH_DP + ) + + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_WHITE, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(actualWireframe?.border).isEqualTo(expectedShapeBorder) + assertThat(actualWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M return the unchecked wireframe W map { unchecked, reflection resolution failure }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.Off) + + whenever(mockSemanticsUtils.resolveBorderColor(mockSemanticsNode)) + .thenReturn(null) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val actualWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + + val expectedShapeBorder = MobileSegment.ShapeBorder( + color = DEFAULT_COLOR_BLACK, + width = BOX_BORDER_WIDTH_DP + ) + + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_WHITE, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(actualWireframe?.border).isEqualTo(expectedShapeBorder) + assertThat(actualWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M return fallback W map { checked, reflection resolution failure, got fill color }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(null) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val backgroundWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + + val expectedBgShapeBorder = MobileSegment.ShapeBorder( + color = fakeBorderColorHexString, + width = BOX_BORDER_WIDTH_DP + ) + + val expectedBgShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_WHITE, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(backgroundWireframe?.border).isEqualTo(expectedBgShapeBorder) + assertThat(backgroundWireframe?.shapeStyle).isEqualTo(expectedBgShapeStyle) + } + + @Test + fun `M return the correct fallback fg wireframe W map { checked, reflection resolution failure }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(null) + + whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) + .thenReturn(fakeFillColor) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_WHITE, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M return fallback fg W map { checked, resolution fail, no fill }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) + .thenReturn(null) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(null) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_BLACK, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M return fallback W map { checked, resolution fail, image wireframe creation fail }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(mockPath) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_BLACK, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M return fallback W map { checked, resolution fail, bmp creation from path fail }`() { + // Given + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(mockPath) + + // When + val semanticsWireframe = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = DEFAULT_COLOR_BLACK, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + // Then + assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + } + + @Test + fun `M call image wireframe creation W map { checked, reflection resolution success }`() { + // Given + whenever(mockSemanticsUtils.resolveInnerBounds(mockSemanticsNode)) doReturn fakeGlobalBounds + + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockPathUtils.asAndroidPathSafe(any())).thenReturn(mock()) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(mockPath) + + whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) + .thenReturn(fakeFillColor) + + whenever( + mockPathUtils.convertPathToBitmap( + checkPath = eq(mockPath), + fillColor = anyOrNull(), + checkmarkColor = anyOrNull() + ) + ) doReturn mockBitmap + + whenever( + mockUiContext.imageWireframeHelper.createImageWireframeByBitmap( + id = any(), + globalBounds = eq(fakeGlobalBounds), + bitmap = eq(mockBitmap), + density = eq(fakeDensity), + isContextualImage = eq(false), + imagePrivacy = eq(ImagePrivacy.MASK_NONE), + asyncJobStatusCallback = eq(mockAsyncJobStatusCallback), + clipping = eq(null), + shapeStyle = eq(null), + border = eq(null) + ) + ).thenReturn(mock()) + + // When + testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + val expectedShapeStyle = MobileSegment.ShapeStyle( + backgroundColor = fakeFillColorHexString, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ) + + val expectedBorder = MobileSegment.ShapeBorder( + color = fakeFillColorHexString, + width = BOX_BORDER_WIDTH_DP + ) + + // Then + verify(mockUiContext.imageWireframeHelper).createImageWireframeByBitmap( + id = any(), + globalBounds = eq(fakeGlobalBounds), + bitmap = eq(mockBitmap), + density = eq(fakeDensity), + isContextualImage = eq(false), + imagePrivacy = eq(ImagePrivacy.MASK_NONE), + asyncJobStatusCallback = eq(mockAsyncJobStatusCallback), + clipping = eq(null), + shapeStyle = eq(expectedShapeStyle), + border = eq(expectedBorder) + ) + } + + @Test + fun `M return image wireframe W map { checked, reflection resolution success }`() { + // Given + whenever(mockSemanticsUtils.resolveInnerBounds(mockSemanticsNode)) doReturn fakeGlobalBounds + + whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) + .thenReturn(ToggleableState.On) + + whenever(mockPathUtils.asAndroidPathSafe(any())).thenReturn(mock()) + + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(mockPath) + + whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) + .thenReturn(fakeFillColor) + + whenever( + mockPathUtils.convertPathToBitmap( + checkPath = eq(mockPath), + fillColor = anyOrNull(), + checkmarkColor = anyOrNull() + ) + ).thenReturn(mockBitmap) + + whenever( + mockUiContext.imageWireframeHelper.createImageWireframeByBitmap( + id = any(), + globalBounds = any(), + bitmap = any(), + density = any(), + isContextualImage = any(), + imagePrivacy = any(), + asyncJobStatusCallback = any(), + clipping = anyOrNull(), + shapeStyle = anyOrNull(), + border = anyOrNull() + ) + ).thenReturn(mock()) + + // When + val wireframes = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + // Then + assertThat(wireframes.wireframes).hasSize(1) + assertThat(wireframes.wireframes[0]).isInstanceOf(MobileSegment.Wireframe.ImageWireframe::class.java) + } +} diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt new file mode 100644 index 0000000000..b840fc1f30 --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt @@ -0,0 +1,143 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.utils + +import android.graphics.Bitmap +import android.graphics.Canvas +import androidx.compose.ui.geometry.Rect +import androidx.compose.ui.graphics.Path +import com.datadog.android.api.InternalLogger +import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper +import fr.xgouchet.elmyr.annotation.LongForgery +import fr.xgouchet.elmyr.junit5.ForgeExtension +import org.assertj.core.api.Assertions.assertThat +import org.junit.jupiter.api.BeforeEach +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.extension.ExtendWith +import org.junit.jupiter.api.extension.Extensions +import org.mockito.Mock +import org.mockito.junit.jupiter.MockitoExtension +import org.mockito.junit.jupiter.MockitoSettings +import org.mockito.kotlin.any +import org.mockito.kotlin.anyOrNull +import org.mockito.kotlin.whenever +import org.mockito.quality.Strictness + +@Extensions( + ExtendWith(MockitoExtension::class), + ExtendWith(ForgeExtension::class) +) +@MockitoSettings(strictness = Strictness.LENIENT) +internal class PathUtilsTest { + private lateinit var testedUtils: PathUtils + + @Mock + private lateinit var mockLogger: InternalLogger + + @Mock + private lateinit var mockCanvasWrapper: CanvasWrapper + + @Mock + private lateinit var mockBitmapWrapper: BitmapWrapper + + @Mock + private lateinit var mockPath: Path + + @Mock + private lateinit var mockBounds: Rect + + @Mock + private lateinit var mockBitmap: Bitmap + + @Mock + private lateinit var mockCanvas: Canvas + + @LongForgery(min = 0xffffffff) + var fakeFillColor: Long = 0L + + @LongForgery(min = 0xffffffff) + var fakeCheckmarkColor: Long = 0L + + @BeforeEach + fun `set up`() { + whenever(mockPath.getBounds()) + .thenReturn(mockBounds) + + testedUtils = PathUtils( + logger = mockLogger, + canvasWrapper = mockCanvasWrapper, + bitmapWrapper = mockBitmapWrapper + ) + } + + @Test + fun `M return input value W convertRgbaToArgb() { length lt 2 }`() { + // Then + assertThat(testedUtils.convertRgbaToArgb("#")).isEqualTo("#") + } + + @Test + fun `M move alpha value W convertRgbaToArgb()`() { + // Then + assertThat(testedUtils.convertRgbaToArgb("#000000FF")).isEqualTo("#FF000000") + } + + @Test + fun `M return null W convertPathToBitmap() { failed to create bitmap }`() { + // Given + whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) + .thenReturn(null) + + // When + val result = testedUtils.convertPathToBitmap( + checkPath = mockPath, + fillColor = fakeFillColor.toInt(), + checkmarkColor = fakeCheckmarkColor.toInt() + ) + + // Then + assertThat(result).isNull() + } + + @Test + fun `M return null W convertPathToBitmap() { failed to create canvas }`() { + // Given + whenever(mockCanvasWrapper.createCanvas(any())) + .thenReturn(null) + + // When + val result = testedUtils.convertPathToBitmap( + checkPath = mockPath, + fillColor = fakeFillColor.toInt(), + checkmarkColor = fakeCheckmarkColor.toInt() + ) + + // Then + assertThat(result).isNull() + } + + @Test + fun `M return bitmap W convertPathToBitmap() { success }`() { + // Given + whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) + .thenReturn(mockBitmap) + + whenever(mockCanvasWrapper.createCanvas(any())) + .thenReturn(mockCanvas) + + // When + val result = testedUtils.convertPathToBitmap( + checkPath = mockPath, + fillColor = fakeFillColor.toInt(), + checkmarkColor = fakeCheckmarkColor.toInt() + ) + + // Then + assertThat(result).isEqualTo(mockBitmap) + } +} diff --git a/features/dd-sdk-android-session-replay/api/apiSurface b/features/dd-sdk-android-session-replay/api/apiSurface index be470a030f..b85ba8dcd4 100644 --- a/features/dd-sdk-android-session-replay/api/apiSurface +++ b/features/dd-sdk-android-session-replay/api/apiSurface @@ -102,6 +102,12 @@ class com.datadog.android.sessionreplay.recorder.resources.DefaultDrawableCopier override fun copy(android.graphics.drawable.Drawable, android.content.res.Resources): android.graphics.drawable.Drawable? interface com.datadog.android.sessionreplay.recorder.resources.DrawableCopier fun copy(android.graphics.drawable.Drawable, android.content.res.Resources): android.graphics.drawable.Drawable? +class com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper + constructor(com.datadog.android.api.InternalLogger = InternalLogger.UNBOUND) + fun createBitmap(Int, Int, android.graphics.Bitmap.Config, android.util.DisplayMetrics? = null): android.graphics.Bitmap? +class com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper + constructor(com.datadog.android.api.InternalLogger) + fun createCanvas(android.graphics.Bitmap): android.graphics.Canvas? open class com.datadog.android.sessionreplay.utils.AndroidMDrawableToColorMapper : LegacyDrawableToColorMapper constructor(List = emptyList()) override fun resolveRippleDrawable(android.graphics.drawable.RippleDrawable, com.datadog.android.api.InternalLogger): Int? diff --git a/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api b/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api index 4884b056b9..67e54151c5 100644 --- a/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api +++ b/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api @@ -1520,6 +1520,19 @@ public abstract interface class com/datadog/android/sessionreplay/recorder/resou public abstract fun copy (Landroid/graphics/drawable/Drawable;Landroid/content/res/Resources;)Landroid/graphics/drawable/Drawable; } +public final class com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper { + public fun ()V + public fun (Lcom/datadog/android/api/InternalLogger;)V + public synthetic fun (Lcom/datadog/android/api/InternalLogger;ILkotlin/jvm/internal/DefaultConstructorMarker;)V + public final fun createBitmap (IILandroid/graphics/Bitmap$Config;Landroid/util/DisplayMetrics;)Landroid/graphics/Bitmap; + public static synthetic fun createBitmap$default (Lcom/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper;IILandroid/graphics/Bitmap$Config;Landroid/util/DisplayMetrics;ILjava/lang/Object;)Landroid/graphics/Bitmap; +} + +public final class com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper { + public fun (Lcom/datadog/android/api/InternalLogger;)V + public final fun createCanvas (Landroid/graphics/Bitmap;)Landroid/graphics/Canvas; +} + public class com/datadog/android/sessionreplay/utils/AndroidMDrawableToColorMapper : com/datadog/android/sessionreplay/utils/LegacyDrawableToColorMapper { public fun ()V public fun (Ljava/util/List;)V diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtils.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtils.kt index 1e6dadf165..501f5fe923 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtils.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtils.kt @@ -18,8 +18,8 @@ import com.datadog.android.api.InternalLogger import com.datadog.android.core.internal.utils.submitSafe import com.datadog.android.sessionreplay.internal.recorder.resources.BitmapCachesManager import com.datadog.android.sessionreplay.internal.recorder.resources.ResourceResolver -import com.datadog.android.sessionreplay.internal.recorder.wrappers.BitmapWrapper -import com.datadog.android.sessionreplay.internal.recorder.wrappers.CanvasWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper import java.util.concurrent.ExecutorService import kotlin.math.sqrt @@ -177,7 +177,12 @@ internal class DrawableUtils( config: Config ): Bitmap? = bitmapCachesManager.getBitmapByProperties(width, height, config) - ?: bitmapWrapper.createBitmap(displayMetrics, width, height, config) + ?: bitmapWrapper.createBitmap( + bitmapWidth = width, + bitmapHeight = height, + config = config, + displayMetrics = displayMetrics + ) internal companion object { @VisibleForTesting diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/BitmapWrapper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt similarity index 75% rename from features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/BitmapWrapper.kt rename to features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt index a14b1f1d6e..687c0bf716 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/BitmapWrapper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt @@ -4,24 +4,39 @@ * Copyright 2016-Present Datadog, Inc. */ -package com.datadog.android.sessionreplay.internal.recorder.wrappers +package com.datadog.android.sessionreplay.recorder.wrappers import android.graphics.Bitmap import android.graphics.Bitmap.Config import android.util.DisplayMetrics import com.datadog.android.api.InternalLogger -internal class BitmapWrapper( +/** + * Wraps the Bitmap class to catch potential crashes. + */ +class BitmapWrapper( private val logger: InternalLogger = InternalLogger.UNBOUND ) { - internal fun createBitmap( - displayMetrics: DisplayMetrics, + /** + * Creates a bitmap with the given parameters. + * @param bitmapWidth the width of the bitmap. + * @param bitmapHeight the height of the bitmap. + * @param config the config of the bitmap. + * @param displayMetrics the optional display metrics to use. + * @return the created bitmap or null if it failed. + */ + fun createBitmap( bitmapWidth: Int, bitmapHeight: Int, - config: Config + config: Config, + displayMetrics: DisplayMetrics? = null ): Bitmap? { return try { - Bitmap.createBitmap(displayMetrics, bitmapWidth, bitmapHeight, config) + if (displayMetrics != null) { + Bitmap.createBitmap(displayMetrics, bitmapWidth, bitmapHeight, config) + } else { + Bitmap.createBitmap(bitmapWidth, bitmapHeight, config) + } } catch (e: IllegalArgumentException) { // should never happen since config is given as valid type and width/height are // normalized to be at least 1 diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt similarity index 80% rename from features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapper.kt rename to features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt index 6f2f5a0e05..3bd929e619 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt @@ -4,17 +4,24 @@ * Copyright 2016-Present Datadog, Inc. */ -package com.datadog.android.sessionreplay.internal.recorder.wrappers +package com.datadog.android.sessionreplay.recorder.wrappers import android.graphics.Bitmap import android.graphics.Canvas import com.datadog.android.api.InternalLogger -internal class CanvasWrapper( - private val logger: InternalLogger = InternalLogger.UNBOUND +/** + * Wraps the Canvas class to catch potential crashes. + */ +class CanvasWrapper( + private val logger: InternalLogger ) { - - internal fun createCanvas(bitmap: Bitmap): Canvas? { + /** + * Creates a canvas with the given bitmap. + * @param bitmap the bitmap to use. + * @return the created canvas or null if it failed. + */ + fun createCanvas(bitmap: Bitmap): Canvas? { if (bitmap.isRecycled || !bitmap.isMutable) { logger.log( level = InternalLogger.Level.ERROR, diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt index aa57de1575..b0b8428e35 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt @@ -173,6 +173,15 @@ internal class DefaultImageWireframeHelperTest { ) ).thenReturn(fakeGeneratedIdentifier) + whenever( + mockViewUtilsInternal.resolveCompoundDrawableBounds( + view = any(), + drawable = any(), + pixelsDensity = any(), + position = any() + ) + ).thenReturn(fakeBounds) + testedHelper = DefaultImageWireframeHelper( logger = mockLogger, resourceResolver = mockResourceResolver, @@ -514,7 +523,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M send telemetry W createImageWireframeByDrawable() { application context is null }`() { + fun `M send telemetry W createImageWireframeByDrawable { application context is null }`() { // Given whenever(mockView.context.applicationContext).thenReturn(null) @@ -544,7 +553,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M log error W createImageWireframeByDrawable() { resources is null }`() { + fun `M log error W createImageWireframeByDrawable { resources is null }`() { // Given whenever(mockView.resources).thenReturn(null) @@ -574,7 +583,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M return null W createImageWireframeByDrawable() { id is null }`() { + fun `M return null W createImageWireframeByDrawable { id is null }`() { // Given whenever(mockViewIdentifierResolver.resolveChildUniqueIdentifier(any(), any())) .thenReturn(null) @@ -602,7 +611,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M return null W createImageWireframeByDrawable() { drawable has no width }`() { + fun `M return null W createImageWireframeByDrawable { drawable has no width }`() { // When val wireframe = testedHelper.createImageWireframeByDrawable( view = mockView, @@ -625,7 +634,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M return null W createImageWireframeByDrawable() { drawable has no height }`() { + fun `M return null W createImageWireframeByDrawable { drawable has no height }`() { // When val wireframe = testedHelper.createImageWireframeByDrawable( view = mockView, @@ -648,7 +657,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M return wireframe W createImageWireframeByDrawable()`( + fun `M return wireframe W createImageWireframeByDrawable`( @Mock mockShapeStyle: MobileSegment.ShapeStyle, @Mock mockBorder: MobileSegment.ShapeBorder, @Mock stubWireframeClip: MobileSegment.WireframeClip @@ -985,7 +994,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M resolve view width and height W createImageWireframe() { RippleDrawable }`( + fun `M resolve view width and height W createImageWireframeByDrawable() { RippleDrawable }`( @Mock mockDrawable: RippleDrawable, @Mock mockInsetDrawable: InsetDrawable, @Mock mockGradientDrawable: GradientDrawable, @@ -1033,7 +1042,7 @@ internal class DefaultImageWireframeHelperTest { } @Test - fun `M resolve drawable width and height W createImageWireframe() { TextView }`() { + fun `M resolve drawable width and height W createImageWireframeByDrawable { TextView }`() { // When testedHelper.createImageWireframeByDrawable( view = mockView, diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtilsTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtilsTest.kt index 60b419f7da..2c20ca02cc 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtilsTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/DrawableUtilsTest.kt @@ -16,8 +16,8 @@ import com.datadog.android.api.InternalLogger import com.datadog.android.sessionreplay.forge.ForgeConfigurator import com.datadog.android.sessionreplay.internal.recorder.resources.BitmapCachesManager import com.datadog.android.sessionreplay.internal.recorder.resources.ResourceResolver -import com.datadog.android.sessionreplay.internal.recorder.wrappers.BitmapWrapper -import com.datadog.android.sessionreplay.internal.recorder.wrappers.CanvasWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper import fr.xgouchet.elmyr.annotation.IntForgery import fr.xgouchet.elmyr.junit5.ForgeConfiguration import fr.xgouchet.elmyr.junit5.ForgeExtension @@ -30,6 +30,7 @@ import org.mockito.Mock import org.mockito.junit.jupiter.MockitoExtension import org.mockito.junit.jupiter.MockitoSettings import org.mockito.kotlin.any +import org.mockito.kotlin.anyOrNull import org.mockito.kotlin.argumentCaptor import org.mockito.kotlin.doAnswer import org.mockito.kotlin.mock @@ -103,7 +104,7 @@ internal class DrawableUtilsTest { whenever(mockDrawable.constantState).thenReturn(mockConstantState) whenever(mockCurrentDrawable.constantState).thenReturn(mockConstantState) whenever(mockDrawable.current).thenReturn(mockCurrentDrawable) - whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), any())) + whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) .thenReturn(mockBitmap) whenever(mockCanvasWrapper.createCanvas(any())) .thenReturn(mockCanvas) @@ -151,10 +152,10 @@ internal class DrawableUtilsTest { // Then verify(mockBitmapWrapper).createBitmap( - displayMetrics = displayMetricsCaptor.capture(), bitmapWidth = argumentCaptor.capture(), bitmapHeight = argumentCaptor.capture(), - config = any() + config = any(), + displayMetrics = displayMetricsCaptor.capture() ) val width = argumentCaptor.firstValue @@ -186,10 +187,10 @@ internal class DrawableUtilsTest { // Then verify(mockBitmapWrapper).createBitmap( - displayMetrics = displayMetricsCaptor.capture(), bitmapWidth = argumentCaptor.capture(), bitmapHeight = argumentCaptor.capture(), - config = any() + config = any(), + displayMetrics = displayMetricsCaptor.capture() ) val width = argumentCaptor.firstValue @@ -221,10 +222,10 @@ internal class DrawableUtilsTest { // Then verify(mockBitmapWrapper).createBitmap( - displayMetrics = displayMetricsCaptor.capture(), bitmapWidth = argumentCaptor.capture(), bitmapHeight = argumentCaptor.capture(), - config = any() + config = any(), + displayMetrics = displayMetricsCaptor.capture() ) val width = argumentCaptor.firstValue @@ -269,7 +270,7 @@ internal class DrawableUtilsTest { whenever(mockDrawable.intrinsicHeight).thenReturn(1) whenever(mockBitmapCachesManager.getBitmapByProperties(any(), any(), any())) .thenReturn(null) - whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), any())) + whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) .thenReturn(null) // When @@ -358,10 +359,10 @@ internal class DrawableUtilsTest { // Then verify(mockBitmapWrapper).createBitmap( - displayMetrics = displayMetricsCaptor.capture(), bitmapWidth = argumentCaptor.capture(), bitmapHeight = argumentCaptor.capture(), - config = any() + config = any(), + displayMetrics = displayMetricsCaptor.capture() ) val width = argumentCaptor.firstValue diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapperTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapperTest.kt similarity index 96% rename from features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapperTest.kt rename to features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapperTest.kt index 658c4ae890..4a1323414c 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/wrappers/CanvasWrapperTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapperTest.kt @@ -4,7 +4,7 @@ * Copyright 2016-Present Datadog, Inc. */ -package com.datadog.android.sessionreplay.internal.recorder.wrappers +package com.datadog.android.sessionreplay.recorder.wrappers import android.graphics.Bitmap import com.datadog.android.api.InternalLogger From c5216f98577f9803e4150846b5e0b4db4cb726cc Mon Sep 17 00:00:00 2001 From: jonathanmos <48201295+jonathanmos@users.noreply.github.com> Date: Wed, 27 Nov 2024 14:04:38 +0200 Subject: [PATCH 2/5] RUM-6195: Add FGM and proguard rules --- detekt_custom.yml | 4 +- .../consumer-rules.pro | 9 ++ .../semantics/CheckboxSemanticsNodeMapper.kt | 100 ++++++++++++++---- .../compose/internal/utils/PathUtils.kt | 29 ++--- .../compose/internal/utils/ReflectionUtils.kt | 30 ++++++ .../compose/internal/utils/SemanticsUtils.kt | 22 ++-- .../CheckboxSemanticsNodeMapperTest.kt | 39 +++++++ .../internal/utils/SemanticsUtilsTest.kt | 76 +++++++++++++ .../recorder/wrappers/BitmapWrapper.kt | 2 + .../recorder/wrappers/CanvasWrapper.kt | 2 + 10 files changed, 258 insertions(+), 55 deletions(-) diff --git a/detekt_custom.yml b/detekt_custom.yml index 5ad1efef3a..72fbfab4a1 100644 --- a/detekt_custom.yml +++ b/detekt_custom.yml @@ -129,11 +129,11 @@ datadog: - "android.database.sqlite.SQLiteDatabase.setTransactionSuccessful():java.lang.IllegalStateException" - "android.graphics.Bitmap.compress(android.graphics.Bitmap.CompressFormat, kotlin.Int, java.io.OutputStream):java.lang.NullPointerException,java.lang.IllegalArgumentException" - "android.graphics.Bitmap.copy(android.graphics.Bitmap.Config, kotlin.Boolean):java.lang.IllegalArgumentException" - - "android.graphics.Bitmap.createBitmap(kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException" - "android.graphics.Bitmap.createBitmap(android.util.DisplayMetrics?, kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException" + - "android.graphics.Bitmap.createBitmap(kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException" - "android.graphics.Bitmap.createScaledBitmap(android.graphics.Bitmap, kotlin.Int, kotlin.Int, kotlin.Boolean):java.lang.IllegalArgumentException" - - "android.graphics.Color.parseColor(kotlin.String?):java.lang.IllegalArgumentException" - "android.graphics.Canvas.constructor(android.graphics.Bitmap):java.lang.IllegalStateException" + - "android.graphics.Color.parseColor(kotlin.String?):java.lang.IllegalArgumentException" - "android.graphics.drawable.LayerDrawable.getDrawable(kotlin.Int):java.lang.IndexOutOfBoundsException" - "android.net.ConnectivityManager.registerDefaultNetworkCallback(android.net.ConnectivityManager.NetworkCallback):java.lang.IllegalArgumentException,java.lang.SecurityException" - "android.net.ConnectivityManager.unregisterNetworkCallback(android.net.ConnectivityManager.NetworkCallback):java.lang.SecurityException" diff --git a/features/dd-sdk-android-session-replay-compose/consumer-rules.pro b/features/dd-sdk-android-session-replay-compose/consumer-rules.pro index 40c7331c5e..dd623a9762 100644 --- a/features/dd-sdk-android-session-replay-compose/consumer-rules.pro +++ b/features/dd-sdk-android-session-replay-compose/consumer-rules.pro @@ -14,6 +14,15 @@ -keepclassmembers class androidx.compose.foundation.text.modifiers.TextStringSimpleElement { ; } +-keepclassmembers class androidx.compose.material.CheckDrawingCache { + ; +} +-keepclassmembers class androidx.compose.material.CheckboxKt { + ; +} +-keepclassmembers class androidx.compose.ui.draw.DrawBehindElement { + ; +} -keepclassmembers class androidx.compose.foundation.BackgroundElement { ; } diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt index 0d19c303c2..09ebd0a93c 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt @@ -11,6 +11,7 @@ import androidx.compose.ui.semantics.SemanticsProperties import androidx.compose.ui.semantics.getOrNull import androidx.compose.ui.state.ToggleableState import com.datadog.android.sessionreplay.ImagePrivacy +import com.datadog.android.sessionreplay.TextAndInputPrivacy import com.datadog.android.sessionreplay.compose.internal.data.SemanticsWireframe import com.datadog.android.sessionreplay.compose.internal.data.UiContext import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils @@ -33,27 +34,70 @@ internal class CheckboxSemanticsNodeMapper( ): SemanticsWireframe { val globalBounds = resolveBounds(semanticsNode) - val wireframes = if (isCheckboxChecked(semanticsNode)) { - createCheckedWireframes( - parentContext = parentContext, - asyncJobStatusCallback = asyncJobStatusCallback, + val checkableWireframes = if (parentContext.textAndInputPrivacy != TextAndInputPrivacy.MASK_SENSITIVE_INPUTS) { + resolveMaskedCheckable( semanticsNode = semanticsNode, globalBounds = globalBounds ) } else { - createUncheckedWireframes( + // Resolves checkable view regardless the state + resolveCheckable( semanticsNode = semanticsNode, - globalBounds = globalBounds, - backgroundColor = DEFAULT_COLOR_WHITE + parentContext = parentContext, + asyncJobStatusCallback = asyncJobStatusCallback, + globalBounds = globalBounds ) } return SemanticsWireframe( uiContext = null, - wireframes = wireframes + wireframes = checkableWireframes ) } + private fun resolveMaskedCheckable( + semanticsNode: SemanticsNode, + globalBounds: GlobalBounds + ): List { + // TODO RUM-5118: Decide how to display masked checkbox, Currently use old unchecked shape wireframe, + return createUncheckedWireframes( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = DEFAULT_COLOR_WHITE, + borderColor = DEFAULT_COLOR_BLACK, + currentIndex = 0 + ) + } + + private fun resolveCheckable( + semanticsNode: SemanticsNode, + parentContext: UiContext, + asyncJobStatusCallback: AsyncJobStatusCallback, + globalBounds: GlobalBounds + ): List = + if (isCheckboxChecked(semanticsNode)) { + createCheckedWireframes( + parentContext = parentContext, + asyncJobStatusCallback = asyncJobStatusCallback, + semanticsNode = semanticsNode, + globalBounds = globalBounds + ) + } else { + val borderColor = + semanticsUtils.resolveBorderColor(semanticsNode) + ?.let { rawColor -> + convertColor(rawColor) + } ?: DEFAULT_COLOR_BLACK + + createUncheckedWireframes( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = DEFAULT_COLOR_WHITE, + borderColor = borderColor, + currentIndex = 0 + ) + } + private fun createCheckedWireframes( parentContext: UiContext, asyncJobStatusCallback: AsyncJobStatusCallback, @@ -119,18 +163,32 @@ internal class CheckboxSemanticsNodeMapper( ): List { val strokeColor = getFallbackCheckmarkColor(backgroundColor) - val background: MobileSegment.Wireframe = createUncheckedWireframes( + val wireframesList = mutableListOf() + var index = 0 + + val borderColor = + semanticsUtils.resolveBorderColor(semanticsNode) + ?.let { rawColor -> + convertColor(rawColor) + } ?: DEFAULT_COLOR_BLACK + + createUncheckedWireframes( semanticsNode = semanticsNode, globalBounds = globalBounds, - backgroundColor = backgroundColor - )[0] + backgroundColor = backgroundColor, + borderColor = borderColor, + currentIndex = 0 + ).firstOrNull()?.let { + wireframesList.add(it) + index++ + } val checkmarkWidth = globalBounds.width * CHECKMARK_SIZE_FACTOR val checkmarkHeight = globalBounds.height * CHECKMARK_SIZE_FACTOR val xPos = globalBounds.x + ((globalBounds.width / 2) - (checkmarkWidth / 2)) val yPos = globalBounds.y + ((globalBounds.height / 2) - (checkmarkHeight / 2)) - val foreground: MobileSegment.Wireframe = MobileSegment.Wireframe.ShapeWireframe( - id = resolveId(semanticsNode, 1), + val foreground = MobileSegment.Wireframe.ShapeWireframe( + id = resolveId(semanticsNode, index), x = xPos.toLong(), y = yPos.toLong(), width = checkmarkWidth.toLong(), @@ -145,23 +203,21 @@ internal class CheckboxSemanticsNodeMapper( width = BOX_BORDER_WIDTH_DP ) ) - return listOf(background, foreground) + + wireframesList.add(foreground) + return wireframesList } private fun createUncheckedWireframes( semanticsNode: SemanticsNode, globalBounds: GlobalBounds, - backgroundColor: String + backgroundColor: String, + borderColor: String, + currentIndex: Int ): List { - val borderColor = - semanticsUtils.resolveBorderColor(semanticsNode) - ?.let { rawColor -> - convertColor(rawColor) - } ?: DEFAULT_COLOR_BLACK - return listOf( MobileSegment.Wireframe.ShapeWireframe( - id = resolveId(semanticsNode, 0), + id = resolveId(semanticsNode, currentIndex), x = globalBounds.x, y = globalBounds.y, width = globalBounds.width, diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt index 739a6996d4..c05143eb10 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt @@ -19,15 +19,16 @@ import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.Chec import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper import java.util.Locale +import android.graphics.Path as AndroidPath internal class PathUtils( private val logger: InternalLogger = InternalLogger.UNBOUND, private val canvasWrapper: CanvasWrapper = CanvasWrapper(logger), private val bitmapWrapper: BitmapWrapper = BitmapWrapper() ) { - @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException internal fun parseColorSafe(color: String): Int? { return try { + @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException Color.parseColor(color) } catch (e: IllegalArgumentException) { logger.log( @@ -56,9 +57,9 @@ internal class PathUtils( return "#$alphaValue$rgbColor" } - @Suppress("UnsafeThirdPartyFunctionCall") // handling UnsupportedOperationException - internal fun asAndroidPathSafe(path: Path): android.graphics.Path? { + internal fun asAndroidPathSafe(path: Path): AndroidPath? { return try { + @Suppress("UnsafeThirdPartyFunctionCall") // handling UnsupportedOperationException path.asAndroidPath() } catch (e: UnsupportedOperationException) { logger.log( @@ -144,28 +145,18 @@ internal class PathUtils( checkmarkColor: Int ): Bitmap? { val canvas = canvasWrapper.createCanvas(bitmap) ?: return null - drawCanvasBackground(canvas, fillColor) - drawCanvasForeground(canvas, scaledPath, checkmarkColor) - return bitmap - } - private fun drawCanvasBackground( - canvas: Canvas, - fillColor: Int - ) { + // draw the background canvas.drawColor(fillColor) - } - private fun drawCanvasForeground( - canvas: Canvas, - path: Path, - checkmarkColor: Int - ) { - drawPathToBitmap(checkmarkColor, path, canvas) + // draw the checkmark + drawPathToBitmap(checkmarkColor, scaledPath, canvas) + + return bitmap } @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException - private fun drawPathSafe(canvas: Canvas?, path: android.graphics.Path, paint: Paint) { + private fun drawPathSafe(canvas: Canvas?, path: AndroidPath, paint: Paint) { try { canvas?.drawPath(path, paint) } catch (e: IllegalArgumentException) { diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ReflectionUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ReflectionUtils.kt index 396803c9cc..f61c70e615 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ReflectionUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ReflectionUtils.kt @@ -9,10 +9,12 @@ package com.datadog.android.sessionreplay.compose.internal.utils import android.graphics.Bitmap import android.text.StaticLayout import android.view.View +import androidx.compose.animation.core.AnimationState import androidx.compose.runtime.Composition import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.ColorProducer +import androidx.compose.ui.graphics.Path import androidx.compose.ui.graphics.Shape import androidx.compose.ui.graphics.painter.BitmapPainter import androidx.compose.ui.graphics.painter.Painter @@ -70,6 +72,10 @@ internal class ReflectionUtils { return ComposeReflection.AsyncImagePainterClass?.isInstance(painter) == true } + fun isDrawBehindElementClass(modifier: Modifier): Boolean { + return ComposeReflection.DrawBehindElementClass?.isInstance(modifier) == true + } + fun getOwner(composition: Composition): Any? { return ComposeReflection.OwnerField?.getSafe(composition) } @@ -171,4 +177,28 @@ internal class ReflectionUtils { fun getInteropView(semanticsNode: SemanticsNode): View? { return GetInteropViewMethod?.invoke(semanticsNode.layoutInfo) as? View } + + fun getOnDraw(modifier: Modifier): Any? { + return ComposeReflection.OnDrawField?.getSafe(modifier) + } + + fun getBoxColor(onDrawInstance: Any): AnimationState<*, *>? { + return ComposeReflection.BoxColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + + fun getCheckColor(onDrawInstance: Any): AnimationState<*, *>? { + return ComposeReflection.CheckColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + + fun getBorderColor(onDrawInstance: Any): AnimationState<*, *>? { + return ComposeReflection.BorderColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + } + + fun getCheckCache(onDrawInstance: Any): Any? { + return ComposeReflection.CheckCacheField?.getSafe(onDrawInstance) + } + + fun getCheckPath(checkCache: Any): Path? { + return ComposeReflection.CheckPathField?.getSafe(checkCache) as? Path + } } diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt index 509be5fa8d..81a5a6a98b 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt @@ -8,7 +8,6 @@ package com.datadog.android.sessionreplay.compose.internal.utils import android.graphics.Bitmap import android.view.View -import androidx.compose.animation.core.AnimationState import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Size @@ -32,8 +31,6 @@ import com.datadog.android.sessionreplay.compose.TextInputSemanticsPropertyKey import com.datadog.android.sessionreplay.compose.TouchSemanticsPropertyKey import com.datadog.android.sessionreplay.compose.internal.data.BitmapInfo import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.TextLayoutInfo -import com.datadog.android.sessionreplay.compose.internal.reflection.ComposeReflection -import com.datadog.android.sessionreplay.compose.internal.reflection.getSafe import com.datadog.android.sessionreplay.utils.GlobalBounds @Suppress("TooManyFunctions") @@ -120,8 +117,8 @@ internal class SemanticsUtils(private val reflectionUtils: ReflectionUtils = Ref internal fun resolveCheckPath(semanticsNode: SemanticsNode): Path? = resolveOnDrawInstance(semanticsNode)?.let { onDraw -> - ComposeReflection.CheckCacheField?.getSafe(onDraw)?.let { checkCache -> - ComposeReflection.CheckPathField?.getSafe(checkCache) as? Path + reflectionUtils.getCheckCache(onDraw)?.let { checkCache -> + reflectionUtils.getCheckPath(checkCache) } } @@ -310,32 +307,33 @@ internal class SemanticsUtils(private val reflectionUtils: ReflectionUtils = Ref private fun resolveOnDrawInstance(semanticsNode: SemanticsNode): Any? { val drawBehindElement = semanticsNode.layoutInfo.getModifierInfo().firstOrNull { modifierInfo -> - ComposeReflection.DrawBehindElementClass?.isInstance(modifierInfo.modifier) == true + reflectionUtils.isDrawBehindElementClass(modifierInfo.modifier) }?.modifier return drawBehindElement?.let { - ComposeReflection.OnDrawField?.getSafe(drawBehindElement) + reflectionUtils.getOnDraw(it) } } private fun resolveReflectedProperty(semanticsNode: SemanticsNode, fieldType: CheckmarkFieldType): Long? { val onDrawInstance = resolveOnDrawInstance(semanticsNode) - val checkmarkColor: AnimationState<*, *>? = onDrawInstance?.let { + val color = onDrawInstance?.let { when (fieldType) { CheckmarkFieldType.FILL_COLOR -> { - ComposeReflection.BoxColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + reflectionUtils.getBoxColor(onDrawInstance) } CheckmarkFieldType.CHECKMARK_COLOR -> { - ComposeReflection.CheckColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + reflectionUtils.getCheckColor(onDrawInstance) } CheckmarkFieldType.BORDER_COLOR -> { - ComposeReflection.BorderColorField?.getSafe(onDrawInstance) as? AnimationState<*, *> + reflectionUtils.getBorderColor(onDrawInstance) } } } - val result = (checkmarkColor?.value as? Color)?.value + val result = (color?.value as? Color) + ?.value return result?.toLong() } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt index a623bd56ba..224d63d577 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt @@ -14,6 +14,7 @@ import androidx.compose.ui.semantics.SemanticsProperties import androidx.compose.ui.semantics.getOrNull import androidx.compose.ui.state.ToggleableState import com.datadog.android.sessionreplay.ImagePrivacy +import com.datadog.android.sessionreplay.TextAndInputPrivacy import com.datadog.android.sessionreplay.compose.internal.data.UiContext import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.BOX_BORDER_WIDTH_DP import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_CORNER_RADIUS @@ -42,6 +43,7 @@ import org.mockito.kotlin.any import org.mockito.kotlin.anyOrNull import org.mockito.kotlin.doReturn import org.mockito.kotlin.eq +import org.mockito.kotlin.never import org.mockito.kotlin.verify import org.mockito.kotlin.whenever import org.mockito.quality.Strictness @@ -107,6 +109,9 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockUiContext.density) .thenReturn(fakeDensity) + whenever(mockUiContext.textAndInputPrivacy) + .thenReturn(TextAndInputPrivacy.MASK_SENSITIVE_INPUTS) + whenever(mockSemanticsUtils.resolveInnerBounds(mockSemanticsNode)) doReturn rectToBounds( fakeBounds, fakeDensity @@ -284,6 +289,7 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest parentContext = mockUiContext, asyncJobStatusCallback = mockAsyncJobStatusCallback ) + assertThat(semanticsWireframe.wireframes).hasSize(2) val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe val expectedShapeStyle = MobileSegment.ShapeStyle( @@ -472,4 +478,37 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest assertThat(wireframes.wireframes).hasSize(1) assertThat(wireframes.wireframes[0]).isInstanceOf(MobileSegment.Wireframe.ImageWireframe::class.java) } + + @Test + fun `M show unchecked wireframe W map() { masked }`() { + // Given + whenever(mockUiContext.textAndInputPrivacy) + .thenReturn(TextAndInputPrivacy.MASK_ALL_INPUTS) + + // When + val wireframes = testedMapper.map( + semanticsNode = mockSemanticsNode, + parentContext = mockUiContext, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + assertThat(wireframes.wireframes).hasSize(1) + val actualWireframe = wireframes.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + assertThat(actualWireframe).isNotNull + + // Then + verify(mockUiContext.imageWireframeHelper, never()).createImageWireframeByBitmap( + id = any(), + globalBounds = any(), + bitmap = any(), + density = any(), + isContextualImage = any(), + imagePrivacy = any(), + asyncJobStatusCallback = any(), + clipping = anyOrNull(), + shapeStyle = anyOrNull(), + border = anyOrNull() + ) + assertThat(actualWireframe?.shapeStyle?.backgroundColor).isEqualTo(DEFAULT_COLOR_WHITE) + assertThat(actualWireframe?.border?.color).isEqualTo(DEFAULT_COLOR_BLACK) + } } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtilsTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtilsTest.kt index 31ae4cc5b8..92d8eba653 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtilsTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtilsTest.kt @@ -8,6 +8,7 @@ package com.datadog.android.sessionreplay.compose.internal.utils import android.graphics.Bitmap import android.view.View +import androidx.compose.animation.core.AnimationState import androidx.compose.foundation.shape.CornerSize import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.runtime.Composition @@ -16,6 +17,7 @@ import androidx.compose.ui.geometry.Offset import androidx.compose.ui.geometry.Rect import androidx.compose.ui.geometry.Size import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.Path import androidx.compose.ui.graphics.Shape import androidx.compose.ui.graphics.painter.BitmapPainter import androidx.compose.ui.graphics.vector.VectorPainter @@ -90,6 +92,12 @@ class SemanticsUtilsTest { @Mock private lateinit var mockModifierInfo: ModifierInfo + @Mock + private lateinit var mockOnDraw: Any + + @Mock + private lateinit var mockCheckCache: Any + @Mock private lateinit var mockModifier: Modifier @@ -109,6 +117,9 @@ class SemanticsUtilsTest { whenever(mockModifierInfo.modifier) doReturn mockModifier whenever(mockLayoutInfo.density) doReturn Density(fakeDensity) whenever(mockSemanticsNode.config) doReturn mockConfig + whenever(mockReflectionUtils.isDrawBehindElementClass(mockModifier)) doReturn true + whenever(mockReflectionUtils.getOnDraw(mockModifier)) doReturn mockOnDraw + whenever(mockReflectionUtils.getCheckCache(mockOnDraw)) doReturn mockCheckCache fakeOffset = Offset(x = forge.aFloat(), y = forge.aFloat()) } @@ -146,6 +157,71 @@ class SemanticsUtilsTest { assertThat(result).isEqualTo(mockShape) } + @Test + fun `M return check path W resolveCheckPath`( + @Mock mockPath: Path + ) { + // Given + whenever(mockReflectionUtils.getCheckPath(mockCheckCache)) doReturn mockPath + + // When + val result = testedSemanticsUtils.resolveCheckPath(mockSemanticsNode) + + // Then + assertThat(result).isEqualTo(mockPath) + } + + @Test + fun `M return checkbox fill color W resolveCheckboxFillColor`( + @IntForgery fakeColorValue: Int + ) { + // Given + val fakeColor = Color(fakeColorValue) + val mockAnimationState = mock>() + whenever(mockReflectionUtils.getBoxColor(mockOnDraw)) doReturn mockAnimationState + whenever(mockAnimationState.value).thenReturn(fakeColor) + + // When + val result = testedSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode) + + // Then + assertThat(result).isEqualTo(fakeColor.value.toLong()) + } + + @Test + fun `M return checkmark color W resolveCheckmarkColor`( + @IntForgery fakeColorValue: Int + ) { + // Given + val fakeColor = Color(fakeColorValue) + val mockAnimationState = mock>() + whenever(mockReflectionUtils.getCheckColor(mockOnDraw)) doReturn mockAnimationState + whenever(mockAnimationState.value).thenReturn(fakeColor) + + // When + val result = testedSemanticsUtils.resolveCheckmarkColor(mockSemanticsNode) + + // Then + assertThat(result).isEqualTo(fakeColor.value.toLong()) + } + + @Test + fun `M return border color W resolveBorderColor`( + @IntForgery fakeColorValue: Int + ) { + // Given + val fakeColor = Color(fakeColorValue) + val mockAnimationState = mock>() + whenever(mockReflectionUtils.getBorderColor(mockOnDraw)) doReturn mockAnimationState + whenever(mockAnimationState.value).thenReturn(fakeColor) + + // When + val result = testedSemanticsUtils.resolveBorderColor(mockSemanticsNode) + + // Then + assertThat(result).isEqualTo(fakeColor.value.toLong()) + } + @Test fun `M return inner bounds W resolveInnerBounds`() { // Given diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt index 687c0bf716..0503935b22 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/BitmapWrapper.kt @@ -10,10 +10,12 @@ import android.graphics.Bitmap import android.graphics.Bitmap.Config import android.util.DisplayMetrics import com.datadog.android.api.InternalLogger +import com.datadog.android.lint.InternalApi /** * Wraps the Bitmap class to catch potential crashes. */ +@InternalApi class BitmapWrapper( private val logger: InternalLogger = InternalLogger.UNBOUND ) { diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt index 3bd929e619..bcde43faa1 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/recorder/wrappers/CanvasWrapper.kt @@ -9,10 +9,12 @@ package com.datadog.android.sessionreplay.recorder.wrappers import android.graphics.Bitmap import android.graphics.Canvas import com.datadog.android.api.InternalLogger +import com.datadog.android.lint.InternalApi /** * Wraps the Canvas class to catch potential crashes. */ +@InternalApi class CanvasWrapper( private val logger: InternalLogger ) { From 6bd64004737875ef2ab0277cea04b02649927266 Mon Sep 17 00:00:00 2001 From: jonathanmos <48201295+jonathanmos@users.noreply.github.com> Date: Wed, 11 Dec 2024 10:21:55 +0200 Subject: [PATCH 3/5] RUM-6195: Implement caching logic for path based images --- detekt_custom.yml | 11 + .../semantics/CheckboxSemanticsNodeMapper.kt | 133 +++++----- .../compose/internal/utils/ColorUtils.kt | 50 ++++ .../compose/internal/utils/PathUtils.kt | 147 +---------- .../CheckboxSemanticsNodeMapperTest.kt | 105 ++++---- .../compose/internal/utils/ColorUtilsTest.kt | 60 +++++ .../api/apiSurface | 1 + .../api/dd-sdk-android-session-replay.api | 2 + .../recorder/SessionReplayRecorder.kt | 2 + .../resources/DefaultImageWireframeHelper.kt | 102 +++++++- .../recorder/resources/ImageTypeResolver.kt | 13 +- .../recorder/resources/ResourceResolver.kt | 183 ++++++++++---- .../sessionreplay/internal/utils/PathUtils.kt | 178 ++++++++++++++ .../utils/ImageWireframeHelper.kt | 43 +++- .../DefaultImageWireframeHelperTest.kt | 231 ++++++++++++++++-- .../resources/ResourceResolverTest.kt | 111 +++++---- .../internal/utils/PathUtilsTest.kt | 117 ++++++--- 17 files changed, 1076 insertions(+), 413 deletions(-) create mode 100644 features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt create mode 100644 features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt create mode 100644 features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt rename features/{dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose => dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay}/internal/utils/PathUtilsTest.kt (52%) diff --git a/detekt_custom.yml b/detekt_custom.yml index 72fbfab4a1..5ae85b0ace 100644 --- a/detekt_custom.yml +++ b/detekt_custom.yml @@ -487,7 +487,14 @@ datadog: - "android.graphics.drawable.Drawable.setTintList(android.content.res.ColorStateList?)" - "android.graphics.drawable.RippleDrawable.findIndexByLayerId(kotlin.Int)" - "android.graphics.drawable.DrawableContainer.DrawableContainerState.getChild(kotlin.Int)" + - "android.graphics.Matrix.constructor()" + - "android.graphics.Matrix.preScale(kotlin.Float, kotlin.Float)" + - "android.graphics.Matrix.preTranslate(kotlin.Float, kotlin.Float)" - "android.graphics.Paint.constructor()" + - "android.graphics.Path.computeBounds(android.graphics.RectF, kotlin.Boolean)" + - "android.graphics.Path.transform(android.graphics.Matrix)" + - "android.graphics.PathMeasure.constructor(android.graphics.Path?, kotlin.Boolean)" + - "android.graphics.PathMeasure.nextContour()" - "android.graphics.Point.constructor()" - "android.graphics.Point.constructor(kotlin.Int, kotlin.Int)" - "android.graphics.Rect.centerX()" @@ -496,6 +503,9 @@ datadog: - "android.graphics.Rect.constructor(kotlin.Int, kotlin.Int, kotlin.Int, kotlin.Int)" - "android.graphics.Rect.height()" - "android.graphics.Rect.width()" + - "android.graphics.RectF.constructor()" + - "android.graphics.RectF.width()" + - "android.graphics.RectF.height()" # endregion # region Androidx APIs - "androidx.appcompat.widget.DatadogActionBarContainerAccessor.constructor(androidx.appcompat.widget.ActionBarContainer)" @@ -1150,6 +1160,7 @@ datadog: - "kotlin.Float.toFloat()" - "kotlin.Float.toInt()" - "kotlin.Float.toLong()" + - "kotlin.FloatArray.constructor(kotlin.Int)" - "kotlin.Int.and(kotlin.Int)" - "kotlin.Int.coerceAtMost(kotlin.Int)" - "kotlin.Int.inv()" diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt index 09ebd0a93c..49fee65d45 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt @@ -10,10 +10,12 @@ import androidx.compose.ui.semantics.SemanticsNode import androidx.compose.ui.semantics.SemanticsProperties import androidx.compose.ui.semantics.getOrNull import androidx.compose.ui.state.ToggleableState +import com.datadog.android.api.InternalLogger import com.datadog.android.sessionreplay.ImagePrivacy import com.datadog.android.sessionreplay.TextAndInputPrivacy import com.datadog.android.sessionreplay.compose.internal.data.SemanticsWireframe import com.datadog.android.sessionreplay.compose.internal.data.UiContext +import com.datadog.android.sessionreplay.compose.internal.utils.ColorUtils import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils import com.datadog.android.sessionreplay.compose.internal.utils.SemanticsUtils import com.datadog.android.sessionreplay.model.MobileSegment @@ -24,7 +26,9 @@ import com.datadog.android.sessionreplay.utils.GlobalBounds internal class CheckboxSemanticsNodeMapper( colorStringFormatter: ColorStringFormatter, private val semanticsUtils: SemanticsUtils = SemanticsUtils(), - private val pathUtils: PathUtils = PathUtils() + private val colorUtils: ColorUtils = ColorUtils(), + private val logger: InternalLogger = InternalLogger.UNBOUND, + private val pathUtils: PathUtils = PathUtils(logger) ) : AbstractSemanticsNodeMapper(colorStringFormatter, semanticsUtils) { override fun map( @@ -35,9 +39,11 @@ internal class CheckboxSemanticsNodeMapper( val globalBounds = resolveBounds(semanticsNode) val checkableWireframes = if (parentContext.textAndInputPrivacy != TextAndInputPrivacy.MASK_SENSITIVE_INPUTS) { - resolveMaskedCheckable( - semanticsNode = semanticsNode, - globalBounds = globalBounds + listOf( + resolveMaskedCheckable( + semanticsNode = semanticsNode, + globalBounds = globalBounds + ) ) } else { // Resolves checkable view regardless the state @@ -58,9 +64,9 @@ internal class CheckboxSemanticsNodeMapper( private fun resolveMaskedCheckable( semanticsNode: SemanticsNode, globalBounds: GlobalBounds - ): List { + ): MobileSegment.Wireframe { // TODO RUM-5118: Decide how to display masked checkbox, Currently use old unchecked shape wireframe, - return createUncheckedWireframes( + return createUncheckedState( semanticsNode = semanticsNode, globalBounds = globalBounds, backgroundColor = DEFAULT_COLOR_WHITE, @@ -74,13 +80,14 @@ internal class CheckboxSemanticsNodeMapper( parentContext: UiContext, asyncJobStatusCallback: AsyncJobStatusCallback, globalBounds: GlobalBounds - ): List = - if (isCheckboxChecked(semanticsNode)) { - createCheckedWireframes( + ): List { + return if (isCheckboxChecked(semanticsNode)) { + createCheckedState( parentContext = parentContext, asyncJobStatusCallback = asyncJobStatusCallback, semanticsNode = semanticsNode, - globalBounds = globalBounds + globalBounds = globalBounds, + currentIndex = 0 ) } else { val borderColor = @@ -89,20 +96,24 @@ internal class CheckboxSemanticsNodeMapper( convertColor(rawColor) } ?: DEFAULT_COLOR_BLACK - createUncheckedWireframes( - semanticsNode = semanticsNode, - globalBounds = globalBounds, - backgroundColor = DEFAULT_COLOR_WHITE, - borderColor = borderColor, - currentIndex = 0 + listOf( + createUncheckedState( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = DEFAULT_COLOR_WHITE, + borderColor = borderColor, + currentIndex = 0 + ) ) } + } - private fun createCheckedWireframes( + private fun createCheckedState( parentContext: UiContext, asyncJobStatusCallback: AsyncJobStatusCallback, semanticsNode: SemanticsNode, - globalBounds: GlobalBounds + globalBounds: GlobalBounds, + currentIndex: Int ): List { val rawFillColor = semanticsUtils.resolveCheckboxFillColor(semanticsNode) val rawCheckmarkColor = semanticsUtils.resolveCheckmarkColor(semanticsNode) @@ -114,24 +125,25 @@ internal class CheckboxSemanticsNodeMapper( convertColor(it) } ?: getFallbackCheckmarkColor(DEFAULT_COLOR_WHITE) - val parsedFillColor = pathUtils.parseColorSafe(fillColorRgba) - val parsedCheckmarkColor = pathUtils.parseColorSafe(checkmarkColorRgba) + val parsedFillColor = colorUtils.parseColorSafe(fillColorRgba) + val parsedCheckmarkColor = colorUtils.parseColorSafe(checkmarkColorRgba) + val wireframes = mutableListOf() if (parsedFillColor != null && parsedCheckmarkColor != null) { - val checkMarkBitmap = semanticsUtils + val androidPath = semanticsUtils .resolveCheckPath(semanticsNode)?.let { checkPath -> - pathUtils.convertPathToBitmap( - checkPath = checkPath, - fillColor = parsedFillColor, - checkmarkColor = parsedCheckmarkColor - ) + pathUtils.asAndroidPathSafe(checkPath) } - if (checkMarkBitmap != null) { - parentContext.imageWireframeHelper.createImageWireframeByBitmap( - id = resolveId(semanticsNode, 0), + if (androidPath != null) { + parentContext.imageWireframeHelper.createImageWireframeByPath( + id = resolveId(semanticsNode, currentIndex), globalBounds = globalBounds, - bitmap = checkMarkBitmap, + path = androidPath, + strokeColor = parsedCheckmarkColor, + strokeWidth = STROKE_WIDTH_DP.toInt(), + targetWidth = CHECKBOX_SIZE_DP, + targetHeight = CHECKBOX_SIZE_DP, density = parentContext.density, isContextualImage = false, imagePrivacy = ImagePrivacy.MASK_NONE, @@ -145,15 +157,24 @@ internal class CheckboxSemanticsNodeMapper( border = MobileSegment.ShapeBorder( color = fillColorRgba, width = BOX_BORDER_WIDTH_DP - ) + ), + customResourceIdCacheKey = null )?.let { imageWireframe -> - return listOf(imageWireframe) + wireframes.add(imageWireframe) } } } + if (wireframes.isNotEmpty()) { + return wireframes + } + // if we failed to create a wireframe from the path - return createManualCheckedWireframe(semanticsNode, globalBounds, fillColorRgba) + return createManualCheckedWireframe( + semanticsNode = semanticsNode, + globalBounds = globalBounds, + backgroundColor = fillColorRgba + ) } private fun createManualCheckedWireframe( @@ -164,7 +185,6 @@ internal class CheckboxSemanticsNodeMapper( val strokeColor = getFallbackCheckmarkColor(backgroundColor) val wireframesList = mutableListOf() - var index = 0 val borderColor = semanticsUtils.resolveBorderColor(semanticsNode) @@ -172,23 +192,22 @@ internal class CheckboxSemanticsNodeMapper( convertColor(rawColor) } ?: DEFAULT_COLOR_BLACK - createUncheckedWireframes( + val background = createUncheckedState( semanticsNode = semanticsNode, globalBounds = globalBounds, backgroundColor = backgroundColor, borderColor = borderColor, currentIndex = 0 - ).firstOrNull()?.let { - wireframesList.add(it) - index++ - } + ) + + wireframesList.add(background) val checkmarkWidth = globalBounds.width * CHECKMARK_SIZE_FACTOR val checkmarkHeight = globalBounds.height * CHECKMARK_SIZE_FACTOR val xPos = globalBounds.x + ((globalBounds.width / 2) - (checkmarkWidth / 2)) val yPos = globalBounds.y + ((globalBounds.height / 2) - (checkmarkHeight / 2)) val foreground = MobileSegment.Wireframe.ShapeWireframe( - id = resolveId(semanticsNode, index), + id = resolveId(semanticsNode, 1), x = xPos.toLong(), y = yPos.toLong(), width = checkmarkWidth.toLong(), @@ -208,29 +227,27 @@ internal class CheckboxSemanticsNodeMapper( return wireframesList } - private fun createUncheckedWireframes( + private fun createUncheckedState( semanticsNode: SemanticsNode, globalBounds: GlobalBounds, backgroundColor: String, borderColor: String, currentIndex: Int - ): List { - return listOf( - MobileSegment.Wireframe.ShapeWireframe( - id = resolveId(semanticsNode, currentIndex), - x = globalBounds.x, - y = globalBounds.y, - width = globalBounds.width, - height = globalBounds.height, - shapeStyle = MobileSegment.ShapeStyle( - backgroundColor = backgroundColor, - opacity = 1f, - cornerRadius = CHECKBOX_CORNER_RADIUS - ), - border = MobileSegment.ShapeBorder( - color = borderColor, - width = BOX_BORDER_WIDTH_DP - ) + ): MobileSegment.Wireframe { + return MobileSegment.Wireframe.ShapeWireframe( + id = resolveId(semanticsNode, currentIndex), + x = globalBounds.x, + y = globalBounds.y, + width = CHECKBOX_SIZE_DP.toLong(), + height = CHECKBOX_SIZE_DP.toLong(), + shapeStyle = MobileSegment.ShapeStyle( + backgroundColor = backgroundColor, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ), + border = MobileSegment.ShapeBorder( + color = borderColor, + width = BOX_BORDER_WIDTH_DP ) ) } diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt new file mode 100644 index 0000000000..5942383824 --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt @@ -0,0 +1,50 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.utils + +import android.graphics.Color +import com.datadog.android.api.InternalLogger +import java.util.Locale + +internal class ColorUtils( + private val logger: InternalLogger = InternalLogger.UNBOUND +) { + internal fun parseColorSafe(color: String): Int? { + return try { + @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException + Color.parseColor(color) + } catch (e: IllegalArgumentException) { + logger.log( + target = InternalLogger.Target.MAINTAINER, + level = InternalLogger.Level.WARN, + messageBuilder = { COLOR_PARSE_ERROR.format(Locale.US, color) }, + throwable = e + ) + null + } + } + + internal fun convertRgbaToArgb(rgbaString: String): String { + if (rgbaString.length < 2) return rgbaString + + // for takeLast: n > 0 + @Suppress("UnsafeThirdPartyFunctionCall") + val alphaValue = rgbaString.takeLast(2) + + // for substring: length is necessarily > 1 at this point + // for dropLast: n > 0 + @Suppress("UnsafeThirdPartyFunctionCall") + val rgbColor = rgbaString + .substring(1) + .dropLast(2) + return "#$alphaValue$rgbColor" + } + + internal companion object { + internal const val COLOR_PARSE_ERROR = "Failed to parse color: %s" + } +} diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt index c05143eb10..d6dc9a4db7 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtils.kt @@ -6,58 +6,14 @@ package com.datadog.android.sessionreplay.compose.internal.utils -import android.graphics.Bitmap -import android.graphics.Canvas -import android.graphics.Color -import android.graphics.Paint -import androidx.compose.ui.graphics.Matrix import androidx.compose.ui.graphics.Path import androidx.compose.ui.graphics.asAndroidPath import com.datadog.android.api.InternalLogger -import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_SIZE_DP -import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.STROKE_WIDTH_DP -import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper -import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper -import java.util.Locale -import android.graphics.Path as AndroidPath internal class PathUtils( - private val logger: InternalLogger = InternalLogger.UNBOUND, - private val canvasWrapper: CanvasWrapper = CanvasWrapper(logger), - private val bitmapWrapper: BitmapWrapper = BitmapWrapper() + private val logger: InternalLogger ) { - internal fun parseColorSafe(color: String): Int? { - return try { - @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException - Color.parseColor(color) - } catch (e: IllegalArgumentException) { - logger.log( - target = InternalLogger.Target.MAINTAINER, - level = InternalLogger.Level.WARN, - messageBuilder = { COLOR_PARSE_ERROR.format(Locale.US, color) }, - throwable = e - ) - null - } - } - - internal fun convertRgbaToArgb(rgbaString: String): String { - if (rgbaString.length < 2) return rgbaString - - // for takeLast: n > 0 - @Suppress("UnsafeThirdPartyFunctionCall") - val alphaValue = rgbaString.takeLast(2) - - // for substring: length is necessarily > 1 at this point - // for dropLast: n > 0 - @Suppress("UnsafeThirdPartyFunctionCall") - val rgbColor = rgbaString - .substring(1) - .dropLast(2) - return "#$alphaValue$rgbColor" - } - - internal fun asAndroidPathSafe(path: Path): AndroidPath? { + internal fun asAndroidPathSafe(path: Path): android.graphics.Path? { return try { @Suppress("UnsafeThirdPartyFunctionCall") // handling UnsupportedOperationException path.asAndroidPath() @@ -72,106 +28,7 @@ internal class PathUtils( } } - internal fun convertPathToBitmap( - checkPath: Path, - fillColor: Int, - checkmarkColor: Int - ): Bitmap? { - val scaledPath = scalePathToBitmapSize(checkPath) - val mutableBitmap = - bitmapWrapper.createBitmap(CHECKBOX_SIZE_DP, CHECKBOX_SIZE_DP, Bitmap.Config.ARGB_8888) - ?: return null - - return drawPathOntoBitmap( - mutableBitmap, - scaledPath, - fillColor, - checkmarkColor - ) - } - - private fun drawPathToBitmap(checkmarkColor: Int, path: Path, canvas: Canvas?) { - val paint = Paint().apply { - color = checkmarkColor - style = Paint.Style.STROKE - strokeWidth = STROKE_WIDTH_DP - isAntiAlias = true - } - - // Draw the Path onto the Canvas - asAndroidPathSafe(path)?.let { - drawPathSafe(canvas, it, paint) - } - } - - private fun scalePathToBitmapSize(path: Path): Path { - // path initial bounds - val originalBounds = path.getBounds() - - // calculate the scale factor - val scaleX = CHECKBOX_SIZE_DP / originalBounds.width - val scaleY = CHECKBOX_SIZE_DP / originalBounds.height - val scaleFactor = minOf(scaleX, scaleY) - - // current center - val currentCenterX = (originalBounds.left + originalBounds.right) / 2 - val currentCenterY = (originalBounds.top + originalBounds.bottom) / 2 - - // new center - val newCenterX = CHECKBOX_SIZE_DP / 2 - val newCenterY = CHECKBOX_SIZE_DP / 2 - - // center changes after scaling - val scaledCenterX = currentCenterX * scaleFactor - val scaledCenterY = currentCenterY * scaleFactor - - // translation needed to recenter - val translateX = newCenterX - scaledCenterX - val translateY = newCenterY - scaledCenterY - - // the order of operations is important - val matrix = Matrix() - matrix.translate(translateX, translateY) - matrix.scale(scaleFactor, scaleFactor) - path.transform(matrix) - - return path - } - - private fun drawPathOntoBitmap( - bitmap: Bitmap, - scaledPath: Path, - fillColor: Int, - checkmarkColor: Int - ): Bitmap? { - val canvas = canvasWrapper.createCanvas(bitmap) ?: return null - - // draw the background - canvas.drawColor(fillColor) - - // draw the checkmark - drawPathToBitmap(checkmarkColor, scaledPath, canvas) - - return bitmap - } - - @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException - private fun drawPathSafe(canvas: Canvas?, path: AndroidPath, paint: Paint) { - try { - canvas?.drawPath(path, paint) - } catch (e: IllegalArgumentException) { - logger.log( - target = InternalLogger.Target.MAINTAINER, - level = InternalLogger.Level.WARN, - messageBuilder = { PATH_DRAW_ERROR }, - throwable = e - ) - } - } - internal companion object { - internal const val COLOR_PARSE_ERROR = "Failed to parse color: %s" internal const val PATH_CONVERSION_ERROR = "Failed to convert Compose Path to Android Path" - internal const val PATH_DRAW_ERROR = "Failed to draw Path to Canvas" } } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt index 224d63d577..e094af601b 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt @@ -6,7 +6,6 @@ package com.datadog.android.sessionreplay.compose.internal.mappers.semantics -import android.graphics.Bitmap import androidx.compose.ui.graphics.Path import androidx.compose.ui.semantics.SemanticsConfiguration import androidx.compose.ui.semantics.SemanticsNode @@ -18,8 +17,11 @@ import com.datadog.android.sessionreplay.TextAndInputPrivacy import com.datadog.android.sessionreplay.compose.internal.data.UiContext import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.BOX_BORDER_WIDTH_DP import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_CORNER_RADIUS +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.CHECKBOX_SIZE_DP import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.DEFAULT_COLOR_BLACK import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.DEFAULT_COLOR_WHITE +import com.datadog.android.sessionreplay.compose.internal.mappers.semantics.CheckboxSemanticsNodeMapper.Companion.STROKE_WIDTH_DP +import com.datadog.android.sessionreplay.compose.internal.utils.ColorUtils import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils import com.datadog.android.sessionreplay.compose.test.elmyr.SessionReplayComposeForgeConfigurator import com.datadog.android.sessionreplay.model.MobileSegment @@ -91,10 +93,10 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest private lateinit var mockPath: Path @Mock - private lateinit var mockBitmap: Bitmap + private lateinit var mockPathUtils: PathUtils @Mock - private lateinit var mockPathUtils: PathUtils + private lateinit var mockColorUtils: ColorUtils @Mock private lateinit var mockImageWireframeHelper: ImageWireframeHelper @@ -122,8 +124,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockSemanticsNode.config).doReturn(mockConfig) } - whenever(mockPath.getBounds()) - .thenReturn(fakeBounds) + whenever(mockPathUtils.asAndroidPathSafe(any())) + .thenReturn(mock()) whenever(mockSemanticsUtils.resolveCheckmarkColor(mockSemanticsNode)) .thenReturn(fakeCheckmarkColor) @@ -131,6 +133,9 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockSemanticsUtils.resolveBorderColor(mockSemanticsNode)) .thenReturn(fakeBorderColor) + whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) + .thenReturn(mockPath) + mockColorStringFormatter(fakeBorderColor, fakeBorderColorHexString) mockColorStringFormatter(fakeFillColor, fakeFillColorHexString) mockColorStringFormatter(fakeCheckmarkColor, fakeCheckmarkColorHexString) @@ -138,7 +143,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest testedMapper = CheckboxSemanticsNodeMapper( colorStringFormatter = mockColorStringFormatter, semanticsUtils = mockSemanticsUtils, - pathUtils = mockPathUtils + pathUtils = mockPathUtils, + colorUtils = mockColorUtils ) } @@ -308,9 +314,6 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) .thenReturn(ToggleableState.On) - whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) - .thenReturn(mockPath) - // When val semanticsWireframe = testedMapper.map( semanticsNode = mockSemanticsNode, @@ -335,9 +338,6 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) .thenReturn(ToggleableState.On) - whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) - .thenReturn(mockPath) - // When val semanticsWireframe = testedMapper.map( semanticsNode = mockSemanticsNode, @@ -364,34 +364,32 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) .thenReturn(ToggleableState.On) - whenever(mockPathUtils.asAndroidPathSafe(any())).thenReturn(mock()) - - whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) - .thenReturn(mockPath) - whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) .thenReturn(fakeFillColor) - whenever( - mockPathUtils.convertPathToBitmap( - checkPath = eq(mockPath), - fillColor = anyOrNull(), - checkmarkColor = anyOrNull() - ) - ) doReturn mockBitmap + whenever(mockColorUtils.parseColorSafe(fakeFillColorHexString)) + .thenReturn(fakeFillColor.toInt()) + + whenever(mockColorUtils.parseColorSafe(fakeCheckmarkColorHexString)) + .thenReturn(fakeCheckmarkColor.toInt()) whenever( - mockUiContext.imageWireframeHelper.createImageWireframeByBitmap( + mockUiContext.imageWireframeHelper.createImageWireframeByPath( id = any(), globalBounds = eq(fakeGlobalBounds), - bitmap = eq(mockBitmap), + path = any(), + strokeColor = eq(fakeCheckmarkColor.toInt()), + strokeWidth = eq(STROKE_WIDTH_DP.toInt()), + targetWidth = eq(CHECKBOX_SIZE_DP), + targetHeight = eq(CHECKBOX_SIZE_DP), density = eq(fakeDensity), - isContextualImage = eq(false), imagePrivacy = eq(ImagePrivacy.MASK_NONE), + isContextualImage = eq(false), asyncJobStatusCallback = eq(mockAsyncJobStatusCallback), clipping = eq(null), shapeStyle = eq(null), - border = eq(null) + border = eq(null), + customResourceIdCacheKey = eq(null) ) ).thenReturn(mock()) @@ -414,17 +412,22 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest ) // Then - verify(mockUiContext.imageWireframeHelper).createImageWireframeByBitmap( + verify(mockUiContext.imageWireframeHelper).createImageWireframeByPath( id = any(), globalBounds = eq(fakeGlobalBounds), - bitmap = eq(mockBitmap), + path = any(), + strokeColor = eq(fakeCheckmarkColor.toInt()), + strokeWidth = eq(STROKE_WIDTH_DP.toInt()), + targetWidth = eq(CHECKBOX_SIZE_DP), + targetHeight = eq(CHECKBOX_SIZE_DP), density = eq(fakeDensity), isContextualImage = eq(false), imagePrivacy = eq(ImagePrivacy.MASK_NONE), asyncJobStatusCallback = eq(mockAsyncJobStatusCallback), clipping = eq(null), shapeStyle = eq(expectedShapeStyle), - border = eq(expectedBorder) + border = eq(expectedBorder), + customResourceIdCacheKey = eq(null) ) } @@ -436,34 +439,32 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest whenever(mockConfig.getOrNull(SemanticsProperties.ToggleableState)) .thenReturn(ToggleableState.On) - whenever(mockPathUtils.asAndroidPathSafe(any())).thenReturn(mock()) - - whenever(mockSemanticsUtils.resolveCheckPath(mockSemanticsNode)) - .thenReturn(mockPath) - whenever(mockSemanticsUtils.resolveCheckboxFillColor(mockSemanticsNode)) .thenReturn(fakeFillColor) - whenever( - mockPathUtils.convertPathToBitmap( - checkPath = eq(mockPath), - fillColor = anyOrNull(), - checkmarkColor = anyOrNull() - ) - ).thenReturn(mockBitmap) + whenever(mockColorUtils.parseColorSafe(fakeFillColorHexString)) + .thenReturn(fakeFillColor.toInt()) + + whenever(mockColorUtils.parseColorSafe(fakeCheckmarkColorHexString)) + .thenReturn(fakeCheckmarkColor.toInt()) whenever( - mockUiContext.imageWireframeHelper.createImageWireframeByBitmap( + mockUiContext.imageWireframeHelper.createImageWireframeByPath( id = any(), - globalBounds = any(), - bitmap = any(), - density = any(), - isContextualImage = any(), - imagePrivacy = any(), - asyncJobStatusCallback = any(), - clipping = anyOrNull(), + globalBounds = eq(fakeGlobalBounds), + path = any(), + strokeColor = eq(fakeCheckmarkColor.toInt()), + strokeWidth = eq(STROKE_WIDTH_DP.toInt()), + targetWidth = eq(CHECKBOX_SIZE_DP), + targetHeight = eq(CHECKBOX_SIZE_DP), + density = eq(fakeDensity), + isContextualImage = eq(false), + imagePrivacy = eq(ImagePrivacy.MASK_NONE), + asyncJobStatusCallback = eq(mockAsyncJobStatusCallback), + clipping = eq(null), shapeStyle = anyOrNull(), - border = anyOrNull() + border = anyOrNull(), + customResourceIdCacheKey = eq(null) ) ).thenReturn(mock()) diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt new file mode 100644 index 0000000000..e10d1ac934 --- /dev/null +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt @@ -0,0 +1,60 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.compose.internal.utils + +import com.datadog.android.sessionreplay.compose.test.elmyr.SessionReplayComposeForgeConfigurator +import fr.xgouchet.elmyr.Case +import fr.xgouchet.elmyr.Forge +import fr.xgouchet.elmyr.junit5.ForgeConfiguration +import fr.xgouchet.elmyr.junit5.ForgeExtension +import org.assertj.core.api.Assertions.assertThat +import org.junit.jupiter.api.BeforeEach +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.extension.ExtendWith +import org.junit.jupiter.api.extension.Extensions +import org.mockito.junit.jupiter.MockitoExtension +import org.mockito.junit.jupiter.MockitoSettings +import org.mockito.quality.Strictness + +@Extensions( + ExtendWith(MockitoExtension::class), + ExtendWith(ForgeExtension::class) +) +@MockitoSettings(strictness = Strictness.LENIENT) +@ForgeConfiguration(SessionReplayComposeForgeConfigurator::class) +internal class ColorUtilsTest { + + private lateinit var testedUtils: ColorUtils + + @BeforeEach + fun `set up`() { + testedUtils = ColorUtils() + } + + @Test + fun `M return input value W convertRgbaToArgb() { length lt 2 }`( + forge: Forge + ) { + // Given + val input = forge.anAsciiString(1) + + // Then + assertThat(testedUtils.convertRgbaToArgb(input)).isEqualTo(input) + } + + @Test + fun `M move alpha value W convertRgbaToArgb()`( + forge: Forge + ) { + // Given + val fakeColorHexString = forge.anHexadecimalString(Case.UPPER, 8) + val expectedResult = "#" + fakeColorHexString.substring(6) + fakeColorHexString.substring(0, 6) + + // Then + assertThat(testedUtils.convertRgbaToArgb("#$fakeColorHexString")).isEqualTo(expectedResult) + } +} diff --git a/features/dd-sdk-android-session-replay/api/apiSurface b/features/dd-sdk-android-session-replay/api/apiSurface index b85ba8dcd4..c073ec3f01 100644 --- a/features/dd-sdk-android-session-replay/api/apiSurface +++ b/features/dd-sdk-android-session-replay/api/apiSurface @@ -140,6 +140,7 @@ interface com.datadog.android.sessionreplay.utils.DrawableToColorMapper data class com.datadog.android.sessionreplay.utils.GlobalBounds constructor(Long, Long, Long, Long) interface com.datadog.android.sessionreplay.utils.ImageWireframeHelper + fun createImageWireframeByPath(Long, GlobalBounds, android.graphics.Path, Int, Int, Int, Int, Float, Boolean, com.datadog.android.sessionreplay.ImagePrivacy, AsyncJobStatusCallback, com.datadog.android.sessionreplay.model.MobileSegment.WireframeClip? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeStyle? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeBorder? = null, String?): com.datadog.android.sessionreplay.model.MobileSegment.Wireframe? fun createImageWireframeByBitmap(Long, GlobalBounds, android.graphics.Bitmap, Float, Boolean, com.datadog.android.sessionreplay.ImagePrivacy, AsyncJobStatusCallback, com.datadog.android.sessionreplay.model.MobileSegment.WireframeClip? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeStyle? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeBorder? = null): com.datadog.android.sessionreplay.model.MobileSegment.Wireframe? fun createImageWireframeByDrawable(android.view.View, com.datadog.android.sessionreplay.ImagePrivacy, Int, Long, Long, Int, Int, Boolean, android.graphics.drawable.Drawable, com.datadog.android.sessionreplay.recorder.resources.DrawableCopier = DefaultDrawableCopier(), AsyncJobStatusCallback, com.datadog.android.sessionreplay.model.MobileSegment.WireframeClip? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeStyle? = null, com.datadog.android.sessionreplay.model.MobileSegment.ShapeBorder? = null, String? = DRAWABLE_CHILD_NAME, String?): com.datadog.android.sessionreplay.model.MobileSegment.Wireframe? fun createCompoundDrawableWireframes(android.widget.TextView, com.datadog.android.sessionreplay.recorder.MappingContext, Int, String?, AsyncJobStatusCallback): MutableList diff --git a/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api b/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api index 67e54151c5..d4021fde85 100644 --- a/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api +++ b/features/dd-sdk-android-session-replay/api/dd-sdk-android-session-replay.api @@ -1617,6 +1617,7 @@ public abstract interface class com/datadog/android/sessionreplay/utils/ImageWir public abstract fun createCompoundDrawableWireframes (Landroid/widget/TextView;Lcom/datadog/android/sessionreplay/recorder/MappingContext;ILjava/lang/String;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;)Ljava/util/List; public abstract fun createImageWireframeByBitmap (JLcom/datadog/android/sessionreplay/utils/GlobalBounds;Landroid/graphics/Bitmap;FZLcom/datadog/android/sessionreplay/ImagePrivacy;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; public abstract fun createImageWireframeByDrawable (Landroid/view/View;Lcom/datadog/android/sessionreplay/ImagePrivacy;IJJIIZLandroid/graphics/drawable/Drawable;Lcom/datadog/android/sessionreplay/recorder/resources/DrawableCopier;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;Ljava/lang/String;Ljava/lang/String;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; + public abstract fun createImageWireframeByPath (JLcom/datadog/android/sessionreplay/utils/GlobalBounds;Landroid/graphics/Path;IIIIFZLcom/datadog/android/sessionreplay/ImagePrivacy;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;Ljava/lang/String;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; } public final class com/datadog/android/sessionreplay/utils/ImageWireframeHelper$Companion { @@ -1625,6 +1626,7 @@ public final class com/datadog/android/sessionreplay/utils/ImageWireframeHelper$ public final class com/datadog/android/sessionreplay/utils/ImageWireframeHelper$DefaultImpls { public static synthetic fun createImageWireframeByBitmap$default (Lcom/datadog/android/sessionreplay/utils/ImageWireframeHelper;JLcom/datadog/android/sessionreplay/utils/GlobalBounds;Landroid/graphics/Bitmap;FZLcom/datadog/android/sessionreplay/ImagePrivacy;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;ILjava/lang/Object;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; public static synthetic fun createImageWireframeByDrawable$default (Lcom/datadog/android/sessionreplay/utils/ImageWireframeHelper;Landroid/view/View;Lcom/datadog/android/sessionreplay/ImagePrivacy;IJJIIZLandroid/graphics/drawable/Drawable;Lcom/datadog/android/sessionreplay/recorder/resources/DrawableCopier;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;Ljava/lang/String;Ljava/lang/String;ILjava/lang/Object;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; + public static synthetic fun createImageWireframeByPath$default (Lcom/datadog/android/sessionreplay/utils/ImageWireframeHelper;JLcom/datadog/android/sessionreplay/utils/GlobalBounds;Landroid/graphics/Path;IIIIFZLcom/datadog/android/sessionreplay/ImagePrivacy;Lcom/datadog/android/sessionreplay/utils/AsyncJobStatusCallback;Lcom/datadog/android/sessionreplay/model/MobileSegment$WireframeClip;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeStyle;Lcom/datadog/android/sessionreplay/model/MobileSegment$ShapeBorder;Ljava/lang/String;ILjava/lang/Object;)Lcom/datadog/android/sessionreplay/model/MobileSegment$Wireframe; } public class com/datadog/android/sessionreplay/utils/LegacyDrawableToColorMapper : com/datadog/android/sessionreplay/utils/DrawableToColorMapper { diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/SessionReplayRecorder.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/SessionReplayRecorder.kt index 4552c232ed..04cf968f3a 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/SessionReplayRecorder.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/SessionReplayRecorder.kt @@ -40,6 +40,7 @@ import com.datadog.android.sessionreplay.internal.resources.ResourceDataStoreMan import com.datadog.android.sessionreplay.internal.storage.RecordWriter import com.datadog.android.sessionreplay.internal.storage.ResourcesWriter import com.datadog.android.sessionreplay.internal.utils.DrawableUtils +import com.datadog.android.sessionreplay.internal.utils.PathUtils import com.datadog.android.sessionreplay.internal.utils.RumContextProvider import com.datadog.android.sessionreplay.internal.utils.TimeProvider import com.datadog.android.sessionreplay.recorder.OptionSelectorDetector @@ -151,6 +152,7 @@ internal class SessionReplayRecorder : OnWindowRefreshedCallback, Recorder { val resourceResolver = ResourceResolver( applicationId = applicationId, recordedDataQueueHandler = recordedDataQueueHandler, + pathUtils = PathUtils(internalLogger, bitmapCachesManager), bitmapCachesManager = bitmapCachesManager, drawableUtils = DrawableUtils( internalLogger, diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelper.kt index d9bae6304e..2f84e4ab0a 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelper.kt @@ -7,6 +7,7 @@ package com.datadog.android.sessionreplay.internal.recorder.resources import android.graphics.Bitmap +import android.graphics.Path import android.graphics.drawable.Drawable import android.graphics.drawable.InsetDrawable import android.graphics.drawable.LayerDrawable @@ -38,6 +39,93 @@ internal class DefaultImageWireframeHelper( private val imageTypeResolver: ImageTypeResolver ) : ImageWireframeHelper { + @Suppress("ReturnCount") + @UiThread + override fun createImageWireframeByPath( + id: Long, + globalBounds: GlobalBounds, + path: Path, + strokeColor: Int, + strokeWidth: Int, + targetWidth: Int, + targetHeight: Int, + density: Float, + isContextualImage: Boolean, + imagePrivacy: ImagePrivacy, + asyncJobStatusCallback: AsyncJobStatusCallback, + clipping: MobileSegment.WireframeClip?, + shapeStyle: MobileSegment.ShapeStyle?, + border: MobileSegment.ShapeBorder?, + customResourceIdCacheKey: String? + ): MobileSegment.Wireframe { + if (imagePrivacy == ImagePrivacy.MASK_ALL) { + return createContentPlaceholderWireframe( + id = id, + x = globalBounds.x, + y = globalBounds.y, + width = targetWidth.toLong(), + height = targetHeight.toLong(), + label = MASK_ALL_CONTENT_LABEL, + clipping = clipping + ) + } + + // in case we suspect the image is PII, return a placeholder + if (shouldMaskContextualImage( + imagePrivacy = imagePrivacy, + usePIIPlaceholder = isContextualImage, + width = targetWidth.densityNormalized(density), + height = targetHeight.densityNormalized(density) + ) + ) { + return createContentPlaceholderWireframe( + id = id, + x = globalBounds.x, + y = globalBounds.y, + width = targetWidth.toLong(), + height = targetHeight.toLong(), + label = MASK_CONTEXTUAL_CONTENT_LABEL, + clipping = clipping + ) + } + + val imageWireframe = + MobileSegment.Wireframe.ImageWireframe( + id = id, + x = globalBounds.x, + y = globalBounds.y, + width = targetWidth.toLong(), + height = targetHeight.toLong(), + shapeStyle = shapeStyle, + border = border, + clip = clipping, + isEmpty = true + ) + + asyncJobStatusCallback.jobStarted() + + resourceResolver.resolveResourceIdFromPath( + path = path, + strokeColor = strokeColor, + strokeWidth = strokeWidth, + desiredWidth = targetWidth, + desiredHeight = targetHeight, + customResourceIdCacheKey = customResourceIdCacheKey, + resourceResolverCallback = object : ResourceResolverCallback { + override fun onSuccess(resourceId: String) { + populateResourceIdInWireframe(resourceId, imageWireframe) + asyncJobStatusCallback.jobFinished() + } + + override fun onFailure() { + asyncJobStatusCallback.jobFinished() + } + } + ) + + return imageWireframe + } + @Suppress("ReturnCount", "LongMethod") @UiThread override fun createImageWireframeByBitmap( @@ -83,7 +171,7 @@ internal class DefaultImageWireframeHelper( asyncJobStatusCallback.jobStarted() - resourceResolver.resolveResourceId( + resourceResolver.resolveResourceIdFromBitmap( bitmap = bitmap, resourceResolverCallback = object : ResourceResolverCallback { override fun onSuccess(resourceId: String) { @@ -200,7 +288,7 @@ internal class DefaultImageWireframeHelper( asyncJobStatusCallback.jobStarted() - resourceResolver.resolveResourceId( + resourceResolver.resolveResourceIdFromDrawable( resources = resources, applicationContext = applicationContext, displayMetrics = displayMetrics, @@ -363,6 +451,16 @@ internal class DefaultImageWireframeHelper( } } + private fun shouldMaskContextualImage( + imagePrivacy: ImagePrivacy, + usePIIPlaceholder: Boolean, + width: Int, + height: Int + ): Boolean = + imagePrivacy == ImagePrivacy.MASK_LARGE_ONLY && + usePIIPlaceholder && + imageTypeResolver.isPIIByDimensions(width, height) + private fun shouldMaskContextualImage( imagePrivacy: ImagePrivacy, usePIIPlaceholder: Boolean, diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ImageTypeResolver.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ImageTypeResolver.kt index 4417eb3927..2898eef039 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ImageTypeResolver.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ImageTypeResolver.kt @@ -14,12 +14,15 @@ import com.datadog.android.internal.utils.densityNormalized internal class ImageTypeResolver { fun isDrawablePII(drawable: Drawable, density: Float): Boolean { val isNotGradient = drawable !is GradientDrawable - val widthAboveThreshold = drawable.intrinsicWidth.densityNormalized(density) >= - IMAGE_DIMEN_CONSIDERED_PII_IN_DP - val heightAboveThreshold = drawable.intrinsicHeight.densityNormalized(density) >= - IMAGE_DIMEN_CONSIDERED_PII_IN_DP + val widthDp = drawable.intrinsicWidth.densityNormalized(density) + val heightDp = drawable.intrinsicHeight.densityNormalized(density) - return isNotGradient && (widthAboveThreshold || heightAboveThreshold) + return isNotGradient && isPIIByDimensions(widthDp, heightDp) + } + + fun isPIIByDimensions(width: Int, height: Int): Boolean { + val isGreaterThan = width >= IMAGE_DIMEN_CONSIDERED_PII_IN_DP || height >= IMAGE_DIMEN_CONSIDERED_PII_IN_DP + return isGreaterThan } internal companion object { diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt index e436a3cebb..b7d757761a 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt @@ -9,6 +9,7 @@ package com.datadog.android.sessionreplay.internal.recorder.resources import android.content.Context import android.content.res.Resources import android.graphics.Bitmap +import android.graphics.Path import android.graphics.drawable.BitmapDrawable import android.graphics.drawable.Drawable import android.util.DisplayMetrics @@ -18,6 +19,7 @@ import com.datadog.android.api.InternalLogger import com.datadog.android.core.internal.utils.executeSafe import com.datadog.android.sessionreplay.internal.async.DataQueueHandler import com.datadog.android.sessionreplay.internal.utils.DrawableUtils +import com.datadog.android.sessionreplay.internal.utils.PathUtils import com.datadog.android.sessionreplay.recorder.resources.DrawableCopier import java.util.concurrent.ExecutorService import java.util.concurrent.LinkedBlockingDeque @@ -27,6 +29,7 @@ import java.util.concurrent.TimeUnit @Suppress("TooManyFunctions") internal class ResourceResolver( private val bitmapCachesManager: BitmapCachesManager, + private val pathUtils: PathUtils, internal val threadPoolExecutor: ExecutorService = THREADPOOL_EXECUTOR, private val drawableUtils: DrawableUtils, private val webPImageCompression: ImageCompression, @@ -40,40 +43,78 @@ internal class ResourceResolver( ) ) { - // region internal - @MainThread - internal fun resolveResourceId( + internal fun resolveResourceIdFromBitmap( bitmap: Bitmap, resourceResolverCallback: ResourceResolverCallback ) { - threadPoolExecutor.executeSafe("resolveResourceId", logger) { - val compressedBitmapBytes = webPImageCompression.compressBitmap(bitmap) + threadPoolExecutor.executeSafe(RESOURCE_RESOLVER_ALIAS, logger) { + getResourceIdFromBitmap(bitmap, resourceResolverCallback) + } + } - // failed to compress bitmap - if (compressedBitmapBytes.isEmpty()) { - resourceResolverCallback.onFailure() - } else { - resolveBitmapHash( - compressedBitmapBytes = compressedBitmapBytes, - resolveResourceCallback = object : ResolveResourceCallback { - override fun onResolved(resourceId: String, resourceData: ByteArray) { - resourceItemCreationHandler.queueItem(resourceId, resourceData) - resourceResolverCallback.onSuccess(resourceId) - } - - override fun onFailed() { - resourceResolverCallback.onFailure() - } + @MainThread + internal fun resolveResourceIdFromPath( + path: Path, + strokeColor: Int, + strokeWidth: Int, + desiredWidth: Int, + desiredHeight: Int, + customResourceIdCacheKey: String?, + resourceResolverCallback: ResourceResolverCallback + ) { + threadPoolExecutor.executeSafe(RESOURCE_RESOLVER_ALIAS, logger) { + val key = + customResourceIdCacheKey + ?: path.let { + pathUtils.generateKeyForPath(it) } - ) + + val resourceId = tryToGetResourceFromCache( + drawable = null, + customResourceIdCacheKey = key + ) + + if (resourceId != null) { + // if we got here it means we saw the bitmap before, + // so we don't need to send the resource again + resourceResolverCallback.onSuccess(resourceId) + return@executeSafe + } + + val bitmap = pathUtils.convertPathToBitmap( + checkPath = path, + checkmarkColor = strokeColor, + desiredWidth = desiredWidth, + desiredHeight = desiredHeight, + strokeWidth = strokeWidth + ) + + if (bitmap == null) { + resourceResolverCallback.onFailure() + return@executeSafe } + + compressAndCacheBitmap( + drawable = null, + bitmap = bitmap, + customResourceIdCacheKey = customResourceIdCacheKey, + resolveResourceCallback = object : ResolveResourceCallback { + override fun onResolved(resourceId: String, resourceData: ByteArray) { + resourceItemCreationHandler.queueItem(resourceId, resourceData) + resourceResolverCallback.onSuccess(resourceId) + } + + override fun onFailed() { + resourceResolverCallback.onFailure() + } + } + ) } } - // endregion @MainThread - internal fun resolveResourceId( + internal fun resolveResourceIdFromDrawable( resources: Resources, applicationContext: Context, displayMetrics: DisplayMetrics, @@ -86,7 +127,8 @@ internal class ResourceResolver( ) { bitmapCachesManager.registerCallbacks(applicationContext) - val resourceId = tryToGetResourceFromCache(drawable = originalDrawable, key = customResourceIdCacheKey) + val resourceId = + tryToGetResourceFromCache(drawable = originalDrawable, customResourceIdCacheKey = customResourceIdCacheKey) if (resourceId != null) { // if we got here it means we saw the bitmap before, @@ -109,8 +151,8 @@ internal class ResourceResolver( } // do in the background - threadPoolExecutor.executeSafe("resolveResourceId", logger) { - createBitmap( + threadPoolExecutor.executeSafe(RESOURCE_RESOLVER_ALIAS, logger) { + createBitmapFromDrawable( drawable = originalDrawable, copiedDrawable = copiedDrawable, drawableWidth = drawableWidth, @@ -137,7 +179,7 @@ internal class ResourceResolver( // region private @WorkerThread - private fun createBitmap( + private fun createBitmapFromDrawable( drawable: Drawable, copiedDrawable: Drawable, drawableWidth: Int, @@ -197,7 +239,7 @@ internal class ResourceResolver( @Suppress("ReturnCount") @WorkerThread private fun resolveResourceHash( - drawable: Drawable, + drawable: Drawable?, bitmap: Bitmap, compressedBitmapBytes: ByteArray, shouldCacheBitmap: Boolean, @@ -236,14 +278,14 @@ internal class ResourceResolver( bitmap: Bitmap, resourceId: String, customResourceIdCacheKey: String?, - drawable: Drawable + drawable: Drawable? ) { if (shouldCacheBitmap) { bitmapCachesManager.putInBitmapPool(bitmap) } val key = customResourceIdCacheKey - ?: bitmapCachesManager.generateResourceKeyFromDrawable(drawable) + ?: generateKey(drawable) ?: return bitmapCachesManager.putInResourceCache(key, resourceId) @@ -267,19 +309,9 @@ internal class ResourceResolver( bitmapCreationCallback = object : BitmapCreationCallback { @WorkerThread override fun onReady(bitmap: Bitmap) { - val compressedBitmapBytes = webPImageCompression.compressBitmap(bitmap) - - // failed to compress bitmap - if (compressedBitmapBytes.isEmpty()) { - resolveResourceCallback.onFailed() - return - } - - resolveResourceHash( + compressAndCacheBitmap( drawable = originalDrawable, bitmap = bitmap, - compressedBitmapBytes = compressedBitmapBytes, - shouldCacheBitmap = true, customResourceIdCacheKey = customResourceIdCacheKey, resolveResourceCallback = resolveResourceCallback ) @@ -293,6 +325,56 @@ internal class ResourceResolver( ) } + @WorkerThread + private fun compressAndCacheBitmap( + drawable: Drawable?, + bitmap: Bitmap, + customResourceIdCacheKey: String?, + resolveResourceCallback: ResolveResourceCallback + ) { + val compressedBitmapBytes = webPImageCompression.compressBitmap(bitmap) + + // failed to compress bitmap + if (compressedBitmapBytes.isEmpty()) { + resolveResourceCallback.onFailed() + return + } + + resolveResourceHash( + drawable = drawable, + bitmap = bitmap, + compressedBitmapBytes = compressedBitmapBytes, + shouldCacheBitmap = true, + customResourceIdCacheKey = customResourceIdCacheKey, + resolveResourceCallback = resolveResourceCallback + ) + } + + @WorkerThread + private fun getResourceIdFromBitmap(bitmap: Bitmap, resourceResolverCallback: ResourceResolverCallback) { + val compressedBitmapBytes = webPImageCompression.compressBitmap(bitmap) + + // failed to compress bitmap + if (compressedBitmapBytes.isEmpty()) { + resourceResolverCallback.onFailure() + return + } else { + resolveBitmapHash( + compressedBitmapBytes = compressedBitmapBytes, + resolveResourceCallback = object : ResolveResourceCallback { + override fun onResolved(resourceId: String, resourceData: ByteArray) { + resourceItemCreationHandler.queueItem(resourceId, resourceData) + resourceResolverCallback.onSuccess(resourceId) + } + + override fun onFailed() { + resourceResolverCallback.onFailure() + } + } + ) + } + } + @WorkerThread @Suppress("ReturnCount") private fun tryToGetBitmapFromBitmapDrawable( @@ -336,14 +418,22 @@ internal class ResourceResolver( } private fun tryToGetResourceFromCache( - drawable: Drawable, - key: String? + drawable: Drawable?, + customResourceIdCacheKey: String? ): String? { - val cacheKey = key - ?: bitmapCachesManager.generateResourceKeyFromDrawable(drawable) + val key = customResourceIdCacheKey + ?: generateKey(drawable) ?: return null - return bitmapCachesManager.getFromResourceCache(cacheKey) + return bitmapCachesManager.getFromResourceCache(key) + } + + private fun generateKey(drawable: Drawable?): String? { + return if (drawable != null) { + bitmapCachesManager.generateResourceKeyFromDrawable(drawable) + } else { + null + } } private fun shouldUseDrawableBitmap(drawable: BitmapDrawable): Boolean { @@ -370,6 +460,7 @@ internal class ResourceResolver( private const val THREAD_POOL_MAX_KEEP_ALIVE_MS = 5000L private const val CORE_DEFAULT_POOL_SIZE = 1 private const val MAX_THREAD_COUNT = 10 + private const val RESOURCE_RESOLVER_ALIAS = "resolveResourceId" @Suppress("UnsafeThirdPartyFunctionCall") // all parameters are non-negative and queue is not null private val THREADPOOL_EXECUTOR = ThreadPoolExecutor( diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt new file mode 100644 index 0000000000..ecb4e08af9 --- /dev/null +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt @@ -0,0 +1,178 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2016-Present Datadog, Inc. + */ + +package com.datadog.android.sessionreplay.internal.utils + +import android.graphics.Bitmap +import android.graphics.Canvas +import android.graphics.Matrix +import android.graphics.Paint +import android.graphics.Path +import android.graphics.PathMeasure +import android.graphics.RectF +import com.datadog.android.api.InternalLogger +import com.datadog.android.sessionreplay.internal.recorder.resources.BitmapCachesManager +import com.datadog.android.sessionreplay.internal.recorder.resources.HashGenerator +import com.datadog.android.sessionreplay.internal.recorder.resources.MD5HashGenerator +import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper +import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper + +internal class PathUtils( + private val logger: InternalLogger = InternalLogger.UNBOUND, + private val bitmapCachesManager: BitmapCachesManager, + private val canvasWrapper: CanvasWrapper = CanvasWrapper(logger), + private val bitmapWrapper: BitmapWrapper = BitmapWrapper(), + private val md5Generator: HashGenerator = MD5HashGenerator(logger) +) { + internal fun convertPathToBitmap( + checkPath: Path, + checkmarkColor: Int, + desiredWidth: Int, + desiredHeight: Int, + strokeWidth: Int + ): Bitmap? { + val scaledPath = scalePathToTargetDimensions(checkPath, desiredWidth, desiredHeight) + + val mutableBitmap = bitmapCachesManager.getBitmapByProperties( + width = desiredWidth, + height = desiredHeight, + config = Bitmap.Config.ARGB_8888 + ) + ?: bitmapWrapper.createBitmap(desiredWidth, desiredHeight, Bitmap.Config.ARGB_8888) + ?: return null + + return drawPathOntoBitmap( + bitmap = mutableBitmap, + scaledPath = scaledPath, + strokeWidth = strokeWidth, + checkmarkColor = checkmarkColor + ) + } + + private fun drawPathToBitmap( + checkmarkColor: Int, + path: Path, + targetStrokeWidth: Int, + canvas: Canvas? + ) { + val paint = Paint().apply { + color = checkmarkColor + style = Paint.Style.STROKE + strokeWidth = targetStrokeWidth.toFloat() + isAntiAlias = true + } + + // Draw the Path onto the Canvas + drawPathSafe(canvas, path, paint) + } + + private fun scalePathToTargetDimensions( + path: Path, + targetWidth: Int, + targetHeight: Int + ): Path { + // path initial bounds + val originalBounds = RectF() + + @Suppress("DEPRECATION") // the new api is flagged as unstable + path.computeBounds(originalBounds, true) + + // calculate the scale factor + val scaleX = targetWidth / originalBounds.width() + val scaleY = targetHeight / originalBounds.height() + val scaleFactor = minOf(scaleX, scaleY) + + // current center + val currentCenterX = (originalBounds.left + originalBounds.right) / 2 + val currentCenterY = (originalBounds.top + originalBounds.bottom) / 2 + + // new center + val newCenterX = targetWidth / 2 + val newCenterY = targetHeight / 2 + + // center changes after scaling + val scaledCenterX = currentCenterX * scaleFactor + val scaledCenterY = currentCenterY * scaleFactor + + // translation needed to recenter + val translateX = newCenterX - scaledCenterX + val translateY = newCenterY - scaledCenterY + + // the order of operations is important + val matrix = Matrix() + matrix.preTranslate(translateX, translateY) + matrix.preScale(scaleFactor, scaleFactor) + path.transform(matrix) + + return path + } + + private fun drawPathOntoBitmap( + bitmap: Bitmap, + scaledPath: Path, + strokeWidth: Int, + checkmarkColor: Int + ): Bitmap? { + val canvas = canvasWrapper.createCanvas(bitmap) ?: return null + + // draw the checkmark + drawPathToBitmap(checkmarkColor, scaledPath, strokeWidth, canvas) + + return bitmap + } + + @Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException + private fun drawPathSafe(canvas: Canvas?, path: Path, paint: Paint) { + try { + canvas?.drawPath(path, paint) + } catch (e: IllegalArgumentException) { + logger.log( + target = InternalLogger.Target.MAINTAINER, + level = InternalLogger.Level.WARN, + messageBuilder = { PATH_DRAW_ERROR }, + throwable = e + ) + } + } + + internal fun generateKeyForPath( + path: Path, + maxPoints: Int = DEFAULT_MAX_PATH_LENGTH, + sampleInterval: Float = DEFAULT_SAMPLE_INTERVAL, + pathMeasure: PathMeasure = PathMeasure(path, false) + ): String? { + val pos = FloatArray(2) + val tan = FloatArray(2) + val sampledPoints = StringBuilder() + var pointCount = 0 + + var distance = 0f + while (distance < pathMeasure.length && pointCount < maxPoints) { + @Suppress("UnsafeThirdPartyFunctionCall") // pos and tan size not lt 2 + pathMeasure.getPosTan(distance, pos, tan) + + sampledPoints.append("${pos[0]},${pos[1]};") + pointCount++ + distance += sampleInterval + if (!pathMeasure.nextContour()) break + } + + val points = sampledPoints.toString() + + return if (points == EMPTY_POINTS) { + null + } else { + md5Generator.generate(points.toByteArray()) + } + } + + internal companion object { + internal const val PATH_DRAW_ERROR = "Failed to draw Path to Canvas" + internal const val EMPTY_POINTS = "0.0,0.0;" + internal const val DEFAULT_MAX_PATH_LENGTH = 1000 + internal const val DEFAULT_SAMPLE_INTERVAL = 10f + } +} diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt index d627d49fe0..0c6790bc6d 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt @@ -7,6 +7,7 @@ package com.datadog.android.sessionreplay.utils import android.graphics.Bitmap +import android.graphics.Path import android.graphics.drawable.Drawable import android.view.View import android.widget.TextView @@ -22,6 +23,44 @@ import com.datadog.android.sessionreplay.recorder.resources.DrawableCopier */ interface ImageWireframeHelper { + /** + * Asks the helper to create an image wireframe based on a given path. + * @param id the unique id for the wireframe. + * @param globalBounds the global bounds of the bitmap. + * @param path the path to use to create the wireframe. + * @param strokeColor the color of the stroke. + * @param strokeWidth the width of the stroke. + * @param targetWidth the target width of the image. + * @param targetHeight the target height of the image. + * @param density the density of the screen. + * @param isContextualImage if the image is contextual. + * @param imagePrivacy defines which images should be hidden. + * @param asyncJobStatusCallback the callback for the async capture process. + * @param clipping the bounds of the image that are actually visible. + * @param shapeStyle provides a custom shape (e.g. rounded corners) to the image wireframe. + * @param border provides a custom border to the image wireframe. + * @param customResourceIdCacheKey an optional custom key with which to cache or retrieve from the resource cache. + * If this key is not provided then one will be generated from the path. + */ + @Suppress("LongParameterList") + fun createImageWireframeByPath( + id: Long, + globalBounds: GlobalBounds, + path: Path, + strokeColor: Int, + strokeWidth: Int, + targetWidth: Int, + targetHeight: Int, + density: Float, + isContextualImage: Boolean, + imagePrivacy: ImagePrivacy, + asyncJobStatusCallback: AsyncJobStatusCallback, + clipping: MobileSegment.WireframeClip? = null, + shapeStyle: MobileSegment.ShapeStyle? = null, + border: MobileSegment.ShapeBorder? = null, + customResourceIdCacheKey: String? + ): MobileSegment.Wireframe? + /** * Asks the helper to create an image wireframe based on a given bitmap. * @param id the unique id for the wireframe. @@ -65,7 +104,7 @@ interface ImageWireframeHelper { * @param shapeStyle provides a custom shape (e.g. rounded corners) to the image wireframe * @param border provides a custom border to the image wireframe * @param prefix a prefix identifying the drawable in the parent view's context - * @param customResourceIdCacheKey an optional key with which to cache or retrieve from the resource cache. + * @param customResourceIdCacheKey an optional custom key with which to cache or retrieve from the resource cache. * If this key is not provided then one will be generated from the drawable. */ // TODO RUM-3666 limit the number of params to this function @@ -93,7 +132,7 @@ interface ImageWireframeHelper { * @param textView the [TextView] to capture the compound drawables from. * @param mappingContext the [MappingContext] for the [TextView]. * @param prevWireframeIndex the index of the previous wireframe in the list of wireframes for the [TextView]. - * @param customResourceIdCacheKey an optional key with which to cache or retrieve from the resource cache. + * @param customResourceIdCacheKey an optional custom key with which to cache or retrieve from the resource cache. * If this key is not provided then one will be generated from the drawable. * @param asyncJobStatusCallback the callback for the async capture process. */ diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt index b0b8428e35..f717801d84 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt @@ -9,6 +9,7 @@ package com.datadog.android.sessionreplay.internal.recorder.resources import android.content.Context import android.content.res.Resources import android.graphics.Bitmap +import android.graphics.Path import android.graphics.drawable.Drawable import android.graphics.drawable.GradientDrawable import android.graphics.drawable.InsetDrawable @@ -23,6 +24,7 @@ import com.datadog.android.sessionreplay.forge.ForgeConfigurator import com.datadog.android.sessionreplay.internal.recorder.ViewUtilsInternal import com.datadog.android.sessionreplay.internal.recorder.resources.DefaultImageWireframeHelper.Companion.APPLICATION_CONTEXT_NULL_ERROR import com.datadog.android.sessionreplay.internal.recorder.resources.DefaultImageWireframeHelper.Companion.RESOURCES_NULL_ERROR +import com.datadog.android.sessionreplay.internal.recorder.resources.ImageTypeResolver.Companion.IMAGE_DIMEN_CONSIDERED_PII_IN_DP import com.datadog.android.sessionreplay.model.MobileSegment import com.datadog.android.sessionreplay.recorder.MappingContext import com.datadog.android.sessionreplay.recorder.SystemInformation @@ -191,6 +193,8 @@ internal class DefaultImageWireframeHelperTest { ) } + // region createImageWireframeByBitmap + @Test fun `M return wireframe W createImageWireframeByBitmap`( @Mock mockShapeStyle: MobileSegment.ShapeStyle, @@ -199,7 +203,7 @@ internal class DefaultImageWireframeHelperTest { ) { // Given whenever( - mockResourceResolver.resolveResourceId( + mockResourceResolver.resolveResourceIdFromBitmap( bitmap = any(), resourceResolverCallback = any() ) @@ -236,7 +240,7 @@ internal class DefaultImageWireframeHelperTest { ) // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromBitmap( bitmap = any(), resourceResolverCallback = any() ) @@ -246,8 +250,6 @@ internal class DefaultImageWireframeHelperTest { assertThat(wireframe).isEqualTo(expectedWireframe) } - // region createImageWireframeByBitmap - @Test fun `M return content placeholder W createImageWireframeByBitmap { ImagePrivacy MASK_ALL }`() { // When @@ -293,7 +295,7 @@ internal class DefaultImageWireframeHelperTest { .thenReturn(fakeGeneratedIdentifier) whenever( - mockResourceResolver.resolveResourceId( + mockResourceResolver.resolveResourceIdFromBitmap( bitmap = any(), resourceResolverCallback = any() ) @@ -331,7 +333,7 @@ internal class DefaultImageWireframeHelperTest { whenever(mockViewIdentifierResolver.resolveChildUniqueIdentifier(any(), any())) .thenReturn(fakeGeneratedIdentifier) whenever( - mockResourceResolver.resolveResourceId( + mockResourceResolver.resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -399,7 +401,7 @@ internal class DefaultImageWireframeHelperTest { ) // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -666,7 +668,7 @@ internal class DefaultImageWireframeHelperTest { whenever(mockViewIdentifierResolver.resolveChildUniqueIdentifier(any(), any())) .thenReturn(fakeGeneratedIdentifier) whenever( - mockResourceResolver.resolveResourceId( + mockResourceResolver.resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -714,7 +716,7 @@ internal class DefaultImageWireframeHelperTest { ) // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -739,7 +741,7 @@ internal class DefaultImageWireframeHelperTest { ) { // Given whenever( - mockResourceResolver.resolveResourceId( + mockResourceResolver.resolveResourceIdFromBitmap( bitmap = any(), resourceResolverCallback = any() ) @@ -776,7 +778,7 @@ internal class DefaultImageWireframeHelperTest { ) // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromBitmap( bitmap = any(), resourceResolverCallback = any() ) @@ -906,7 +908,7 @@ internal class DefaultImageWireframeHelperTest { // Then val argumentCaptor = argumentCaptor() - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -954,7 +956,7 @@ internal class DefaultImageWireframeHelperTest { // Then val argumentCaptor = argumentCaptor() - verify(mockResourceResolver, times(2)).resolveResourceId( + verify(mockResourceResolver, times(2)).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1027,7 +1029,7 @@ internal class DefaultImageWireframeHelperTest { // Then val captor = argumentCaptor() - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1062,7 +1064,7 @@ internal class DefaultImageWireframeHelperTest { // Then val captor = argumentCaptor() - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1148,7 +1150,7 @@ internal class DefaultImageWireframeHelperTest { ) // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1223,7 +1225,7 @@ internal class DefaultImageWireframeHelperTest { val expectedKey = fakeResourceIdCacheKey + "_$index" // Then - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1287,7 +1289,7 @@ internal class DefaultImageWireframeHelperTest { // Then val argumentCaptor = argumentCaptor() - verify(mockResourceResolver).resolveResourceId( + verify(mockResourceResolver).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1335,7 +1337,7 @@ internal class DefaultImageWireframeHelperTest { // Then val argumentCaptor = argumentCaptor() - verify(mockResourceResolver, times(2)).resolveResourceId( + verify(mockResourceResolver, times(2)).resolveResourceIdFromDrawable( resources = any(), applicationContext = any(), displayMetrics = any(), @@ -1375,4 +1377,195 @@ internal class DefaultImageWireframeHelperTest { } // endregion + + // region createImageWireframeByPath + + @Test + fun `M return content placeholder W createImageWireframeByPath { ImagePrivacy MASK_ALL }`( + @Mock mockPath: Path, + forge: Forge + ) { + // Given + val fakeWidth = forge.aPositiveInt() + val fakeHeight = forge.aPositiveInt() + val fakeStrokeWidth = forge.aPositiveInt() + val fakeStrokeColor = forge.aPositiveInt() + + // When + val wireframe = testedHelper.createImageWireframeByPath( + id = fakeViewId, + path = mockPath, + imagePrivacy = ImagePrivacy.MASK_ALL, + isContextualImage = false, + globalBounds = fakeBounds, + shapeStyle = null, + border = null, + density = fakeDensity, + strokeWidth = fakeStrokeWidth, + strokeColor = fakeStrokeColor, + targetWidth = fakeWidth, + targetHeight = fakeHeight, + customResourceIdCacheKey = null, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + // Then + assertThat(wireframe).isInstanceOf(MobileSegment.Wireframe.PlaceholderWireframe::class.java) + } + + @Test + fun `M return content placeholder W createImageWireframeByPath { ImagePrivacy MASK_LARGE_ONLY }`( + @Mock mockPath: Path, + forge: Forge + ) { + // Given + val fakeWidth = forge.anInt(min = IMAGE_DIMEN_CONSIDERED_PII_IN_DP) + val fakeHeight = forge.anInt(min = IMAGE_DIMEN_CONSIDERED_PII_IN_DP) + val fakeStrokeWidth = forge.aPositiveInt() + val fakeStrokeColor = forge.aPositiveInt() + whenever(mockImageTypeResolver.isPIIByDimensions(any(), any())).thenReturn(true) + + // When + val wireframe = testedHelper.createImageWireframeByPath( + id = fakeViewId, + path = mockPath, + imagePrivacy = ImagePrivacy.MASK_LARGE_ONLY, + isContextualImage = true, + globalBounds = fakeBounds, + shapeStyle = null, + border = null, + density = fakeDensity, + strokeWidth = fakeStrokeWidth, + strokeColor = fakeStrokeColor, + targetWidth = fakeWidth, + targetHeight = fakeHeight, + customResourceIdCacheKey = null, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + // Then + assertThat(wireframe).isInstanceOf(MobileSegment.Wireframe.PlaceholderWireframe::class.java) + } + + @Test + fun `M return image wireframe W createImageWireframeByPath`( + @Mock mockPath: Path, + forge: Forge + ) { + // Given + val fakeWidth = forge.aPositiveInt() + val fakeHeight = forge.aPositiveInt() + val fakeStrokeWidth = forge.aPositiveInt() + val fakeStrokeColor = forge.aPositiveInt() + + whenever( + mockResourceResolver.resolveResourceIdFromPath( + path = any(), + strokeColor = any(), + strokeWidth = any(), + desiredWidth = any(), + desiredHeight = any(), + customResourceIdCacheKey = anyOrNull(), + resourceResolverCallback = any() + ) + ).thenAnswer { + val callback = it.arguments[6] as ResourceResolverCallback + callback.onSuccess(fakeResourceId) + } + + // When + val wireframe = testedHelper.createImageWireframeByPath( + id = fakeViewId, + path = mockPath, + imagePrivacy = ImagePrivacy.MASK_NONE, + isContextualImage = false, + globalBounds = fakeBounds, + shapeStyle = null, + border = null, + density = fakeDensity, + strokeWidth = fakeStrokeWidth, + strokeColor = fakeStrokeColor, + targetWidth = fakeWidth, + targetHeight = fakeHeight, + customResourceIdCacheKey = null, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + // Then + assertThat(wireframe).isInstanceOf(MobileSegment.Wireframe.ImageWireframe::class.java) + verify(mockResourceResolver).resolveResourceIdFromPath( + path = any(), + strokeColor = eq(fakeStrokeColor), + strokeWidth = eq(fakeStrokeWidth), + desiredWidth = eq(fakeWidth), + desiredHeight = eq(fakeHeight), + customResourceIdCacheKey = anyOrNull(), + resourceResolverCallback = any() + ) + verify(mockAsyncJobStatusCallback).jobStarted() + verify(mockAsyncJobStatusCallback).jobFinished() + verifyNoMoreInteractions(mockAsyncJobStatusCallback) + } + + @Test + fun `M call jobFinished W createImageWireframeFromPath { failure }`( + @Mock mockPath: Path, + forge: Forge + ) { + // Given + val fakeWidth = forge.aPositiveInt() + val fakeHeight = forge.aPositiveInt() + val fakeStrokeWidth = forge.aPositiveInt() + val fakeStrokeColor = forge.aPositiveInt() + + whenever( + mockResourceResolver.resolveResourceIdFromPath( + path = any(), + strokeColor = any(), + strokeWidth = any(), + desiredWidth = any(), + desiredHeight = any(), + customResourceIdCacheKey = anyOrNull(), + resourceResolverCallback = any() + ) + ).thenAnswer { + val callback = it.arguments[6] as ResourceResolverCallback + callback.onFailure() + } + + // When + val wireframe = testedHelper.createImageWireframeByPath( + id = fakeViewId, + path = mockPath, + imagePrivacy = ImagePrivacy.MASK_NONE, + isContextualImage = false, + globalBounds = fakeBounds, + shapeStyle = null, + border = null, + density = fakeDensity, + strokeWidth = fakeStrokeWidth, + strokeColor = fakeStrokeColor, + targetWidth = fakeWidth, + targetHeight = fakeHeight, + customResourceIdCacheKey = null, + asyncJobStatusCallback = mockAsyncJobStatusCallback + ) + + // Then + assertThat(wireframe).isInstanceOf(MobileSegment.Wireframe.ImageWireframe::class.java) + verify(mockResourceResolver).resolveResourceIdFromPath( + path = any(), + strokeColor = eq(fakeStrokeColor), + strokeWidth = eq(fakeStrokeWidth), + desiredWidth = eq(fakeWidth), + desiredHeight = eq(fakeHeight), + customResourceIdCacheKey = anyOrNull(), + resourceResolverCallback = any() + ) + verify(mockAsyncJobStatusCallback).jobStarted() + verify(mockAsyncJobStatusCallback).jobFinished() + verifyNoMoreInteractions(mockAsyncJobStatusCallback) + } + + // endregion } diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolverTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolverTest.kt index 61a04739f2..b77fc6a346 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolverTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolverTest.kt @@ -18,6 +18,7 @@ import com.datadog.android.api.InternalLogger import com.datadog.android.sessionreplay.forge.ForgeConfigurator import com.datadog.android.sessionreplay.internal.async.RecordedDataQueueHandler import com.datadog.android.sessionreplay.internal.utils.DrawableUtils +import com.datadog.android.sessionreplay.internal.utils.PathUtils import com.datadog.android.sessionreplay.recorder.resources.DrawableCopier import fr.xgouchet.elmyr.Forge import fr.xgouchet.elmyr.annotation.Forgery @@ -105,6 +106,9 @@ internal class ResourceResolverTest { @Mock lateinit var mockBitmapDrawable: BitmapDrawable + @Mock + lateinit var mockPathUtils: PathUtils + @Mock lateinit var mockResources: Resources @@ -173,7 +177,7 @@ internal class ResourceResolverTest { } @Test - fun `M get data from cache W resolveResourceId() { cache hit with resourceId }`() { + fun `M get data from cache W resolveResourceIdFromDrawable() { cache hit with resourceId }`() { // Given whenever(mockBitmapCachesManager.getFromResourceCache(fakeResourceKey)).thenReturn(fakeResourceId) @@ -181,7 +185,7 @@ internal class ResourceResolverTest { .thenReturn(fakeImageCompressionByteArray) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -199,7 +203,7 @@ internal class ResourceResolverTest { } @Test - fun `M retry image creation only once W resolveResourceId() { image was recycled while working on it }`() { + fun `M retry image creation once W resolveResourceIdFromDrawable() { image recycled while working on it }`() { // Given whenever(mockDrawableUtils.createScaledBitmap(any(), anyOrNull())) .thenReturn(mockBitmap) @@ -219,7 +223,7 @@ internal class ResourceResolverTest { .thenReturn(fakeImageCompressionByteArray) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -244,7 +248,7 @@ internal class ResourceResolverTest { } @Test - fun `M send onReady W resolveResourceId(Drawable) { failed to get image data }`() { + fun `M send onReady W resolveResourceIdFromDrawable(Drawable) { failed to get image data }`() { // Given whenever(mockBitmap.isRecycled) .thenReturn(true) @@ -256,7 +260,7 @@ internal class ResourceResolverTest { .thenReturn(emptyByteArray) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -283,7 +287,7 @@ internal class ResourceResolverTest { ).doReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -300,7 +304,7 @@ internal class ResourceResolverTest { } @Test - fun `M send onReady W resolveResourceId(Bitmap) { failed to get image data }`() { + fun `M send onReady W resolveResourceIdFromDrawable(Bitmap) { failed to get image data }`() { // Given whenever(mockBitmap.isRecycled) .thenReturn(true) @@ -312,7 +316,7 @@ internal class ResourceResolverTest { .thenReturn(emptyByteArray) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromBitmap( bitmap = mockBitmap, resourceResolverCallback = mockSerializerCallback ) @@ -322,12 +326,12 @@ internal class ResourceResolverTest { } @Test - fun `M calculate resourceId W resolveResourceId() { cache miss }`() { + fun `M calculate resourceId W resolveResourceIdFromDrawable() { cache miss }`() { // Given whenever(mockResourcesLRUCache.get(fakeResourceKey)).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -352,7 +356,7 @@ internal class ResourceResolverTest { } @Test - fun `M return failure W resolveResourceId { createBitmapOfApproxSizeFromDrawable failed }`() { + fun `M return failure W resolveResourceIdFromDrawable { createBitmapOfApproxSizeFromDrawable failed }`() { // Given whenever(mockResourcesLRUCache.get(fakeResourceKey)).thenReturn(null) whenever( @@ -370,7 +374,7 @@ internal class ResourceResolverTest { } // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -395,6 +399,7 @@ internal class ResourceResolverTest { webPImageCompression = mockWebPImageCompression, drawableUtils = mockDrawableUtils, logger = mockLogger, + pathUtils = mockPathUtils, md5HashGenerator = mockMD5HashGenerator, bitmapCachesManager = mockBitmapCachesManager ) @@ -404,6 +409,7 @@ internal class ResourceResolverTest { webPImageCompression = mockWebPImageCompression, drawableUtils = mockDrawableUtils, logger = mockLogger, + pathUtils = mockPathUtils, md5HashGenerator = mockMD5HashGenerator, bitmapCachesManager = mockBitmapCachesManager ) @@ -415,12 +421,12 @@ internal class ResourceResolverTest { } @Test - fun `M not try to cache resourceId W resolveResourceId() { and did not get resourceId }`() { + fun `M not try to cache resourceId W resolveResourceIdFromDrawable() { and did not get resourceId }`() { // Given whenever(mockMD5HashGenerator.generate(any())).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -437,12 +443,12 @@ internal class ResourceResolverTest { } @Test - fun `M not use bitmap from bitmapDrawable W resolveResourceId() { no bitmap }`() { + fun `M not use bitmap from bitmapDrawable W resolveResourceIdFromDrawable() { no bitmap }`() { // Given whenever(mockBitmapDrawable.bitmap).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -467,12 +473,12 @@ internal class ResourceResolverTest { } @Test - fun `M not use bitmap from bitmapDrawable W resolveResourceId() { bitmap was recycled }`() { + fun `M not use bitmap from bitmapDrawable W resolveResourceIdFromDrawable() { bitmap was recycled }`() { // Given whenever(mockBitmap.isRecycled).thenReturn(true) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -497,9 +503,9 @@ internal class ResourceResolverTest { } @Test - fun `M use scaled bitmap from bitmapDrawable W resolveResourceId() { has bitmap }`() { + fun `M use scaled bitmap from bitmapDrawable W resolveResourceIdFromDrawable() { has bitmap }`() { // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -519,12 +525,12 @@ internal class ResourceResolverTest { } @Test - fun `M draw bitmap W resolveResourceId() { bitmapDrawable where bitmap has no width }`() { + fun `M draw bitmap W resolveResourceIdFromDrawable() { bitmapDrawable where bitmap has no width }`() { // Given whenever(mockBitmap.width).thenReturn(0) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -553,12 +559,12 @@ internal class ResourceResolverTest { } @Test - fun `M draw bitmap W resolveResourceId() { bitmapDrawable where bitmap has no height }`() { + fun `M draw bitmap W resolveResourceIdFromDrawable() { bitmapDrawable where bitmap has no height }`() { // Given whenever(mockBitmap.height).thenReturn(0) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -587,13 +593,13 @@ internal class ResourceResolverTest { } @Test - fun `M not cache bitmap W resolveResourceId() { BitmapDrawable with bitmap not resized }`() { + fun `M not cache bitmap W resolveResourceIdFromDrawable() { BitmapDrawable with bitmap not resized }`() { // Given whenever(mockDrawableUtils.createScaledBitmap(any(), anyOrNull())) .thenReturn(mockBitmap) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -610,7 +616,7 @@ internal class ResourceResolverTest { } @Test - fun `M cache bitmap W resolveResourceId() { BitmapDrawable width was resized }`( + fun `M cache bitmap W resolveResourceIdFromDrawable() { BitmapDrawable width was resized }`( @Mock mockResizedBitmap: Bitmap, @StringForgery fakeString: String ) { @@ -626,7 +632,7 @@ internal class ResourceResolverTest { whenever(mockDrawableUtils.createScaledBitmap(any(), anyOrNull())).thenReturn(mockResizedBitmap) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -643,7 +649,7 @@ internal class ResourceResolverTest { } @Test - fun `M cache bitmap W resolveResourceId() { BitmapDrawable height was resized }`( + fun `M cache bitmap W resolveResourceIdFromDrawable() { BitmapDrawable height was resized }`( @Mock mockResizedBitmap: Bitmap, @StringForgery fakeString: String ) { @@ -659,7 +665,7 @@ internal class ResourceResolverTest { whenever(mockDrawableUtils.createScaledBitmap(any(), anyOrNull())).thenReturn(mockResizedBitmap) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -676,14 +682,14 @@ internal class ResourceResolverTest { } @Test - fun `M cache bitmap W resolveResourceId() { from BitmapDrawable with null bitmap }`() { + fun `M cache bitmap W resolveResourceIdFromDrawable() { from BitmapDrawable with null bitmap }`() { // Given whenever(mockBitmapCachesManager.getFromResourceCache(fakeResourceKey)) .thenReturn(null) whenever(mockBitmapDrawable.bitmap).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -700,12 +706,12 @@ internal class ResourceResolverTest { } @Test - fun `M cache bitmap W resolveResourceId() { not a BitmapDrawable }`() { + fun `M cache bitmap W resolveResourceIdFromDrawable() { not a BitmapDrawable }`() { // Given val mockLayerDrawable = mock() whenever(mockDrawableCopier.copy(any(), any())).thenReturn(mockLayerDrawable) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -722,7 +728,7 @@ internal class ResourceResolverTest { } @Test - fun `M return all callbacks W resolveResourceId(Drawable) { multiple threads, first takes longer }`( + fun `M return all callbacks W resolveResourceIdFromDrawable(Drawable) { multiple threads, first takes longer }`( @Mock mockFirstCallback: ResourceResolverCallback, @Mock mockSecondCallback: ResourceResolverCallback, @Mock mockFirstDrawable: Drawable, @@ -742,7 +748,7 @@ internal class ResourceResolverTest { val countDownLatch = CountDownLatch(2) val thread1 = Thread { - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -757,7 +763,7 @@ internal class ResourceResolverTest { countDownLatch.countDown() } val thread2 = Thread { - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -783,7 +789,7 @@ internal class ResourceResolverTest { } @Test - fun `M return all callbacks W resolveResourceId(Bitmap) { multiple threads, first takes longer }`( + fun `M return all callbacks W resolveResourceIdFromDrawable(Bitmap) { multiple threads, first takes longer }`( @Mock mockFirstCallback: ResourceResolverCallback, @Mock mockSecondCallback: ResourceResolverCallback, @Mock mockFirstBitmap: Bitmap, @@ -801,7 +807,7 @@ internal class ResourceResolverTest { whenever(mockMD5HashGenerator.generate(secondBitmapCompression)).thenReturn(fakeSecondResourceId) val countDownLatch = CountDownLatch(2) val thread1 = Thread { - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromBitmap( bitmap = mockFirstBitmap, resourceResolverCallback = mockFirstCallback ) @@ -809,7 +815,7 @@ internal class ResourceResolverTest { countDownLatch.countDown() } val thread2 = Thread { - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromBitmap( bitmap = mockSecondBitmap, resourceResolverCallback = mockSecondCallback ) @@ -828,7 +834,7 @@ internal class ResourceResolverTest { } @Test - fun `M failover to bitmap creation W resolveResourceId() { bitmapDrawable returned empty bytearray }`( + fun `M failover to bitmap creation W resolveResourceIdFromDrawable() { bitmapDrawable returned empty bytearray }`( @Mock mockCreatedBitmap: Bitmap ) { // Given @@ -852,7 +858,7 @@ internal class ResourceResolverTest { .thenReturn(mockCreatedBitmap) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -877,7 +883,7 @@ internal class ResourceResolverTest { } @Test - fun `M only send resource once W resolveResourceId() { call twice on the same image }`( + fun `M only send resource once W resolveResourceIdFromDrawable() { call twice on the same image }`( @Mock mockCreatedBitmap: Bitmap, @StringForgery fakeResourceId: String, @StringForgery fakeResource: String @@ -904,7 +910,7 @@ internal class ResourceResolverTest { .thenReturn(mockCreatedBitmap) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -919,7 +925,7 @@ internal class ResourceResolverTest { // Then // second time - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -938,7 +944,7 @@ internal class ResourceResolverTest { ) // second time - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -963,7 +969,7 @@ internal class ResourceResolverTest { whenever(mockBitmapCachesManager.generateResourceKeyFromDrawable(mockDrawable)).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -987,7 +993,7 @@ internal class ResourceResolverTest { whenever(mockBitmapCachesManager.getFromResourceCache(fakeCacheKey)).thenReturn(fakeResourceId) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -1011,7 +1017,7 @@ internal class ResourceResolverTest { whenever(mockBitmapCachesManager.getFromResourceCache(fakeCacheKey)).thenReturn(null) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -1029,6 +1035,7 @@ internal class ResourceResolverTest { private fun createResourceResolver(): ResourceResolver = ResourceResolver( logger = mockLogger, + pathUtils = mockPathUtils, threadPoolExecutor = mockExecutorService, drawableUtils = mockDrawableUtils, webPImageCompression = mockWebPImageCompression, @@ -1046,7 +1053,7 @@ internal class ResourceResolverTest { whenever(mockDrawableCopier.copy(mockDrawable, mockResources)).thenReturn(mockCopiedDrawable) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, @@ -1070,7 +1077,7 @@ internal class ResourceResolverTest { whenever(mockDrawableCopier.copy(mockDrawable, mockResources)).thenReturn(mockCopiedDrawable) // When - testedResourceResolver.resolveResourceId( + testedResourceResolver.resolveResourceIdFromDrawable( resources = mockResources, applicationContext = mockApplicationContext, displayMetrics = mockDisplayMetrics, diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtilsTest.kt similarity index 52% rename from features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt rename to features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtilsTest.kt index b840fc1f30..7900d9effd 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/PathUtilsTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtilsTest.kt @@ -4,16 +4,22 @@ * Copyright 2016-Present Datadog, Inc. */ -package com.datadog.android.sessionreplay.compose.internal.utils +package com.datadog.android.sessionreplay.internal.utils import android.graphics.Bitmap import android.graphics.Canvas -import androidx.compose.ui.geometry.Rect -import androidx.compose.ui.graphics.Path +import android.graphics.Path +import android.graphics.PathMeasure import com.datadog.android.api.InternalLogger +import com.datadog.android.sessionreplay.internal.recorder.resources.BitmapCachesManager +import com.datadog.android.sessionreplay.internal.recorder.resources.HashGenerator +import com.datadog.android.sessionreplay.internal.utils.PathUtils.Companion.DEFAULT_MAX_PATH_LENGTH import com.datadog.android.sessionreplay.recorder.wrappers.BitmapWrapper import com.datadog.android.sessionreplay.recorder.wrappers.CanvasWrapper +import fr.xgouchet.elmyr.Forge +import fr.xgouchet.elmyr.annotation.IntForgery import fr.xgouchet.elmyr.annotation.LongForgery +import fr.xgouchet.elmyr.annotation.StringForgery import fr.xgouchet.elmyr.junit5.ForgeExtension import org.assertj.core.api.Assertions.assertThat import org.junit.jupiter.api.BeforeEach @@ -46,10 +52,7 @@ internal class PathUtilsTest { private lateinit var mockBitmapWrapper: BitmapWrapper @Mock - private lateinit var mockPath: Path - - @Mock - private lateinit var mockBounds: Rect + private lateinit var mockBitmapCachesManager: BitmapCachesManager @Mock private lateinit var mockBitmap: Bitmap @@ -57,38 +60,41 @@ internal class PathUtilsTest { @Mock private lateinit var mockCanvas: Canvas - @LongForgery(min = 0xffffffff) - var fakeFillColor: Long = 0L - @LongForgery(min = 0xffffffff) var fakeCheckmarkColor: Long = 0L + @Mock lateinit var mockGenerator: HashGenerator + + @Mock lateinit var mockPathMeasure: PathMeasure + + @Mock lateinit var mockPath: Path + + @StringForgery + lateinit var fakeHash: String + @BeforeEach - fun `set up`() { - whenever(mockPath.getBounds()) - .thenReturn(mockBounds) + fun `set up`(forge: Forge) { + whenever(mockPathMeasure.getPosTan(any(), any(), any())) + .thenReturn(true) + + val fakeContourLength = forge.aFloat(min = DEFAULT_MAX_PATH_LENGTH.toFloat()) + whenever(mockPathMeasure.length).thenReturn(fakeContourLength) testedUtils = PathUtils( logger = mockLogger, canvasWrapper = mockCanvasWrapper, - bitmapWrapper = mockBitmapWrapper + bitmapWrapper = mockBitmapWrapper, + bitmapCachesManager = mockBitmapCachesManager, + md5Generator = mockGenerator ) } @Test - fun `M return input value W convertRgbaToArgb() { length lt 2 }`() { - // Then - assertThat(testedUtils.convertRgbaToArgb("#")).isEqualTo("#") - } - - @Test - fun `M move alpha value W convertRgbaToArgb()`() { - // Then - assertThat(testedUtils.convertRgbaToArgb("#000000FF")).isEqualTo("#FF000000") - } - - @Test - fun `M return null W convertPathToBitmap() { failed to create bitmap }`() { + fun `M return null W convertPathToBitmap() { failed to create bitmap }`( + @IntForgery fakeWidth: Int, + @IntForgery fakeHeight: Int, + @IntForgery fakeStrokeWidth: Int + ) { // Given whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) .thenReturn(null) @@ -96,7 +102,9 @@ internal class PathUtilsTest { // When val result = testedUtils.convertPathToBitmap( checkPath = mockPath, - fillColor = fakeFillColor.toInt(), + desiredWidth = fakeWidth, + desiredHeight = fakeHeight, + strokeWidth = fakeStrokeWidth, checkmarkColor = fakeCheckmarkColor.toInt() ) @@ -105,7 +113,11 @@ internal class PathUtilsTest { } @Test - fun `M return null W convertPathToBitmap() { failed to create canvas }`() { + fun `M return null W convertPathToBitmap() { failed to create canvas }`( + @IntForgery fakeWidth: Int, + @IntForgery fakeHeight: Int, + @IntForgery fakeStrokeWidth: Int + ) { // Given whenever(mockCanvasWrapper.createCanvas(any())) .thenReturn(null) @@ -113,7 +125,9 @@ internal class PathUtilsTest { // When val result = testedUtils.convertPathToBitmap( checkPath = mockPath, - fillColor = fakeFillColor.toInt(), + desiredWidth = fakeWidth, + desiredHeight = fakeHeight, + strokeWidth = fakeStrokeWidth, checkmarkColor = fakeCheckmarkColor.toInt() ) @@ -122,7 +136,11 @@ internal class PathUtilsTest { } @Test - fun `M return bitmap W convertPathToBitmap() { success }`() { + fun `M return bitmap W convertPathToBitmap() { success }`( + @IntForgery fakeWidth: Int, + @IntForgery fakeHeight: Int, + @IntForgery fakeStrokeWidth: Int + ) { // Given whenever(mockBitmapWrapper.createBitmap(any(), any(), any(), anyOrNull())) .thenReturn(mockBitmap) @@ -133,11 +151,46 @@ internal class PathUtilsTest { // When val result = testedUtils.convertPathToBitmap( checkPath = mockPath, - fillColor = fakeFillColor.toInt(), + desiredWidth = fakeWidth, + desiredHeight = fakeHeight, + strokeWidth = fakeStrokeWidth, checkmarkColor = fakeCheckmarkColor.toInt() ) // Then assertThat(result).isEqualTo(mockBitmap) } + + @Test + fun `M return path W generateKeyForPath`() { + // Given + whenever(mockGenerator.generate(any())).thenReturn(fakeHash) + whenever(mockPathMeasure.nextContour()).thenReturn(true) + + // When + val result = testedUtils.generateKeyForPath( + path = mockPath, + pathMeasure = mockPathMeasure + ) + + // Then + assertThat(result).isEqualTo(fakeHash) + } + + @Test + fun `M return null W generateKeyForPath { empty points }`() { + // Given + val emptyPoints = "0.0,0.0;" + whenever(mockGenerator.generate(emptyPoints.toByteArray())).thenReturn("") + whenever(mockPathMeasure.nextContour()).thenReturn(false) + + // When + val result = testedUtils.generateKeyForPath( + path = mockPath, + pathMeasure = mockPathMeasure + ) + + // Then + assertThat(result).isNull() + } } From 1f3e4af13ff87594171f9b21c08b6f0141eae119 Mon Sep 17 00:00:00 2001 From: jonathanmos <48201295+jonathanmos@users.noreply.github.com> Date: Thu, 12 Dec 2024 14:55:47 +0200 Subject: [PATCH 4/5] RUM-6195: Improve background color on unchecked state --- .../semantics/CheckboxSemanticsNodeMapper.kt | 106 +++++++----------- .../CheckboxSemanticsNodeMapperTest.kt | 2 +- .../semantics/RootSemanticsNodeMapperTest.kt | 27 ++++- 3 files changed, 70 insertions(+), 65 deletions(-) diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt index 49fee65d45..5475fc437a 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt @@ -38,7 +38,7 @@ internal class CheckboxSemanticsNodeMapper( ): SemanticsWireframe { val globalBounds = resolveBounds(semanticsNode) - val checkableWireframes = if (parentContext.textAndInputPrivacy != TextAndInputPrivacy.MASK_SENSITIVE_INPUTS) { + val checkableWireframes = if (isCheckboxMasked(parentContext)) { listOf( resolveMaskedCheckable( semanticsNode = semanticsNode, @@ -46,12 +46,12 @@ internal class CheckboxSemanticsNodeMapper( ) ) } else { - // Resolves checkable view regardless the state - resolveCheckable( - semanticsNode = semanticsNode, + createCheckboxWireframe( parentContext = parentContext, asyncJobStatusCallback = asyncJobStatusCallback, - globalBounds = globalBounds + semanticsNode = semanticsNode, + globalBounds = globalBounds, + currentIndex = 0 ) } @@ -61,6 +61,9 @@ internal class CheckboxSemanticsNodeMapper( ) } + private fun isCheckboxMasked(parentContext: UiContext): Boolean = + parentContext.textAndInputPrivacy != TextAndInputPrivacy.MASK_SENSITIVE_INPUTS + private fun resolveMaskedCheckable( semanticsNode: SemanticsNode, globalBounds: GlobalBounds @@ -75,72 +78,39 @@ internal class CheckboxSemanticsNodeMapper( ) } - private fun resolveCheckable( - semanticsNode: SemanticsNode, - parentContext: UiContext, - asyncJobStatusCallback: AsyncJobStatusCallback, - globalBounds: GlobalBounds - ): List { - return if (isCheckboxChecked(semanticsNode)) { - createCheckedState( - parentContext = parentContext, - asyncJobStatusCallback = asyncJobStatusCallback, - semanticsNode = semanticsNode, - globalBounds = globalBounds, - currentIndex = 0 - ) - } else { - val borderColor = - semanticsUtils.resolveBorderColor(semanticsNode) - ?.let { rawColor -> - convertColor(rawColor) - } ?: DEFAULT_COLOR_BLACK - - listOf( - createUncheckedState( - semanticsNode = semanticsNode, - globalBounds = globalBounds, - backgroundColor = DEFAULT_COLOR_WHITE, - borderColor = borderColor, - currentIndex = 0 - ) - ) - } - } - - private fun createCheckedState( + private fun createCheckboxWireframe( parentContext: UiContext, asyncJobStatusCallback: AsyncJobStatusCallback, semanticsNode: SemanticsNode, globalBounds: GlobalBounds, currentIndex: Int ): List { + val borderColor = resolveBorderColor(semanticsNode) val rawFillColor = semanticsUtils.resolveCheckboxFillColor(semanticsNode) val rawCheckmarkColor = semanticsUtils.resolveCheckmarkColor(semanticsNode) - - val fillColorRgba = rawFillColor?.let { - convertColor(it) - } ?: DEFAULT_COLOR_WHITE - val checkmarkColorRgba = rawCheckmarkColor?.let { - convertColor(it) - } ?: getFallbackCheckmarkColor(DEFAULT_COLOR_WHITE) - + val fillColorRgba = rawFillColor?.let { convertColor(it) } ?: DEFAULT_COLOR_WHITE + val checkmarkColorRgba = rawCheckmarkColor?.let { convertColor(it) } + ?: getFallbackCheckmarkColor(DEFAULT_COLOR_WHITE) val parsedFillColor = colorUtils.parseColorSafe(fillColorRgba) - val parsedCheckmarkColor = colorUtils.parseColorSafe(checkmarkColorRgba) + val isChecked = isCheckboxChecked(semanticsNode) + val checkmarkColor = resolveCheckmarkColor(isChecked, checkmarkColorRgba, parsedFillColor) + val wireframes = mutableListOf() - if (parsedFillColor != null && parsedCheckmarkColor != null) { - val androidPath = semanticsUtils - .resolveCheckPath(semanticsNode)?.let { checkPath -> - pathUtils.asAndroidPathSafe(checkPath) - } + if (parsedFillColor != null && checkmarkColor != null) { + val composePath = semanticsUtils + .resolveCheckPath(semanticsNode) + + val androidPath = composePath?.let { checkPath -> + pathUtils.asAndroidPathSafe(checkPath) + } if (androidPath != null) { parentContext.imageWireframeHelper.createImageWireframeByPath( id = resolveId(semanticsNode, currentIndex), globalBounds = globalBounds, path = androidPath, - strokeColor = parsedCheckmarkColor, + strokeColor = checkmarkColor, strokeWidth = STROKE_WIDTH_DP.toInt(), targetWidth = CHECKBOX_SIZE_DP, targetHeight = CHECKBOX_SIZE_DP, @@ -155,7 +125,7 @@ internal class CheckboxSemanticsNodeMapper( cornerRadius = CHECKBOX_CORNER_RADIUS ), border = MobileSegment.ShapeBorder( - color = fillColorRgba, + color = borderColor, width = BOX_BORDER_WIDTH_DP ), customResourceIdCacheKey = null @@ -173,25 +143,35 @@ internal class CheckboxSemanticsNodeMapper( return createManualCheckedWireframe( semanticsNode = semanticsNode, globalBounds = globalBounds, - backgroundColor = fillColorRgba + backgroundColor = fillColorRgba, + borderColor = borderColor ) } + private fun resolveCheckmarkColor(isChecked: Boolean, checkmarkColorRgba: String, fillColor: Int?): Int? = + if (isChecked) { + colorUtils.parseColorSafe(checkmarkColorRgba) + } else { + fillColor + } + + private fun resolveBorderColor(semanticsNode: SemanticsNode): String { + return semanticsUtils.resolveBorderColor(semanticsNode) + ?.let { rawColor -> + convertColor(rawColor) + } ?: DEFAULT_COLOR_BLACK + } + private fun createManualCheckedWireframe( semanticsNode: SemanticsNode, globalBounds: GlobalBounds, - backgroundColor: String + backgroundColor: String, + borderColor: String ): List { val strokeColor = getFallbackCheckmarkColor(backgroundColor) val wireframesList = mutableListOf() - val borderColor = - semanticsUtils.resolveBorderColor(semanticsNode) - ?.let { rawColor -> - convertColor(rawColor) - } ?: DEFAULT_COLOR_BLACK - val background = createUncheckedState( semanticsNode = semanticsNode, globalBounds = globalBounds, diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt index e094af601b..6c26e38b49 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt @@ -407,7 +407,7 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest ) val expectedBorder = MobileSegment.ShapeBorder( - color = fakeFillColorHexString, + color = fakeBorderColorHexString, width = BOX_BORDER_WIDTH_DP ) diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt index d3aa409f86..12f6a764b0 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt @@ -71,6 +71,9 @@ class RootSemanticsNodeMapperTest { @Mock private lateinit var mockImageSemanticsNodeMapper: ImageSemanticsNodeMapper + @Mock + private lateinit var mockCheckboxSemanticsNodeMapper: CheckboxSemanticsNodeMapper + @Mock private lateinit var mockComposeHiddenMapper: ComposeHiddenMapper @@ -91,7 +94,8 @@ class RootSemanticsNodeMapperTest { Role.RadioButton to mockRadioButtonSemanticsNodeMapper, Role.Tab to mockTabSemanticsNodeMapper, Role.Button to mockButtonSemanticsNodeMapper, - Role.Image to mockImageSemanticsNodeMapper + Role.Image to mockImageSemanticsNodeMapper, + Role.Checkbox to mockCheckboxSemanticsNodeMapper ), textSemanticsNodeMapper = mockTextSemanticsNodeMapper, containerSemanticsNodeMapper = mockContainerSemanticsNodeMapper, @@ -204,6 +208,27 @@ class RootSemanticsNodeMapperTest { ) } + @Test + fun `M use CheckboxSemanticsNodeMapper W map { role is Checkbox }`() { + // Given + val mockSemanticsNode = mockSemanticsNode(Role.Checkbox) + + // When + testedRootSemanticsNodeMapper.createComposeWireframes( + mockSemanticsNode, + fakeMappingContext.systemInformation.screenDensity, + fakeMappingContext, + mockAsyncJobStatusCallback + ) + + // Then + verify(mockCheckboxSemanticsNodeMapper).map( + eq(mockSemanticsNode), + any(), + eq(mockAsyncJobStatusCallback) + ) + } + @Test fun `M use ComposeHideMapper W node is hidden`(forge: Forge) { // Given From c1b4f66c26e891548c0d4587a4c978ce14c94afa Mon Sep 17 00:00:00 2001 From: jonathanmos <48201295+jonathanmos@users.noreply.github.com> Date: Wed, 18 Dec 2024 10:31:56 +0000 Subject: [PATCH 5/5] RUM-6195: Small refactoring --- .../android/internal/utils/ImageViewUtils.kt | 14 ++--- .../semantics/CheckboxSemanticsNodeMapper.kt | 48 +++++++------- .../compose/internal/utils/ColorUtils.kt | 16 ----- .../compose/internal/utils/SemanticsUtils.kt | 3 +- .../CheckboxSemanticsNodeMapperTest.kt | 63 ++++++++++--------- .../semantics/RootSemanticsNodeMapperTest.kt | 12 ++-- .../compose/internal/utils/ColorUtilsTest.kt | 60 ------------------ .../recorder/resources/ResourceResolver.kt | 2 + .../sessionreplay/internal/utils/PathUtils.kt | 2 +- .../utils/ImageWireframeHelper.kt | 6 +- .../DefaultImageWireframeHelperTest.kt | 4 +- 11 files changed, 77 insertions(+), 153 deletions(-) delete mode 100644 features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt diff --git a/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt b/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt index 73dbcaed52..cf3f148758 100644 --- a/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt +++ b/dd-sdk-android-internal/src/main/java/com/datadog/android/internal/utils/ImageViewUtils.kt @@ -19,7 +19,7 @@ import android.widget.ImageView object ImageViewUtils { /** * Resolves the absolute position on the screen of the given [View]. - * @param view: the [View]. + * @param view the [View]. * @return the [Rect] representing the absolute position of the view. */ fun resolveParentRectAbsPosition(view: View): Rect { @@ -42,9 +42,9 @@ object ImageViewUtils { /** * Calculates the clipping [Rect] of the given child [Rect] using its parent [Rect] and * the screen density. - * @param parentRect: the parent [Rect]. - * @param childRect: the child [Rect]. - * @param density: the screen density. + * @param parentRect the parent [Rect]. + * @param childRect the child [Rect]. + * @param density the screen density. * @return the clipping [Rect]. */ fun calculateClipping(parentRect: Rect, childRect: Rect, density: Float): Rect { @@ -78,9 +78,9 @@ object ImageViewUtils { /** * Resolves the [Drawable] content [Rect] using the given [ImageView] scale type. - * @param imageView: the [ImageView]. - * @param drawable: the [Drawable]. - * @param customScaleType: optional custom [ImageView.ScaleType]. + * @param imageView the [ImageView]. + * @param drawable the [Drawable]. + * @param customScaleType optional custom [ImageView.ScaleType]. * @return the resolved content [Rect]. */ fun resolveContentRectWithScaling( diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt index 5475fc437a..ed717b9e4b 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapper.kt @@ -46,7 +46,7 @@ internal class CheckboxSemanticsNodeMapper( ) ) } else { - createCheckboxWireframe( + createCheckboxWireframes( parentContext = parentContext, asyncJobStatusCallback = asyncJobStatusCallback, semanticsNode = semanticsNode, @@ -78,7 +78,7 @@ internal class CheckboxSemanticsNodeMapper( ) } - private fun createCheckboxWireframe( + private fun createCheckboxWireframes( parentContext: UiContext, asyncJobStatusCallback: AsyncJobStatusCallback, semanticsNode: SemanticsNode, @@ -140,7 +140,7 @@ internal class CheckboxSemanticsNodeMapper( } // if we failed to create a wireframe from the path - return createManualCheckedWireframe( + return createManualCheckedWireframes( semanticsNode = semanticsNode, globalBounds = globalBounds, backgroundColor = fillColorRgba, @@ -162,7 +162,7 @@ internal class CheckboxSemanticsNodeMapper( } ?: DEFAULT_COLOR_BLACK } - private fun createManualCheckedWireframe( + private fun createManualCheckedWireframes( semanticsNode: SemanticsNode, globalBounds: GlobalBounds, backgroundColor: String, @@ -170,7 +170,7 @@ internal class CheckboxSemanticsNodeMapper( ): List { val strokeColor = getFallbackCheckmarkColor(backgroundColor) - val wireframesList = mutableListOf() + val wireframes = mutableListOf() val background = createUncheckedState( semanticsNode = semanticsNode, @@ -180,7 +180,7 @@ internal class CheckboxSemanticsNodeMapper( currentIndex = 0 ) - wireframesList.add(background) + wireframes.add(background) val checkmarkWidth = globalBounds.width * CHECKMARK_SIZE_FACTOR val checkmarkHeight = globalBounds.height * CHECKMARK_SIZE_FACTOR @@ -203,8 +203,8 @@ internal class CheckboxSemanticsNodeMapper( ) ) - wireframesList.add(foreground) - return wireframesList + wireframes.add(foreground) + return wireframes } private fun createUncheckedState( @@ -213,24 +213,22 @@ internal class CheckboxSemanticsNodeMapper( backgroundColor: String, borderColor: String, currentIndex: Int - ): MobileSegment.Wireframe { - return MobileSegment.Wireframe.ShapeWireframe( - id = resolveId(semanticsNode, currentIndex), - x = globalBounds.x, - y = globalBounds.y, - width = CHECKBOX_SIZE_DP.toLong(), - height = CHECKBOX_SIZE_DP.toLong(), - shapeStyle = MobileSegment.ShapeStyle( - backgroundColor = backgroundColor, - opacity = 1f, - cornerRadius = CHECKBOX_CORNER_RADIUS - ), - border = MobileSegment.ShapeBorder( - color = borderColor, - width = BOX_BORDER_WIDTH_DP - ) + ) = MobileSegment.Wireframe.ShapeWireframe( + id = resolveId(semanticsNode, currentIndex), + x = globalBounds.x, + y = globalBounds.y, + width = CHECKBOX_SIZE_DP.toLong(), + height = CHECKBOX_SIZE_DP.toLong(), + shapeStyle = MobileSegment.ShapeStyle( + backgroundColor = backgroundColor, + opacity = 1f, + cornerRadius = CHECKBOX_CORNER_RADIUS + ), + border = MobileSegment.ShapeBorder( + color = borderColor, + width = BOX_BORDER_WIDTH_DP ) - } + ) private fun isCheckboxChecked(semanticsNode: SemanticsNode): Boolean = semanticsNode.config.getOrNull(SemanticsProperties.ToggleableState) == ToggleableState.On diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt index 5942383824..84caf13c0b 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtils.kt @@ -28,22 +28,6 @@ internal class ColorUtils( } } - internal fun convertRgbaToArgb(rgbaString: String): String { - if (rgbaString.length < 2) return rgbaString - - // for takeLast: n > 0 - @Suppress("UnsafeThirdPartyFunctionCall") - val alphaValue = rgbaString.takeLast(2) - - // for substring: length is necessarily > 1 at this point - // for dropLast: n > 0 - @Suppress("UnsafeThirdPartyFunctionCall") - val rgbColor = rgbaString - .substring(1) - .dropLast(2) - return "#$alphaValue$rgbColor" - } - internal companion object { internal const val COLOR_PARSE_ERROR = "Failed to parse color: %s" } diff --git a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt index 81a5a6a98b..b140ac7ca4 100644 --- a/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt +++ b/features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/SemanticsUtils.kt @@ -332,8 +332,7 @@ internal class SemanticsUtils(private val reflectionUtils: ReflectionUtils = Ref } } - val result = (color?.value as? Color) - ?.value + val result = (color?.value as? Color)?.value return result?.toLong() } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt index 6c26e38b49..1f0e983b14 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/CheckboxSemanticsNodeMapperTest.kt @@ -161,7 +161,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val actualWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val actualWireframe = semanticsWireframe.wireframes[0] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeBorder = MobileSegment.ShapeBorder( color = fakeBorderColorHexString, @@ -174,9 +175,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(actualWireframe?.border).isEqualTo(expectedShapeBorder) - assertThat(actualWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(actualWireframe.border).isEqualTo(expectedShapeBorder) + assertThat(actualWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -195,7 +195,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val actualWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val actualWireframe = semanticsWireframe.wireframes[0] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeBorder = MobileSegment.ShapeBorder( color = DEFAULT_COLOR_BLACK, @@ -208,9 +209,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(actualWireframe?.border).isEqualTo(expectedShapeBorder) - assertThat(actualWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(actualWireframe.border).isEqualTo(expectedShapeBorder) + assertThat(actualWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -229,7 +229,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val backgroundWireframe = semanticsWireframe.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val backgroundWireframe = semanticsWireframe.wireframes[0] as MobileSegment.Wireframe.ShapeWireframe val expectedBgShapeBorder = MobileSegment.ShapeBorder( color = fakeBorderColorHexString, @@ -242,9 +243,8 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(backgroundWireframe?.border).isEqualTo(expectedBgShapeBorder) - assertThat(backgroundWireframe?.shapeStyle).isEqualTo(expectedBgShapeStyle) + assertThat(backgroundWireframe.border).isEqualTo(expectedBgShapeBorder) + assertThat(backgroundWireframe.shapeStyle).isEqualTo(expectedBgShapeStyle) } @Test @@ -266,15 +266,15 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val foregroundWireframe = semanticsWireframe.wireframes[1] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeStyle = MobileSegment.ShapeStyle( backgroundColor = DEFAULT_COLOR_WHITE, opacity = 1f, cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(foregroundWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -295,17 +295,18 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest parentContext = mockUiContext, asyncJobStatusCallback = mockAsyncJobStatusCallback ) + + // Then assertThat(semanticsWireframe.wireframes).hasSize(2) - val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + val foregroundWireframe = semanticsWireframe.wireframes[1] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeStyle = MobileSegment.ShapeStyle( backgroundColor = DEFAULT_COLOR_BLACK, opacity = 1f, cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(foregroundWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -321,15 +322,15 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val foregroundWireframe = semanticsWireframe.wireframes[1] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeStyle = MobileSegment.ShapeStyle( backgroundColor = DEFAULT_COLOR_BLACK, opacity = 1f, cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(foregroundWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -345,15 +346,15 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) - val foregroundWireframe = semanticsWireframe.wireframes[1] as? MobileSegment.Wireframe.ShapeWireframe + // Then + val foregroundWireframe = semanticsWireframe.wireframes[1] as MobileSegment.Wireframe.ShapeWireframe val expectedShapeStyle = MobileSegment.ShapeStyle( backgroundColor = DEFAULT_COLOR_BLACK, opacity = 1f, cornerRadius = CHECKBOX_CORNER_RADIUS ) - // Then - assertThat(foregroundWireframe?.shapeStyle).isEqualTo(expectedShapeStyle) + assertThat(foregroundWireframe.shapeStyle).isEqualTo(expectedShapeStyle) } @Test @@ -400,6 +401,7 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest asyncJobStatusCallback = mockAsyncJobStatusCallback ) + // Then val expectedShapeStyle = MobileSegment.ShapeStyle( backgroundColor = fakeFillColorHexString, opacity = 1f, @@ -411,7 +413,6 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest width = BOX_BORDER_WIDTH_DP ) - // Then verify(mockUiContext.imageWireframeHelper).createImageWireframeByPath( id = any(), globalBounds = eq(fakeGlobalBounds), @@ -487,16 +488,16 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest .thenReturn(TextAndInputPrivacy.MASK_ALL_INPUTS) // When - val wireframes = testedMapper.map( + val checkboxWireframe = testedMapper.map( semanticsNode = mockSemanticsNode, parentContext = mockUiContext, asyncJobStatusCallback = mockAsyncJobStatusCallback ) - assertThat(wireframes.wireframes).hasSize(1) - val actualWireframe = wireframes.wireframes[0] as? MobileSegment.Wireframe.ShapeWireframe - assertThat(actualWireframe).isNotNull // Then + assertThat(checkboxWireframe.wireframes).hasSize(1) + val actualWireframe = checkboxWireframe.wireframes[0] as MobileSegment.Wireframe.ShapeWireframe + verify(mockUiContext.imageWireframeHelper, never()).createImageWireframeByBitmap( id = any(), globalBounds = any(), @@ -509,7 +510,7 @@ internal class CheckboxSemanticsNodeMapperTest : AbstractSemanticsNodeMapperTest shapeStyle = anyOrNull(), border = anyOrNull() ) - assertThat(actualWireframe?.shapeStyle?.backgroundColor).isEqualTo(DEFAULT_COLOR_WHITE) - assertThat(actualWireframe?.border?.color).isEqualTo(DEFAULT_COLOR_BLACK) + assertThat(actualWireframe.shapeStyle?.backgroundColor).isEqualTo(DEFAULT_COLOR_WHITE) + assertThat(actualWireframe.border?.color).isEqualTo(DEFAULT_COLOR_BLACK) } } diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt index 12f6a764b0..a8e401c6c0 100644 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt +++ b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapperTest.kt @@ -104,7 +104,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use ContainerSemanticsNodeMapper W map { role is missing }`() { + fun `M use ContainerSemanticsNodeMapper W createComposeWireframes { role is missing }`() { // Given val mockSemanticsNode = mockSemanticsNode(null) @@ -125,7 +125,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use ButtonSemanticsNodeMapper W map { role is Button }`() { + fun `M use ButtonSemanticsNodeMapper W createComposeWireframes { role is Button }`() { // Given val mockSemanticsNode = mockSemanticsNode(Role.Button) @@ -146,7 +146,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use RadioButtonSemanticsNodeMapper W map { role is RadioButton }`() { + fun `M use RadioButtonSemanticsNodeMapper W createComposeWireframes { role is RadioButton }`() { // Given val mockSemanticsNode = mockSemanticsNode(Role.RadioButton) @@ -167,7 +167,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use TabSemanticsNodeMapper W map { role is Tab }`() { + fun `M use TabSemanticsNodeMapper W createComposeWireframes { role is Tab }`() { // Given val mockSemanticsNode = mockSemanticsNode(Role.Tab) @@ -188,7 +188,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use ImageSemanticsNodeMapper W map { role is Image }`() { + fun `M use ImageSemanticsNodeMapper W createComposeWireframes { role is Image }`() { // Given val mockSemanticsNode = mockSemanticsNode(Role.Image) @@ -209,7 +209,7 @@ class RootSemanticsNodeMapperTest { } @Test - fun `M use CheckboxSemanticsNodeMapper W map { role is Checkbox }`() { + fun `M use CheckboxSemanticsNodeMapper W createComposeWireframes { role is Checkbox }`() { // Given val mockSemanticsNode = mockSemanticsNode(Role.Checkbox) diff --git a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt b/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt deleted file mode 100644 index e10d1ac934..0000000000 --- a/features/dd-sdk-android-session-replay-compose/src/test/kotlin/com/datadog/android/sessionreplay/compose/internal/utils/ColorUtilsTest.kt +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. - * This product includes software developed at Datadog (https://www.datadoghq.com/). - * Copyright 2016-Present Datadog, Inc. - */ - -package com.datadog.android.sessionreplay.compose.internal.utils - -import com.datadog.android.sessionreplay.compose.test.elmyr.SessionReplayComposeForgeConfigurator -import fr.xgouchet.elmyr.Case -import fr.xgouchet.elmyr.Forge -import fr.xgouchet.elmyr.junit5.ForgeConfiguration -import fr.xgouchet.elmyr.junit5.ForgeExtension -import org.assertj.core.api.Assertions.assertThat -import org.junit.jupiter.api.BeforeEach -import org.junit.jupiter.api.Test -import org.junit.jupiter.api.extension.ExtendWith -import org.junit.jupiter.api.extension.Extensions -import org.mockito.junit.jupiter.MockitoExtension -import org.mockito.junit.jupiter.MockitoSettings -import org.mockito.quality.Strictness - -@Extensions( - ExtendWith(MockitoExtension::class), - ExtendWith(ForgeExtension::class) -) -@MockitoSettings(strictness = Strictness.LENIENT) -@ForgeConfiguration(SessionReplayComposeForgeConfigurator::class) -internal class ColorUtilsTest { - - private lateinit var testedUtils: ColorUtils - - @BeforeEach - fun `set up`() { - testedUtils = ColorUtils() - } - - @Test - fun `M return input value W convertRgbaToArgb() { length lt 2 }`( - forge: Forge - ) { - // Given - val input = forge.anAsciiString(1) - - // Then - assertThat(testedUtils.convertRgbaToArgb(input)).isEqualTo(input) - } - - @Test - fun `M move alpha value W convertRgbaToArgb()`( - forge: Forge - ) { - // Given - val fakeColorHexString = forge.anHexadecimalString(Case.UPPER, 8) - val expectedResult = "#" + fakeColorHexString.substring(6) + fakeColorHexString.substring(0, 6) - - // Then - assertThat(testedUtils.convertRgbaToArgb("#$fakeColorHexString")).isEqualTo(expectedResult) - } -} diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt index b7d757761a..84294b0796 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/ResourceResolver.kt @@ -43,6 +43,8 @@ internal class ResourceResolver( ) ) { + // region internal + @MainThread internal fun resolveResourceIdFromBitmap( bitmap: Bitmap, diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt index ecb4e08af9..0c07ee6bc4 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/internal/utils/PathUtils.kt @@ -77,7 +77,7 @@ internal class PathUtils( // path initial bounds val originalBounds = RectF() - @Suppress("DEPRECATION") // the new api is flagged as unstable + @Suppress("DEPRECATION") // # TODO RUM-7784 replace when possible path.computeBounds(originalBounds, true) // calculate the scale factor diff --git a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt index 0c6790bc6d..8eb79247b8 100644 --- a/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt +++ b/features/dd-sdk-android-session-replay/src/main/kotlin/com/datadog/android/sessionreplay/utils/ImageWireframeHelper.kt @@ -24,7 +24,7 @@ import com.datadog.android.sessionreplay.recorder.resources.DrawableCopier interface ImageWireframeHelper { /** - * Asks the helper to create an image wireframe based on a given path. + * Creates an image wireframe based on a given path. * @param id the unique id for the wireframe. * @param globalBounds the global bounds of the bitmap. * @param path the path to use to create the wireframe. @@ -62,7 +62,7 @@ interface ImageWireframeHelper { ): MobileSegment.Wireframe? /** - * Asks the helper to create an image wireframe based on a given bitmap. + * Creates an image wireframe based on a given bitmap. * @param id the unique id for the wireframe. * @param globalBounds the global bounds of the bitmap. * @param bitmap the bitmap to capture. @@ -88,7 +88,7 @@ interface ImageWireframeHelper { ): MobileSegment.Wireframe? /** - * Asks the helper to create an image wireframe, and process the provided drawable in the background. + * Creates an image wireframe, and process the provided drawable in the background. * @param view the view owning the drawable * @param imagePrivacy defines which images should be hidden * @param currentWireframeIndex the index of the wireframe in the list of wireframes for the view diff --git a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt index f717801d84..0471d5d111 100644 --- a/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt +++ b/features/dd-sdk-android-session-replay/src/test/kotlin/com/datadog/android/sessionreplay/internal/recorder/resources/DefaultImageWireframeHelperTest.kt @@ -1469,7 +1469,7 @@ internal class DefaultImageWireframeHelperTest { resourceResolverCallback = any() ) ).thenAnswer { - val callback = it.arguments[6] as ResourceResolverCallback + val callback = it.getArgument(6) callback.onSuccess(fakeResourceId) } @@ -1529,7 +1529,7 @@ internal class DefaultImageWireframeHelperTest { resourceResolverCallback = any() ) ).thenAnswer { - val callback = it.arguments[6] as ResourceResolverCallback + val callback = it.getArgument(6) callback.onFailure() }