Skip to content

Commit

Permalink
ANDROAPP-5621-mobile-ui-Create-dropdown-component (#99)
Browse files Browse the repository at this point in the history
  • Loading branch information
siddh1004 authored Oct 16, 2023
1 parent a597283 commit 2b15310
Show file tree
Hide file tree
Showing 5 changed files with 408 additions and 0 deletions.
2 changes: 2 additions & 0 deletions common/src/commonMain/kotlin/org/hisp/dhis/common/App.kt
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ import org.hisp.dhis.common.screens.ImageBlockScreen
import org.hisp.dhis.common.screens.InputAgeScreen
import org.hisp.dhis.common.screens.InputBarCodeScreen
import org.hisp.dhis.common.screens.InputCheckBoxScreen
import org.hisp.dhis.common.screens.InputDropDownScreen
import org.hisp.dhis.common.screens.InputEmailScreen
import org.hisp.dhis.common.screens.InputIntegerScreen
import org.hisp.dhis.common.screens.InputLetterScreen
Expand Down Expand Up @@ -177,6 +178,7 @@ fun Main() {
Components.INPUT_POLYGON -> InputPolygonScreen()
Components.INPUT_ORG_UNIT -> InputOrgUnitScreen()
Components.IMAGE_BLOCK -> ImageBlockScreen()
Components.INPUT_DROPDOWN -> InputDropDownScreen()
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -49,4 +49,5 @@ enum class Components(val label: String) {
INPUT_POLYGON("Input Polygon"),
INPUT_ORG_UNIT("Input Org. Unit"),
IMAGE_BLOCK("Image Block"),
INPUT_DROPDOWN("Input Dropdown"),
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
package org.hisp.dhis.common.screens

import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.size
import androidx.compose.material3.DropdownMenu
import androidx.compose.material3.DropdownMenuItem
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue
import androidx.compose.ui.Modifier
import org.hisp.dhis.mobile.ui.designsystem.component.ColumnComponentContainer
import org.hisp.dhis.mobile.ui.designsystem.component.InputDropDown
import org.hisp.dhis.mobile.ui.designsystem.component.InputShellState
import org.hisp.dhis.mobile.ui.designsystem.component.SubTitle
import org.hisp.dhis.mobile.ui.designsystem.component.Title
import org.hisp.dhis.mobile.ui.designsystem.theme.Spacing
import org.hisp.dhis.mobile.ui.designsystem.theme.TextColor

@Composable
fun InputDropDownScreen() {
ColumnComponentContainer {
val options = listOf("Option 1", "Option 2", "Option 3", "Option 4", "Option 5", "Option 6", "Option 7")
var expanded by rememberSaveable { mutableStateOf(false) }

Title("Input Dropdown", textColor = TextColor.OnSurfaceVariant)

SubTitle("Basic Input Dropdown ", textColor = TextColor.OnSurfaceVariant)
var selectedItem by rememberSaveable { mutableStateOf<String?>(null) }
Box {
InputDropDown(
title = "Label",
state = InputShellState.UNFOCUSED,
selectedItem = selectedItem,
onResetButtonClicked = {
selectedItem = null
},
onArrowDropDownButtonClicked = {
expanded = !expanded
},
)
DropdownMenu(
expanded = expanded,
onDismissRequest = { expanded = false },
) {
options.forEach {
DropdownMenuItem(
text = { Text(it) },
onClick = {
selectedItem = it
expanded = false
},
)
}
}
}
Spacer(Modifier.size(Spacing.Spacing18))

SubTitle("Basic Input Dropdown with content ", textColor = TextColor.OnSurfaceVariant)
var selectedItem1 by rememberSaveable { mutableStateOf<String?>(options[0]) }
InputDropDown(
title = "Label",
state = InputShellState.UNFOCUSED,
selectedItem = selectedItem1,
onResetButtonClicked = {
selectedItem1 = null
},
onArrowDropDownButtonClicked = {
},
)
Spacer(Modifier.size(Spacing.Spacing18))

SubTitle("Error Input Dropdown ", textColor = TextColor.OnSurfaceVariant)
var selectedItem2 by rememberSaveable { mutableStateOf<String?>(null) }
InputDropDown(
title = "Label",
state = InputShellState.ERROR,
selectedItem = selectedItem2,
onResetButtonClicked = {
selectedItem2 = null
},
onArrowDropDownButtonClicked = {
},
)
Spacer(Modifier.size(Spacing.Spacing18))

SubTitle("Disabled Input Dropdown with content ", textColor = TextColor.OnSurfaceVariant)
var selectedItem3 by rememberSaveable { mutableStateOf<String?>(options[1]) }
InputDropDown(
title = "Label",
state = InputShellState.DISABLED,
selectedItem = selectedItem3,
onResetButtonClicked = {
selectedItem3 = null
},
onArrowDropDownButtonClicked = {
expanded = !expanded
},
)
Spacer(Modifier.size(Spacing.Spacing18))
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
package org.hisp.dhis.mobile.ui.designsystem.component

import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.ArrowDropDown
import androidx.compose.material.icons.outlined.Cancel
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.alpha
import androidx.compose.ui.focus.FocusRequester
import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.focus.onFocusChanged
import androidx.compose.ui.platform.testTag
import org.hisp.dhis.mobile.ui.designsystem.theme.TextColor

/**
* DHIS2 Input dropdown. Wraps DHIS · [InputShell].
* @param title controls the text to be shown for the title
* @param state Manages the InputShell state
* @param selectedItem manages the value of the selected item
* @param supportingTextData is a list of SupportingTextData that
* manages all the messages to be shown
* @param legendData manages the legendComponent
* @param isRequiredField controls whether the field is mandatory or not
* @param onFocusChanged gives access to the onFocusChanged returns true if
* item is focused
* @param modifier allows a modifier to be passed externally
* @param onResetButtonClicked callback to when reset button is clicked
* @param onArrowDropDownButtonClicked callback to when arrow drop down button is clicked
*/
@Composable
fun InputDropDown(
title: String,
state: InputShellState,
selectedItem: String? = null,
supportingTextData: List<SupportingTextData>? = null,
legendData: LegendData? = null,
isRequiredField: Boolean = false,
modifier: Modifier = Modifier,
onFocusChanged: ((Boolean) -> Unit)? = null,
onResetButtonClicked: () -> Unit,
onArrowDropDownButtonClicked: () -> Unit,
) {
val focusRequester = remember { FocusRequester() }
InputShell(
modifier = modifier
.testTag("INPUT_DROPDOWN")
.focusRequester(focusRequester),
title = title,
state = state,
isRequiredField = isRequiredField,
onFocusChanged = onFocusChanged,
supportingText = {
supportingTextData?.forEach { label ->
SupportingText(
label.text,
label.state,
modifier = modifier.testTag("INPUT_DROPDOWN_SUPPORTING_TEXT"),
)
}
},
legend = {
legendData?.let {
Legend(legendData, modifier.testTag("INPUT_DROPDOWN_LEGEND"))
}
},
inputField = {
Box {
Text(
modifier = Modifier
.testTag("INPUT_DROPDOWN_TEXT")
.fillMaxWidth(),
text = selectedItem ?: "",
style = MaterialTheme.typography.bodyLarge.copy(
color = if (state != InputShellState.DISABLED) {
TextColor.OnSurface
} else {
TextColor.OnDisabledSurface
},
),
)
Box(
modifier = Modifier
.matchParentSize()
.alpha(0f)
.clickable(
enabled = state != InputShellState.DISABLED,
onClick = {
focusRequester.requestFocus()
onArrowDropDownButtonClicked.invoke()
},
),
)
}
},
primaryButton = {
IconButton(
modifier = Modifier.testTag("INPUT_DROPDOWN_ARROW_BUTTON").onFocusChanged {
onFocusChanged?.invoke(it.isFocused)
},
enabled = state != InputShellState.DISABLED,
icon = {
Icon(
imageVector = Icons.Outlined.ArrowDropDown,
contentDescription = "Dropdown Button",
)
},
onClick = {
focusRequester.requestFocus()
onArrowDropDownButtonClicked.invoke()
},
)
},
secondaryButton =
if (!selectedItem.isNullOrEmpty() && state != InputShellState.DISABLED) {
{
IconButton(
modifier = Modifier.testTag("INPUT_DROPDOWN_RESET_BUTTON"),
icon = {
Icon(
imageVector = Icons.Outlined.Cancel,
contentDescription = "Reset Button",
)
},
onClick = onResetButtonClicked,
)
}
} else {
null
},
)
}
Loading

0 comments on commit 2b15310

Please sign in to comment.