Professional Documents
Culture Documents
Lab 2
Lab 2
Завдання: Архітектура MVI, Додаток конвертера одиниць об'єму, UI - InputField, action switcher
Вхідної одиниці, action switcher Вихідної одиниці, TextLabel з результатом
App.kt
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material.DropdownMenu
import androidx.compose.material.DropdownMenuItem
import androidx.compose.material.MaterialTheme
import androidx.compose.material.Text
import androidx.compose.material.TextField
import androidx.compose.runtime.Composable
import androidx.compose.runtime.Immutable
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.text.input.KeyboardType
import kotlin.jvm.JvmInline
@Immutable
data class UiState(
val inputVolumeField: VolumeFieldUiState,
val outputVolumeField: VolumeFieldUiState,
)
@JvmInline
value class OutputVolumeFieldIntent(val event: VolumeFieldIntent) : Intent
@JvmInline
value class InputVolumeFieldIntent(val event: VolumeFieldIntent) : Intent
}
@Composable
fun App() {
val uiState by viewModel.uiState.collectAsState()
val obtainIntent = viewModel::obtainIntent
App(
uiState = uiState,
obtainIntent = obtainIntent
)
}
@Composable
fun App(uiState: UiState, obtainIntent: (Intent) -> Unit) {
MaterialTheme {
Box(
modifier = Modifier.fillMaxSize(),
contentAlignment = Alignment.Center
) {
Column {
VolumeField(
uiState = uiState.inputVolumeField,
obtainEvent = {
obtainIntent(Intent.InputVolumeFieldIntent(it)) },
name = "Input volume",
readOnly = false
)
VolumeField(
uiState = uiState.outputVolumeField,
obtainEvent = {
obtainIntent(Intent.OutputVolumeFieldIntent(it)) },
name = "Output volume",
readOnly = true
)
}
}
}
}
@Immutable
data class VolumeFieldUiState(
val volume: String,
val volumeType: VolumeType,
) {
enum class VolumeType(val decimals: Int, val title: String) {
Santimeters(-2, "cm^3"),
Meters(0, "m^3"),
Decimeters(-1, "dm^3"),
Kilometers(3, "km^3"),
}
}
@JvmInline
value class VolumeChange(val volume: String) : VolumeFieldIntent
@JvmInline
value class VolumeTypeChange(val type: VolumeFieldUiState.VolumeType) :
VolumeFieldIntent
}
@Composable
private fun VolumeField(
uiState: VolumeFieldUiState,
obtainEvent: (VolumeFieldIntent) -> Unit,
name: String,
readOnly: Boolean,
) {
Row(modifier = Modifier.fillMaxWidth(0.8f)) {
TextField(
value = uiState.volume,
onValueChange = { obtainEvent(VolumeFieldIntent.VolumeChange(it)) },
modifier = Modifier.weight(1f),
readOnly = readOnly,
label = { Text(name) },
keyboardOptions = KeyboardOptions(keyboardType =
KeyboardType.Number)
)
DropdownMenu(
expanded = dropdownMenuExpanded,
onDismissRequest = {
dropdownMenuExpanded = false
}
) {
VolumeFieldUiState.VolumeType.entries.forEach { type ->
DropdownMenuItem(onClick = {
obtainEvent(VolumeFieldIntent.VolumeTypeChange(type)) }) {
Text(type.name)
}
}
}
}
}
}
AppViewModel
import androidx.lifecycle.ViewModel
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.asStateFlow
import kotlinx.coroutines.flow.update
import kotlin.jvm.JvmInline
import kotlin.math.pow
)
)
val uiState = _uiState.asStateFlow()
is VolumeFieldIntent.VolumeTypeChange -> {
sendMessage(Message.InputVolumeTypeChanged(event.type))
}
}
@JvmInline
value class InputVolumeTypeChanged(val type:
VolumeFieldUiState.VolumeType) : Message
@JvmInline
value class OutputVolumeTypeChanged(val type:
VolumeFieldUiState.VolumeType) : Message
}