From 72304e98f78613327af2d6c2899be3ae2e06713a Mon Sep 17 00:00:00 2001 From: Sasikanth Miriyampalli Date: Fri, 6 Oct 2023 18:02:59 +0530 Subject: [PATCH] Add `InputAge` component --- .../dhis/common/screens/InputAgeScreen.kt | 86 ++++++- .../designsystem/component/AgeFieldHelper.kt | 14 +- .../ui/designsystem/component/InputAge.kt | 217 ++++++++++++++++++ .../ui/designsystem/component/InputField.kt | 18 +- .../component/internal/StringUtils.kt | 56 +++++ 5 files changed, 382 insertions(+), 9 deletions(-) create mode 100644 designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputAge.kt diff --git a/common/src/commonMain/kotlin/org/hisp/dhis/common/screens/InputAgeScreen.kt b/common/src/commonMain/kotlin/org/hisp/dhis/common/screens/InputAgeScreen.kt index 0b9b6aa6d..f55210902 100644 --- a/common/src/commonMain/kotlin/org/hisp/dhis/common/screens/InputAgeScreen.kt +++ b/common/src/commonMain/kotlin/org/hisp/dhis/common/screens/InputAgeScreen.kt @@ -5,7 +5,10 @@ import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.setValue +import org.hisp.dhis.mobile.ui.designsystem.component.AgeInputType import org.hisp.dhis.mobile.ui.designsystem.component.ColumnComponentContainer +import org.hisp.dhis.mobile.ui.designsystem.component.InputAge +import org.hisp.dhis.mobile.ui.designsystem.component.InputShellState import org.hisp.dhis.mobile.ui.designsystem.component.Orientation import org.hisp.dhis.mobile.ui.designsystem.component.RadioButtonData import org.hisp.dhis.mobile.ui.designsystem.component.SubTitle @@ -14,7 +17,7 @@ import org.hisp.dhis.mobile.ui.designsystem.component.TimeUnitValues @Composable fun InputAgeScreen() { - ColumnComponentContainer("Age Field components") { + ColumnComponentContainer { SubTitle("Horizontal Age Field Helper") var selectedFieldHorizontal by remember { mutableStateOf(RadioButtonData("0", selected = true, enabled = true, textInput = TimeUnitValues.YEARS.value)) @@ -22,5 +25,86 @@ fun InputAgeScreen() { TimeUnitSelector(Orientation.HORIZONTAL, TimeUnitValues.YEARS.value) { selectedFieldHorizontal = it } + + SubTitle("Input Age Component - Idle") + var inputType by remember { mutableStateOf(AgeInputType.None) } + + InputAge( + title = "Label", + inputType = inputType, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { newInputType -> + inputType = newInputType + }, + ) + + SubTitle("Input Age Component - Idle Disabled") + InputAge( + title = "Label", + inputType = AgeInputType.None, + state = InputShellState.DISABLED, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { newInputType -> + inputType = newInputType + }, + ) + + SubTitle("Input Age Component - Date Of Birth") + InputAge( + title = "Label", + inputType = AgeInputType.DateOfBirth("01011985"), + state = InputShellState.DISABLED, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { newInputType -> + inputType = newInputType + }, + ) + + SubTitle("Input Age Component - Date Of Birth Required Error") + InputAge( + title = "Label", + inputType = AgeInputType.DateOfBirth("010"), + state = InputShellState.ERROR, + isRequired = true, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { + // no-op + }, + ) + + SubTitle("Input Age Component - Age Disabled") + InputAge( + title = "Label", + inputType = AgeInputType.Age(value = "56", unit = TimeUnitValues.YEARS), + state = InputShellState.DISABLED, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { newInputType -> + inputType = newInputType + }, + ) + + SubTitle("Input Age Component - Age Required Error") + InputAge( + title = "Label", + inputType = AgeInputType.Age(value = "56", unit = TimeUnitValues.YEARS), + state = InputShellState.ERROR, + isRequired = true, + onCalendarActionClicked = { + // no-op + }, + onValueChanged = { + // no-op + }, + ) } } diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/AgeFieldHelper.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/AgeFieldHelper.kt index 5daf918a1..704c2ff37 100644 --- a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/AgeFieldHelper.kt +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/AgeFieldHelper.kt @@ -25,18 +25,26 @@ import org.hisp.dhis.mobile.ui.designsystem.theme.SurfaceColor fun TimeUnitSelector( orientation: Orientation, optionSelected: String, + modifier: Modifier = Modifier, + enabled: Boolean = true, onClick: (RadioButtonData) -> Unit, ) { + val backgroundColor = if (enabled) { + SurfaceColor.Surface + } else { + SurfaceColor.DisabledSurface + } + RowComponentContainer( - modifier = Modifier - .background(color = SurfaceColor.Surface, Shape.SmallBottom) + modifier = modifier + .background(color = backgroundColor, Shape.SmallBottom) .padding( start = Spacing.Spacing8, end = Spacing.Spacing8, ), ) { val options = TimeUnitValues.values().map { - RadioButtonData(it.value, optionSelected == it.value, true, provideStringResource(it.value)) + RadioButtonData(it.value, optionSelected == it.value, enabled, provideStringResource(it.value)) } val selectedItem = options.find { it.selected diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputAge.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputAge.kt new file mode 100644 index 000000000..3b3c70d0d --- /dev/null +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputAge.kt @@ -0,0 +1,217 @@ +package org.hisp.dhis.mobile.ui.designsystem.component + +import androidx.compose.foundation.focusable +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.Event +import androidx.compose.material.icons.outlined.Cancel +import androidx.compose.material3.Icon +import androidx.compose.runtime.Composable +import androidx.compose.runtime.remember +import androidx.compose.ui.Modifier +import androidx.compose.ui.platform.testTag +import org.hisp.dhis.mobile.ui.designsystem.component.AgeInputType.Age +import org.hisp.dhis.mobile.ui.designsystem.component.AgeInputType.DateOfBirth +import org.hisp.dhis.mobile.ui.designsystem.component.AgeInputType.None +import org.hisp.dhis.mobile.ui.designsystem.component.TimeUnitValues.YEARS +import org.hisp.dhis.mobile.ui.designsystem.component.internal.RegExValidations +import org.hisp.dhis.mobile.ui.designsystem.resource.provideStringResource +import org.hisp.dhis.mobile.ui.designsystem.theme.Spacing + +// Update [DateOfBirthTransformation] when updating the mask +// Check the usages before modifying +private const val DATE_OF_BIRTH_MASK = "DDMMYYYY" + +/** + * Input filed to enter date-of-birth or age + * + * @param title: Label of the component. + * @param inputType: The type of input [None], [DateOfBirth], [Age]. + * @param onCalendarActionClicked: Callback to handle the action when the calendar icon is clicked. + * @param state: [InputShellState] + * @param supportingText: List of [SupportingTextData] that manages all the messages to be shown. + * @param isRequired: Mark this input as marked + * @param onValueChanged: Callback to receive changes in the input + */ +@Composable +fun InputAge( + title: String, + inputType: AgeInputType = None, + onCalendarActionClicked: () -> Unit, + modifier: Modifier = Modifier, + state: InputShellState = InputShellState.UNFOCUSED, + supportingText: List? = null, + isRequired: Boolean = false, + testTag: String = "", + dateOfBirthLabel: String = provideStringResource("date_birth"), + orLabel: String = provideStringResource("or"), + ageLabel: String = provideStringResource("age"), + onValueChanged: (AgeInputType) -> Unit, +) { + val maxAgeCharLimit = 3 + val allowedCharacters = RegExValidations.DATE_OF_BIRTH.regex + + val helperText = remember(inputType) { + when (inputType) { + None -> null + is DateOfBirth -> DATE_OF_BIRTH_MASK + is Age -> inputType.unit.value + } + } + val helperStyle = remember(inputType) { + when (inputType) { + None -> InputStyle.NONE + is DateOfBirth -> InputStyle.WITH_DATE_OF_BIRTH_HELPER + is Age -> InputStyle.WITH_HELPER_AFTER + } + } + + val calendarButton: (@Composable () -> Unit)? = if (inputType is DateOfBirth) { + @Composable { + SquareIconButton( + modifier = Modifier.testTag("INPUT_AGE_${testTag}_OPEN_CALENDAR_BUTTON"), + icon = { + Icon( + imageVector = Icons.Filled.Event, + contentDescription = null, + ) + }, + onClick = onCalendarActionClicked, + enabled = state != InputShellState.DISABLED, + ) + } + } else { + null + } + + InputShell( + modifier = modifier.testTag("INPUT_AGE_$testTag"), + title = title, + state = state, + isRequiredField = isRequired, + inputField = { + when (inputType) { + None -> { + TextButtonSelector( + modifier = Modifier.focusable(true), + firstOptionText = dateOfBirthLabel, + onClickFirstOption = { + onValueChanged.invoke(DateOfBirth.EMPTY) + }, + middleText = orLabel, + secondOptionText = ageLabel, + onClickSecondOption = { + onValueChanged.invoke(Age.EMPTY) + }, + enabled = state != InputShellState.DISABLED, + ) + } + is DateOfBirth, is Age -> { + BasicTextField( + modifier = Modifier + .testTag("INPUT_" + testTag + "_FIELD") + .fillMaxWidth(), + inputText = transformInputText(inputType), + helper = helperText, + isSingleLine = true, + helperStyle = helperStyle, + onInputChanged = { newText -> + if (newText.length > maxAgeCharLimit && inputType is Age) { + return@BasicTextField + } + + @Suppress("KotlinConstantConditions") + val newInputType: AgeInputType = when (inputType) { + is Age -> inputType.copy(value = newText) + is DateOfBirth -> updateDateOfBirth(inputType, newText) + None -> None + } + + if (allowedCharacters.containsMatchIn(newText) || newText.isBlank()) { + onValueChanged.invoke(newInputType) + } + }, + enabled = state != InputShellState.DISABLED, + state = state, + ) + } + } + }, + primaryButton = { + if (inputType != None) { + IconButton( + modifier = Modifier.testTag("INPUT_AGE_" + testTag + "_RESET_BUTTON").padding(Spacing.Spacing0), + icon = { + Icon( + imageVector = Icons.Outlined.Cancel, + contentDescription = "Icon Button", + ) + }, + onClick = { + onValueChanged.invoke(None) + }, + enabled = state != InputShellState.DISABLED, + ) + } + }, + secondaryButton = calendarButton, + supportingText = { + supportingText?.forEach { label -> + SupportingText( + label.text, + label.state, + modifier = Modifier.testTag("INPUT_AGE_" + testTag + "_SUPPORTING_TEXT"), + ) + } + }, + legend = { + if (inputType is Age) { + TimeUnitSelector( + modifier = Modifier.fillMaxWidth(), + orientation = Orientation.HORIZONTAL, + optionSelected = YEARS.value, + enabled = state != InputShellState.DISABLED, + onClick = { itemData -> + val timeUnit = TimeUnitValues.values() + .first { it.value.contains(itemData.textInput!!, ignoreCase = true) } + + onValueChanged.invoke(inputType.copy(unit = timeUnit)) + }, + ) + } + }, + ) +} + +private fun transformInputText(inputType: AgeInputType): String { + return when (inputType) { + is Age -> inputType.value + is DateOfBirth -> inputType.value + None -> "" + } +} + +private fun updateDateOfBirth(inputType: DateOfBirth, newText: String): AgeInputType { + return if (newText.length <= DATE_OF_BIRTH_MASK.replace("/", "").length) { + inputType.copy(value = newText) + } else { + inputType + } +} + +sealed interface AgeInputType { + data object None : AgeInputType + + data class DateOfBirth(val value: String) : AgeInputType { + companion object { + val EMPTY = DateOfBirth("") + } + } + + data class Age(val value: String, val unit: TimeUnitValues) : AgeInputType { + companion object { + val EMPTY = Age("", YEARS) + } + } +} diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputField.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputField.kt index 3449dad06..c575a1bd6 100644 --- a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputField.kt +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/InputField.kt @@ -28,6 +28,7 @@ import androidx.compose.ui.graphics.drawscope.Stroke import androidx.compose.ui.platform.LocalSoftwareKeyboardController import androidx.compose.ui.text.input.ImeAction import androidx.compose.ui.text.input.VisualTransformation +import org.hisp.dhis.mobile.ui.designsystem.component.internal.DateOfBirthTransformation import org.hisp.dhis.mobile.ui.designsystem.component.internal.PrefixTransformation import org.hisp.dhis.mobile.ui.designsystem.component.internal.SuffixTransformer import org.hisp.dhis.mobile.ui.designsystem.theme.Color.Blue300 @@ -91,11 +92,17 @@ fun BasicTextField( var visualTransformation = VisualTransformation.None if (helperStyle != InputStyle.NONE) { - if (helperStyle == InputStyle.WITH_HELPER_BEFORE) { - helper?.let { visualTransformation = PrefixTransformation(it, enabled) } - } else { - helper?.let { - visualTransformation = SuffixTransformer(it) + when (helperStyle) { + InputStyle.WITH_HELPER_BEFORE -> { + helper?.let { visualTransformation = PrefixTransformation(it, enabled) } + } + InputStyle.WITH_DATE_OF_BIRTH_HELPER -> { + helper?.let { visualTransformation = DateOfBirthTransformation(it) } + } + else -> { + helper?.let { + visualTransformation = SuffixTransformer(it) + } } } } @@ -153,5 +160,6 @@ fun BasicTextField( enum class InputStyle { WITH_HELPER_AFTER, WITH_HELPER_BEFORE, + WITH_DATE_OF_BIRTH_HELPER, NONE, } diff --git a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/internal/StringUtils.kt b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/internal/StringUtils.kt index cdea45c03..8e803781d 100644 --- a/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/internal/StringUtils.kt +++ b/designsystem/src/commonMain/kotlin/org/hisp/dhis/mobile/ui/designsystem/component/internal/StringUtils.kt @@ -1,6 +1,7 @@ package org.hisp.dhis.mobile.ui.designsystem.component.internal import androidx.compose.ui.text.AnnotatedString +import androidx.compose.ui.text.buildAnnotatedString import androidx.compose.ui.text.input.OffsetMapping import androidx.compose.ui.text.input.TransformedText import androidx.compose.ui.text.input.VisualTransformation @@ -62,6 +63,60 @@ internal class SuffixTransformer(val suffix: String) : VisualTransformation { } } +internal class DateOfBirthTransformation(private val mask: String) : VisualTransformation { + + companion object { + private const val SEPARATOR = "/" + } + + override fun filter(text: AnnotatedString): TransformedText { + return dateFilter(text) + } + + private fun dateFilter(text: AnnotatedString): TransformedText { + val trimmed = if (text.text.length > mask.length) text.text.substring(0..mask.length) else text.text + val output = buildAnnotatedString { + for (i in mask.indices) { + val dateChar = trimmed.getOrNull(i) + if (dateChar == null) { + append(AnnotatedString(mask[i].toString(), DHIS2SCustomTextStyles.inputFieldHelper)) + } else { + append(trimmed[i]) + } + + if (i % 2 == 1 && i < 4) { + val separator = if (dateChar != null) { + SEPARATOR + } else { + AnnotatedString(SEPARATOR, DHIS2SCustomTextStyles.inputFieldHelper) + } + append(separator) + } + } + } + + val offsetMapping = object : OffsetMapping { + override fun originalToTransformed(offset: Int): Int { + if (trimmed.lastIndex >= 0) { + if (offset <= 1) return offset + if (offset <= 3) return offset + 1 + if (offset <= 8) return offset + 2 + return 10 + } else { + return 0 + } + } + + override fun transformedToOriginal(offset: Int): Int { + if (offset > text.length) return text.length + return offset + } + } + + return TransformedText(output, offsetMapping) + } +} + enum class RegExValidations(val regex: Regex) { BRITISH_DECIMAL_NOTATION("""^(?!\.)(?!.*-[^0-9])(?!(?:[^.]*\.){3})[-0-9]*(?:\.[0-9]*)?$""".toRegex()), EUROPEAN_DECIMAL_NOTATION("""^(?!.*,.+,|.*-.*-)[0-9,-]*$""".toRegex()), @@ -74,4 +129,5 @@ enum class RegExValidations(val regex: Regex) { PHONE_NUMBER("^[+0-9-()]+$".toRegex()), LINK("((https?|ftp|smtp)://)?(www\\.)?[a-zA-Z0-9@:%._+~#=-]{2,256}\\.[a-z]{2,6}\\b([-a-zA-Z0-9@:%_+.~#?&/=-]*)".toRegex()), EMAIL("^[a-zA-Z0-9._%-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,6}\$".toRegex()), + DATE_OF_BIRTH("^[0-9]+$".toRegex()), }