Skip to content

Commit f9bc2b5

Browse files
authoredDec 18, 2024··
Merge pull request #2414 from DataDog/jmoskovich/rum-6195/checkbox-semantics-mapper
RUM-6195: Add support for Compose Checkbox
2 parents 4e7d937 + c1b4f66 commit f9bc2b5

File tree

28 files changed

+2118
-168
lines changed

28 files changed

+2118
-168
lines changed
 

‎detekt_custom.yml

+20
Original file line numberDiff line numberDiff line change
@@ -130,8 +130,10 @@ datadog:
130130
- "android.graphics.Bitmap.compress(android.graphics.Bitmap.CompressFormat, kotlin.Int, java.io.OutputStream):java.lang.NullPointerException,java.lang.IllegalArgumentException"
131131
- "android.graphics.Bitmap.copy(android.graphics.Bitmap.Config, kotlin.Boolean):java.lang.IllegalArgumentException"
132132
- "android.graphics.Bitmap.createBitmap(android.util.DisplayMetrics?, kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException"
133+
- "android.graphics.Bitmap.createBitmap(kotlin.Int, kotlin.Int, android.graphics.Bitmap.Config):java.lang.IllegalArgumentException"
133134
- "android.graphics.Bitmap.createScaledBitmap(android.graphics.Bitmap, kotlin.Int, kotlin.Int, kotlin.Boolean):java.lang.IllegalArgumentException"
134135
- "android.graphics.Canvas.constructor(android.graphics.Bitmap):java.lang.IllegalStateException"
136+
- "android.graphics.Color.parseColor(kotlin.String?):java.lang.IllegalArgumentException"
135137
- "android.graphics.drawable.LayerDrawable.getDrawable(kotlin.Int):java.lang.IndexOutOfBoundsException"
136138
- "android.net.ConnectivityManager.registerDefaultNetworkCallback(android.net.ConnectivityManager.NetworkCallback):java.lang.IllegalArgumentException,java.lang.SecurityException"
137139
- "android.net.ConnectivityManager.unregisterNetworkCallback(android.net.ConnectivityManager.NetworkCallback):java.lang.SecurityException"
@@ -468,6 +470,7 @@ datadog:
468470
# endregion
469471
# region Android Graphics
470472
- "android.graphics.Bitmap.recycle()"
473+
- "android.graphics.Canvas.drawColor(kotlin.Int)"
471474
- "android.graphics.Canvas.drawColor(kotlin.Int, android.graphics.PorterDuff.Mode)"
472475
- "android.graphics.Color.argb(kotlin.Int, kotlin.Int, kotlin.Int, kotlin.Int)"
473476
- "android.graphics.Color.blue(kotlin.Int)"
@@ -484,6 +487,14 @@ datadog:
484487
- "android.graphics.drawable.Drawable.setTintList(android.content.res.ColorStateList?)"
485488
- "android.graphics.drawable.RippleDrawable.findIndexByLayerId(kotlin.Int)"
486489
- "android.graphics.drawable.DrawableContainer.DrawableContainerState.getChild(kotlin.Int)"
490+
- "android.graphics.Matrix.constructor()"
491+
- "android.graphics.Matrix.preScale(kotlin.Float, kotlin.Float)"
492+
- "android.graphics.Matrix.preTranslate(kotlin.Float, kotlin.Float)"
493+
- "android.graphics.Paint.constructor()"
494+
- "android.graphics.Path.computeBounds(android.graphics.RectF, kotlin.Boolean)"
495+
- "android.graphics.Path.transform(android.graphics.Matrix)"
496+
- "android.graphics.PathMeasure.constructor(android.graphics.Path?, kotlin.Boolean)"
497+
- "android.graphics.PathMeasure.nextContour()"
487498
- "android.graphics.Point.constructor()"
488499
- "android.graphics.Point.constructor(kotlin.Int, kotlin.Int)"
489500
- "android.graphics.Rect.centerX()"
@@ -492,6 +503,9 @@ datadog:
492503
- "android.graphics.Rect.constructor(kotlin.Int, kotlin.Int, kotlin.Int, kotlin.Int)"
493504
- "android.graphics.Rect.height()"
494505
- "android.graphics.Rect.width()"
506+
- "android.graphics.RectF.constructor()"
507+
- "android.graphics.RectF.width()"
508+
- "android.graphics.RectF.height()"
495509
# endregion
496510
# region Androidx APIs
497511
- "androidx.appcompat.widget.DatadogActionBarContainerAccessor.constructor(androidx.appcompat.widget.ActionBarContainer)"
@@ -510,6 +524,11 @@ datadog:
510524
- "androidx.compose.runtime.tooling.CompositionGroup.stableId()"
511525
- "androidx.compose.ui.graphics.Color(kotlin.Long)"
512526
- "androidx.compose.ui.graphics.Color.toArgb()"
527+
- "androidx.compose.ui.graphics.Matrix.constructor(kotlin.FloatArray)"
528+
- "androidx.compose.ui.graphics.Matrix.scale(kotlin.Float, kotlin.Float, kotlin.Float)"
529+
- "androidx.compose.ui.graphics.Matrix.translate(kotlin.Float, kotlin.Float, kotlin.Float)"
530+
- "androidx.compose.ui.graphics.Path.getBounds()"
531+
- "androidx.compose.ui.graphics.Path.transform(androidx.compose.ui.graphics.Matrix)"
513532
- "androidx.compose.ui.layout.LayoutCoordinates.positionInWindow()"
514533
- "androidx.compose.ui.layout.LayoutInfo.getModifierInfo()"
515534
- "androidx.compose.ui.unit.Density(kotlin.Float, kotlin.Float)"
@@ -1141,6 +1160,7 @@ datadog:
11411160
- "kotlin.Float.toFloat()"
11421161
- "kotlin.Float.toInt()"
11431162
- "kotlin.Float.toLong()"
1163+
- "kotlin.FloatArray.constructor(kotlin.Int)"
11441164
- "kotlin.Int.and(kotlin.Int)"
11451165
- "kotlin.Int.coerceAtMost(kotlin.Int)"
11461166
- "kotlin.Int.inv()"

‎features/dd-sdk-android-session-replay-compose/consumer-rules.pro

+9
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,15 @@
1414
-keepclassmembers class androidx.compose.foundation.text.modifiers.TextStringSimpleElement {
1515
<fields>;
1616
}
17+
-keepclassmembers class androidx.compose.material.CheckDrawingCache {
18+
<fields>;
19+
}
20+
-keepclassmembers class androidx.compose.material.CheckboxKt {
21+
<fields>;
22+
}
23+
-keepclassmembers class androidx.compose.ui.draw.DrawBehindElement {
24+
<fields>;
25+
}
1726
-keepclassmembers class androidx.compose.foundation.BackgroundElement {
1827
<fields>;
1928
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,256 @@
1+
/*
2+
* Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0.
3+
* This product includes software developed at Datadog (https://www.datadoghq.com/).
4+
* Copyright CHECKBOX_SIZE16-Present Datadog, Inc.
5+
*/
6+
7+
package com.datadog.android.sessionreplay.compose.internal.mappers.semantics
8+
9+
import androidx.compose.ui.semantics.SemanticsNode
10+
import androidx.compose.ui.semantics.SemanticsProperties
11+
import androidx.compose.ui.semantics.getOrNull
12+
import androidx.compose.ui.state.ToggleableState
13+
import com.datadog.android.api.InternalLogger
14+
import com.datadog.android.sessionreplay.ImagePrivacy
15+
import com.datadog.android.sessionreplay.TextAndInputPrivacy
16+
import com.datadog.android.sessionreplay.compose.internal.data.SemanticsWireframe
17+
import com.datadog.android.sessionreplay.compose.internal.data.UiContext
18+
import com.datadog.android.sessionreplay.compose.internal.utils.ColorUtils
19+
import com.datadog.android.sessionreplay.compose.internal.utils.PathUtils
20+
import com.datadog.android.sessionreplay.compose.internal.utils.SemanticsUtils
21+
import com.datadog.android.sessionreplay.model.MobileSegment
22+
import com.datadog.android.sessionreplay.utils.AsyncJobStatusCallback
23+
import com.datadog.android.sessionreplay.utils.ColorStringFormatter
24+
import com.datadog.android.sessionreplay.utils.GlobalBounds
25+
26+
internal class CheckboxSemanticsNodeMapper(
27+
colorStringFormatter: ColorStringFormatter,
28+
private val semanticsUtils: SemanticsUtils = SemanticsUtils(),
29+
private val colorUtils: ColorUtils = ColorUtils(),
30+
private val logger: InternalLogger = InternalLogger.UNBOUND,
31+
private val pathUtils: PathUtils = PathUtils(logger)
32+
) : AbstractSemanticsNodeMapper(colorStringFormatter, semanticsUtils) {
33+
34+
override fun map(
35+
semanticsNode: SemanticsNode,
36+
parentContext: UiContext,
37+
asyncJobStatusCallback: AsyncJobStatusCallback
38+
): SemanticsWireframe {
39+
val globalBounds = resolveBounds(semanticsNode)
40+
41+
val checkableWireframes = if (isCheckboxMasked(parentContext)) {
42+
listOf(
43+
resolveMaskedCheckable(
44+
semanticsNode = semanticsNode,
45+
globalBounds = globalBounds
46+
)
47+
)
48+
} else {
49+
createCheckboxWireframes(
50+
parentContext = parentContext,
51+
asyncJobStatusCallback = asyncJobStatusCallback,
52+
semanticsNode = semanticsNode,
53+
globalBounds = globalBounds,
54+
currentIndex = 0
55+
)
56+
}
57+
58+
return SemanticsWireframe(
59+
uiContext = null,
60+
wireframes = checkableWireframes
61+
)
62+
}
63+
64+
private fun isCheckboxMasked(parentContext: UiContext): Boolean =
65+
parentContext.textAndInputPrivacy != TextAndInputPrivacy.MASK_SENSITIVE_INPUTS
66+
67+
private fun resolveMaskedCheckable(
68+
semanticsNode: SemanticsNode,
69+
globalBounds: GlobalBounds
70+
): MobileSegment.Wireframe {
71+
// TODO RUM-5118: Decide how to display masked checkbox, Currently use old unchecked shape wireframe,
72+
return createUncheckedState(
73+
semanticsNode = semanticsNode,
74+
globalBounds = globalBounds,
75+
backgroundColor = DEFAULT_COLOR_WHITE,
76+
borderColor = DEFAULT_COLOR_BLACK,
77+
currentIndex = 0
78+
)
79+
}
80+
81+
private fun createCheckboxWireframes(
82+
parentContext: UiContext,
83+
asyncJobStatusCallback: AsyncJobStatusCallback,
84+
semanticsNode: SemanticsNode,
85+
globalBounds: GlobalBounds,
86+
currentIndex: Int
87+
): List<MobileSegment.Wireframe> {
88+
val borderColor = resolveBorderColor(semanticsNode)
89+
val rawFillColor = semanticsUtils.resolveCheckboxFillColor(semanticsNode)
90+
val rawCheckmarkColor = semanticsUtils.resolveCheckmarkColor(semanticsNode)
91+
val fillColorRgba = rawFillColor?.let { convertColor(it) } ?: DEFAULT_COLOR_WHITE
92+
val checkmarkColorRgba = rawCheckmarkColor?.let { convertColor(it) }
93+
?: getFallbackCheckmarkColor(DEFAULT_COLOR_WHITE)
94+
val parsedFillColor = colorUtils.parseColorSafe(fillColorRgba)
95+
val isChecked = isCheckboxChecked(semanticsNode)
96+
val checkmarkColor = resolveCheckmarkColor(isChecked, checkmarkColorRgba, parsedFillColor)
97+
98+
val wireframes = mutableListOf<MobileSegment.Wireframe>()
99+
100+
if (parsedFillColor != null && checkmarkColor != null) {
101+
val composePath = semanticsUtils
102+
.resolveCheckPath(semanticsNode)
103+
104+
val androidPath = composePath?.let { checkPath ->
105+
pathUtils.asAndroidPathSafe(checkPath)
106+
}
107+
108+
if (androidPath != null) {
109+
parentContext.imageWireframeHelper.createImageWireframeByPath(
110+
id = resolveId(semanticsNode, currentIndex),
111+
globalBounds = globalBounds,
112+
path = androidPath,
113+
strokeColor = checkmarkColor,
114+
strokeWidth = STROKE_WIDTH_DP.toInt(),
115+
targetWidth = CHECKBOX_SIZE_DP,
116+
targetHeight = CHECKBOX_SIZE_DP,
117+
density = parentContext.density,
118+
isContextualImage = false,
119+
imagePrivacy = ImagePrivacy.MASK_NONE,
120+
asyncJobStatusCallback = asyncJobStatusCallback,
121+
clipping = null,
122+
shapeStyle = MobileSegment.ShapeStyle(
123+
backgroundColor = fillColorRgba,
124+
opacity = 1f,
125+
cornerRadius = CHECKBOX_CORNER_RADIUS
126+
),
127+
border = MobileSegment.ShapeBorder(
128+
color = borderColor,
129+
width = BOX_BORDER_WIDTH_DP
130+
),
131+
customResourceIdCacheKey = null
132+
)?.let { imageWireframe ->
133+
wireframes.add(imageWireframe)
134+
}
135+
}
136+
}
137+
138+
if (wireframes.isNotEmpty()) {
139+
return wireframes
140+
}
141+
142+
// if we failed to create a wireframe from the path
143+
return createManualCheckedWireframes(
144+
semanticsNode = semanticsNode,
145+
globalBounds = globalBounds,
146+
backgroundColor = fillColorRgba,
147+
borderColor = borderColor
148+
)
149+
}
150+
151+
private fun resolveCheckmarkColor(isChecked: Boolean, checkmarkColorRgba: String, fillColor: Int?): Int? =
152+
if (isChecked) {
153+
colorUtils.parseColorSafe(checkmarkColorRgba)
154+
} else {
155+
fillColor
156+
}
157+
158+
private fun resolveBorderColor(semanticsNode: SemanticsNode): String {
159+
return semanticsUtils.resolveBorderColor(semanticsNode)
160+
?.let { rawColor ->
161+
convertColor(rawColor)
162+
} ?: DEFAULT_COLOR_BLACK
163+
}
164+
165+
private fun createManualCheckedWireframes(
166+
semanticsNode: SemanticsNode,
167+
globalBounds: GlobalBounds,
168+
backgroundColor: String,
169+
borderColor: String
170+
): List<MobileSegment.Wireframe> {
171+
val strokeColor = getFallbackCheckmarkColor(backgroundColor)
172+
173+
val wireframes = mutableListOf<MobileSegment.Wireframe>()
174+
175+
val background = createUncheckedState(
176+
semanticsNode = semanticsNode,
177+
globalBounds = globalBounds,
178+
backgroundColor = backgroundColor,
179+
borderColor = borderColor,
180+
currentIndex = 0
181+
)
182+
183+
wireframes.add(background)
184+
185+
val checkmarkWidth = globalBounds.width * CHECKMARK_SIZE_FACTOR
186+
val checkmarkHeight = globalBounds.height * CHECKMARK_SIZE_FACTOR
187+
val xPos = globalBounds.x + ((globalBounds.width / 2) - (checkmarkWidth / 2))
188+
val yPos = globalBounds.y + ((globalBounds.height / 2) - (checkmarkHeight / 2))
189+
val foreground = MobileSegment.Wireframe.ShapeWireframe(
190+
id = resolveId(semanticsNode, 1),
191+
x = xPos.toLong(),
192+
y = yPos.toLong(),
193+
width = checkmarkWidth.toLong(),
194+
height = checkmarkHeight.toLong(),
195+
shapeStyle = MobileSegment.ShapeStyle(
196+
backgroundColor = strokeColor,
197+
opacity = 1f,
198+
cornerRadius = CHECKBOX_CORNER_RADIUS
199+
),
200+
border = MobileSegment.ShapeBorder(
201+
color = DEFAULT_COLOR_BLACK,
202+
width = BOX_BORDER_WIDTH_DP
203+
)
204+
)
205+
206+
wireframes.add(foreground)
207+
return wireframes
208+
}
209+
210+
private fun createUncheckedState(
211+
semanticsNode: SemanticsNode,
212+
globalBounds: GlobalBounds,
213+
backgroundColor: String,
214+
borderColor: String,
215+
currentIndex: Int
216+
) = MobileSegment.Wireframe.ShapeWireframe(
217+
id = resolveId(semanticsNode, currentIndex),
218+
x = globalBounds.x,
219+
y = globalBounds.y,
220+
width = CHECKBOX_SIZE_DP.toLong(),
221+
height = CHECKBOX_SIZE_DP.toLong(),
222+
shapeStyle = MobileSegment.ShapeStyle(
223+
backgroundColor = backgroundColor,
224+
opacity = 1f,
225+
cornerRadius = CHECKBOX_CORNER_RADIUS
226+
),
227+
border = MobileSegment.ShapeBorder(
228+
color = borderColor,
229+
width = BOX_BORDER_WIDTH_DP
230+
)
231+
)
232+
233+
private fun isCheckboxChecked(semanticsNode: SemanticsNode): Boolean =
234+
semanticsNode.config.getOrNull(SemanticsProperties.ToggleableState) == ToggleableState.On
235+
236+
private fun getFallbackCheckmarkColor(backgroundColor: String?) =
237+
if (backgroundColor == DEFAULT_COLOR_WHITE) {
238+
DEFAULT_COLOR_BLACK
239+
} else {
240+
DEFAULT_COLOR_WHITE
241+
}
242+
243+
internal companion object {
244+
internal const val DEFAULT_COLOR_BLACK = "#000000FF"
245+
internal const val DEFAULT_COLOR_WHITE = "#FFFFFFFF"
246+
247+
// when we create the checkmark manually, what % of the checkbox size should it be
248+
internal const val CHECKMARK_SIZE_FACTOR = 0.5
249+
250+
// values from Compose Checkbox sourcecode
251+
internal const val BOX_BORDER_WIDTH_DP = 2L
252+
internal const val STROKE_WIDTH_DP = 2f
253+
internal const val CHECKBOX_SIZE_DP = 20
254+
internal const val CHECKBOX_CORNER_RADIUS = 2f
255+
}
256+
}

‎features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/mappers/semantics/RootSemanticsNodeMapper.kt

+2-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,8 @@ internal class RootSemanticsNodeMapper(
3131
Role.RadioButton to RadioButtonSemanticsNodeMapper(colorStringFormatter, semanticsUtils),
3232
Role.Tab to TabSemanticsNodeMapper(colorStringFormatter, semanticsUtils),
3333
Role.Button to ButtonSemanticsNodeMapper(colorStringFormatter, semanticsUtils),
34-
Role.Image to ImageSemanticsNodeMapper(colorStringFormatter, semanticsUtils)
34+
Role.Image to ImageSemanticsNodeMapper(colorStringFormatter, semanticsUtils),
35+
Role.Checkbox to CheckboxSemanticsNodeMapper(colorStringFormatter, semanticsUtils)
3536
),
3637
// Text doesn't have a role in semantics, so it should be a fallback mapper.
3738
private val textSemanticsNodeMapper: TextSemanticsNodeMapper = TextSemanticsNodeMapper(

‎features/dd-sdk-android-session-replay-compose/src/main/kotlin/com/datadog/android/sessionreplay/compose/internal/reflection/ComposeReflection.kt

+10
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,16 @@ internal object ComposeReflection {
4040
val ColorField = BackgroundElementClass?.getDeclaredFieldSafe("color")
4141
val ShapeField = BackgroundElementClass?.getDeclaredFieldSafe("shape")
4242

43+
val CheckDrawingCacheClass = getClassSafe("androidx.compose.material.CheckDrawingCache")
44+
val CheckboxKtClass = getClassSafe("androidx.compose.material.CheckboxKt\$CheckboxImpl\$1\$1")
45+
val DrawBehindElementClass = getClassSafe("androidx.compose.ui.draw.DrawBehindElement")
46+
val BorderColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$borderColor\$delegate")
47+
val BoxColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$boxColor\$delegate")
48+
val CheckCacheField = CheckboxKtClass?.getDeclaredFieldSafe("\$checkCache")
49+
val CheckColorField = CheckboxKtClass?.getDeclaredFieldSafe("\$checkColor\$delegate")
50+
val CheckPathField = CheckDrawingCacheClass?.getDeclaredFieldSafe("checkPath")
51+
val OnDrawField = DrawBehindElementClass?.getDeclaredFieldSafe("onDraw")
52+
4353
val PaddingElementClass = getClassSafe("androidx.compose.foundation.layout.PaddingElement")
4454
val StartField = PaddingElementClass?.getDeclaredFieldSafe("start")
4555
val EndField = PaddingElementClass?.getDeclaredFieldSafe("end")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
/*
2+
* Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0.
3+
* This product includes software developed at Datadog (https://www.datadoghq.com/).
4+
* Copyright 2016-Present Datadog, Inc.
5+
*/
6+
7+
package com.datadog.android.sessionreplay.compose.internal.utils
8+
9+
import android.graphics.Color
10+
import com.datadog.android.api.InternalLogger
11+
import java.util.Locale
12+
13+
internal class ColorUtils(
14+
private val logger: InternalLogger = InternalLogger.UNBOUND
15+
) {
16+
internal fun parseColorSafe(color: String): Int? {
17+
return try {
18+
@Suppress("UnsafeThirdPartyFunctionCall") // handling IllegalArgumentException
19+
Color.parseColor(color)
20+
} catch (e: IllegalArgumentException) {
21+
logger.log(
22+
target = InternalLogger.Target.MAINTAINER,
23+
level = InternalLogger.Level.WARN,
24+
messageBuilder = { COLOR_PARSE_ERROR.format(Locale.US, color) },
25+
throwable = e
26+
)
27+
null
28+
}
29+
}
30+
31+
internal companion object {
32+
internal const val COLOR_PARSE_ERROR = "Failed to parse color: %s"
33+
}
34+
}

0 commit comments

Comments
 (0)