Download as docx, pdf, or txt
Download as docx, pdf, or txt
You are on page 1of 7

Федчук Тамара БС-15, 11 варіант

Завдання: Архітектура 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,
)

sealed interface Intent {

@JvmInline
value class OutputVolumeFieldIntent(val event: VolumeFieldIntent) : Intent

@JvmInline
value class InputVolumeFieldIntent(val event: VolumeFieldIntent) : Intent
}

private val viewModel = AppViewModel()

@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"),

}
}

sealed interface VolumeFieldIntent {

@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)
)

var dropdownMenuExpanded by remember { mutableStateOf(false) }


Box {
Text(uiState.volumeType.title, modifier = Modifier.clickable {
dropdownMenuExpanded = true })

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

class AppViewModel : ViewModel() {

private val _uiState = MutableStateFlow(


UiState(
inputVolumeField = VolumeFieldUiState(
volume = "0",
volumeType = VolumeFieldUiState.VolumeType.Santimeters
),
outputVolumeField = VolumeFieldUiState(
volume = "0",
volumeType = VolumeFieldUiState.VolumeType.Meters
)

)
)
val uiState = _uiState.asStateFlow()

private val reducer = Reducer()

private val executor = Executor(


sendLabel = { label -> TODO("Ooops, you haven't implemented freshly
added label") },
sendMessage = { message ->
_uiState.update { uiStateValue -> reducer.reduce(uiStateValue,
message) }
}
)

fun obtainIntent(intent: Intent) = executor.execute(intent)

private class Executor(


private val sendLabel: (Label) -> Unit,
private val sendMessage: (Message) -> Unit
) {
fun execute(intent: Intent) {
when (intent) {
is Intent.InputVolumeFieldIntent -> when (val event =
intent.event) {
is VolumeFieldIntent.VolumeChange -> {
sendMessage(Message.InputVolumeChanged(event.volume))
}

is VolumeFieldIntent.VolumeTypeChange -> {
sendMessage(Message.InputVolumeTypeChanged(event.type))
}
}

is Intent.OutputVolumeFieldIntent -> when (val event =


intent.event) {
is VolumeFieldIntent.VolumeChange -> TODO("Ooops, you set
read only to true in output volume field")

is VolumeFieldIntent.VolumeTypeChange -> sendMessage(


Message.OutputVolumeTypeChanged(
event.type
)
)
}
}
}
}

sealed interface Label

sealed interface Message {


@JvmInline
value class InputVolumeChanged(val volume: String) : Message

@JvmInline
value class InputVolumeTypeChanged(val type:
VolumeFieldUiState.VolumeType) : Message

@JvmInline
value class OutputVolumeTypeChanged(val type:
VolumeFieldUiState.VolumeType) : Message
}

private class Reducer {

private fun calculateVolume(


volume: String,
from: VolumeFieldUiState.VolumeType,
to: VolumeFieldUiState.VolumeType
): String {
val inPower = from.decimals
val outPower = to.decimals
val power = inPower - outPower
return (volume.toDouble() * 10.0.pow(power * 3)).toString()
}

fun reduce(state: UiState, message: Message): UiState {


return when (message) {
is Message.InputVolumeChanged -> state.copy(
inputVolumeField = state.inputVolumeField.copy(
volume = message.volume
),
outputVolumeField = state.outputVolumeField.copy(
volume = calculateVolume(
volume = message.volume,
from = state.inputVolumeField.volumeType,
to = state.outputVolumeField.volumeType
)
)
)

is Message.InputVolumeTypeChanged -> state.copy(


inputVolumeField = state.inputVolumeField.copy(
volumeType = message.type
),
outputVolumeField = state.outputVolumeField.copy(
volume = calculateVolume(
volume = state.inputVolumeField.volume,
from = message.type,
to = state.outputVolumeField.volumeType
)
)
)

is Message.OutputVolumeTypeChanged -> state.copy(


outputVolumeField = state.outputVolumeField.copy(
volumeType = message.type,
volume = calculateVolume(
volume = state.inputVolumeField.volume,
from = state.inputVolumeField.volumeType,
to = message.type
)
),
)
}
}
}
}
Результати:

You might also like