mirror of
https://github.com/Ashinch/ReadYou.git
synced 2025-01-27 09:41:16 +01:00
Improve FeedOptionView
This commit is contained in:
parent
43bbb87280
commit
efba776db3
4
app/proguard-rules.pro
vendored
4
app/proguard-rules.pro
vendored
@ -14,11 +14,11 @@
|
||||
|
||||
# Uncomment this to preserve the line number information for
|
||||
# debugging stack traces.
|
||||
#-keepattributes SourceFile,LineNumberTable
|
||||
-keepattributes SourceFile,LineNumberTable
|
||||
|
||||
# If you keep the line number information, uncomment this to
|
||||
# hide the original source file name.
|
||||
#-renamesourcefileattribute SourceFile
|
||||
-renamesourcefileattribute SourceFile
|
||||
|
||||
-dontobfuscate
|
||||
|
||||
|
@ -2,6 +2,7 @@ package me.ash.reader.ui.component
|
||||
|
||||
import androidx.compose.foundation.background
|
||||
import androidx.compose.foundation.layout.*
|
||||
import androidx.compose.foundation.shape.CircleShape
|
||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||
import androidx.compose.material.ExperimentalMaterialApi
|
||||
import androidx.compose.material.ModalBottomSheetDefaults
|
||||
@ -12,6 +13,7 @@ import androidx.compose.material3.Surface
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.draw.clip
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.compose.ui.zIndex
|
||||
|
||||
@ -55,7 +57,8 @@ fun BottomDrawer(
|
||||
) {
|
||||
Row(
|
||||
modifier = modifier
|
||||
.size(38.dp, 4.dp)
|
||||
.size(30.dp, 4.dp)
|
||||
.clip(CircleShape)
|
||||
.background(MaterialTheme.colorScheme.outline.copy(alpha = 0.2f))
|
||||
.zIndex(1f)
|
||||
) {}
|
||||
|
@ -0,0 +1,83 @@
|
||||
package me.ash.reader.ui.component
|
||||
|
||||
import androidx.compose.foundation.horizontalScroll
|
||||
import androidx.compose.foundation.layout.Column
|
||||
import androidx.compose.foundation.layout.Spacer
|
||||
import androidx.compose.foundation.layout.height
|
||||
import androidx.compose.foundation.layout.padding
|
||||
import androidx.compose.foundation.rememberScrollState
|
||||
import androidx.compose.foundation.text.KeyboardActionScope
|
||||
import androidx.compose.foundation.text.KeyboardActions
|
||||
import androidx.compose.foundation.text.KeyboardOptions
|
||||
import androidx.compose.foundation.text.selection.SelectionContainer
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.focus.FocusManager
|
||||
import androidx.compose.ui.text.input.ImeAction
|
||||
import androidx.compose.ui.unit.dp
|
||||
|
||||
@Composable
|
||||
fun ClipboardTextField(
|
||||
modifier: Modifier = Modifier,
|
||||
readOnly: Boolean = false,
|
||||
value: String = "",
|
||||
onValueChange: (String) -> Unit = {},
|
||||
placeholder: String = "",
|
||||
errorText: String = "",
|
||||
imeAction: ImeAction = ImeAction.Done,
|
||||
focusManager: FocusManager? = null,
|
||||
onConfirm: (String) -> Unit = {},
|
||||
) {
|
||||
Column(modifier = modifier) {
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
TextField(
|
||||
readOnly = readOnly,
|
||||
value = value,
|
||||
onValueChange = onValueChange,
|
||||
placeholder = placeholder,
|
||||
errorMessage = errorText,
|
||||
keyboardActions = KeyboardActions(
|
||||
onDone = if (imeAction == ImeAction.Done)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
onGo = if (imeAction == ImeAction.Go)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
onNext = if (imeAction == ImeAction.Next)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
onPrevious = if (imeAction == ImeAction.Previous)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
onSearch = if (imeAction == ImeAction.Search)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
onSend = if (imeAction == ImeAction.Send)
|
||||
action(focusManager, onConfirm, value) else null,
|
||||
),
|
||||
keyboardOptions = KeyboardOptions(
|
||||
imeAction = imeAction
|
||||
),
|
||||
)
|
||||
if (errorText.isNotEmpty()) {
|
||||
SelectionContainer {
|
||||
Text(
|
||||
modifier = Modifier
|
||||
.padding(start = 16.dp)
|
||||
.horizontalScroll(rememberScrollState()),
|
||||
text = errorText,
|
||||
color = MaterialTheme.colorScheme.error,
|
||||
maxLines = 1,
|
||||
softWrap = false,
|
||||
)
|
||||
}
|
||||
}
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
}
|
||||
}
|
||||
|
||||
private fun action(
|
||||
focusManager: FocusManager?,
|
||||
onConfirm: (String) -> Unit,
|
||||
value: String
|
||||
): KeyboardActionScope.() -> Unit = {
|
||||
focusManager?.clearFocus()
|
||||
onConfirm(value)
|
||||
}
|
@ -44,7 +44,7 @@ fun SelectionChip(
|
||||
val focusManager = LocalFocusManager.current
|
||||
|
||||
FilterChip(
|
||||
modifier = modifier,
|
||||
modifier = modifier.defaultMinSize(minHeight = 36.dp),
|
||||
colors = ChipDefaults.filterChipColors(
|
||||
backgroundColor = MaterialTheme.colorScheme.surfaceVariant,
|
||||
contentColor = MaterialTheme.colorScheme.onSurfaceVariant,
|
||||
|
90
app/src/main/java/me/ash/reader/ui/component/TextField.kt
Normal file
90
app/src/main/java/me/ash/reader/ui/component/TextField.kt
Normal file
@ -0,0 +1,90 @@
|
||||
package me.ash.reader.ui.component
|
||||
|
||||
import androidx.compose.foundation.text.KeyboardActions
|
||||
import androidx.compose.foundation.text.KeyboardOptions
|
||||
import androidx.compose.material.TextFieldDefaults
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.rounded.Close
|
||||
import androidx.compose.material.icons.rounded.ContentPaste
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.IconButton
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.LaunchedEffect
|
||||
import androidx.compose.runtime.remember
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.focus.FocusRequester
|
||||
import androidx.compose.ui.focus.focusRequester
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.platform.LocalClipboardManager
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import kotlinx.coroutines.delay
|
||||
import me.ash.reader.R
|
||||
|
||||
@Composable
|
||||
fun TextField(
|
||||
readOnly: Boolean,
|
||||
value: String,
|
||||
onValueChange: (String) -> Unit,
|
||||
placeholder: String,
|
||||
errorMessage: String,
|
||||
keyboardOptions: KeyboardOptions = KeyboardOptions.Default,
|
||||
keyboardActions: KeyboardActions = KeyboardActions(),
|
||||
) {
|
||||
val clipboardManager = LocalClipboardManager.current
|
||||
val focusRequester = remember { FocusRequester() }
|
||||
|
||||
LaunchedEffect(Unit) {
|
||||
delay(100) // ???
|
||||
focusRequester.requestFocus()
|
||||
}
|
||||
|
||||
androidx.compose.material.TextField(
|
||||
modifier = Modifier.focusRequester(focusRequester),
|
||||
colors = TextFieldDefaults.textFieldColors(
|
||||
backgroundColor = Color.Transparent,
|
||||
cursorColor = MaterialTheme.colorScheme.onSurface,
|
||||
textColor = MaterialTheme.colorScheme.onSurface,
|
||||
focusedIndicatorColor = MaterialTheme.colorScheme.primary,
|
||||
),
|
||||
enabled = !readOnly,
|
||||
value = value,
|
||||
onValueChange = {
|
||||
if (!readOnly) onValueChange(it)
|
||||
},
|
||||
placeholder = {
|
||||
Text(
|
||||
text = placeholder,
|
||||
color = MaterialTheme.colorScheme.outline.copy(alpha = 0.7f)
|
||||
)
|
||||
},
|
||||
isError = errorMessage.isNotEmpty(),
|
||||
singleLine = true,
|
||||
trailingIcon = {
|
||||
if (value.isNotEmpty()) {
|
||||
IconButton(onClick = {
|
||||
if (!readOnly) onValueChange("")
|
||||
}) {
|
||||
Icon(
|
||||
imageVector = Icons.Rounded.Close,
|
||||
contentDescription = stringResource(R.string.clear),
|
||||
tint = MaterialTheme.colorScheme.outline.copy(alpha = 0.5f),
|
||||
)
|
||||
}
|
||||
} else {
|
||||
IconButton(onClick = {
|
||||
onValueChange(clipboardManager.getText()?.text ?: "")
|
||||
}) {
|
||||
Icon(
|
||||
imageVector = Icons.Rounded.ContentPaste,
|
||||
contentDescription = stringResource(R.string.paste),
|
||||
tint = MaterialTheme.colorScheme.primary
|
||||
)
|
||||
}
|
||||
}
|
||||
},
|
||||
keyboardOptions = keyboardOptions,
|
||||
keyboardActions = keyboardActions,
|
||||
)
|
||||
}
|
@ -0,0 +1,92 @@
|
||||
package me.ash.reader.ui.component
|
||||
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.material3.TextButton
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.graphics.vector.ImageVector
|
||||
import androidx.compose.ui.platform.LocalFocusManager
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.input.ImeAction
|
||||
import androidx.compose.ui.text.style.TextOverflow
|
||||
import androidx.compose.ui.window.DialogProperties
|
||||
import com.google.accompanist.pager.ExperimentalPagerApi
|
||||
import me.ash.reader.R
|
||||
|
||||
@OptIn(ExperimentalPagerApi::class)
|
||||
@Composable
|
||||
fun TextFieldDialog(
|
||||
modifier: Modifier = Modifier,
|
||||
properties: DialogProperties = DialogProperties(),
|
||||
visible: Boolean = false,
|
||||
readOnly: Boolean = false,
|
||||
title: String = "",
|
||||
icon: ImageVector? = null,
|
||||
value: String = "",
|
||||
placeholder: String = "",
|
||||
errorText: String = "",
|
||||
dismissText: String = stringResource(R.string.cancel),
|
||||
confirmText: String = stringResource(R.string.confirm),
|
||||
onValueChange: (String) -> Unit = {},
|
||||
onDismissRequest: () -> Unit = {},
|
||||
onConfirm: (String) -> Unit = {},
|
||||
imeAction: ImeAction = ImeAction.Done,
|
||||
) {
|
||||
val focusManager = LocalFocusManager.current
|
||||
|
||||
Dialog(
|
||||
modifier = modifier,
|
||||
visible = visible,
|
||||
onDismissRequest = onDismissRequest,
|
||||
icon = {
|
||||
icon?.let {
|
||||
Icon(
|
||||
imageVector = icon,
|
||||
contentDescription = title,
|
||||
)
|
||||
}
|
||||
},
|
||||
title = {
|
||||
Text(text = title, maxLines = 1, overflow = TextOverflow.Ellipsis)
|
||||
},
|
||||
text = {
|
||||
ClipboardTextField(
|
||||
modifier = modifier,
|
||||
readOnly = readOnly,
|
||||
value = value,
|
||||
onValueChange = onValueChange,
|
||||
placeholder = placeholder,
|
||||
errorText = errorText,
|
||||
imeAction = imeAction,
|
||||
focusManager = focusManager,
|
||||
onConfirm = onConfirm,
|
||||
)
|
||||
},
|
||||
confirmButton = {
|
||||
TextButton(
|
||||
enabled = value.isNotBlank(),
|
||||
onClick = {
|
||||
focusManager.clearFocus()
|
||||
onConfirm(value)
|
||||
}
|
||||
) {
|
||||
Text(
|
||||
text = confirmText,
|
||||
color = if (value.isNotBlank()) {
|
||||
Color.Unspecified
|
||||
} else {
|
||||
MaterialTheme.colorScheme.outline.copy(alpha = 0.7f)
|
||||
}
|
||||
)
|
||||
}
|
||||
},
|
||||
dismissButton = {
|
||||
TextButton(onClick = onDismissRequest) {
|
||||
Text(text = dismissText)
|
||||
}
|
||||
},
|
||||
)
|
||||
}
|
@ -67,6 +67,10 @@ fun HomePage(
|
||||
|
||||
BackHandler(true) {
|
||||
val currentPage = viewState.pagerState.currentPage
|
||||
if (currentPage == 0) {
|
||||
context.findActivity()?.moveTaskToBack(false)
|
||||
return@BackHandler
|
||||
}
|
||||
homeViewModel.dispatch(
|
||||
HomeViewAction.ScrollToPage(
|
||||
scope = scope,
|
||||
|
@ -2,7 +2,7 @@ package me.ash.reader.ui.page.home.drawer.feed
|
||||
|
||||
import android.widget.Toast
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.rounded.DeleteOutline
|
||||
import androidx.compose.material.icons.outlined.DeleteForever
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.material3.TextButton
|
||||
@ -36,7 +36,7 @@ fun DeleteFeedDialog(
|
||||
},
|
||||
icon = {
|
||||
Icon(
|
||||
imageVector = Icons.Rounded.DeleteOutline,
|
||||
imageVector = Icons.Outlined.DeleteForever,
|
||||
contentDescription = stringResource(R.string.subscribe),
|
||||
)
|
||||
},
|
||||
|
@ -1,16 +1,19 @@
|
||||
package me.ash.reader.ui.page.home.drawer.feed
|
||||
|
||||
import androidx.compose.foundation.BorderStroke
|
||||
import androidx.compose.foundation.layout.*
|
||||
import androidx.compose.foundation.layout.Column
|
||||
import androidx.compose.foundation.layout.Spacer
|
||||
import androidx.compose.foundation.layout.fillMaxWidth
|
||||
import androidx.compose.foundation.layout.height
|
||||
import androidx.compose.material.ExperimentalMaterialApi
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.rounded.DeleteOutline
|
||||
import androidx.compose.material.icons.outlined.CreateNewFolder
|
||||
import androidx.compose.material.icons.rounded.RssFeed
|
||||
import androidx.compose.material3.*
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.style.TextAlign
|
||||
import androidx.compose.ui.text.style.TextOverflow
|
||||
@ -18,7 +21,7 @@ import androidx.compose.ui.unit.dp
|
||||
import androidx.hilt.navigation.compose.hiltViewModel
|
||||
import me.ash.reader.R
|
||||
import me.ash.reader.ui.component.BottomDrawer
|
||||
import me.ash.reader.ui.component.Subtitle
|
||||
import me.ash.reader.ui.component.TextFieldDialog
|
||||
import me.ash.reader.ui.ext.collectAsStateValue
|
||||
import me.ash.reader.ui.ext.roundClick
|
||||
import me.ash.reader.ui.page.home.feeds.subscribe.ResultView
|
||||
@ -27,10 +30,10 @@ import me.ash.reader.ui.page.home.feeds.subscribe.ResultView
|
||||
@Composable
|
||||
fun FeedOptionDrawer(
|
||||
modifier: Modifier = Modifier,
|
||||
viewModel: FeedOptionViewModel = hiltViewModel(),
|
||||
feedOptionViewModel: FeedOptionViewModel = hiltViewModel(),
|
||||
content: @Composable () -> Unit = {},
|
||||
) {
|
||||
val viewState = viewModel.viewState.collectAsStateValue()
|
||||
val viewState = feedOptionViewModel.viewState.collectAsStateValue()
|
||||
val feed = viewState.feed
|
||||
|
||||
BottomDrawer(
|
||||
@ -65,53 +68,24 @@ fun FeedOptionDrawer(
|
||||
groups = viewState.groups,
|
||||
selectedAllowNotificationPreset = viewState.feed?.isNotification ?: false,
|
||||
selectedParseFullContentPreset = viewState.feed?.isFullContent ?: false,
|
||||
showUnsubscribe = true,
|
||||
selectedGroupId = viewState.feed?.groupId ?: "",
|
||||
newGroupContent = viewState.newGroupContent,
|
||||
onNewGroupValueChange = {
|
||||
viewModel.dispatch(FeedOptionViewAction.InputNewGroup(it))
|
||||
},
|
||||
newGroupSelected = viewState.newGroupSelected,
|
||||
changeNewGroupSelected = {
|
||||
viewModel.dispatch(FeedOptionViewAction.SelectedNewGroup(it))
|
||||
},
|
||||
allowNotificationPresetOnClick = {
|
||||
viewModel.dispatch(FeedOptionViewAction.ChangeAllowNotificationPreset)
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.ChangeAllowNotificationPreset)
|
||||
},
|
||||
parseFullContentPresetOnClick = {
|
||||
viewModel.dispatch(FeedOptionViewAction.ChangeParseFullContentPreset)
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.ChangeParseFullContentPreset)
|
||||
},
|
||||
unsubscribeOnClick = {
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.ShowDeleteDialog)
|
||||
},
|
||||
onGroupClick = {
|
||||
viewModel.dispatch(FeedOptionViewAction.SelectedGroup(it))
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.SelectedGroup(it))
|
||||
},
|
||||
onKeyboardAction = { },
|
||||
)
|
||||
Spacer(modifier = Modifier.height(20.dp))
|
||||
Subtitle(text = stringResource(R.string.options))
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
Button(
|
||||
colors = ButtonDefaults.buttonColors(
|
||||
containerColor = Color.Transparent,
|
||||
contentColor = MaterialTheme.colorScheme.error,
|
||||
),
|
||||
border = BorderStroke(
|
||||
width = 1.dp,
|
||||
color = MaterialTheme.colorScheme.error,
|
||||
),
|
||||
onClick = {
|
||||
viewModel.dispatch(FeedOptionViewAction.ShowDeleteDialog)
|
||||
onAddNewGroup = {
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.ShowNewGroupDialog)
|
||||
}
|
||||
) {
|
||||
Icon(
|
||||
modifier = Modifier.size(ButtonDefaults.IconSize),
|
||||
imageVector = Icons.Rounded.DeleteOutline,
|
||||
contentDescription = stringResource(R.string.delete),
|
||||
)
|
||||
Spacer(Modifier.size(ButtonDefaults.IconSpacing))
|
||||
Text(
|
||||
text = stringResource(R.string.unsubscribe),
|
||||
style = MaterialTheme.typography.titleSmall,
|
||||
)
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
) {
|
||||
@ -119,4 +93,21 @@ fun FeedOptionDrawer(
|
||||
}
|
||||
|
||||
DeleteFeedDialog(feedName = feed?.name ?: "")
|
||||
|
||||
TextFieldDialog(
|
||||
visible = viewState.newGroupDialogVisible,
|
||||
title = stringResource(R.string.create_new_group),
|
||||
icon = Icons.Outlined.CreateNewFolder,
|
||||
value = viewState.newGroupContent,
|
||||
placeholder = stringResource(R.string.name),
|
||||
onValueChange = {
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.InputNewGroup(it))
|
||||
},
|
||||
onDismissRequest = {
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.HideNewGroupDialog)
|
||||
},
|
||||
onConfirm = {
|
||||
feedOptionViewModel.dispatch(FeedOptionViewAction.AddNewGroup)
|
||||
}
|
||||
)
|
||||
}
|
@ -49,12 +49,14 @@ class FeedOptionViewModel @Inject constructor(
|
||||
is FeedOptionViewAction.Hide -> hide(action.scope)
|
||||
is FeedOptionViewAction.SelectedGroup -> selectedGroup(action.groupId)
|
||||
is FeedOptionViewAction.InputNewGroup -> inputNewGroup(action.content)
|
||||
is FeedOptionViewAction.SelectedNewGroup -> selectedNewGroup(action.selected)
|
||||
is FeedOptionViewAction.ChangeAllowNotificationPreset -> changeAllowNotificationPreset()
|
||||
is FeedOptionViewAction.ChangeParseFullContentPreset -> changeParseFullContentPreset()
|
||||
is FeedOptionViewAction.ShowDeleteDialog -> showDeleteDialog()
|
||||
is FeedOptionViewAction.HideDeleteDialog -> hideDeleteDialog()
|
||||
is FeedOptionViewAction.Delete -> delete(action.callback)
|
||||
is FeedOptionViewAction.AddNewGroup -> addNewGroup()
|
||||
is FeedOptionViewAction.ShowNewGroupDialog -> changeNewGroupDialogVisible(true)
|
||||
is FeedOptionViewAction.HideNewGroupDialog -> changeNewGroupDialogVisible(false)
|
||||
}
|
||||
}
|
||||
|
||||
@ -81,6 +83,15 @@ class FeedOptionViewModel @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun changeNewGroupDialogVisible(visible: Boolean) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
newGroupDialogVisible = visible,
|
||||
newGroupContent = "",
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private fun inputNewGroup(content: String) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
@ -89,6 +100,15 @@ class FeedOptionViewModel @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun addNewGroup() {
|
||||
if (_viewState.value.newGroupContent.isNotBlank()) {
|
||||
viewModelScope.launch {
|
||||
selectedGroup(rssRepository.get().addGroup(_viewState.value.newGroupContent))
|
||||
changeNewGroupDialogVisible(false)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private fun selectedGroup(groupId: String) {
|
||||
viewModelScope.launch(Dispatchers.IO) {
|
||||
_viewState.value.feed?.let {
|
||||
@ -102,14 +122,6 @@ class FeedOptionViewModel @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun selectedNewGroup(selected: Boolean) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
newGroupSelected = selected,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private fun changeParseFullContentPreset() {
|
||||
viewModelScope.launch(Dispatchers.IO) {
|
||||
_viewState.value.feed?.let {
|
||||
@ -170,7 +182,7 @@ data class FeedOptionViewState(
|
||||
val feed: Feed? = null,
|
||||
val selectedGroupId: String = "",
|
||||
val newGroupContent: String = "",
|
||||
val newGroupSelected: Boolean = false,
|
||||
val newGroupDialogVisible: Boolean = false,
|
||||
val groups: List<Group> = emptyList(),
|
||||
val deleteDialogVisible: Boolean = false,
|
||||
)
|
||||
@ -196,14 +208,14 @@ sealed class FeedOptionViewAction {
|
||||
val content: String
|
||||
) : FeedOptionViewAction()
|
||||
|
||||
data class SelectedNewGroup(
|
||||
val selected: Boolean
|
||||
) : FeedOptionViewAction()
|
||||
|
||||
data class Delete(
|
||||
val callback: () -> Unit = {}
|
||||
) : FeedOptionViewAction()
|
||||
|
||||
object ShowDeleteDialog : FeedOptionViewAction()
|
||||
object HideDeleteDialog : FeedOptionViewAction()
|
||||
|
||||
object ShowNewGroupDialog : FeedOptionViewAction()
|
||||
object HideNewGroupDialog : FeedOptionViewAction()
|
||||
object AddNewGroup : FeedOptionViewAction()
|
||||
}
|
||||
|
@ -10,9 +10,9 @@ import androidx.compose.foundation.lazy.LazyColumn
|
||||
import androidx.compose.foundation.lazy.itemsIndexed
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.outlined.KeyboardArrowRight
|
||||
import androidx.compose.material.icons.outlined.Settings
|
||||
import androidx.compose.material.icons.rounded.Add
|
||||
import androidx.compose.material.icons.rounded.Refresh
|
||||
import androidx.compose.material.icons.rounded.Tune
|
||||
import androidx.compose.material3.*
|
||||
import androidx.compose.runtime.*
|
||||
import androidx.compose.ui.Modifier
|
||||
@ -106,7 +106,8 @@ fun FeedsPage(
|
||||
navigationIcon = {
|
||||
FeedbackIconButton(
|
||||
isHaptic = false,
|
||||
imageVector = Icons.Rounded.Tune,
|
||||
modifier = Modifier.size(20.dp),
|
||||
imageVector = Icons.Outlined.Settings,
|
||||
contentDescription = stringResource(R.string.settings),
|
||||
tint = MaterialTheme.colorScheme.onSurface,
|
||||
) {
|
||||
|
@ -3,18 +3,27 @@ package me.ash.reader.ui.page.home.feeds.subscribe
|
||||
import android.content.Intent
|
||||
import android.net.Uri
|
||||
import androidx.compose.animation.animateContentSize
|
||||
import androidx.compose.foundation.background
|
||||
import androidx.compose.foundation.clickable
|
||||
import androidx.compose.foundation.layout.*
|
||||
import androidx.compose.foundation.lazy.LazyRow
|
||||
import androidx.compose.foundation.lazy.items
|
||||
import androidx.compose.foundation.rememberScrollState
|
||||
import androidx.compose.foundation.shape.CircleShape
|
||||
import androidx.compose.foundation.text.selection.SelectionContainer
|
||||
import androidx.compose.foundation.verticalScroll
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.outlined.Add
|
||||
import androidx.compose.material.icons.outlined.Article
|
||||
import androidx.compose.material.icons.outlined.Notifications
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.LaunchedEffect
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.draw.clip
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.style.TextOverflow
|
||||
@ -24,7 +33,6 @@ import com.google.accompanist.flowlayout.MainAxisAlignment
|
||||
import me.ash.reader.R
|
||||
import me.ash.reader.data.entity.Group
|
||||
import me.ash.reader.ui.component.SelectionChip
|
||||
import me.ash.reader.ui.component.SelectionEditorChip
|
||||
import me.ash.reader.ui.component.Subtitle
|
||||
import me.ash.reader.ui.ext.roundClick
|
||||
|
||||
@ -35,41 +43,39 @@ fun ResultView(
|
||||
groups: List<Group> = emptyList(),
|
||||
selectedAllowNotificationPreset: Boolean = false,
|
||||
selectedParseFullContentPreset: Boolean = false,
|
||||
showUnsubscribe: Boolean = false,
|
||||
selectedGroupId: String = "",
|
||||
newGroupContent: String = "",
|
||||
newGroupSelected: Boolean,
|
||||
onNewGroupValueChange: (String) -> Unit = {},
|
||||
changeNewGroupSelected: (Boolean) -> Unit = {},
|
||||
allowNotificationPresetOnClick: () -> Unit = {},
|
||||
parseFullContentPresetOnClick: () -> Unit = {},
|
||||
unsubscribeOnClick: () -> Unit = {},
|
||||
onGroupClick: (groupId: String) -> Unit = {},
|
||||
onKeyboardAction: () -> Unit = {},
|
||||
onAddNewGroup: () -> Unit = {},
|
||||
) {
|
||||
LaunchedEffect(Unit) {
|
||||
if (groups.isNotEmpty()) onGroupClick(groups.first().id)
|
||||
}
|
||||
|
||||
Column(
|
||||
modifier = modifier.verticalScroll(rememberScrollState())
|
||||
) {
|
||||
Link(
|
||||
text = link
|
||||
)
|
||||
Link(text = link)
|
||||
Spacer(modifier = Modifier.height(26.dp))
|
||||
|
||||
Preset(
|
||||
selectedAllowNotificationPreset = selectedAllowNotificationPreset,
|
||||
selectedParseFullContentPreset = selectedParseFullContentPreset,
|
||||
showUnsubscribe = showUnsubscribe,
|
||||
allowNotificationPresetOnClick = allowNotificationPresetOnClick,
|
||||
parseFullContentPresetOnClick = parseFullContentPresetOnClick,
|
||||
unsubscribeOnClick = unsubscribeOnClick,
|
||||
)
|
||||
Spacer(modifier = Modifier.height(26.dp))
|
||||
|
||||
AddToGroup(
|
||||
groups = groups,
|
||||
selectedGroupId = selectedGroupId,
|
||||
newGroupContent = newGroupContent,
|
||||
newGroupSelected = newGroupSelected,
|
||||
onNewGroupValueChange = onNewGroupValueChange,
|
||||
changeNewGroupSelected = changeNewGroupSelected,
|
||||
onGroupClick = onGroupClick,
|
||||
onKeyboardAction = onKeyboardAction,
|
||||
onAddNewGroup = onAddNewGroup,
|
||||
)
|
||||
Spacer(modifier = Modifier.height(6.dp))
|
||||
}
|
||||
@ -105,8 +111,10 @@ private fun Link(
|
||||
private fun Preset(
|
||||
selectedAllowNotificationPreset: Boolean = false,
|
||||
selectedParseFullContentPreset: Boolean = false,
|
||||
showUnsubscribe: Boolean = false,
|
||||
allowNotificationPresetOnClick: () -> Unit = {},
|
||||
parseFullContentPresetOnClick: () -> Unit = {},
|
||||
unsubscribeOnClick: () -> Unit = {},
|
||||
) {
|
||||
Subtitle(text = stringResource(R.string.preset))
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
@ -147,6 +155,15 @@ private fun Preset(
|
||||
) {
|
||||
parseFullContentPresetOnClick()
|
||||
}
|
||||
if (showUnsubscribe) {
|
||||
SelectionChip(
|
||||
modifier = Modifier.animateContentSize(),
|
||||
content = stringResource(R.string.unsubscribe),
|
||||
selected = false,
|
||||
) {
|
||||
unsubscribeOnClick()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -154,39 +171,61 @@ private fun Preset(
|
||||
private fun AddToGroup(
|
||||
groups: List<Group>,
|
||||
selectedGroupId: String,
|
||||
newGroupContent: String,
|
||||
newGroupSelected: Boolean,
|
||||
onNewGroupValueChange: (String) -> Unit = {},
|
||||
changeNewGroupSelected: (Boolean) -> Unit = {},
|
||||
onGroupClick: (groupId: String) -> Unit = {},
|
||||
onKeyboardAction: () -> Unit = {},
|
||||
onAddNewGroup: () -> Unit = {},
|
||||
) {
|
||||
Subtitle(text = stringResource(R.string.add_to_group))
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
FlowRow(
|
||||
mainAxisAlignment = MainAxisAlignment.Start,
|
||||
crossAxisSpacing = 10.dp,
|
||||
mainAxisSpacing = 10.dp,
|
||||
) {
|
||||
groups.forEach {
|
||||
SelectionChip(
|
||||
modifier = Modifier.animateContentSize(),
|
||||
content = it.name,
|
||||
selected = !newGroupSelected && it.id == selectedGroupId,
|
||||
) {
|
||||
changeNewGroupSelected(false)
|
||||
onGroupClick(it.id)
|
||||
}
|
||||
}
|
||||
|
||||
SelectionEditorChip(
|
||||
modifier = Modifier.animateContentSize(),
|
||||
content = newGroupContent,
|
||||
onValueChange = onNewGroupValueChange,
|
||||
selected = newGroupSelected,
|
||||
onKeyboardAction = onKeyboardAction,
|
||||
if (groups.size > 6) {
|
||||
LazyRow {
|
||||
items(groups) {
|
||||
SelectionChip(
|
||||
modifier = Modifier.animateContentSize(),
|
||||
content = it.name,
|
||||
selected = it.id == selectedGroupId,
|
||||
) {
|
||||
onGroupClick(it.id)
|
||||
}
|
||||
Spacer(modifier = Modifier.width(10.dp))
|
||||
}
|
||||
item { NewGroupButton(onAddNewGroup) }
|
||||
}
|
||||
} else {
|
||||
FlowRow(
|
||||
mainAxisAlignment = MainAxisAlignment.Start,
|
||||
crossAxisSpacing = 10.dp,
|
||||
mainAxisSpacing = 10.dp,
|
||||
) {
|
||||
changeNewGroupSelected(true)
|
||||
groups.forEach {
|
||||
SelectionChip(
|
||||
modifier = Modifier.animateContentSize(),
|
||||
content = it.name,
|
||||
selected = it.id == selectedGroupId,
|
||||
) {
|
||||
onGroupClick(it.id)
|
||||
}
|
||||
}
|
||||
NewGroupButton(onAddNewGroup)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun NewGroupButton(onAddNewGroup: () -> Unit) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(36.dp)
|
||||
.clip(CircleShape)
|
||||
.background(MaterialTheme.colorScheme.surfaceVariant)
|
||||
.clickable { onAddNewGroup() },
|
||||
contentAlignment = Alignment.Center,
|
||||
) {
|
||||
Icon(
|
||||
modifier = Modifier.size(20.dp),
|
||||
imageVector = Icons.Outlined.Add,
|
||||
contentDescription = stringResource(R.string.create_new_group),
|
||||
tint = MaterialTheme.colorScheme.onSurfaceVariant.copy(alpha = 0.7f),
|
||||
)
|
||||
}
|
||||
}
|
@ -1,125 +0,0 @@
|
||||
package me.ash.reader.ui.page.home.feeds.subscribe
|
||||
|
||||
import androidx.compose.foundation.horizontalScroll
|
||||
import androidx.compose.foundation.layout.Column
|
||||
import androidx.compose.foundation.layout.Spacer
|
||||
import androidx.compose.foundation.layout.height
|
||||
import androidx.compose.foundation.layout.padding
|
||||
import androidx.compose.foundation.rememberScrollState
|
||||
import androidx.compose.foundation.text.KeyboardActions
|
||||
import androidx.compose.foundation.text.KeyboardOptions
|
||||
import androidx.compose.foundation.text.selection.SelectionContainer
|
||||
import androidx.compose.material.TextField
|
||||
import androidx.compose.material.TextFieldDefaults
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.rounded.Close
|
||||
import androidx.compose.material.icons.rounded.ContentPaste
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.IconButton
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.LaunchedEffect
|
||||
import androidx.compose.runtime.remember
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.focus.FocusRequester
|
||||
import androidx.compose.ui.focus.focusRequester
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.platform.LocalClipboardManager
|
||||
import androidx.compose.ui.platform.LocalFocusManager
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.input.ImeAction
|
||||
import androidx.compose.ui.unit.dp
|
||||
import kotlinx.coroutines.delay
|
||||
import me.ash.reader.R
|
||||
|
||||
@Composable
|
||||
fun SearchView(
|
||||
modifier: Modifier = Modifier,
|
||||
readOnly: Boolean = false,
|
||||
inputLink: String = "",
|
||||
errorMessage: String = "",
|
||||
onLinkValueChange: (String) -> Unit = {},
|
||||
onKeyboardAction: () -> Unit = {},
|
||||
) {
|
||||
val focusManager = LocalFocusManager.current
|
||||
val clipboardManager = LocalClipboardManager.current
|
||||
val focusRequester = remember { FocusRequester() }
|
||||
|
||||
LaunchedEffect(Unit) {
|
||||
delay(100) // ???
|
||||
focusRequester.requestFocus()
|
||||
}
|
||||
|
||||
Column(modifier = modifier) {
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
TextField(
|
||||
modifier = Modifier.focusRequester(focusRequester),
|
||||
colors = TextFieldDefaults.textFieldColors(
|
||||
backgroundColor = Color.Transparent,
|
||||
cursorColor = MaterialTheme.colorScheme.onSurface,
|
||||
textColor = MaterialTheme.colorScheme.onSurface,
|
||||
focusedIndicatorColor = MaterialTheme.colorScheme.primary,
|
||||
),
|
||||
enabled = !readOnly,
|
||||
value = inputLink,
|
||||
onValueChange = {
|
||||
if (!readOnly) onLinkValueChange(it)
|
||||
},
|
||||
placeholder = {
|
||||
Text(
|
||||
text = stringResource(R.string.feed_or_site_url),
|
||||
color = MaterialTheme.colorScheme.outline.copy(alpha = 0.7f)
|
||||
)
|
||||
},
|
||||
isError = errorMessage.isNotEmpty(),
|
||||
singleLine = true,
|
||||
trailingIcon = {
|
||||
if (inputLink.isNotEmpty()) {
|
||||
IconButton(onClick = {
|
||||
if (!readOnly) onLinkValueChange("")
|
||||
}) {
|
||||
Icon(
|
||||
imageVector = Icons.Rounded.Close,
|
||||
contentDescription = stringResource(R.string.clear),
|
||||
tint = MaterialTheme.colorScheme.outline.copy(alpha = 0.5f),
|
||||
)
|
||||
}
|
||||
} else {
|
||||
IconButton(onClick = {
|
||||
onLinkValueChange(clipboardManager.getText()?.text ?: "")
|
||||
}) {
|
||||
Icon(
|
||||
imageVector = Icons.Rounded.ContentPaste,
|
||||
contentDescription = stringResource(R.string.paste),
|
||||
tint = MaterialTheme.colorScheme.primary
|
||||
)
|
||||
}
|
||||
}
|
||||
},
|
||||
keyboardActions = KeyboardActions(
|
||||
onSearch = {
|
||||
focusManager.clearFocus()
|
||||
onKeyboardAction()
|
||||
}
|
||||
),
|
||||
keyboardOptions = KeyboardOptions(
|
||||
imeAction = ImeAction.Search
|
||||
),
|
||||
)
|
||||
if (errorMessage.isNotEmpty()) {
|
||||
SelectionContainer {
|
||||
Text(
|
||||
modifier = Modifier
|
||||
.padding(start = 16.dp)
|
||||
.horizontalScroll(rememberScrollState()),
|
||||
text = errorMessage,
|
||||
color = MaterialTheme.colorScheme.error,
|
||||
maxLines = 1,
|
||||
softWrap = false,
|
||||
)
|
||||
}
|
||||
}
|
||||
Spacer(modifier = Modifier.height(10.dp))
|
||||
}
|
||||
}
|
@ -5,6 +5,7 @@ import androidx.activity.result.contract.ActivityResultContracts
|
||||
import androidx.compose.animation.*
|
||||
import androidx.compose.foundation.layout.padding
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.outlined.CreateNewFolder
|
||||
import androidx.compose.material.icons.rounded.RssFeed
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
@ -18,12 +19,16 @@ import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.compose.ui.platform.LocalFocusManager
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.input.ImeAction
|
||||
import androidx.compose.ui.text.style.TextOverflow
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.compose.ui.window.DialogProperties
|
||||
import androidx.hilt.navigation.compose.hiltViewModel
|
||||
import me.ash.reader.R
|
||||
import me.ash.reader.ui.component.ClipboardTextField
|
||||
import me.ash.reader.ui.component.Dialog
|
||||
import me.ash.reader.ui.ext.*
|
||||
import me.ash.reader.ui.component.TextFieldDialog
|
||||
import me.ash.reader.ui.ext.collectAsStateValue
|
||||
|
||||
@OptIn(
|
||||
androidx.compose.ui.ExperimentalComposeUiApi::class,
|
||||
@ -45,15 +50,9 @@ fun SubscribeDialog(
|
||||
}
|
||||
}
|
||||
}
|
||||
val readYouString = stringResource(R.string.read_you)
|
||||
val defaultString = stringResource(R.string.defaults)
|
||||
|
||||
LaunchedEffect(viewState.visible) {
|
||||
if (viewState.visible) {
|
||||
val defaultGroupId = context.dataStore
|
||||
.get(DataStoreKeys.CurrentAccountId)!!
|
||||
.spacerDollar(readYouString + defaultString)
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.SelectedGroup(defaultGroupId))
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.Init)
|
||||
} else {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.Reset)
|
||||
@ -77,11 +76,13 @@ fun SubscribeDialog(
|
||||
},
|
||||
title = {
|
||||
Text(
|
||||
if (viewState.isSearchPage) {
|
||||
text = if (viewState.isSearchPage) {
|
||||
viewState.title
|
||||
} else {
|
||||
viewState.feed?.name ?: stringResource(R.string.unknown)
|
||||
}
|
||||
},
|
||||
maxLines = 1,
|
||||
overflow = TextOverflow.Ellipsis,
|
||||
)
|
||||
},
|
||||
text = {
|
||||
@ -93,14 +94,17 @@ fun SubscribeDialog(
|
||||
}
|
||||
) { targetExpanded ->
|
||||
if (targetExpanded) {
|
||||
SearchView(
|
||||
ClipboardTextField(
|
||||
readOnly = viewState.lockLinkInput,
|
||||
inputLink = viewState.linkContent,
|
||||
errorMessage = viewState.errorMessage,
|
||||
onLinkValueChange = {
|
||||
value = viewState.linkContent,
|
||||
onValueChange = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.InputLink(it))
|
||||
},
|
||||
onKeyboardAction = {
|
||||
placeholder = stringResource(R.string.feed_or_site_url),
|
||||
errorText = viewState.errorMessage,
|
||||
imeAction = ImeAction.Search,
|
||||
focusManager = focusManager,
|
||||
onConfirm = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.Search)
|
||||
},
|
||||
)
|
||||
@ -111,14 +115,6 @@ fun SubscribeDialog(
|
||||
selectedAllowNotificationPreset = viewState.allowNotificationPreset,
|
||||
selectedParseFullContentPreset = viewState.parseFullContentPreset,
|
||||
selectedGroupId = viewState.selectedGroupId,
|
||||
newGroupContent = viewState.newGroupContent,
|
||||
onNewGroupValueChange = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.InputNewGroup(it))
|
||||
},
|
||||
newGroupSelected = viewState.newGroupSelected,
|
||||
changeNewGroupSelected = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.SelectedNewGroup(it))
|
||||
},
|
||||
allowNotificationPresetOnClick = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.ChangeAllowNotificationPreset)
|
||||
},
|
||||
@ -128,8 +124,8 @@ fun SubscribeDialog(
|
||||
onGroupClick = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.SelectedGroup(it))
|
||||
},
|
||||
onKeyboardAction = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.Subscribe)
|
||||
onAddNewGroup = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.ShowNewGroupDialog)
|
||||
},
|
||||
)
|
||||
}
|
||||
@ -138,7 +134,7 @@ fun SubscribeDialog(
|
||||
confirmButton = {
|
||||
if (viewState.isSearchPage) {
|
||||
TextButton(
|
||||
enabled = viewState.linkContent.isNotEmpty()
|
||||
enabled = viewState.linkContent.isNotBlank()
|
||||
&& viewState.title != stringResource(R.string.searching),
|
||||
onClick = {
|
||||
focusManager.clearFocus()
|
||||
@ -147,7 +143,7 @@ fun SubscribeDialog(
|
||||
) {
|
||||
Text(
|
||||
text = stringResource(R.string.search),
|
||||
color = if (viewState.linkContent.isNotEmpty()) {
|
||||
color = if (viewState.linkContent.isNotBlank()) {
|
||||
Color.Unspecified
|
||||
} else {
|
||||
MaterialTheme.colorScheme.outline.copy(alpha = 0.7f)
|
||||
@ -188,4 +184,21 @@ fun SubscribeDialog(
|
||||
}
|
||||
},
|
||||
)
|
||||
|
||||
TextFieldDialog(
|
||||
visible = viewState.newGroupDialogVisible,
|
||||
title = stringResource(R.string.create_new_group),
|
||||
icon = Icons.Outlined.CreateNewFolder,
|
||||
value = viewState.newGroupContent,
|
||||
placeholder = stringResource(R.string.name),
|
||||
onValueChange = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.InputNewGroup(it))
|
||||
},
|
||||
onDismissRequest = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.HideNewGroupDialog)
|
||||
},
|
||||
onConfirm = {
|
||||
subscribeViewModel.dispatch(SubscribeViewAction.AddNewGroup)
|
||||
}
|
||||
)
|
||||
}
|
@ -40,6 +40,8 @@ class SubscribeViewModel @Inject constructor(
|
||||
is SubscribeViewAction.Reset -> reset()
|
||||
is SubscribeViewAction.Show -> changeVisible(true)
|
||||
is SubscribeViewAction.Hide -> changeVisible(false)
|
||||
is SubscribeViewAction.ShowNewGroupDialog -> changeNewGroupDialogVisible(true)
|
||||
is SubscribeViewAction.HideNewGroupDialog -> changeNewGroupDialogVisible(false)
|
||||
is SubscribeViewAction.SwitchPage -> switchPage(action.isSearchPage)
|
||||
is SubscribeViewAction.ImportFromInputStream -> importFromInputStream(action.inputStream)
|
||||
is SubscribeViewAction.InputLink -> inputLink(action.content)
|
||||
@ -50,7 +52,7 @@ class SubscribeViewModel @Inject constructor(
|
||||
changeParseFullContentPreset()
|
||||
is SubscribeViewAction.SelectedGroup -> selectedGroup(action.groupId)
|
||||
is SubscribeViewAction.InputNewGroup -> inputNewGroup(action.content)
|
||||
is SubscribeViewAction.SelectedNewGroup -> selectedNewGroup(action.selected)
|
||||
is SubscribeViewAction.AddNewGroup -> addNewGroup()
|
||||
is SubscribeViewAction.Subscribe -> subscribe()
|
||||
}
|
||||
}
|
||||
@ -90,14 +92,7 @@ class SubscribeViewModel @Inject constructor(
|
||||
val articles = _viewState.value.articles
|
||||
viewModelScope.launch(Dispatchers.IO) {
|
||||
val groupId = async {
|
||||
if (
|
||||
_viewState.value.newGroupSelected &&
|
||||
_viewState.value.newGroupContent.isNotBlank()
|
||||
) {
|
||||
rssRepository.get().addGroup(_viewState.value.newGroupContent)
|
||||
} else {
|
||||
_viewState.value.selectedGroupId
|
||||
}
|
||||
_viewState.value.selectedGroupId
|
||||
}
|
||||
rssRepository.get().subscribe(
|
||||
feed.copy(
|
||||
@ -118,11 +113,17 @@ class SubscribeViewModel @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun selectedNewGroup(selected: Boolean) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
newGroupSelected = selected,
|
||||
)
|
||||
private fun addNewGroup() {
|
||||
if (_viewState.value.newGroupContent.isNotBlank()) {
|
||||
viewModelScope.launch {
|
||||
selectedGroup(rssRepository.get().addGroup(_viewState.value.newGroupContent))
|
||||
changeNewGroupDialogVisible(false)
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
newGroupContent = "",
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -224,6 +225,14 @@ class SubscribeViewModel @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun changeNewGroupDialogVisible(visible: Boolean) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
newGroupDialogVisible = visible,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private fun switchPage(isSearchPage: Boolean) {
|
||||
_viewState.update {
|
||||
it.copy(
|
||||
@ -245,8 +254,8 @@ data class SubscribeViewState(
|
||||
val allowNotificationPreset: Boolean = false,
|
||||
val parseFullContentPreset: Boolean = false,
|
||||
val selectedGroupId: String = "",
|
||||
val newGroupDialogVisible: Boolean = false,
|
||||
val newGroupContent: String = "",
|
||||
val newGroupSelected: Boolean = false,
|
||||
val groups: Flow<List<Group>> = emptyFlow(),
|
||||
val isSearchPage: Boolean = true,
|
||||
)
|
||||
@ -258,6 +267,10 @@ sealed class SubscribeViewAction {
|
||||
object Show : SubscribeViewAction()
|
||||
object Hide : SubscribeViewAction()
|
||||
|
||||
object ShowNewGroupDialog : SubscribeViewAction()
|
||||
object HideNewGroupDialog : SubscribeViewAction()
|
||||
object AddNewGroup : SubscribeViewAction()
|
||||
|
||||
data class SwitchPage(
|
||||
val isSearchPage: Boolean
|
||||
) : SubscribeViewAction()
|
||||
@ -270,7 +283,7 @@ sealed class SubscribeViewAction {
|
||||
val content: String
|
||||
) : SubscribeViewAction()
|
||||
|
||||
object Search: SubscribeViewAction()
|
||||
object Search : SubscribeViewAction()
|
||||
|
||||
object ChangeAllowNotificationPreset : SubscribeViewAction()
|
||||
object ChangeParseFullContentPreset : SubscribeViewAction()
|
||||
@ -283,9 +296,5 @@ sealed class SubscribeViewAction {
|
||||
val content: String
|
||||
) : SubscribeViewAction()
|
||||
|
||||
data class SelectedNewGroup(
|
||||
val selected: Boolean
|
||||
) : SubscribeViewAction()
|
||||
|
||||
object Subscribe : SubscribeViewAction()
|
||||
}
|
||||
|
@ -31,13 +31,14 @@
|
||||
<string name="allow_notification">允许通知</string>
|
||||
<string name="parse_full_content">全文解析</string>
|
||||
<string name="add_to_group">添加到组</string>
|
||||
<string name="new_group">新建分组</string>
|
||||
<string name="create_new_group">新建分组</string>
|
||||
<string name="name">名称</string>
|
||||
<string name="open_with">打开 %1$s</string>
|
||||
<string name="options">选项</string>
|
||||
<string name="delete">删除</string>
|
||||
<string name="has_been_deleted">“%1$s” 已被删除</string>
|
||||
<string name="has_been_deleted">\"%1$s\" 已被删除</string>
|
||||
<string name="unsubscribe">取消订阅</string>
|
||||
<string name="unsubscribe_tip">不再订阅 “%1$s”,同时删除其所有已归档的文章。</string>
|
||||
<string name="unsubscribe_tip">不再订阅 \"%1$s\",同时删除其所有已归档的文章。</string>
|
||||
<string name="today">今天</string>
|
||||
<string name="yesterday">昨天</string>
|
||||
<string name="date_at_time">%1$s %2$s</string>
|
||||
|
@ -31,13 +31,14 @@
|
||||
<string name="allow_notification">Allow Notification</string>
|
||||
<string name="parse_full_content">Parse Full Content</string>
|
||||
<string name="add_to_group">Add to Group</string>
|
||||
<string name="new_group">New Group</string>
|
||||
<string name="create_new_group">Create New Group</string>
|
||||
<string name="name">Name</string>
|
||||
<string name="open_with">Open %1$s</string>
|
||||
<string name="options">Options</string>
|
||||
<string name="delete">Delete</string>
|
||||
<string name="has_been_deleted">"%1$s" has been deleted</string>
|
||||
<string name="has_been_deleted">\"%1$s\" has been deleted</string>
|
||||
<string name="unsubscribe">Unsubscribe</string>
|
||||
<string name="unsubscribe_tip">Unsubscribe "%1$s" and delete all its archived articles.</string>
|
||||
<string name="unsubscribe_tip">Unsubscribe \"%1$s\" and delete all its archived articles.</string>
|
||||
<string name="today">Today</string>
|
||||
<string name="yesterday">Yesterday</string>
|
||||
<string name="date_at_time">%1$s At %2$s</string>
|
||||
|
Loading…
x
Reference in New Issue
Block a user