From f4b927312fc2cb22b0bef7aaf8a0f02099d81b27 Mon Sep 17 00:00:00 2001 From: Xavier Molloy <44061143+xavimolloy@users.noreply.github.com> Date: Wed, 8 Nov 2023 14:38:10 +0100 Subject: [PATCH] [ANDROAPP-5631] correct ripple effect for app implementation (#141) --- .../ui/designsystem/component/Legend.kt | 115 ++++++------- .../ui/designsystem/component/ListCard.kt | 153 ++++++++++-------- 2 files changed, 147 insertions(+), 121 deletions(-) diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/Legend.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/Legend.kt index 303a30d9c..8ac357f72 100644 --- a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/Legend.kt +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/Legend.kt @@ -2,6 +2,7 @@ package org.hisp.dhis.mobile.ui.designsystem.component import androidx.compose.foundation.background import androidx.compose.foundation.clickable +import androidx.compose.foundation.interaction.MutableInteractionSource import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column @@ -14,15 +15,15 @@ import androidx.compose.foundation.shape.CircleShape import androidx.compose.material.icons.Icons import androidx.compose.material.icons.outlined.HelpOutline import androidx.compose.material.icons.outlined.Info -import androidx.compose.material.ripple.LocalRippleTheme +import androidx.compose.material.ripple.rememberRipple import androidx.compose.material3.Divider import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable -import androidx.compose.runtime.CompositionLocalProvider import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment @@ -30,9 +31,9 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip import androidx.compose.ui.graphics.Color import androidx.compose.ui.platform.testTag +import androidx.compose.ui.semantics.Role import org.hisp.dhis.mobile.ui.designsystem.theme.Border import org.hisp.dhis.mobile.ui.designsystem.theme.InternalSizeValues -import org.hisp.dhis.mobile.ui.designsystem.theme.Ripple import org.hisp.dhis.mobile.ui.designsystem.theme.Spacing import org.hisp.dhis.mobile.ui.designsystem.theme.SurfaceColor import org.hisp.dhis.mobile.ui.designsystem.theme.hoverPointerIcon @@ -44,64 +45,68 @@ fun Legend( ) { var showBottomSheetShell by rememberSaveable { mutableStateOf(false) } - CompositionLocalProvider(LocalRippleTheme provides Ripple.CustomDHISRippleTheme) { - val hasPopupLegendDescriptionData = legendData.popUpLegendDescriptionData.orEmpty().isNotEmpty() - val clickableModifier = if (hasPopupLegendDescriptionData) { - Modifier - .clickable( - onClick = { - legendData.popUpLegendDescriptionData?.let { - showBottomSheetShell = true - } - }, - ) - .hoverPointerIcon(true) - } else { - Modifier - } + val hasPopupLegendDescriptionData = legendData.popUpLegendDescriptionData.orEmpty().isNotEmpty() + val interactionSource = remember { MutableInteractionSource() } + val clickableModifier = if (hasPopupLegendDescriptionData) { + Modifier + .clickable( + onClick = { + legendData.popUpLegendDescriptionData?.let { + showBottomSheetShell = true + } + }, + role = Role.Button, + interactionSource = interactionSource, + indication = rememberRipple( + color = SurfaceColor.Primary, + ), + ) + .hoverPointerIcon(true) + } else { + Modifier + } - Column( - modifier = modifier - .then(clickableModifier) - .testTag("LEGEND"), + Column( + modifier = modifier + .then(clickableModifier) + .testTag("LEGEND"), + ) { + Row( + modifier = Modifier + .padding(Spacing.Spacing16, Spacing.Spacing8, Spacing.Spacing8, Spacing.Spacing6), + horizontalArrangement = Arrangement.Center, + verticalAlignment = Alignment.CenterVertically, ) { - Row( - modifier = Modifier - .padding(Spacing.Spacing16, Spacing.Spacing8, Spacing.Spacing8, Spacing.Spacing6), - horizontalArrangement = Arrangement.Center, - verticalAlignment = Alignment.CenterVertically, - ) { - Column(modifier = Modifier.align(Alignment.Top)) { - Spacer(modifier = Modifier.size(Spacing.Spacing4).padding(end = Spacing.Spacing8)) - Box( - modifier = Modifier.size(InternalSizeValues.Size12) - .clip(CircleShape) - .background(legendData.color), - ) - } - Text( - legendData.title, - Modifier.padding(start = Spacing.Spacing8) - .weight(2f, true), - style = MaterialTheme.typography.bodyMedium, + Column(modifier = Modifier.align(Alignment.Top)) { + Spacer(modifier = Modifier.size(Spacing.Spacing4).padding(end = Spacing.Spacing8)) + Box( + modifier = Modifier.size(InternalSizeValues.Size12) + .clip(CircleShape) + .background(legendData.color), ) - - if (hasPopupLegendDescriptionData) { - Icon( - imageVector = Icons.Outlined.HelpOutline, - contentDescription = "Legend Icon", - modifier = Modifier.size(InternalSizeValues.Size18), - ) - } } - Divider( - modifier = Modifier - .fillMaxWidth() - .padding(), - thickness = Border.Regular, - color = legendData.color, + Text( + legendData.title, + Modifier.padding(start = Spacing.Spacing8) + .weight(2f, true), + style = MaterialTheme.typography.bodyMedium, ) + + if (hasPopupLegendDescriptionData) { + Icon( + imageVector = Icons.Outlined.HelpOutline, + contentDescription = "Legend Icon", + modifier = Modifier.size(InternalSizeValues.Size18), + ) + } } + Divider( + modifier = Modifier + .fillMaxWidth() + .padding(), + thickness = Border.Regular, + color = legendData.color, + ) } if (showBottomSheetShell) { diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/ListCard.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/ListCard.kt index 55f55814c..8cdb5da83 100644 --- a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/ListCard.kt +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/ListCard.kt @@ -6,6 +6,7 @@ import androidx.compose.animation.shrinkVertically import androidx.compose.foundation.Image import androidx.compose.foundation.background import androidx.compose.foundation.clickable +import androidx.compose.foundation.interaction.MutableInteractionSource import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column @@ -21,12 +22,11 @@ import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.KeyboardArrowDown import androidx.compose.material.icons.filled.KeyboardArrowUp import androidx.compose.material.icons.outlined.Sync -import androidx.compose.material.ripple.LocalRippleTheme +import androidx.compose.material.ripple.rememberRipple import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable -import androidx.compose.runtime.CompositionLocalProvider import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember @@ -38,13 +38,13 @@ import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.painter.Painter import androidx.compose.ui.layout.ContentScale import androidx.compose.ui.platform.testTag +import androidx.compose.ui.semantics.Role import androidx.compose.ui.text.style.TextOverflow import org.hisp.dhis.mobile.ui.designsystem.component.internal.conditional import org.hisp.dhis.mobile.ui.designsystem.resource.provideDHIS2Icon import org.hisp.dhis.mobile.ui.designsystem.resource.provideStringResource import org.hisp.dhis.mobile.ui.designsystem.theme.InternalSizeValues import org.hisp.dhis.mobile.ui.designsystem.theme.Radius -import org.hisp.dhis.mobile.ui.designsystem.theme.Ripple import org.hisp.dhis.mobile.ui.designsystem.theme.Spacing import org.hisp.dhis.mobile.ui.designsystem.theme.Spacing.Spacing4 import org.hisp.dhis.mobile.ui.designsystem.theme.SurfaceColor @@ -90,52 +90,59 @@ fun ListCard( expandableItemList.add(item) } } - CompositionLocalProvider(LocalRippleTheme provides Ripple.CustomDHISRippleTheme) { - Row( - modifier = modifier - .background(color = TextColor.OnPrimary) - .clip(shape = RoundedCornerShape(Radius.S)) - .clickable(onClick = onCardClick) - .padding(Spacing.Spacing8) - .hoverPointerIcon(true), - ) { - listAvatar?.let { - it.invoke() - Spacer(Modifier.size(Spacing.Spacing16)) - } - Column(Modifier.fillMaxWidth().weight(1f)) { - Row(horizontalArrangement = Arrangement.SpaceBetween) { - // Row with header and last updated - ListCardTitle(text = title, modifier.weight(1f)) - if (lastUpdated != null) { - ListCardLastUpdated(lastUpdated) - } - } - AdditionalInfoColumn( - expandableItems = expandableItemList, - constantItems = constantItemList, - modifier = Modifier.testTag("LIST_CARD_ADDITIONAL_INFO_COLUMN"), - expandLabelText = expandLabelText, - shrinkLabelText = shrinkLabelText, - syncProgressItem = AdditionalInfoItem( - icon = { - Icon( - imageVector = Icons.Outlined.Sync, - contentDescription = "Icon Button", - tint = SurfaceColor.Primary, - ) - }, - value = "Syncing...", - color = SurfaceColor.Primary, - isConstantItem = false, - ), - showLoading = showLoading, + val interactionSource = remember { MutableInteractionSource() } - ) - actionButton?.invoke() + Row( + modifier = modifier + .background(color = TextColor.OnPrimary) + .clip(shape = RoundedCornerShape(Radius.S)) + .clickable( + role = Role.Button, + interactionSource = interactionSource, + indication = rememberRipple( + color = SurfaceColor.Primary, + ), + onClick = onCardClick, + ) + .padding(Spacing.Spacing8) + .hoverPointerIcon(true), + ) { + listAvatar?.let { + it.invoke() + Spacer(Modifier.size(Spacing.Spacing16)) + } + Column(Modifier.fillMaxWidth().weight(1f)) { + Row(horizontalArrangement = Arrangement.SpaceBetween) { + // Row with header and last updated + ListCardTitle(text = title, modifier.weight(1f)) + if (lastUpdated != null) { + ListCardLastUpdated(lastUpdated) + } } - // rest of items here (KeyValue component) + AdditionalInfoColumn( + expandableItems = expandableItemList, + constantItems = constantItemList, + modifier = Modifier.testTag("LIST_CARD_ADDITIONAL_INFO_COLUMN"), + expandLabelText = expandLabelText, + shrinkLabelText = shrinkLabelText, + syncProgressItem = AdditionalInfoItem( + icon = { + Icon( + imageVector = Icons.Outlined.Sync, + contentDescription = "Icon Button", + tint = SurfaceColor.Primary, + ) + }, + value = "Syncing...", + color = SurfaceColor.Primary, + isConstantItem = false, + ), + showLoading = showLoading, + + ) + actionButton?.invoke() } + // rest of items here (KeyValue component) } } @@ -321,6 +328,7 @@ private fun AdditionalInfoColumn( if (expandableItems != null && expandableItems.size > 3) { val expandText = mutableStateOf(if (sectionState == SectionState.OPEN) shrinkLabelText else expandLabelText) + val interactionSource = remember { MutableInteractionSource() } val iconVector = if (sectionState == SectionState.CLOSE) { Icons.Filled.KeyboardArrowDown @@ -329,28 +337,33 @@ private fun AdditionalInfoColumn( } val verticalPadding = if (isDetailCard) Spacing.Spacing10 else Spacing.Spacing0 val expandTextColor = if (isDetailCard) TextColor.OnSurfaceLight else SurfaceColor.Primary - CompositionLocalProvider(LocalRippleTheme provides Ripple.CustomDHISRippleTheme) { - Row( - Modifier - .clip(RoundedCornerShape(Radius.M)) - .clickable(onClick = { + Row( + Modifier + .clip(RoundedCornerShape(Radius.M)) + .clickable( + onClick = { sectionState = if (sectionState == SectionState.CLOSE) SectionState.OPEN else SectionState.CLOSE - }) - .padding(top = verticalPadding, end = Spacing.Spacing2, bottom = verticalPadding), - ) { - Icon( - imageVector = iconVector, - contentDescription = "Button", - tint = expandTextColor, - ) - Text( - text = expandText.value, - color = expandTextColor, - style = MaterialTheme.typography.bodyMedium, - modifier = Modifier.padding(horizontal = Spacing4), + }, + role = Role.Button, + interactionSource = interactionSource, + indication = rememberRipple( + color = SurfaceColor.Primary, + ), ) - } + .padding(top = verticalPadding, end = Spacing.Spacing2, bottom = verticalPadding), + ) { + Icon( + imageVector = iconVector, + contentDescription = "Button", + tint = expandTextColor, + ) + Text( + text = expandText.value, + color = expandTextColor, + style = MaterialTheme.typography.bodyMedium, + modifier = Modifier.padding(horizontal = Spacing4), + ) } } } @@ -371,6 +384,7 @@ private fun KeyValue( ) { val keyColor: Color var valueColor: Color + val interactionSource = remember { MutableInteractionSource() } if (isDetailCard) { keyColor = AdditionalInfoItemColor.DEFAULT_KEY.color @@ -394,7 +408,14 @@ private fun KeyValue( modifier = Modifier .clip(shape = RoundedCornerShape(Radius.XS)) .conditional(additionalInfoItem.action != null, { - clickable(onClick = additionalInfoItem.action ?: {}) + clickable( + role = Role.Button, + interactionSource = interactionSource, + indication = rememberRipple( + color = SurfaceColor.Primary, + ), + onClick = additionalInfoItem.action ?: {}, + ) }), ) { Spacer(Modifier.size(Spacing4))