diff --git a/app/build.gradle.kts b/app/build.gradle.kts index cdecf77..8eeb007 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -21,7 +21,7 @@ android { minSdk = 24 targetSdk = 34 // 版本号为x.y.z则versionCode为x*1000000+y*10000+z*100+debug版本号(开发需要时迭代, 两位数) - versionCode = 1_00_00_010 + versionCode = 1_00_00_013 versionName = "1.0.0" testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/components/SettingsEntry.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/components/SettingsEntry.kt index 9db0c6b..aa4334a 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/components/SettingsEntry.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/components/SettingsEntry.kt @@ -96,13 +96,13 @@ fun SettingsSwitchEntry( Box( modifier = Modifier .fillMaxHeight() - .padding(end = 8.dp), + .padding(end = 12.dp), contentAlignment = Alignment.Center ) { Icon( modifier = Modifier.size(24.dp), painter = painterResource(iconRes), - tint = MaterialTheme.colorScheme.onSecondaryContainer, + tint = MaterialTheme.colorScheme.onSurfaceVariant, contentDescription = "Icon" ) } @@ -203,7 +203,7 @@ private fun SettingsSliderEntry( Icon( modifier = Modifier.size(24.dp), painter = painterResource(iconRes), - tint = MaterialTheme.colorScheme.onSecondaryContainer, + tint = MaterialTheme.colorScheme.onSurfaceVariant, contentDescription = "Icon" ) } @@ -277,13 +277,13 @@ fun SettingsMenuEntry( Box( modifier = Modifier .fillMaxHeight() - .padding(end = 8.dp), + .padding(end = 12.dp), contentAlignment = Alignment.Center ) { Icon( modifier = Modifier.size(24.dp), painter = painterResource(iconRes), - tint = MaterialTheme.colorScheme.onSecondaryContainer, + tint = MaterialTheme.colorScheme.onSurfaceVariant, contentDescription = "Icon" ) } @@ -405,13 +405,13 @@ fun SettingsClickableEntry( Box( modifier = Modifier .fillMaxHeight() - .padding(end = 8.dp), + .padding(end = 12.dp), contentAlignment = Alignment.Center ) { Icon( modifier = Modifier.size(24.dp), painter = painterResource(iconRes), - tint = MaterialTheme.colorScheme.onSecondaryContainer, + tint = MaterialTheme.colorScheme.onSurfaceVariant, contentDescription = "Icon" ) } diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfHomeScreen.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfHomeScreen.kt index ca81afd..421fb6c 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfHomeScreen.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfHomeScreen.kt @@ -61,12 +61,10 @@ import androidx.compose.ui.draw.clip import androidx.compose.ui.draw.rotate import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.painter.Painter -import androidx.compose.ui.hapticfeedback.HapticFeedbackType import androidx.compose.ui.input.nestedscroll.nestedScroll import androidx.compose.ui.layout.onGloballyPositioned import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalDensity -import androidx.compose.ui.platform.LocalHapticFeedback import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.font.FontWeight @@ -117,7 +115,6 @@ fun BookshelfHomeScreen( ) { val scope = rememberCoroutineScope() val context = LocalContext.current - val haptic = LocalHapticFeedback.current val workManager = WorkManager.getInstance(context) val enterAlwaysScrollBehavior = TopAppBarDefaults.enterAlwaysScrollBehavior() val animatedBackgroundColor by animateColorAsState( @@ -276,7 +273,6 @@ fun BookshelfHomeScreen( val onLongPress: (Int) -> Unit = { bookId -> onClickEnableSelectMode.invoke() changeBookSelectState(bookId) - haptic.performHapticFeedback(HapticFeedbackType.LongPress) } LazyColumn( @@ -287,7 +283,7 @@ fun BookshelfHomeScreen( verticalArrangement = Arrangement.spacedBy(10.dp), state = lazyListState ) { - if (uiState.selectedBookshelf.updatedBookIds.isNotEmpty()) + if (uiState.selectedBookshelf.updatedBookIds.isNotEmpty() && !uiState.selectMode) item { CollapseGroupTitle( modifier = Modifier.animateItem(), @@ -302,7 +298,6 @@ fun BookshelfHomeScreen( uiState.bookInformationMap[updatedBookId]?.let { BookCardItem( bookInformation = it, - haptic = haptic, selected = uiState.selectedBookIds.contains(it.id), latestChapterTitle = uiState.bookLastChapterTitleMap[updatedBookId], onClick = { @@ -316,7 +311,7 @@ fun BookshelfHomeScreen( } } } - if (uiState.selectedBookshelf.pinnedBookIds.isNotEmpty() && !uiState.selectMode) + if (uiState.selectedBookshelf.pinnedBookIds.isNotEmpty()) item { CollapseGroupTitle( modifier = Modifier.animateItem(), @@ -331,7 +326,6 @@ fun BookshelfHomeScreen( uiState.bookInformationMap[pinnedBookId]?.let { BookCardItem( bookInformation = it, - haptic = haptic, selected = uiState.selectedBookIds.contains(it.id), onClick = { if (!uiState.selectMode) @@ -359,7 +353,6 @@ fun BookshelfHomeScreen( uiState.bookInformationMap[bookId]?.let { BookCardItem( bookInformation = it, - haptic = haptic, selected = uiState.selectedBookIds.contains(it.id), onClick = { if (!uiState.selectMode) diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfUIComponents.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfUIComponents.kt index 85c2952..1639867 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfUIComponents.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/bookshelf/home/BookshelfUIComponents.kt @@ -35,7 +35,6 @@ import androidx.compose.material3.SwipeToDismissBoxValue.EndToStart import androidx.compose.material3.SwipeToDismissBoxValue.Settled import androidx.compose.material3.SwipeToDismissBoxValue.StartToEnd import androidx.compose.material3.Text -import androidx.compose.material3.rememberSwipeToDismissBoxState import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.getValue @@ -43,14 +42,16 @@ import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip +import androidx.compose.ui.draw.scale import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.graphicsLayer -import androidx.compose.ui.hapticfeedback.HapticFeedback import androidx.compose.ui.hapticfeedback.HapticFeedbackType import androidx.compose.ui.platform.LocalDensity +import androidx.compose.ui.platform.LocalHapticFeedback import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.text.style.TextOverflow import androidx.compose.ui.unit.Density import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.sp @@ -58,10 +59,12 @@ import indi.dmzz_yyhyy.lightnovelreader.R import indi.dmzz_yyhyy.lightnovelreader.data.book.BookInformation import indi.dmzz_yyhyy.lightnovelreader.ui.components.Cover import indi.dmzz_yyhyy.lightnovelreader.utils.SwipeAction +import indi.dmzz_yyhyy.lightnovelreader.utils.withHaptic @Composable fun BookCardContent( selected: Boolean, + collected: Boolean, modifier: Modifier = Modifier, bookInformation: BookInformation, latestChapterTitle: String? = null @@ -83,16 +86,33 @@ fun BookCardContent( url = bookInformation.coverUrl, rounded = 8.dp ) - if (latestChapterTitle != null) { + if (latestChapterTitle != null) { // 有可用更新 Badge Box( modifier = Modifier.padding(4.dp) .align(Alignment.TopEnd) ) { Badge( + containerColor = MaterialTheme.colorScheme.error, modifier = Modifier.size(12.dp), ) } } + if (collected) { + Box( + modifier = Modifier.padding(4.dp) + .align(Alignment.TopStart) + .size(20.dp) + .clip(RoundedCornerShape(4.dp)) + .background(MaterialTheme.colorScheme.secondaryContainer) + ) { + Icon( + modifier = Modifier.scale(0.75f, 0.75f), + painter = painterResource(R.drawable.filled_bookmark_24px), + contentDescription = "collected_indicator", + tint = MaterialTheme.colorScheme.onSurfaceVariant, + ) + } + } } androidx.compose.animation.AnimatedVisibility( @@ -134,10 +154,11 @@ fun BookCardContent( val titleLineHeight = 20.sp Text( modifier = Modifier.height( - with(LocalDensity.current) { (titleLineHeight * 2).toDp() } + with(LocalDensity.current) { (titleLineHeight * 2.2f).toDp() } ).wrapContentHeight(Alignment.CenterVertically), text = bookInformation.title, maxLines = 2, + overflow = TextOverflow.Ellipsis, fontWeight = FontWeight.Bold, fontSize = 16.sp, lineHeight = titleLineHeight, @@ -190,23 +211,20 @@ fun BookCardContent( ) } else { Column { - Row { - - Text( - text = "已更新至: ", - fontSize = 14.sp, - lineHeight = 18.sp, - ) - } - Row { - Text( - text = latestChapterTitle, - fontWeight = FontWeight.Bold, - color = MaterialTheme.colorScheme.primary, - fontSize = 14.sp, - lineHeight = 18.sp, - ) - } + Text( + text = "已更新至: ", + fontSize = 14.sp, + lineHeight = 18.sp, + ) + Text( + text = latestChapterTitle, + fontWeight = FontWeight.Bold, + color = MaterialTheme.colorScheme.primary, + fontSize = 14.sp, + lineHeight = 18.sp, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) } } } @@ -255,15 +273,15 @@ fun BookCardItem( modifier: Modifier = Modifier, bookInformation: BookInformation, selected: Boolean = false, + collected: Boolean = false, onClick: () -> Unit, onLongPress: () -> Unit, latestChapterTitle: String? = null, swipeToRightAction: SwipeAction = SwipeAction.None, swipeToLeftAction: SwipeAction = SwipeAction.None, progress: (SwipeAction) -> Unit?, - haptic: HapticFeedback ){ - + val haptic = LocalHapticFeedback.current val dismissState = rememberNoFlingSwipeToDismissBoxState( positionalThreshold = { it * 0.6f }, confirmValueChange = { @@ -313,11 +331,12 @@ fun BookCardItem( .background(backgroundColor) .combinedClickable( onClick = onClick, - onLongClick = onLongPress, + onLongClick = withHaptic { onLongPress() }, ) ) { BookCardContent( selected = selected, + collected = collected, latestChapterTitle = latestChapterTitle, bookInformation = bookInformation ) diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/Exploration.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/Exploration.kt index bcc5f9a..e9a90a0 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/Exploration.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/Exploration.kt @@ -3,22 +3,15 @@ package indi.dmzz_yyhyy.lightnovelreader.ui.home.exploration import androidx.compose.animation.AnimatedVisibility import androidx.compose.animation.fadeIn import androidx.compose.animation.fadeOut -import androidx.compose.foundation.clickable import androidx.compose.foundation.layout.Arrangement -import androidx.compose.foundation.layout.Column -import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.WindowInsets import androidx.compose.foundation.layout.WindowInsetsSides import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.only -import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.safeDrawing import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.material3.ExperimentalMaterial3Api -import androidx.compose.material3.Icon -import androidx.compose.material3.IconButton import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.material3.TopAppBar @@ -31,17 +24,12 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.text.font.FontWeight -import androidx.compose.ui.text.style.TextOverflow -import androidx.compose.ui.unit.dp -import androidx.compose.ui.unit.sp import androidx.hilt.navigation.compose.hiltViewModel import androidx.navigation.compose.NavHost import androidx.navigation.compose.composable import androidx.navigation.compose.rememberNavController import indi.dmzz_yyhyy.lightnovelreader.R -import indi.dmzz_yyhyy.lightnovelreader.data.book.BookInformation import indi.dmzz_yyhyy.lightnovelreader.ui.Screen -import indi.dmzz_yyhyy.lightnovelreader.ui.components.Cover import indi.dmzz_yyhyy.lightnovelreader.ui.components.EmptyPage import indi.dmzz_yyhyy.lightnovelreader.ui.components.NavItem import indi.dmzz_yyhyy.lightnovelreader.ui.home.exploration.expanded.ExpandedPageScreen @@ -143,7 +131,7 @@ fun Exploration( ExplorationSearchScreen( topBar = topBar, requestAddBookToBookshelf = requestAddBookToBookshelf, - onCLickBack = { navController.popBackStack() }, + onClickBack = { navController.popBackStack() }, init = explorationSearchViewModel::init, onChangeSearchType = { explorationSearchViewModel.changeSearchType(it) }, onSearch = { explorationSearchViewModel.search(it) }, @@ -177,78 +165,4 @@ fun Exploration( } } } -} - -@Composable -fun ExplorationBookCard( - modifier: Modifier = Modifier, - bookInformation: BookInformation, - allBookshelfBookIds: List<Int>, - requestAddBookToBookshelf: (Int) -> Unit, - onClickBook: (Int) -> Unit -) { - Row( - modifier = modifier - .height(125.dp) - .clickable { - onClickBook(bookInformation.id) - } - ) { - Cover( - width = 82.dp, - height = 125.dp, - url = bookInformation.coverUrl, - rounded = 8.dp - ) - Column ( - modifier = Modifier - .fillMaxWidth() - .padding(8.dp, 2.dp, 14.dp, 5.dp), - verticalArrangement = Arrangement.spacedBy(4.dp) - ) { - Row( - modifier = Modifier.fillMaxWidth() - ) { - Text( - modifier = Modifier.weight(2f), - text = bookInformation.title, - style = MaterialTheme.typography.labelLarge, - fontWeight = FontWeight.W700, - fontSize = 16.sp, - maxLines = 2 - ) - IconButton( - onClick = { requestAddBookToBookshelf(bookInformation.id) }, - modifier = Modifier.height(40.dp) - ) { - Icon( - painter = - if (!allBookshelfBookIds.contains(bookInformation.id)) - painterResource(R.drawable.outline_bookmark_24px) - else painterResource(R.drawable.filled_bookmark_24px), - contentDescription = "mark", - tint = MaterialTheme.colorScheme.onSurface - ) - } - } - Text( - text = stringResource( - id = R.string.book_info_detailed, - bookInformation.author, - bookInformation.publishingHouse, - bookInformation.lastUpdated.year, - bookInformation.lastUpdated.monthValue, - bookInformation.lastUpdated.dayOfMonth, - bookInformation.wordCount, - if (bookInformation.isComplete) stringResource(R.string.book_completed) - else stringResource(R.string.book_ongoing), - bookInformation.description.trim() - ), - style = MaterialTheme.typography.labelLarge, - fontSize = 13.sp, - lineHeight = 17.sp, - overflow = TextOverflow.Ellipsis - ) - } - } } \ No newline at end of file diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/expanded/ExpandedPageScreen.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/expanded/ExpandedPageScreen.kt index c3019f0..942c57e 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/expanded/ExpandedPageScreen.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/expanded/ExpandedPageScreen.kt @@ -5,6 +5,7 @@ import androidx.compose.animation.fadeIn import androidx.compose.animation.fadeOut import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.PaddingValues import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding @@ -41,7 +42,8 @@ import androidx.lifecycle.compose.LifecycleEventEffect import indi.dmzz_yyhyy.lightnovelreader.R import indi.dmzz_yyhyy.lightnovelreader.ui.components.Component import indi.dmzz_yyhyy.lightnovelreader.ui.components.Loading -import indi.dmzz_yyhyy.lightnovelreader.ui.home.exploration.ExplorationBookCard +import indi.dmzz_yyhyy.lightnovelreader.ui.home.bookshelf.home.BookCardItem +import indi.dmzz_yyhyy.lightnovelreader.utils.withHaptic import kotlinx.coroutines.launch @OptIn(ExperimentalMaterial3Api::class) @@ -94,6 +96,7 @@ fun ExpandedPageScreen( modifier = Modifier .fillMaxSize() .nestedScroll(enterAlwaysScrollBehavior.nestedScrollConnection), + contentPadding = PaddingValues(horizontal = 16.dp, vertical = 3.dp), verticalArrangement = Arrangement.spacedBy(8.dp), ) { item { @@ -108,14 +111,12 @@ fun ExpandedPageScreen( Box(Modifier.height(3.dp)) } itemsIndexed(uiState.bookList) { index, bookInformation -> - ExplorationBookCard( - modifier = Modifier - .padding(start = 19.dp, end = 10.dp) - .animateItem(), + BookCardItem( bookInformation = bookInformation, - requestAddBookToBookshelf = requestAddBookToBookshelf, - allBookshelfBookIds = uiState.allBookshelfBookIds, - onClickBook = onClickBook + onClick = { onClickBook(bookInformation.id) }, + onLongPress = withHaptic {}, + collected = uiState.allBookshelfBookIds.contains(bookInformation.id), + progress = {}, ) LaunchedEffect(uiState.bookList.size) { if (uiState.bookList.size - index == 3) { @@ -137,7 +138,7 @@ fun TopBar( MediumTopAppBar( title = { Text( - text = "${stringResource(id = R.string.nav_exploration)} · $title", + text = stringResource(id = R.string.nav_exploration_child, title), style = MaterialTheme.typography.titleLarge, fontWeight = FontWeight.W600, color = MaterialTheme.colorScheme.onSurface, diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/home/ExplorationHomeScreen.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/home/ExplorationHomeScreen.kt index 380a7e3..021ebc3 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/home/ExplorationHomeScreen.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/home/ExplorationHomeScreen.kt @@ -6,6 +6,7 @@ import androidx.compose.animation.fadeIn import androidx.compose.animation.fadeOut import androidx.compose.animation.togetherWith import androidx.compose.foundation.clickable +import androidx.compose.foundation.gestures.snapping.rememberSnapFlingBehavior import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column @@ -18,6 +19,8 @@ import androidx.compose.foundation.layout.width import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.lazy.LazyRow import androidx.compose.foundation.lazy.items +import androidx.compose.foundation.lazy.rememberLazyListState +import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.HorizontalDivider import androidx.compose.material3.Icon @@ -39,6 +42,7 @@ import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip import androidx.compose.ui.input.nestedscroll.NestedScrollConnection import androidx.compose.ui.input.nestedscroll.nestedScroll import androidx.compose.ui.res.painterResource @@ -185,23 +189,29 @@ fun ExplorationPage( modifier = Modifier.animateItem() ) { Row( - modifier = Modifier.fillMaxWidth().height(48.dp) - .padding(start = 16.dp), + modifier = Modifier + .padding(vertical = 4.dp) + .fillMaxWidth() + .height(46.dp) + .padding(horizontal = 16.dp), verticalAlignment = Alignment.CenterVertically ) { Text( modifier = Modifier.weight(2f), text = explorationBooksRow.title, - style = MaterialTheme.typography.titleMedium, - fontWeight = FontWeight.W700, + fontSize = 17.sp, + fontWeight = FontWeight.Bold, color = MaterialTheme.colorScheme.onSurface ) if (explorationBooksRow.expandable) { - IconButton(onClick = { - explorationBooksRow.expandedPageDataSourceId?.let { - onClickExpand(it) + IconButton( + modifier = Modifier.size(40.dp), + onClick = { + explorationBooksRow.expandedPageDataSourceId?.let { + onClickExpand(it) + } } - }) { + ) { Icon( painter = painterResource(id = R.drawable.arrow_forward_24px), contentDescription = "expand" @@ -209,40 +219,80 @@ fun ExplorationPage( } } } + val lazyRowState = rememberLazyListState() + LazyRow( - modifier = Modifier.fillMaxWidth() - .padding(bottom = 11.dp, start = 8.dp), - horizontalArrangement = Arrangement.spacedBy(8.dp) + modifier = Modifier + .fillMaxWidth() + .padding(bottom = 12.dp), + horizontalArrangement = Arrangement.spacedBy(8.dp), + state = lazyRowState, + flingBehavior = rememberSnapFlingBehavior(lazyRowState) ) { + item { + Box(modifier = Modifier.width(10.dp)) + } + items(explorationBooksRow.bookList) { explorationDisplayBook -> Column( - modifier = Modifier.clickable { - onClickBook(explorationDisplayBook.id) - } + modifier = Modifier + .clip(RoundedCornerShape(10.dp)) + .clickable { + onClickBook(explorationDisplayBook.id) + } ) { - Cover( - width = 88.dp, - height = 125.dp, - url = explorationDisplayBook.coverUrl - ) - Text( - modifier = Modifier.width(88.dp), - text = explorationDisplayBook.title, - style = MaterialTheme.typography.labelLarge, - fontWeight = FontWeight.W500, - fontSize = 12.sp, - lineHeight = 14.sp, - maxLines = 2, - overflow = TextOverflow.Ellipsis - ) + Box( + modifier = Modifier.padding(horizontal = 4.dp) + ) { + Cover( + width = 106.dp, + height = 162.dp, + url = explorationDisplayBook.coverUrl, + rounded = 10.dp + ) + } + Column( + modifier = Modifier + .width(108.dp) + .padding(horizontal = 2.dp) + .padding(top = 4.dp, bottom = 2.dp), + verticalArrangement = Arrangement.spacedBy(2.dp) + ) { + Text( + text = explorationDisplayBook.title, + fontWeight = FontWeight.Bold, + fontSize = 14.sp, + lineHeight = 18.sp, + maxLines = 2, + overflow = TextOverflow.Ellipsis + ) + /* 可选作者(未实现) + Text( + text = explorationDisplayBook.author.toString(), + fontSize = 13.sp, + lineHeight = 18.sp, + color = MaterialTheme.colorScheme.secondary, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + */ + } } } + + item { + Box(modifier = Modifier.width(12.dp)) + } } - Box(Modifier.fillMaxWidth().padding(horizontal = 10.dp, vertical = 0.dp)) { + Box( + Modifier.fillMaxWidth() + .padding(horizontal = 16.dp) + ) { HorizontalDivider() } } } } + } } \ No newline at end of file diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/search/ExplorationSearchScreen.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/search/ExplorationSearchScreen.kt index e7b4e54..19c55d1 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/search/ExplorationSearchScreen.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/exploration/search/ExplorationSearchScreen.kt @@ -17,6 +17,8 @@ import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.lazy.items import androidx.compose.foundation.rememberScrollState import androidx.compose.foundation.verticalScroll +import androidx.compose.material3.Button +import androidx.compose.material3.ButtonDefaults import androidx.compose.material3.DropdownMenu import androidx.compose.material3.DropdownMenuItem import androidx.compose.material3.ExperimentalMaterial3Api @@ -36,6 +38,7 @@ import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Rect +import androidx.compose.ui.graphics.Color import androidx.compose.ui.layout.boundsInParent import androidx.compose.ui.layout.onGloballyPositioned import androidx.compose.ui.res.painterResource @@ -53,14 +56,15 @@ import indi.dmzz_yyhyy.lightnovelreader.R import indi.dmzz_yyhyy.lightnovelreader.ui.components.AnimatedText import indi.dmzz_yyhyy.lightnovelreader.ui.components.EmptyPage import indi.dmzz_yyhyy.lightnovelreader.ui.components.Loading -import indi.dmzz_yyhyy.lightnovelreader.ui.home.exploration.ExplorationBookCard +import indi.dmzz_yyhyy.lightnovelreader.ui.home.bookshelf.home.BookCardItem +import indi.dmzz_yyhyy.lightnovelreader.utils.withHaptic @OptIn(ExperimentalMaterial3Api::class) @Composable fun ExplorationSearchScreen( topBar: (@Composable () -> Unit) -> Unit, requestAddBookToBookshelf: (Int) -> Unit, - onCLickBack: () -> Unit, + onClickBack: () -> Unit, init: () -> Unit, onChangeSearchType: (String) -> Unit, onSearch: (String) -> Unit, @@ -124,7 +128,7 @@ fun ExplorationSearchScreen( color = MaterialTheme.colorScheme.onSurfaceVariant, ) }, leadingIcon = { - IconButton(onClick = onCLickBack) { + IconButton(onClick = onClickBack) { Icon(painter = painterResource(R.drawable.arrow_back_24px), contentDescription = "back") } }, @@ -146,7 +150,7 @@ fun ExplorationSearchScreen( ) }, expanded = searchBarExpanded, - onExpandedChange = { if (!it) onCLickBack.invoke() } + onExpandedChange = { if (!it) onClickBack.invoke() } ) { AnimatedVisibility( visible = uiState.historyList.isEmpty() || uiState.historyList.all { it.isEmpty() }, @@ -164,46 +168,55 @@ fun ExplorationSearchScreen( enter = fadeIn(), exit = fadeOut() ) { - Column(Modifier.verticalScroll(rememberScrollState())) { + Column(Modifier + .padding(vertical = 8.dp) + .verticalScroll(rememberScrollState()) + ) { Row( - modifier = Modifier.fillMaxWidth() + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 16.dp), + verticalAlignment = Alignment.CenterVertically, ) { Text( - modifier = Modifier.padding(16.dp, 8.dp), text = stringResource(id = R.string.search_history), - style = MaterialTheme.typography.displayLarge, fontSize = 16.sp, fontWeight = FontWeight.W700, lineHeight = 16.sp, - letterSpacing = 0.5.sp, - color = MaterialTheme.colorScheme.onSurfaceVariant ) + Box(Modifier.weight(2f)) - Text( - modifier = Modifier - .padding(16.dp, 8.dp) - .clickable(onClick = onClickClearAllHistory), - text = stringResource(id = R.string.search_history_clear), - style = MaterialTheme.typography.displayLarge, - fontSize = 16.sp, - fontWeight = FontWeight.W700, - lineHeight = 16.sp, - letterSpacing = 0.5.sp, - color = MaterialTheme.colorScheme.primary - ) + + Button( + onClick = onClickClearAllHistory, + colors = ButtonDefaults.buttonColors( + containerColor = Color.Transparent, + contentColor = MaterialTheme.colorScheme.primary + ), + elevation = ButtonDefaults.buttonElevation(0.dp) + ) { + Text( + text = stringResource(id = R.string.search_history_clear), + fontSize = 16.sp, + fontWeight = FontWeight.Bold, + lineHeight = 16.sp, + ) + } } + Box(Modifier.height(8.dp)) + uiState.historyList.forEach { history -> if (history.isEmpty()) return@forEach AnimatedContent( targetState = history, - label = "HistoryItemAnime" + label = "HistoryItemAnimation" ) { Row ( modifier = Modifier .fillMaxWidth() .height(46.dp) - .padding(horizontal = 16.dp ) + .padding(horizontal = 16.dp) .clickable { searchKeyword = it searchBarExpanded = false @@ -212,9 +225,9 @@ fun ExplorationSearchScreen( verticalAlignment = Alignment.CenterVertically ) { Text( + modifier = Modifier.padding(start = 8.dp), text = it, - style = MaterialTheme.typography.bodyLarge, - fontWeight = FontWeight.W400, + fontSize = 16.sp, color = MaterialTheme.colorScheme.onSurface ) Box(Modifier.weight(2f)) @@ -276,21 +289,22 @@ fun ExplorationSearchScreen( ) } items(uiState.searchResult) { - ExplorationBookCard( - modifier = Modifier.animateItem(), - allBookshelfBookIds = uiState.allBookshelfBookIds, + BookCardItem( bookInformation = it, - requestAddBookToBookshelf = requestAddBookToBookshelf, - onClickBook = onClickBook + onClick = { onClickBook(it.id) }, + onLongPress = withHaptic {}, + collected = uiState.allBookshelfBookIds.contains(it.id), + progress = {}, ) } - item {AnimatedVisibility( - visible = !uiState.isLoadingComplete, - enter = fadeIn(), - exit = fadeOut() - ) { + item { + AnimatedVisibility( + visible = !uiState.isLoadingComplete, + enter = fadeIn(), + exit = fadeOut() + ) { LinearProgressIndicator( - modifier = Modifier.fillMaxWidth().padding(8.dp), + modifier = Modifier.fillMaxWidth().padding(8.dp) ) } } diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/SettingsScreen.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/SettingsScreen.kt index 3e6df71..c2b45a0 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/SettingsScreen.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/SettingsScreen.kt @@ -74,7 +74,7 @@ fun SettingsScreen( TopBar(pinnedScrollBehavior) } AnimatedVisibility( - visible = viewModel.settingState != null, + visible = viewModel.settingState != null, enter = fadeIn(), exit = fadeOut() ) { @@ -168,7 +168,7 @@ fun SettingsCategory( var expanded by remember { mutableStateOf(true) } Card( modifier = Modifier.fillMaxWidth() - .padding(vertical = 8.dp, horizontal = 14.dp), + .padding(vertical = 8.dp, horizontal = 8.dp), colors = CardDefaults.cardColors( containerColor = MaterialTheme.colorScheme.surfaceContainerLow ), @@ -218,7 +218,7 @@ fun SettingsCategory( } AnimatedVisibility(visible = expanded) { Box( - modifier = Modifier.padding(top = 0.dp, end = 14.dp, start = 14.dp, bottom = 14.dp) + modifier = Modifier.padding(top = 0.dp, end = 12.dp, start = 12.dp, bottom = 12.dp) ) { Column( modifier = Modifier.clip(RoundedCornerShape(16.dp)), diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AboutSettingsList.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AboutSettingsList.kt index 038945e..52a96f4 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AboutSettingsList.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AboutSettingsList.kt @@ -13,13 +13,13 @@ fun AboutSettingsList( settingState: SettingState) { val appInfo: String = buildString { appendLine(BuildConfig.APPLICATION_ID) - append(BuildConfig.VERSION_NAME).append(" (").append(BuildConfig.VERSION_CODE).append(")") + append("${BuildConfig.VERSION_NAME} (${BuildConfig.VERSION_CODE}) - ") + .append(if (BuildConfig.DEBUG) "debug" else "release") } val buildInfo: String = buildString { appendLine(stringResource(R.string.info_build_date)) - appendLine(stringResource(R.string.info_build_host)) - append(if (BuildConfig.DEBUG) "DEV (DEBUG)" else "RELEASE") + append(stringResource(R.string.info_build_host)) } SettingsClickableEntry( title = stringResource(R.string.app_name), @@ -31,15 +31,18 @@ fun AboutSettingsList( } ) SettingsClickableEntry( + iconRes = R.drawable.handyman_24px, title = stringResource(R.string.settings_app_build), description = buildInfo, ) SettingsClickableEntry( + iconRes = R.drawable.archive_24px, title = stringResource(R.string.settings_github_repo), description = stringResource(R.string.settings_github_repo_desc), openUrl = "https://github.com/dmzz-yyhyy/LightNovelReader" ) SettingsClickableEntry( + iconRes = R.drawable.group_24px, title = stringResource(R.string.settings_communication), description = stringResource(R.string.settings_communication_desc), openUrl = "https://qm.qq.com/q/Tp80Hf9Oms" diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AppSettingsList.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AppSettingsList.kt index c5f3307..b582d44 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AppSettingsList.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/AppSettingsList.kt @@ -15,12 +15,14 @@ fun AppSettingsList( checkUpdate: () -> Unit ) { SettingsSwitchEntry( + iconRes = R.drawable.cloud_download_24px, title = stringResource(R.string.settings_auto_check_updates), description = stringResource(R.string.settings_auto_check_updates_desc), checked = settingState.checkUpdate, booleanUserData = settingState.checkUpdateUserData ) SettingsMenuEntry( + iconRes = R.drawable.alt_route_24px, title = stringResource(R.string.settings_update_channel), description = stringResource(R.string.settings_update_channel_desc), options = MenuOptions.UpdateChannelOptions, @@ -28,6 +30,7 @@ fun AppSettingsList( onOptionChange = settingState.updateChannelKeyUserData::asynchronousSet ) SettingsClickableEntry( + iconRes = R.drawable.deployed_code_update_24px, title = stringResource(R.string.settings_get_updates), description = stringResource(R.string.settings_get_updates_desc), onClick = { checkUpdate() } diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DataSettingsList.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DataSettingsList.kt index de86c46..88057a1 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DataSettingsList.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DataSettingsList.kt @@ -20,6 +20,7 @@ import androidx.compose.ui.platform.LocalContext import androidx.work.OneTimeWorkRequest import androidx.work.WorkInfo import androidx.work.WorkManager +import indi.dmzz_yyhyy.lightnovelreader.R import indi.dmzz_yyhyy.lightnovelreader.ui.components.ExportContext import indi.dmzz_yyhyy.lightnovelreader.ui.components.ExportDialog import indi.dmzz_yyhyy.lightnovelreader.ui.components.MutableExportContext @@ -111,16 +112,19 @@ fun DataSettingsList( } } SettingsClickableEntry( + iconRes = R.drawable.output_24px, title = "导出数据", description = "将当前应用内的用户数据导出为.lnr文件", onClick = { displayExportDialog = true } ) SettingsClickableEntry( + iconRes = R.drawable.input_24px, title = "导入数据", description = "从外部.lnr文件内导入数据至软件", onClick = { selectDataFile(importDataLauncher) } ) SettingsClickableEntry( + iconRes = R.drawable.public_24px, title = "切换数据源", description = "切换软件的网络数据提供源,但这会导致你的用户数据被暂存,将在下次切换到此数据源后恢复。但是你的缓存数据会被永久删除。", onClick = { displaySourceChangeDialog = true } diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DisplaySettingsList.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DisplaySettingsList.kt index 2f99e75..4820db7 100644 --- a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DisplaySettingsList.kt +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/ui/home/settings/list/DisplaySettingsList.kt @@ -14,6 +14,7 @@ fun DisplaySettingsList( settingState: SettingState ) { SettingsMenuEntry( + iconRes = R.drawable.dark_mode_24px, title = stringResource(R.string.settings_dark_theme), description = stringResource(R.string.settings_dark_theme_desc), options = MenuOptions.DarkModeOptions, @@ -21,6 +22,7 @@ fun DisplaySettingsList( onOptionChange = settingState.darkModeKeyUserData::asynchronousSet ) SettingsSwitchEntry( + iconRes = R.drawable.format_color_fill_24px, title = stringResource(R.string.settings_dynamic_colors), description = stringResource(R.string.settings_dynamic_colors_desc), checked = settingState.dynamicColorsKey, @@ -28,6 +30,7 @@ fun DisplaySettingsList( disabled = Build.VERSION.SDK_INT < Build.VERSION_CODES.S ) SettingsMenuEntry( + iconRes = R.drawable.translate_24px, title = stringResource(R.string.settings_characters_variant), description = stringResource(R.string.settings_characters_variant_desc), options = MenuOptions.AppLocaleOptions, diff --git a/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/utils/Haptic.kt b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/utils/Haptic.kt new file mode 100644 index 0000000..94a0373 --- /dev/null +++ b/app/src/main/kotlin/indi/dmzz_yyhyy/lightnovelreader/utils/Haptic.kt @@ -0,0 +1,16 @@ +package indi.dmzz_yyhyy.lightnovelreader.utils + +import androidx.compose.runtime.Composable +import androidx.compose.ui.hapticfeedback.HapticFeedbackType +import androidx.compose.ui.platform.LocalHapticFeedback + +@Composable +fun withHaptic(action: (() -> Unit)?): () -> Unit { + val haptic = LocalHapticFeedback.current + return { + action?.let { + haptic.performHapticFeedback(HapticFeedbackType.LongPress) + it() + } + } +} \ No newline at end of file diff --git a/app/src/main/res/drawable/alt_route_24px.xml b/app/src/main/res/drawable/alt_route_24px.xml new file mode 100644 index 0000000..748b552 --- /dev/null +++ b/app/src/main/res/drawable/alt_route_24px.xml @@ -0,0 +1,11 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal" + android:autoMirrored="true"> + <path + android:fillColor="@android:color/white" + android:pathData="M440,880L440,680Q440,624 423,597Q406,570 378,544L435,487Q447,498 458,510.5Q469,523 480,537Q494,518 508.5,503.5Q523,489 538,475Q576,440 607,394Q638,348 640,233L577,296L520,240L680,80L840,240L784,296L720,233Q718,376 676,436.5Q634,497 592,535Q560,564 540,591.5Q520,619 520,680L520,880L440,880ZM248,327Q244,307 242.5,283Q241,259 240,233L176,296L120,240L280,80L440,240L383,296L320,234Q320,255 322,273.5Q324,292 326,308L248,327ZM334,503Q314,482 295.5,454Q277,426 263,385L340,366Q350,393 363,412Q376,431 391,446L334,503Z"/> +</vector> diff --git a/app/src/main/res/drawable/archive_24px.xml b/app/src/main/res/drawable/archive_24px.xml new file mode 100644 index 0000000..a7d0bf2 --- /dev/null +++ b/app/src/main/res/drawable/archive_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M480,720L640,560L584,504L520,568L520,400L440,400L440,568L376,504L320,560L480,720ZM200,320L200,760Q200,760 200,760Q200,760 200,760L760,760Q760,760 760,760Q760,760 760,760L760,320L200,320ZM200,840Q167,840 143.5,816.5Q120,793 120,760L120,261Q120,247 124.5,234Q129,221 138,210L188,149Q199,135 215.5,127.5Q232,120 250,120L710,120Q728,120 744.5,127.5Q761,135 772,149L822,210Q831,221 835.5,234Q840,247 840,261L840,760Q840,793 816.5,816.5Q793,840 760,840L200,840ZM216,240L744,240L710,200Q710,200 710,200Q710,200 710,200L250,200Q250,200 250,200Q250,200 250,200L216,240ZM480,540L480,540L480,540Q480,540 480,540Q480,540 480,540L480,540Q480,540 480,540Q480,540 480,540Z"/> +</vector> diff --git a/app/src/main/res/drawable/dark_mode_24px.xml b/app/src/main/res/drawable/dark_mode_24px.xml new file mode 100644 index 0000000..b578c20 --- /dev/null +++ b/app/src/main/res/drawable/dark_mode_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M480,840Q330,840 225,735Q120,630 120,480Q120,330 225,225Q330,120 480,120Q494,120 507.5,121Q521,122 534,124Q493,153 468.5,199.5Q444,246 444,300Q444,390 507,453Q570,516 660,516Q715,516 761,491.5Q807,467 836,426Q838,439 839,452.5Q840,466 840,480Q840,630 735,735Q630,840 480,840ZM480,760Q568,760 638,711.5Q708,663 740,585Q720,590 700,593Q680,596 660,596Q537,596 450.5,509.5Q364,423 364,300Q364,280 367,260Q370,240 375,220Q297,252 248.5,322Q200,392 200,480Q200,596 282,678Q364,760 480,760ZM470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Q470,490 470,490Z"/> +</vector> diff --git a/app/src/main/res/drawable/deployed_code_update_24px.xml b/app/src/main/res/drawable/deployed_code_update_24px.xml new file mode 100644 index 0000000..9ae209f --- /dev/null +++ b/app/src/main/res/drawable/deployed_code_update_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M720,880L840,760L812,732L740,804L740,640L700,640L700,804L628,732L600,760L720,880ZM480,160L243,297L480,434L717,297L480,160Q480,160 480,160Q480,160 480,160ZM120,639L120,321Q120,299 130.5,281Q141,263 160,252L440,91Q450,86 459.5,83Q469,80 480,80Q491,80 501,83Q511,86 520,91L800,252Q819,263 829.5,281Q840,299 840,321L840,480Q822,480 800,480Q778,480 760,480L760,364L479,526L200,364L200,638Q200,638 200,638Q200,638 200,638L440,777Q440,798 440,823Q440,848 440,869Q440,869 440,869Q440,869 440,869L160,708Q141,697 130.5,679Q120,661 120,639ZM720,960Q637,960 578.5,901.5Q520,843 520,760Q520,677 578.5,618.5Q637,560 720,560Q803,560 861.5,618.5Q920,677 920,760Q920,843 861.5,901.5Q803,960 720,960ZM480,469Q480,469 480,469Q480,469 480,469Q480,469 480,469Q480,469 480,469Q480,469 480,469Q480,469 480,469L480,469Q480,469 480,469Q480,469 480,469L480,469Q480,469 480,469Q480,469 480,469L480,469Q480,469 480,469Q480,469 480,469L480,469Q480,469 480,469Q480,469 480,469Z"/> +</vector> diff --git a/app/src/main/res/drawable/format_color_fill_24px.xml b/app/src/main/res/drawable/format_color_fill_24px.xml new file mode 100644 index 0000000..b224581 --- /dev/null +++ b/app/src/main/res/drawable/format_color_fill_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M247,56L304,0L647,343Q670,366 670,400Q670,434 647,457L457,647Q434,670 400,670Q366,670 343,647L153,457Q130,434 130,400Q130,366 153,343L343,152L247,56ZM400,209L209,400Q209,400 209,400Q209,400 209,400L591,400Q591,400 591,400Q591,400 591,400L400,209ZM760,680Q727,680 703.5,656.5Q680,633 680,600Q680,579 692.5,555Q705,531 720,510Q729,498 739,485Q749,472 760,460Q771,472 781,485Q791,498 800,510Q815,531 827.5,555Q840,579 840,600Q840,633 816.5,656.5Q793,680 760,680ZM80,960L80,800L880,800L880,960L80,960Z"/> +</vector> diff --git a/app/src/main/res/drawable/group_24px.xml b/app/src/main/res/drawable/group_24px.xml new file mode 100644 index 0000000..556499a --- /dev/null +++ b/app/src/main/res/drawable/group_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M40,800L40,688Q40,654 57.5,625.5Q75,597 104,582Q166,551 230,535.5Q294,520 360,520Q426,520 490,535.5Q554,551 616,582Q645,597 662.5,625.5Q680,654 680,688L680,800L40,800ZM760,800L760,680Q760,636 735.5,595.5Q711,555 666,526Q717,532 762,546.5Q807,561 846,582Q882,602 901,626.5Q920,651 920,680L920,800L760,800ZM360,480Q294,480 247,433Q200,386 200,320Q200,254 247,207Q294,160 360,160Q426,160 473,207Q520,254 520,320Q520,386 473,433Q426,480 360,480ZM760,320Q760,386 713,433Q666,480 600,480Q589,480 572,477.5Q555,475 544,472Q571,440 585.5,401Q600,362 600,320Q600,278 585.5,239Q571,200 544,168Q558,163 572,161.5Q586,160 600,160Q666,160 713,207Q760,254 760,320ZM120,720L600,720L600,688Q600,677 594.5,668Q589,659 580,654Q526,627 471,613.5Q416,600 360,600Q304,600 249,613.5Q194,627 140,654Q131,659 125.5,668Q120,677 120,688L120,720ZM360,400Q393,400 416.5,376.5Q440,353 440,320Q440,287 416.5,263.5Q393,240 360,240Q327,240 303.5,263.5Q280,287 280,320Q280,353 303.5,376.5Q327,400 360,400ZM360,720L360,720L360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720Q360,720 360,720ZM360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Q360,320 360,320Z"/> +</vector> diff --git a/app/src/main/res/drawable/handyman_24px.xml b/app/src/main/res/drawable/handyman_24px.xml new file mode 100644 index 0000000..6255c60 --- /dev/null +++ b/app/src/main/res/drawable/handyman_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M754,879Q746,879 739,876.5Q732,874 726,868L522,664Q516,658 513.5,651Q511,644 511,636Q511,628 513.5,621Q516,614 522,608L607,523Q613,517 620,514.5Q627,512 635,512Q643,512 650,514.5Q657,517 663,523L867,727Q873,733 875.5,740Q878,747 878,755Q878,763 875.5,770Q873,777 867,783L782,868Q776,874 769,876.5Q762,879 754,879ZM754,784L783,755L636,608L607,637L754,784ZM205,880Q197,880 189.5,877Q182,874 176,868L92,784Q86,778 83,770.5Q80,763 80,755Q80,747 83,740Q86,733 92,727L304,515L389,515L423,481L258,316L201,316L80,195L193,82L314,203L314,260L479,425L595,309L552,266L608,210L495,210L467,182L609,40L637,68L637,181L693,125L835,267Q852,284 861,305.5Q870,327 870,351Q870,375 861,397Q852,419 835,436L750,351L694,407L652,365L445,572L445,656L233,868Q227,874 220,877Q213,880 205,880ZM205,784L375,614L375,585L346,585L176,755L205,784ZM205,784L176,755L176,755L191,769L205,784ZM754,784L754,784L783,755L783,755L754,784Z"/> +</vector> diff --git a/app/src/main/res/drawable/input_24px.xml b/app/src/main/res/drawable/input_24px.xml new file mode 100644 index 0000000..55c2537 --- /dev/null +++ b/app/src/main/res/drawable/input_24px.xml @@ -0,0 +1,11 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal" + android:autoMirrored="true"> + <path + android:fillColor="@android:color/white" + android:pathData="M160,800Q127,800 103.5,776.5Q80,753 80,720L80,600L160,600L160,720Q160,720 160,720Q160,720 160,720L800,720Q800,720 800,720Q800,720 800,720L800,240Q800,240 800,240Q800,240 800,240L160,240Q160,240 160,240Q160,240 160,240L160,360L80,360L80,240Q80,207 103.5,183.5Q127,160 160,160L800,160Q833,160 856.5,183.5Q880,207 880,240L880,720Q880,753 856.5,776.5Q833,800 800,800L160,800ZM460,660L404,602L487,520L80,520L80,440L487,440L404,358L460,300L640,480L460,660Z"/> +</vector> diff --git a/app/src/main/res/drawable/translate_24px.xml b/app/src/main/res/drawable/translate_24px.xml new file mode 100644 index 0000000..5792d2c --- /dev/null +++ b/app/src/main/res/drawable/translate_24px.xml @@ -0,0 +1,10 @@ +<vector xmlns:android="http://schemas.android.com/apk/res/android" + android:width="24dp" + android:height="24dp" + android:viewportWidth="960" + android:viewportHeight="960" + android:tint="?attr/colorControlNormal"> + <path + android:fillColor="@android:color/white" + android:pathData="M476,880L658,400L742,400L924,880L840,880L797,758L603,758L560,880L476,880ZM160,760L104,704L306,502Q271,467 242.5,422Q214,377 190,320L274,320Q294,359 314,388Q334,417 362,446Q395,413 430.5,353.5Q466,294 484,240L40,240L40,160L320,160L320,80L400,80L400,160L680,160L680,240L564,240Q543,312 501,388Q459,464 418,504L514,602L484,684L362,559L160,760ZM628,688L772,688L700,484L628,688Z"/> +</vector> diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml index 010657a..e625cd6 100644 --- a/app/src/main/res/values-zh-rCN/strings.xml +++ b/app/src/main/res/values-zh-rCN/strings.xml @@ -5,6 +5,7 @@ <string name="nav_bookshelf">书架</string> <string name="nav_bookshelf_select_mode">选择 书架 (%d)</string> <string name="nav_exploration">探索</string> + <string name="nav_exploration_child">探索 · %s</string> <string name="nav_exploration_homepage">首页</string> <string name="nav_exploration_categories">分类</string> <string name="nav_settings">设置</string> diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 6ec23af..f8a30ae 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -6,6 +6,7 @@ <string name="nav_bookshelf">Bookshelf</string> <string name="nav_bookshelf_select_mode">Select Bookshelf (%d)</string> <string name="nav_exploration">Exploration</string> + <string name="nav_exploration_child">%s - Exploration</string> <string name="nav_exploration_homepage">Homepage</string> <string name="nav_exploration_categories">Categories</string> <string name="nav_settings">Settings</string>