In today’s tutorial, we will learn about the Chip in KMM KMP Compose Multiplatform for both Android & iOS platforms. Currently, there are 2 Chip components available in Kotlin multiplatform-Kotlin multiplatform mobile. Chip component and FilterChip component. We will learn about both, including designing custom chips like the Assist, Input, Suggestion, and Elevated chips.
Chip in KMM KMP Compose Multiplatform Tutorial:
The Chip component is a UI element used for actions, filtering content, showing user selections, and showing emails or contacts uniquely. We all have seen Chip in many mobile and web applications. In the mobile, when we select a filter option in the filter, our selection starts showing at the top in Chips format. When we send an email to multiple contacts, it will show in Chip format after typing each email. These are the basic examples of Chip. The Chip component displays Text and Icons or even images.
Creating Simple Chip in KMP(KMM):
In KMP, we will use the Chip keyword to create a Chip component.
1 2 3 4 5 |
Chip( onClick = { /* Do something! */ } ) { Text("Simple Chip") } |
Code explanation:
- onClick: Invokes when the user clicks or taps on the Chip.
- Text: The body part Text component.
Creating Chip with Icons:
To create a Chip component with vector icons, we have to use the leadingIcon property.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
Chip( onClick = { println("Chip Clicked...!") }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors(), leadingIcon = { Icon( Icons.Filled.Settings, contentDescription = "Localized description" ) }) { Text("Change settings") } |
Code explanation:
- border: To show a simple line border around the Chip.
- colors: To manage Chip colors.
- leadingIcon: Pass any component you want at the start of the chip. We are showing Icon.
Complete code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 |
package com.app.test import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.padding import androidx.compose.material.Chip import androidx.compose.material.ChipDefaults import androidx.compose.material.ExperimentalMaterialApi import androidx.compose.material.Icon import androidx.compose.material.MaterialTheme import androidx.compose.material.Text import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Settings import androidx.compose.runtime.* import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp @OptIn(ExperimentalMaterialApi::class) @Composable fun App() { MaterialTheme { Column( modifier = Modifier .padding(12.dp) .fillMaxWidth(), verticalArrangement = Arrangement.Top, horizontalAlignment = Alignment.CenterHorizontally ) { Text( text = "Chip in KMP(KMM)", style = MaterialTheme.typography.h6, color = Color.Black, textAlign = TextAlign.Center ) Chip( onClick = { println("Chip Clicked...!") }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors(), leadingIcon = { Icon( Icons.Filled.Settings, contentDescription = "Localized description" ) }) { Text("Change settings") } } } } |
Screenshot:
Creating Chip with Custom Colors:
The Chip component has a property called ChipDefaults.outlinedChipColors() has a few different sub-properties, which are used to set every color in a Chip.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
Chip( onClick = { /* Do something! */ }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors( disabledLeadingIconContentColor = Color.LightGray, disabledContentColor = Color.LightGray, contentColor = Color(0xffFFFFFF), backgroundColor = Color(0xff00B8D4), disabledBackgroundColor = Color(0xffCFD8DC), leadingIconContentColor = Color(0xffFFFFFF) ), leadingIcon = { Icon( Icons.Filled.Settings, contentDescription = "Localized description" ) }) { Text("Change settings") } |
Code explanation:
- disabledLeadingIconContentColor: When the Chip is disabled, this color will be applied to the chip leading component.
- disabledContentColor: Applied on the Chip content when it is disabled.
- contentColor: Default chip content Text color.
- backgroundColor: Background color of complete Chip.
- disabledBackgroundColor: The background color of the Chip when it is disabled.
- leadingIconContentColor: Default chip leading icon or widget tint color.
Screenshot:
Dynamically generating multiple Chip:
We use the repeat() in the FlowRow() component to create Chip dynamically. For testing purposes, we generate 10 chips and print the index position for each chip.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
FlowRow( modifier = Modifier .fillMaxWidth() .wrapContentHeight(align = Alignment.Top), horizontalArrangement = Arrangement.Start ) { repeat(10) { index -> Chip( modifier = Modifier .padding(horizontal = 4.dp) .align(Alignment.CenterVertically), onClick = { /* do something */ } ) { Text("Chip $index") } } } |
Screenshot:
Creating Disabled Chip:
To create a Disabled Chip, we have to use the enabled property. This supports value in Boolean format. In the disabled state, the onClick method will not work.
- enabled = false :- Will disable the Chip.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
Chip( onClick = { /* Do something! */ }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors( disabledLeadingIconContentColor = Color.White, disabledContentColor = Color.White, contentColor = Color(0xffFFFFFF), backgroundColor = Color(0xff00B8D4), disabledBackgroundColor = Color(0xffCFD8DC), leadingIconContentColor = Color(0xffFFFFFF) ), leadingIcon = { Icon( Icons.Filled.Settings, contentDescription = "Localized description" ) }, enabled = false ) { Text("Change settings") } |
Screenshot:
Creating Filter Chip:
Filter chips apply different types of filters to data in Kotlin Compose Multiplatform. In this example, we show selected filter chip items in a Text.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
val chipItems = listOf("Option 1", "Option 2", "Option 3", "Option 4") val selectedChips = remember { mutableStateListOf<String>() } chipItems.forEach { chipLabel -> FilterChip( selected = selectedChips.contains(chipLabel), onClick = { if (selectedChips.contains(chipLabel)) { selectedChips.remove(chipLabel) } else { selectedChips.add(chipLabel) } }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedFilterChipColors(), ) { Text(chipLabel) } } |
Code explanation:
- chipItems: This state contains all the Chip items.
- selectedChips: All the selected Chip items will be stored in this state.
- chipItems.forEach{}: We will run a for each loop as per chip items so it will draw all of these items on the screen.
- filterChip: Filter chip widget.
- selected: Contains selected chips.
- onClick: Invokes every time the user Taps on a Chip, then we will check if that item exists or not in the State and update data.
Complete code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 |
package com.app.test import androidx.compose.foundation.layout.* import androidx.compose.foundation.rememberScrollState import androidx.compose.foundation.horizontalScroll import androidx.compose.material.* import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Settings import androidx.compose.runtime.* import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp @OptIn(ExperimentalMaterialApi::class) @Composable fun App() { MaterialTheme { Column( modifier = Modifier .padding(12.dp) .fillMaxWidth(), verticalArrangement = Arrangement.Top, horizontalAlignment = Alignment.CenterHorizontally ) { Text( text = "Filter Chips in Jetpack Compose", style = MaterialTheme.typography.h6, color = Color.Black, textAlign = TextAlign.Center ) Spacer(modifier = Modifier.height(8.dp)) val chipItems = listOf("Option 1", "Option 2", "Option 3", "Option 4") val selectedChips = remember { mutableStateListOf<String>() } // Row containing Filter Chips Row( modifier = Modifier .fillMaxWidth() .horizontalScroll(rememberScrollState()), horizontalArrangement = Arrangement.spacedBy(8.dp) ) { chipItems.forEach { chipLabel -> FilterChip( selected = selectedChips.contains(chipLabel), onClick = { if (selectedChips.contains(chipLabel)) { selectedChips.remove(chipLabel) } else { selectedChips.add(chipLabel) } }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedFilterChipColors(), ) { Text(chipLabel) } } } Spacer(modifier = Modifier.height(16.dp)) // Display Selected Chips Text( text = if (selectedChips.isEmpty()) "No filter selected" else "Selected Filters: ${selectedChips.joinToString()}", style = MaterialTheme.typography.body1, color = Color.Black, textAlign = TextAlign.Center ) } } } |
Screenshots:
Input Chip in KMP KMM:
Input chips are dynamically generated at application run time using TextField. We have seen these types of Chips in Emails and messages when we add multiple contacts.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 |
var inputText by remember { mutableStateOf("") } val chipList = remember { mutableStateListOf<String>() } OutlinedTextField( value = inputText, onValueChange = { inputText = it }, label = { Text("Enter Chip Data") }, modifier = Modifier.padding(16.dp) ) Spacer(modifier = Modifier.height(12.dp)) Button( onClick = { if (inputText.isNotBlank()) { chipList.add(inputText) inputText = "" } } ) { Text("Create Chip") } Spacer(modifier = Modifier.height(12.dp)) // Display Chips FlowRow( modifier = Modifier.fillMaxWidth().wrapContentHeight(align = Alignment.Top), horizontalArrangement = Arrangement.Center ) { chipList.forEach { chipText -> Chip( onClick = { /* Clicked on Chip */ }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors() ) { Row( verticalAlignment = Alignment.CenterVertically ) { Text(chipText) Spacer(modifier = Modifier.width(3.dp)) IconButton( onClick = { chipList.remove(chipText) } ) { Icon( Icons.Filled.Close, contentDescription = "Remove Chip" ) } } } } } |
Code explanation:
- inputText: Will contains the TextField typed text.
- chipList: Stores the Chip list.
- OutlinedTextField: Text Input component to take input from the user.
Complete code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 |
package com.app.test import androidx.compose.foundation.layout.* import androidx.compose.material.* import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Close import androidx.compose.material.icons.filled.Settings import androidx.compose.runtime.* import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp @OptIn(ExperimentalMaterialApi::class, ExperimentalLayoutApi::class) @Composable fun App() { MaterialTheme { Column( modifier = Modifier .padding(12.dp) .fillMaxWidth(), verticalArrangement = Arrangement.Top, horizontalAlignment = Alignment.CenterHorizontally ) { Text( text = "Input Chip Example in KMM KMP", style = MaterialTheme.typography.h6, color = Color.Black, textAlign = TextAlign.Center ) Spacer(modifier = Modifier.height(8.dp)) var inputText by remember { mutableStateOf("") } val chipList = remember { mutableStateListOf<String>() } OutlinedTextField( value = inputText, onValueChange = { inputText = it }, label = { Text("Enter Chip Data") }, modifier = Modifier.padding(16.dp) ) Spacer(modifier = Modifier.height(12.dp)) Button( onClick = { if (inputText.isNotBlank()) { chipList.add(inputText) inputText = "" } } ) { Text("Create Chip") } Spacer(modifier = Modifier.height(12.dp)) // Display Chips FlowRow( modifier = Modifier.fillMaxWidth().wrapContentHeight(align = Alignment.Top), horizontalArrangement = Arrangement.Center ) { chipList.forEach { chipText -> Chip( onClick = { /* Clicked on Chip */ }, border = ChipDefaults.outlinedBorder, colors = ChipDefaults.outlinedChipColors() ) { Row( verticalAlignment = Alignment.CenterVertically ) { Text(chipText) Spacer(modifier = Modifier.width(3.dp)) IconButton( onClick = { chipList.remove(chipText) } ) { Icon( Icons.Filled.Close, contentDescription = "Remove Chip" ) } } } } } } } } |
Screenshots:
Screenshot in iOS: