Move date time pickers

This commit is contained in:
Sad Ellie 2023-10-09 12:40:07 +03:00
parent ea6e6b566e
commit 25cbd82534
4 changed files with 172 additions and 135 deletions

View File

@ -16,60 +16,33 @@
* along with this program. If not, see <https://www.gnu.org/licenses/>. * along with this program. If not, see <https://www.gnu.org/licenses/>.
*/ */
package com.sadellie.unitto.core.ui.common package com.sadellie.unitto.core.ui.common.datetimepicker
import android.text.format.DateFormat
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.IntrinsicSize
import androidx.compose.foundation.layout.PaddingValues import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.heightIn import androidx.compose.foundation.layout.heightIn
import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.requiredWidth import androidx.compose.foundation.layout.requiredWidth
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.layout.width
import androidx.compose.foundation.layout.wrapContentHeight import androidx.compose.foundation.layout.wrapContentHeight
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Keyboard
import androidx.compose.material.icons.outlined.Schedule
import androidx.compose.material3.AlertDialog import androidx.compose.material3.AlertDialog
import androidx.compose.material3.DatePicker import androidx.compose.material3.DatePicker
import androidx.compose.material3.DatePickerDefaults import androidx.compose.material3.DatePickerDefaults
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Surface import androidx.compose.material3.Surface
import androidx.compose.material3.Text import androidx.compose.material3.Text
import androidx.compose.material3.TextButton import androidx.compose.material3.TextButton
import androidx.compose.material3.TimeInput
import androidx.compose.material3.TimePicker
import androidx.compose.material3.rememberDatePickerState import androidx.compose.material3.rememberDatePickerState
import androidx.compose.material3.rememberTimePickerState
import androidx.compose.runtime.Composable 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.Alignment import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.layout.Layout import androidx.compose.ui.layout.Layout
import androidx.compose.ui.layout.Placeable import androidx.compose.ui.layout.Placeable
import androidx.compose.ui.platform.LocalConfiguration
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import androidx.compose.ui.semantics.isTraversalGroup
import androidx.compose.ui.semantics.semantics
import androidx.compose.ui.unit.Dp import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.DialogProperties import androidx.compose.ui.window.DialogProperties
import androidx.compose.ui.zIndex
import com.sadellie.unitto.core.base.R import com.sadellie.unitto.core.base.R
import java.time.Instant import java.time.Instant
import java.time.LocalDateTime import java.time.LocalDateTime
@ -77,106 +50,6 @@ import java.time.ZoneId
import java.time.ZonedDateTime import java.time.ZonedDateTime
import kotlin.math.max import kotlin.math.max
// https://cs.android.com/androidx/platform/tools/dokka-devsite-plugin/+/master:testData/compose/samples/material3/samples/TimePickerSamples.kt
@Composable
fun TimePickerDialog(
hour: Int,
minute: Int,
onCancel: () -> Unit,
onConfirm: (hour: Int, minute: Int) -> Unit,
confirmLabel: String = stringResource(R.string.ok_label),
) {
val pickerState = rememberTimePickerState(
initialHour = hour,
initialMinute = minute,
is24Hour = DateFormat.is24HourFormat(LocalContext.current)
)
val configuration = LocalConfiguration.current
var showingPicker by rememberSaveable { mutableStateOf(true) }
AlertDialog(
onDismissRequest = onCancel,
properties = DialogProperties(
usePlatformDefaultWidth = false
),
) {
Surface(
shape = MaterialTheme.shapes.extraLarge,
tonalElevation = 6.dp,
modifier = Modifier
.width(IntrinsicSize.Min)
.height(IntrinsicSize.Min)
.background(
shape = MaterialTheme.shapes.extraLarge,
color = MaterialTheme.colorScheme.surface
),
) {
if (configuration.screenHeightDp > 400) {
// Make this take the entire viewport. This will guarantee that Screen readers
// focus the toggle first.
Box(
Modifier
.fillMaxSize()
.semantics {
isTraversalGroup = true
}
) {
IconButton(
modifier = Modifier
// This is a workaround so that the Icon comes up first
// in the talkback traversal order. So that users of a11y
// services can use the text input. When talkback traversal
// order is customizable we can remove this.
.size(64.dp, 72.dp)
.align(Alignment.BottomStart)
.zIndex(5f),
onClick = { showingPicker = !showingPicker }) {
val icon = if (showingPicker) {
Icons.Outlined.Keyboard
} else {
Icons.Outlined.Schedule
}
Icon(
imageVector = icon,
contentDescription = stringResource(R.string.select_time_label)
)
}
}
}
Column(
modifier = Modifier.padding(24.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier
.fillMaxWidth()
.padding(bottom = 20.dp),
text = stringResource(R.string.select_time_label),
style = MaterialTheme.typography.labelMedium
)
if (showingPicker && configuration.screenHeightDp > 400) {
TimePicker(state = pickerState)
} else {
TimeInput(state = pickerState)
}
Row(
modifier = Modifier
.height(40.dp)
.fillMaxWidth()
) {
Spacer(modifier = Modifier.weight(1f))
TextButton(
onClick = onCancel
) { Text(stringResource(R.string.cancel_label)) }
TextButton(
onClick = { onConfirm(pickerState.hour, pickerState.minute) }
) { Text(confirmLabel) }
}
}
}
}
}
@Composable @Composable
fun DatePickerDialog( fun DatePickerDialog(
modifier: Modifier = Modifier, modifier: Modifier = Modifier,
@ -204,9 +77,11 @@ fun DatePickerDialog(
Column(verticalArrangement = Arrangement.SpaceBetween) { Column(verticalArrangement = Arrangement.SpaceBetween) {
DatePicker(state = pickerState) DatePicker(state = pickerState)
Box(modifier = Modifier Box(
modifier = Modifier
.align(Alignment.End) .align(Alignment.End)
.padding(_dialogButtonsPadding)) { .padding(_dialogButtonsPadding)
) {
AlertDialogFlowRow( AlertDialogFlowRow(
mainAxisSpacing = _dialogButtonsMainAxisSpacing, mainAxisSpacing = _dialogButtonsMainAxisSpacing,

View File

@ -0,0 +1,162 @@
/*
* Unitto is a unit converter for Android
* Copyright (c) 2023 Elshan Agaev
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package com.sadellie.unitto.core.ui.common.datetimepicker
import android.text.format.DateFormat
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.IntrinsicSize
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.layout.width
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Keyboard
import androidx.compose.material.icons.outlined.Schedule
import androidx.compose.material3.AlertDialog
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.material3.TextButton
import androidx.compose.material3.TimeInput
import androidx.compose.material3.TimePicker
import androidx.compose.material3.rememberTimePickerState
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.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalConfiguration
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.semantics.isTraversalGroup
import androidx.compose.ui.semantics.semantics
import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.DialogProperties
import androidx.compose.ui.zIndex
import com.sadellie.unitto.core.base.R
// https://cs.android.com/androidx/platform/tools/dokka-devsite-plugin/+/master:testData/compose/samples/material3/samples/TimePickerSamples.kt
@Composable
fun TimePickerDialog(
hour: Int,
minute: Int,
onCancel: () -> Unit,
onConfirm: (hour: Int, minute: Int) -> Unit,
confirmLabel: String = stringResource(R.string.ok_label),
) {
val pickerState = rememberTimePickerState(
initialHour = hour,
initialMinute = minute,
is24Hour = DateFormat.is24HourFormat(LocalContext.current)
)
val configuration = LocalConfiguration.current
var showingPicker by rememberSaveable { mutableStateOf(true) }
AlertDialog(
onDismissRequest = onCancel,
properties = DialogProperties(
usePlatformDefaultWidth = false
),
) {
Surface(
shape = MaterialTheme.shapes.extraLarge,
tonalElevation = 6.dp,
modifier = Modifier
.width(IntrinsicSize.Min)
.height(IntrinsicSize.Min)
.background(
shape = MaterialTheme.shapes.extraLarge,
color = MaterialTheme.colorScheme.surface
),
) {
if (configuration.screenHeightDp > 400) {
// Make this take the entire viewport. This will guarantee that Screen readers
// focus the toggle first.
Box(
Modifier
.fillMaxSize()
.semantics {
isTraversalGroup = true
}
) {
IconButton(
modifier = Modifier
// This is a workaround so that the Icon comes up first
// in the talkback traversal order. So that users of a11y
// services can use the text input. When talkback traversal
// order is customizable we can remove this.
.size(64.dp, 72.dp)
.align(Alignment.BottomStart)
.zIndex(5f),
onClick = { showingPicker = !showingPicker }) {
val icon = if (showingPicker) {
Icons.Outlined.Keyboard
} else {
Icons.Outlined.Schedule
}
Icon(
imageVector = icon,
contentDescription = stringResource(R.string.select_time_label)
)
}
}
}
Column(
modifier = Modifier.padding(24.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier
.fillMaxWidth()
.padding(bottom = 20.dp),
text = stringResource(R.string.select_time_label),
style = MaterialTheme.typography.labelMedium
)
if (showingPicker && configuration.screenHeightDp > 400) {
TimePicker(state = pickerState)
} else {
TimeInput(state = pickerState)
}
Row(
modifier = Modifier
.height(40.dp)
.fillMaxWidth()
) {
Spacer(modifier = Modifier.weight(1f))
TextButton(
onClick = onCancel
) { Text(stringResource(R.string.cancel_label)) }
TextButton(
onClick = { onConfirm(pickerState.hour, pickerState.minute) }
) { Text(confirmLabel) }
}
}
}
}
}

View File

@ -21,8 +21,8 @@ package com.sadellie.unitto.feature.datecalculator.components
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import com.sadellie.unitto.core.base.R import com.sadellie.unitto.core.base.R
import com.sadellie.unitto.core.ui.common.DatePickerDialog import com.sadellie.unitto.core.ui.common.datetimepicker.DatePickerDialog
import com.sadellie.unitto.core.ui.common.TimePickerDialog import com.sadellie.unitto.core.ui.common.datetimepicker.TimePickerDialog
import java.time.ZonedDateTime import java.time.ZonedDateTime
@Composable @Composable

View File

@ -75,7 +75,7 @@ import androidx.lifecycle.compose.collectAsStateWithLifecycle
import com.sadellie.unitto.core.base.R import com.sadellie.unitto.core.base.R
import com.sadellie.unitto.core.ui.common.MenuButton import com.sadellie.unitto.core.ui.common.MenuButton
import com.sadellie.unitto.core.ui.common.SettingsButton import com.sadellie.unitto.core.ui.common.SettingsButton
import com.sadellie.unitto.core.ui.common.TimePickerDialog import com.sadellie.unitto.core.ui.common.datetimepicker.TimePickerDialog
import com.sadellie.unitto.core.ui.common.UnittoEmptyScreen import com.sadellie.unitto.core.ui.common.UnittoEmptyScreen
import com.sadellie.unitto.core.ui.common.UnittoScreenWithTopBar import com.sadellie.unitto.core.ui.common.UnittoScreenWithTopBar
import com.sadellie.unitto.data.model.timezone.FavoriteZone import com.sadellie.unitto.data.model.timezone.FavoriteZone