Touch up manga grid/list items (#8307)
* Touch up library item touch indicator Now the touch indicator has the same coverage as the selection indicator. Experimental Modifier.Node API is used to draw the selection indicator * Unify library and browse source list item layouts
This commit is contained in:
parent
e36d31bf0f
commit
87391832ba
@ -1,28 +1,22 @@
|
|||||||
package eu.kanade.presentation.browse.components
|
package eu.kanade.presentation.browse.components
|
||||||
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
import androidx.compose.foundation.layout.Column
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.lazy.grid.GridCells
|
import androidx.compose.foundation.lazy.grid.GridCells
|
||||||
import androidx.compose.foundation.lazy.grid.GridItemSpan
|
import androidx.compose.foundation.lazy.grid.GridItemSpan
|
||||||
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
|
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.runtime.State
|
import androidx.compose.runtime.State
|
||||||
import androidx.compose.runtime.getValue
|
import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.draw.drawWithContent
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.paging.LoadState
|
import androidx.paging.LoadState
|
||||||
import androidx.paging.compose.LazyPagingItems
|
import androidx.paging.compose.LazyPagingItems
|
||||||
import eu.kanade.domain.manga.model.Manga
|
import eu.kanade.domain.manga.model.Manga
|
||||||
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
import eu.kanade.presentation.components.Badge
|
import eu.kanade.presentation.components.Badge
|
||||||
import eu.kanade.presentation.components.MangaCover
|
import eu.kanade.presentation.components.CommonMangaItemDefaults
|
||||||
import eu.kanade.presentation.library.components.MangaGridComfortableText
|
import eu.kanade.presentation.components.MangaComfortableGridItem
|
||||||
import eu.kanade.presentation.library.components.MangaGridCover
|
|
||||||
import eu.kanade.presentation.util.plus
|
import eu.kanade.presentation.util.plus
|
||||||
import eu.kanade.tachiyomi.R
|
import eu.kanade.tachiyomi.R
|
||||||
|
|
||||||
@ -37,9 +31,9 @@ fun BrowseSourceComfortableGrid(
|
|||||||
) {
|
) {
|
||||||
LazyVerticalGrid(
|
LazyVerticalGrid(
|
||||||
columns = columns,
|
columns = columns,
|
||||||
contentPadding = PaddingValues(8.dp, 4.dp) + contentPadding,
|
contentPadding = contentPadding + PaddingValues(8.dp),
|
||||||
horizontalArrangement = Arrangement.spacedBy(8.dp),
|
verticalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridVerticalSpacer),
|
||||||
verticalArrangement = Arrangement.spacedBy(8.dp),
|
horizontalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridHorizontalSpacer),
|
||||||
) {
|
) {
|
||||||
if (mangaList.loadState.prepend is LoadState.Loading) {
|
if (mangaList.loadState.prepend is LoadState.Loading) {
|
||||||
item(span = { GridItemSpan(maxLineSpan) }) {
|
item(span = { GridItemSpan(maxLineSpan) }) {
|
||||||
@ -71,36 +65,22 @@ fun BrowseSourceComfortableGridItem(
|
|||||||
onClick: () -> Unit = {},
|
onClick: () -> Unit = {},
|
||||||
onLongClick: () -> Unit = onClick,
|
onLongClick: () -> Unit = onClick,
|
||||||
) {
|
) {
|
||||||
val overlayColor = MaterialTheme.colorScheme.background.copy(alpha = 0.66f)
|
MangaComfortableGridItem(
|
||||||
Column(
|
title = manga.title,
|
||||||
modifier = Modifier
|
coverData = MangaCover(
|
||||||
.combinedClickable(
|
mangaId = manga.id,
|
||||||
onClick = onClick,
|
sourceId = manga.source,
|
||||||
onLongClick = onLongClick,
|
isMangaFavorite = manga.favorite,
|
||||||
|
url = manga.thumbnailUrl,
|
||||||
|
lastModified = manga.coverLastModified,
|
||||||
),
|
),
|
||||||
) {
|
coverAlpha = if (manga.favorite) CommonMangaItemDefaults.BrowseFavoriteCoverAlpha else 1f,
|
||||||
MangaGridCover(
|
coverBadgeStart = {
|
||||||
cover = {
|
|
||||||
MangaCover.Book(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.drawWithContent {
|
|
||||||
drawContent()
|
|
||||||
if (manga.favorite) {
|
|
||||||
drawRect(overlayColor)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
data = manga.thumbnailUrl,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
badgesStart = {
|
|
||||||
if (manga.favorite) {
|
if (manga.favorite) {
|
||||||
Badge(text = stringResource(R.string.in_library))
|
Badge(text = stringResource(R.string.in_library))
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
onLongClick = onLongClick,
|
||||||
|
onClick = onClick,
|
||||||
)
|
)
|
||||||
MangaGridComfortableText(
|
|
||||||
text = manga.title,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,35 +1,22 @@
|
|||||||
package eu.kanade.presentation.browse.components
|
package eu.kanade.presentation.browse.components
|
||||||
|
|
||||||
import androidx.compose.foundation.background
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
import androidx.compose.foundation.layout.Box
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.fillMaxHeight
|
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.lazy.grid.GridCells
|
import androidx.compose.foundation.lazy.grid.GridCells
|
||||||
import androidx.compose.foundation.lazy.grid.GridItemSpan
|
import androidx.compose.foundation.lazy.grid.GridItemSpan
|
||||||
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
|
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.runtime.State
|
import androidx.compose.runtime.State
|
||||||
import androidx.compose.runtime.getValue
|
import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.draw.clip
|
|
||||||
import androidx.compose.ui.draw.drawWithContent
|
|
||||||
import androidx.compose.ui.graphics.Brush
|
|
||||||
import androidx.compose.ui.graphics.Color
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.paging.LoadState
|
import androidx.paging.LoadState
|
||||||
import androidx.paging.compose.LazyPagingItems
|
import androidx.paging.compose.LazyPagingItems
|
||||||
import eu.kanade.domain.manga.model.Manga
|
import eu.kanade.domain.manga.model.Manga
|
||||||
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
import eu.kanade.presentation.components.Badge
|
import eu.kanade.presentation.components.Badge
|
||||||
import eu.kanade.presentation.components.MangaCover
|
import eu.kanade.presentation.components.CommonMangaItemDefaults
|
||||||
import eu.kanade.presentation.library.components.MangaGridCompactText
|
import eu.kanade.presentation.components.MangaCompactGridItem
|
||||||
import eu.kanade.presentation.library.components.MangaGridCover
|
|
||||||
import eu.kanade.presentation.util.plus
|
import eu.kanade.presentation.util.plus
|
||||||
import eu.kanade.tachiyomi.R
|
import eu.kanade.tachiyomi.R
|
||||||
|
|
||||||
@ -44,12 +31,12 @@ fun BrowseSourceCompactGrid(
|
|||||||
) {
|
) {
|
||||||
LazyVerticalGrid(
|
LazyVerticalGrid(
|
||||||
columns = columns,
|
columns = columns,
|
||||||
contentPadding = PaddingValues(8.dp, 4.dp) + contentPadding,
|
contentPadding = contentPadding + PaddingValues(8.dp),
|
||||||
horizontalArrangement = Arrangement.spacedBy(8.dp),
|
verticalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridVerticalSpacer),
|
||||||
verticalArrangement = Arrangement.spacedBy(8.dp),
|
horizontalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridHorizontalSpacer),
|
||||||
) {
|
) {
|
||||||
item(span = { GridItemSpan(maxLineSpan) }) {
|
|
||||||
if (mangaList.loadState.prepend is LoadState.Loading) {
|
if (mangaList.loadState.prepend is LoadState.Loading) {
|
||||||
|
item(span = { GridItemSpan(maxLineSpan) }) {
|
||||||
BrowseSourceLoadingItem()
|
BrowseSourceLoadingItem()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -64,8 +51,8 @@ fun BrowseSourceCompactGrid(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
item(span = { GridItemSpan(maxLineSpan) }) {
|
|
||||||
if (mangaList.loadState.refresh is LoadState.Loading || mangaList.loadState.append is LoadState.Loading) {
|
if (mangaList.loadState.refresh is LoadState.Loading || mangaList.loadState.append is LoadState.Loading) {
|
||||||
|
item(span = { GridItemSpan(maxLineSpan) }) {
|
||||||
BrowseSourceLoadingItem()
|
BrowseSourceLoadingItem()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -73,57 +60,27 @@ fun BrowseSourceCompactGrid(
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
fun BrowseSourceCompactGridItem(
|
private fun BrowseSourceCompactGridItem(
|
||||||
manga: Manga,
|
manga: Manga,
|
||||||
onClick: () -> Unit = {},
|
onClick: () -> Unit = {},
|
||||||
onLongClick: () -> Unit = onClick,
|
onLongClick: () -> Unit = onClick,
|
||||||
) {
|
) {
|
||||||
val overlayColor = MaterialTheme.colorScheme.background.copy(alpha = 0.66f)
|
MangaCompactGridItem(
|
||||||
MangaGridCover(
|
title = manga.title,
|
||||||
modifier = Modifier
|
coverData = MangaCover(
|
||||||
.combinedClickable(
|
mangaId = manga.id,
|
||||||
onClick = onClick,
|
sourceId = manga.source,
|
||||||
onLongClick = onLongClick,
|
isMangaFavorite = manga.favorite,
|
||||||
|
url = manga.thumbnailUrl,
|
||||||
|
lastModified = manga.coverLastModified,
|
||||||
),
|
),
|
||||||
cover = {
|
coverAlpha = if (manga.favorite) CommonMangaItemDefaults.BrowseFavoriteCoverAlpha else 1f,
|
||||||
MangaCover.Book(
|
coverBadgeStart = {
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxHeight()
|
|
||||||
.drawWithContent {
|
|
||||||
drawContent()
|
|
||||||
if (manga.favorite) {
|
|
||||||
drawRect(overlayColor)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
data = eu.kanade.domain.manga.model.MangaCover(
|
|
||||||
manga.id,
|
|
||||||
manga.source,
|
|
||||||
manga.favorite,
|
|
||||||
manga.thumbnailUrl,
|
|
||||||
manga.coverLastModified,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
},
|
|
||||||
badgesStart = {
|
|
||||||
if (manga.favorite) {
|
if (manga.favorite) {
|
||||||
Badge(text = stringResource(R.string.in_library))
|
Badge(text = stringResource(R.string.in_library))
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
content = {
|
onLongClick = onLongClick,
|
||||||
Box(
|
onClick = onClick,
|
||||||
modifier = Modifier
|
|
||||||
.clip(RoundedCornerShape(bottomStart = 4.dp, bottomEnd = 4.dp))
|
|
||||||
.background(
|
|
||||||
Brush.verticalGradient(
|
|
||||||
0f to Color.Transparent,
|
|
||||||
1f to Color(0xAA000000),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
.fillMaxHeight(0.33f)
|
|
||||||
.fillMaxWidth()
|
|
||||||
.align(Alignment.BottomCenter),
|
|
||||||
)
|
|
||||||
MangaGridCompactText(manga.title)
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -1,26 +1,21 @@
|
|||||||
package eu.kanade.presentation.browse.components
|
package eu.kanade.presentation.browse.components
|
||||||
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.fillMaxHeight
|
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.runtime.State
|
import androidx.compose.runtime.State
|
||||||
import androidx.compose.runtime.getValue
|
import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.draw.drawWithContent
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.paging.LoadState
|
import androidx.paging.LoadState
|
||||||
import androidx.paging.compose.LazyPagingItems
|
import androidx.paging.compose.LazyPagingItems
|
||||||
import androidx.paging.compose.items
|
import androidx.paging.compose.items
|
||||||
import eu.kanade.domain.manga.model.Manga
|
import eu.kanade.domain.manga.model.Manga
|
||||||
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
import eu.kanade.presentation.components.Badge
|
import eu.kanade.presentation.components.Badge
|
||||||
|
import eu.kanade.presentation.components.CommonMangaItemDefaults
|
||||||
import eu.kanade.presentation.components.LazyColumn
|
import eu.kanade.presentation.components.LazyColumn
|
||||||
import eu.kanade.presentation.components.MangaCover
|
import eu.kanade.presentation.components.MangaListItem
|
||||||
import eu.kanade.presentation.library.components.MangaListItem
|
|
||||||
import eu.kanade.presentation.library.components.MangaListItemContent
|
|
||||||
import eu.kanade.presentation.util.plus
|
import eu.kanade.presentation.util.plus
|
||||||
import eu.kanade.presentation.util.verticalPadding
|
|
||||||
import eu.kanade.tachiyomi.R
|
import eu.kanade.tachiyomi.R
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
@ -32,7 +27,7 @@ fun BrowseSourceList(
|
|||||||
onMangaLongClick: (Manga) -> Unit,
|
onMangaLongClick: (Manga) -> Unit,
|
||||||
) {
|
) {
|
||||||
LazyColumn(
|
LazyColumn(
|
||||||
contentPadding = contentPadding,
|
contentPadding = contentPadding + PaddingValues(vertical = 8.dp),
|
||||||
) {
|
) {
|
||||||
item {
|
item {
|
||||||
if (mangaList.loadState.prepend is LoadState.Loading) {
|
if (mangaList.loadState.prepend is LoadState.Loading) {
|
||||||
@ -64,31 +59,22 @@ fun BrowseSourceListItem(
|
|||||||
onClick: () -> Unit = {},
|
onClick: () -> Unit = {},
|
||||||
onLongClick: () -> Unit = onClick,
|
onLongClick: () -> Unit = onClick,
|
||||||
) {
|
) {
|
||||||
val overlayColor = MaterialTheme.colorScheme.background.copy(alpha = 0.66f)
|
|
||||||
MangaListItem(
|
MangaListItem(
|
||||||
coverContent = {
|
title = manga.title,
|
||||||
MangaCover.Square(
|
coverData = MangaCover(
|
||||||
modifier = Modifier
|
mangaId = manga.id,
|
||||||
.padding(vertical = verticalPadding)
|
sourceId = manga.source,
|
||||||
.fillMaxHeight()
|
isMangaFavorite = manga.favorite,
|
||||||
.drawWithContent {
|
url = manga.thumbnailUrl,
|
||||||
drawContent()
|
lastModified = manga.coverLastModified,
|
||||||
if (manga.favorite) {
|
),
|
||||||
drawRect(overlayColor)
|
coverAlpha = if (manga.favorite) CommonMangaItemDefaults.BrowseFavoriteCoverAlpha else 1f,
|
||||||
}
|
badge = {
|
||||||
},
|
|
||||||
data = manga.thumbnailUrl,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
badges = {
|
|
||||||
if (manga.favorite) {
|
if (manga.favorite) {
|
||||||
Badge(text = stringResource(R.string.in_library))
|
Badge(text = stringResource(R.string.in_library))
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
content = {
|
onLongClick = onLongClick,
|
||||||
MangaListItemContent(text = manga.title)
|
onClick = onClick,
|
||||||
},
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,317 @@
|
|||||||
|
package eu.kanade.presentation.components
|
||||||
|
|
||||||
|
import androidx.compose.foundation.background
|
||||||
|
import androidx.compose.foundation.combinedClickable
|
||||||
|
import androidx.compose.foundation.layout.Box
|
||||||
|
import androidx.compose.foundation.layout.BoxScope
|
||||||
|
import androidx.compose.foundation.layout.Column
|
||||||
|
import androidx.compose.foundation.layout.Row
|
||||||
|
import androidx.compose.foundation.layout.RowScope
|
||||||
|
import androidx.compose.foundation.layout.aspectRatio
|
||||||
|
import androidx.compose.foundation.layout.fillMaxHeight
|
||||||
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
|
import androidx.compose.foundation.layout.height
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||||
|
import androidx.compose.material3.LocalContentColor
|
||||||
|
import androidx.compose.material3.MaterialTheme
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.CompositionLocalProvider
|
||||||
|
import androidx.compose.ui.Alignment
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.draw.alpha
|
||||||
|
import androidx.compose.ui.draw.clip
|
||||||
|
import androidx.compose.ui.graphics.Brush
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import androidx.compose.ui.graphics.Shadow
|
||||||
|
import androidx.compose.ui.graphics.drawscope.ContentDrawScope
|
||||||
|
import androidx.compose.ui.node.DrawModifierNode
|
||||||
|
import androidx.compose.ui.node.modifierElementOf
|
||||||
|
import androidx.compose.ui.text.TextStyle
|
||||||
|
import androidx.compose.ui.text.style.TextOverflow
|
||||||
|
import androidx.compose.ui.unit.dp
|
||||||
|
import androidx.compose.ui.unit.sp
|
||||||
|
import eu.kanade.presentation.util.selectedBackground
|
||||||
|
|
||||||
|
object CommonMangaItemDefaults {
|
||||||
|
val GridHorizontalSpacer = 4.dp
|
||||||
|
val GridVerticalSpacer = 4.dp
|
||||||
|
|
||||||
|
const val BrowseFavoriteCoverAlpha = 0.34f
|
||||||
|
}
|
||||||
|
|
||||||
|
private const val GridSelectedCoverAlpha = 0.76f
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Layout of grid list item with title overlaying the cover.
|
||||||
|
* Accepts null [title] for a cover-only view.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun MangaCompactGridItem(
|
||||||
|
isSelected: Boolean = false,
|
||||||
|
title: String? = null,
|
||||||
|
coverData: eu.kanade.domain.manga.model.MangaCover,
|
||||||
|
coverAlpha: Float = 1f,
|
||||||
|
coverBadgeStart: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
coverBadgeEnd: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
onLongClick: () -> Unit,
|
||||||
|
onClick: () -> Unit,
|
||||||
|
) {
|
||||||
|
GridItemSelectable(
|
||||||
|
isSelected = isSelected,
|
||||||
|
onClick = onClick,
|
||||||
|
onLongClick = onLongClick,
|
||||||
|
) {
|
||||||
|
MangaGridCover(
|
||||||
|
cover = {
|
||||||
|
MangaCover.Book(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.alpha(if (isSelected) GridSelectedCoverAlpha else coverAlpha),
|
||||||
|
data = coverData,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
badgesStart = coverBadgeStart,
|
||||||
|
badgesEnd = coverBadgeEnd,
|
||||||
|
content = {
|
||||||
|
if (title != null) {
|
||||||
|
CoverTextOverlay(title = title)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Title overlay for [MangaCompactGridItem]
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
private fun BoxScope.CoverTextOverlay(title: String) {
|
||||||
|
Box(
|
||||||
|
modifier = Modifier
|
||||||
|
.clip(RoundedCornerShape(bottomStart = 4.dp, bottomEnd = 4.dp))
|
||||||
|
.background(
|
||||||
|
Brush.verticalGradient(
|
||||||
|
0f to Color.Transparent,
|
||||||
|
1f to Color(0xAA000000),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.fillMaxHeight(0.33f)
|
||||||
|
.fillMaxWidth()
|
||||||
|
.align(Alignment.BottomCenter),
|
||||||
|
)
|
||||||
|
GridItemTitle(
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(8.dp)
|
||||||
|
.align(Alignment.BottomStart),
|
||||||
|
title = title,
|
||||||
|
style = MaterialTheme.typography.titleSmall.copy(
|
||||||
|
color = Color.White,
|
||||||
|
shadow = Shadow(
|
||||||
|
color = Color.Black,
|
||||||
|
blurRadius = 4f,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Layout of grid list item with title below the cover.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun MangaComfortableGridItem(
|
||||||
|
isSelected: Boolean = false,
|
||||||
|
title: String,
|
||||||
|
coverData: eu.kanade.domain.manga.model.MangaCover,
|
||||||
|
coverAlpha: Float = 1f,
|
||||||
|
coverBadgeStart: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
coverBadgeEnd: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
onLongClick: () -> Unit,
|
||||||
|
onClick: () -> Unit,
|
||||||
|
) {
|
||||||
|
GridItemSelectable(
|
||||||
|
isSelected = isSelected,
|
||||||
|
onClick = onClick,
|
||||||
|
onLongClick = onLongClick,
|
||||||
|
) {
|
||||||
|
Column {
|
||||||
|
MangaGridCover(
|
||||||
|
cover = {
|
||||||
|
MangaCover.Book(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.alpha(if (isSelected) GridSelectedCoverAlpha else coverAlpha),
|
||||||
|
data = coverData,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
badgesStart = coverBadgeStart,
|
||||||
|
badgesEnd = coverBadgeEnd,
|
||||||
|
)
|
||||||
|
GridItemTitle(
|
||||||
|
modifier = Modifier.padding(4.dp),
|
||||||
|
title = title,
|
||||||
|
style = MaterialTheme.typography.titleSmall,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Common cover layout to add contents to be drawn on top of the cover.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
private fun MangaGridCover(
|
||||||
|
modifier: Modifier = Modifier,
|
||||||
|
cover: @Composable BoxScope.() -> Unit = {},
|
||||||
|
badgesStart: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
badgesEnd: (@Composable RowScope.() -> Unit)? = null,
|
||||||
|
content: @Composable (BoxScope.() -> Unit)? = null,
|
||||||
|
) {
|
||||||
|
Box(
|
||||||
|
modifier = modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.aspectRatio(MangaCover.Book.ratio),
|
||||||
|
) {
|
||||||
|
cover()
|
||||||
|
content?.invoke(this)
|
||||||
|
if (badgesStart != null) {
|
||||||
|
BadgeGroup(
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(4.dp)
|
||||||
|
.align(Alignment.TopStart),
|
||||||
|
content = badgesStart,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (badgesEnd != null) {
|
||||||
|
BadgeGroup(
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(4.dp)
|
||||||
|
.align(Alignment.TopEnd),
|
||||||
|
content = badgesEnd,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun GridItemTitle(
|
||||||
|
modifier: Modifier,
|
||||||
|
title: String,
|
||||||
|
style: TextStyle,
|
||||||
|
) {
|
||||||
|
Text(
|
||||||
|
modifier = modifier,
|
||||||
|
text = title,
|
||||||
|
fontSize = 12.sp,
|
||||||
|
lineHeight = 18.sp,
|
||||||
|
maxLines = 2,
|
||||||
|
overflow = TextOverflow.Ellipsis,
|
||||||
|
style = style,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrapper for grid items to handle selection state, click and long click.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
private fun GridItemSelectable(
|
||||||
|
modifier: Modifier = Modifier,
|
||||||
|
isSelected: Boolean,
|
||||||
|
onClick: () -> Unit,
|
||||||
|
onLongClick: () -> Unit,
|
||||||
|
content: @Composable () -> Unit,
|
||||||
|
) {
|
||||||
|
Box(
|
||||||
|
modifier = modifier
|
||||||
|
.clip(RoundedCornerShape(8.dp))
|
||||||
|
.combinedClickable(
|
||||||
|
onClick = onClick,
|
||||||
|
onLongClick = onLongClick,
|
||||||
|
)
|
||||||
|
.selectedOutline(isSelected = isSelected, color = MaterialTheme.colorScheme.secondary)
|
||||||
|
.padding(4.dp),
|
||||||
|
) {
|
||||||
|
val contentColor = if (isSelected) {
|
||||||
|
MaterialTheme.colorScheme.onSecondary
|
||||||
|
} else {
|
||||||
|
LocalContentColor.current
|
||||||
|
}
|
||||||
|
CompositionLocalProvider(LocalContentColor provides contentColor) {
|
||||||
|
content()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see GridItemSelectable
|
||||||
|
*/
|
||||||
|
private fun Modifier.selectedOutline(
|
||||||
|
isSelected: Boolean,
|
||||||
|
color: Color,
|
||||||
|
): Modifier {
|
||||||
|
class SelectedOutlineNode(var selected: Boolean, var color: Color) : DrawModifierNode, Modifier.Node() {
|
||||||
|
override fun ContentDrawScope.draw() {
|
||||||
|
if (selected) drawRect(color)
|
||||||
|
drawContent()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return this then modifierElementOf(
|
||||||
|
params = isSelected.hashCode() + color.hashCode(),
|
||||||
|
create = { SelectedOutlineNode(isSelected, color) },
|
||||||
|
update = {
|
||||||
|
it.selected = isSelected
|
||||||
|
it.color = color
|
||||||
|
},
|
||||||
|
definitions = {
|
||||||
|
name = "selectionOutline"
|
||||||
|
properties["isSelected"] = isSelected
|
||||||
|
properties["color"] = color
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Layout of list item.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun MangaListItem(
|
||||||
|
isSelected: Boolean = false,
|
||||||
|
title: String,
|
||||||
|
coverData: eu.kanade.domain.manga.model.MangaCover,
|
||||||
|
coverAlpha: Float = 1f,
|
||||||
|
badge: @Composable RowScope.() -> Unit,
|
||||||
|
onLongClick: () -> Unit,
|
||||||
|
onClick: () -> Unit,
|
||||||
|
) {
|
||||||
|
Row(
|
||||||
|
modifier = Modifier
|
||||||
|
.selectedBackground(isSelected)
|
||||||
|
.height(56.dp)
|
||||||
|
.combinedClickable(
|
||||||
|
onClick = onClick,
|
||||||
|
onLongClick = onLongClick,
|
||||||
|
)
|
||||||
|
.padding(horizontal = 16.dp, vertical = 8.dp),
|
||||||
|
verticalAlignment = Alignment.CenterVertically,
|
||||||
|
) {
|
||||||
|
MangaCover.Square(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxHeight()
|
||||||
|
.alpha(coverAlpha),
|
||||||
|
data = coverData,
|
||||||
|
)
|
||||||
|
Text(
|
||||||
|
text = title,
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(horizontal = 16.dp)
|
||||||
|
.weight(1f),
|
||||||
|
maxLines = 2,
|
||||||
|
overflow = TextOverflow.Ellipsis,
|
||||||
|
style = MaterialTheme.typography.bodyMedium,
|
||||||
|
)
|
||||||
|
BadgeGroup(content = badge)
|
||||||
|
}
|
||||||
|
}
|
@ -12,6 +12,7 @@ import androidx.compose.ui.Modifier
|
|||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.compose.ui.zIndex
|
import androidx.compose.ui.zIndex
|
||||||
|
import eu.kanade.presentation.components.CommonMangaItemDefaults
|
||||||
import eu.kanade.presentation.components.FastScrollLazyVerticalGrid
|
import eu.kanade.presentation.components.FastScrollLazyVerticalGrid
|
||||||
import eu.kanade.presentation.util.plus
|
import eu.kanade.presentation.util.plus
|
||||||
import eu.kanade.tachiyomi.R
|
import eu.kanade.tachiyomi.R
|
||||||
@ -26,9 +27,9 @@ fun LazyLibraryGrid(
|
|||||||
FastScrollLazyVerticalGrid(
|
FastScrollLazyVerticalGrid(
|
||||||
columns = if (columns == 0) GridCells.Adaptive(128.dp) else GridCells.Fixed(columns),
|
columns = if (columns == 0) GridCells.Adaptive(128.dp) else GridCells.Fixed(columns),
|
||||||
modifier = modifier,
|
modifier = modifier,
|
||||||
contentPadding = contentPadding + PaddingValues(12.dp),
|
contentPadding = contentPadding + PaddingValues(8.dp),
|
||||||
verticalArrangement = Arrangement.spacedBy(12.dp),
|
verticalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridVerticalSpacer),
|
||||||
horizontalArrangement = Arrangement.spacedBy(12.dp),
|
horizontalArrangement = Arrangement.spacedBy(CommonMangaItemDefaults.GridHorizontalSpacer),
|
||||||
content = content,
|
content = content,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -1,21 +1,14 @@
|
|||||||
package eu.kanade.presentation.library.components
|
package eu.kanade.presentation.library.components
|
||||||
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.Column
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.fillMaxSize
|
import androidx.compose.foundation.layout.fillMaxSize
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.foundation.lazy.grid.items
|
import androidx.compose.foundation.lazy.grid.items
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.material3.Text
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.text.style.TextOverflow
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
import androidx.compose.ui.unit.sp
|
|
||||||
import androidx.compose.ui.util.fastAny
|
import androidx.compose.ui.util.fastAny
|
||||||
import eu.kanade.domain.library.model.LibraryManga
|
import eu.kanade.domain.library.model.LibraryManga
|
||||||
import eu.kanade.domain.manga.model.MangaCover
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
|
import eu.kanade.presentation.components.MangaComfortableGridItem
|
||||||
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
@ -44,76 +37,37 @@ fun LibraryComfortableGrid(
|
|||||||
items = items,
|
items = items,
|
||||||
contentType = { "library_comfortable_grid_item" },
|
contentType = { "library_comfortable_grid_item" },
|
||||||
) { libraryItem ->
|
) { libraryItem ->
|
||||||
LibraryComfortableGridItem(
|
val manga = libraryItem.libraryManga.manga
|
||||||
item = libraryItem,
|
MangaComfortableGridItem(
|
||||||
showDownloadBadge = showDownloadBadges,
|
|
||||||
showUnreadBadge = showUnreadBadges,
|
|
||||||
showLocalBadge = showLocalBadges,
|
|
||||||
showLanguageBadge = showLanguageBadges,
|
|
||||||
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
||||||
onClick = onClick,
|
title = manga.title,
|
||||||
onLongClick = onLongClick,
|
coverData = MangaCover(
|
||||||
)
|
mangaId = manga.id,
|
||||||
}
|
sourceId = manga.source,
|
||||||
}
|
isMangaFavorite = manga.favorite,
|
||||||
}
|
url = manga.thumbnailUrl,
|
||||||
|
lastModified = manga.coverLastModified,
|
||||||
@Composable
|
|
||||||
fun LibraryComfortableGridItem(
|
|
||||||
item: LibraryItem,
|
|
||||||
showDownloadBadge: Boolean,
|
|
||||||
showUnreadBadge: Boolean,
|
|
||||||
showLocalBadge: Boolean,
|
|
||||||
showLanguageBadge: Boolean,
|
|
||||||
isSelected: Boolean,
|
|
||||||
onClick: (LibraryManga) -> Unit,
|
|
||||||
onLongClick: (LibraryManga) -> Unit,
|
|
||||||
) {
|
|
||||||
val libraryManga = item.libraryManga
|
|
||||||
val manga = libraryManga.manga
|
|
||||||
LibraryGridItemSelectable(isSelected = isSelected) {
|
|
||||||
Column(
|
|
||||||
modifier = Modifier
|
|
||||||
.combinedClickable(
|
|
||||||
onClick = {
|
|
||||||
onClick(libraryManga)
|
|
||||||
},
|
|
||||||
onLongClick = {
|
|
||||||
onLongClick(libraryManga)
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
) {
|
coverBadgeStart = {
|
||||||
LibraryGridCover(
|
DownloadsBadge(
|
||||||
mangaCover = MangaCover(
|
enabled = showDownloadBadges,
|
||||||
manga.id,
|
item = libraryItem,
|
||||||
manga.source,
|
|
||||||
manga.favorite,
|
|
||||||
manga.thumbnailUrl,
|
|
||||||
manga.coverLastModified,
|
|
||||||
),
|
|
||||||
item = item,
|
|
||||||
showDownloadBadge = showDownloadBadge,
|
|
||||||
showUnreadBadge = showUnreadBadge,
|
|
||||||
showLocalBadge = showLocalBadge,
|
|
||||||
showLanguageBadge = showLanguageBadge,
|
|
||||||
)
|
)
|
||||||
MangaGridComfortableText(
|
UnreadBadge(
|
||||||
text = manga.title,
|
enabled = showUnreadBadges,
|
||||||
|
item = libraryItem,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
coverBadgeEnd = {
|
||||||
|
LanguageBadge(
|
||||||
|
showLanguage = showLanguageBadges,
|
||||||
|
showLocal = showLocalBadges,
|
||||||
|
item = libraryItem,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
onLongClick = { onLongClick(libraryItem.libraryManga) },
|
||||||
|
onClick = { onClick(libraryItem.libraryManga) },
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun MangaGridComfortableText(
|
|
||||||
text: String,
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
modifier = Modifier.padding(4.dp),
|
|
||||||
text = text,
|
|
||||||
fontSize = 12.sp,
|
|
||||||
maxLines = 2,
|
|
||||||
overflow = TextOverflow.Ellipsis,
|
|
||||||
style = MaterialTheme.typography.titleSmall,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
@ -1,35 +1,20 @@
|
|||||||
package eu.kanade.presentation.library.components
|
package eu.kanade.presentation.library.components
|
||||||
|
|
||||||
import androidx.compose.foundation.background
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.Box
|
|
||||||
import androidx.compose.foundation.layout.BoxScope
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.fillMaxHeight
|
|
||||||
import androidx.compose.foundation.layout.fillMaxSize
|
import androidx.compose.foundation.layout.fillMaxSize
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.foundation.lazy.grid.items
|
import androidx.compose.foundation.lazy.grid.items
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.material3.Text
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.draw.clip
|
|
||||||
import androidx.compose.ui.graphics.Brush
|
|
||||||
import androidx.compose.ui.graphics.Color
|
|
||||||
import androidx.compose.ui.graphics.Shadow
|
|
||||||
import androidx.compose.ui.text.style.TextOverflow
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
import androidx.compose.ui.unit.sp
|
|
||||||
import androidx.compose.ui.util.fastAny
|
import androidx.compose.ui.util.fastAny
|
||||||
import eu.kanade.domain.library.model.LibraryManga
|
import eu.kanade.domain.library.model.LibraryManga
|
||||||
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
|
import eu.kanade.presentation.components.MangaCompactGridItem
|
||||||
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
fun LibraryCompactGrid(
|
fun LibraryCompactGrid(
|
||||||
items: List<LibraryItem>,
|
items: List<LibraryItem>,
|
||||||
|
showTitle: Boolean,
|
||||||
showDownloadBadges: Boolean,
|
showDownloadBadges: Boolean,
|
||||||
showUnreadBadges: Boolean,
|
showUnreadBadges: Boolean,
|
||||||
showLocalBadges: Boolean,
|
showLocalBadges: Boolean,
|
||||||
@ -53,92 +38,37 @@ fun LibraryCompactGrid(
|
|||||||
items = items,
|
items = items,
|
||||||
contentType = { "library_compact_grid_item" },
|
contentType = { "library_compact_grid_item" },
|
||||||
) { libraryItem ->
|
) { libraryItem ->
|
||||||
LibraryCompactGridItem(
|
val manga = libraryItem.libraryManga.manga
|
||||||
item = libraryItem,
|
MangaCompactGridItem(
|
||||||
showDownloadBadge = showDownloadBadges,
|
|
||||||
showUnreadBadge = showUnreadBadges,
|
|
||||||
showLocalBadge = showLocalBadges,
|
|
||||||
showLanguageBadge = showLanguageBadges,
|
|
||||||
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
||||||
onClick = onClick,
|
title = manga.title.takeIf { showTitle },
|
||||||
onLongClick = onLongClick,
|
coverData = MangaCover(
|
||||||
|
mangaId = manga.id,
|
||||||
|
sourceId = manga.source,
|
||||||
|
isMangaFavorite = manga.favorite,
|
||||||
|
url = manga.thumbnailUrl,
|
||||||
|
lastModified = manga.coverLastModified,
|
||||||
|
),
|
||||||
|
coverBadgeStart = {
|
||||||
|
DownloadsBadge(
|
||||||
|
enabled = showDownloadBadges,
|
||||||
|
item = libraryItem,
|
||||||
|
)
|
||||||
|
UnreadBadge(
|
||||||
|
enabled = showUnreadBadges,
|
||||||
|
item = libraryItem,
|
||||||
)
|
)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryCompactGridItem(
|
|
||||||
item: LibraryItem,
|
|
||||||
showDownloadBadge: Boolean,
|
|
||||||
showUnreadBadge: Boolean,
|
|
||||||
showLocalBadge: Boolean,
|
|
||||||
showLanguageBadge: Boolean,
|
|
||||||
isSelected: Boolean,
|
|
||||||
onClick: (LibraryManga) -> Unit,
|
|
||||||
onLongClick: (LibraryManga) -> Unit,
|
|
||||||
) {
|
|
||||||
val libraryManga = item.libraryManga
|
|
||||||
val manga = libraryManga.manga
|
|
||||||
LibraryGridCover(
|
|
||||||
modifier = Modifier
|
|
||||||
.selectedOutline(isSelected)
|
|
||||||
.combinedClickable(
|
|
||||||
onClick = {
|
|
||||||
onClick(libraryManga)
|
|
||||||
},
|
},
|
||||||
onLongClick = {
|
coverBadgeEnd = {
|
||||||
onLongClick(libraryManga)
|
LanguageBadge(
|
||||||
|
showLanguage = showLanguageBadges,
|
||||||
|
showLocal = showLocalBadges,
|
||||||
|
item = libraryItem,
|
||||||
|
)
|
||||||
},
|
},
|
||||||
),
|
onLongClick = { onLongClick(libraryItem.libraryManga) },
|
||||||
mangaCover = eu.kanade.domain.manga.model.MangaCover(
|
onClick = { onClick(libraryItem.libraryManga) },
|
||||||
manga.id,
|
|
||||||
manga.source,
|
|
||||||
manga.favorite,
|
|
||||||
manga.thumbnailUrl,
|
|
||||||
manga.coverLastModified,
|
|
||||||
),
|
|
||||||
item = item,
|
|
||||||
showDownloadBadge = showDownloadBadge,
|
|
||||||
showUnreadBadge = showUnreadBadge,
|
|
||||||
showLocalBadge = showLocalBadge,
|
|
||||||
showLanguageBadge = showLanguageBadge,
|
|
||||||
) {
|
|
||||||
Box(
|
|
||||||
modifier = Modifier
|
|
||||||
.clip(RoundedCornerShape(bottomStart = 4.dp, bottomEnd = 4.dp))
|
|
||||||
.background(
|
|
||||||
Brush.verticalGradient(
|
|
||||||
0f to Color.Transparent,
|
|
||||||
1f to Color(0xAA000000),
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
.fillMaxHeight(0.33f)
|
}
|
||||||
.fillMaxWidth()
|
|
||||||
.align(Alignment.BottomCenter),
|
|
||||||
)
|
|
||||||
MangaGridCompactText(manga.title)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun BoxScope.MangaGridCompactText(
|
|
||||||
text: String,
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
text = text,
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(8.dp)
|
|
||||||
.align(Alignment.BottomStart),
|
|
||||||
color = Color.White,
|
|
||||||
fontSize = 12.sp,
|
|
||||||
maxLines = 2,
|
|
||||||
overflow = TextOverflow.Ellipsis,
|
|
||||||
style = MaterialTheme.typography.titleSmall.copy(
|
|
||||||
shadow = Shadow(
|
|
||||||
color = Color.Black,
|
|
||||||
blurRadius = 4f,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
@ -1,90 +0,0 @@
|
|||||||
package eu.kanade.presentation.library.components
|
|
||||||
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
|
||||||
import androidx.compose.foundation.layout.fillMaxSize
|
|
||||||
import androidx.compose.foundation.lazy.grid.items
|
|
||||||
import androidx.compose.runtime.Composable
|
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.util.fastAny
|
|
||||||
import eu.kanade.domain.library.model.LibraryManga
|
|
||||||
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryCoverOnlyGrid(
|
|
||||||
items: List<LibraryItem>,
|
|
||||||
showDownloadBadges: Boolean,
|
|
||||||
showUnreadBadges: Boolean,
|
|
||||||
showLocalBadges: Boolean,
|
|
||||||
showLanguageBadges: Boolean,
|
|
||||||
columns: Int,
|
|
||||||
contentPadding: PaddingValues,
|
|
||||||
selection: List<LibraryManga>,
|
|
||||||
onClick: (LibraryManga) -> Unit,
|
|
||||||
onLongClick: (LibraryManga) -> Unit,
|
|
||||||
searchQuery: String?,
|
|
||||||
onGlobalSearchClicked: () -> Unit,
|
|
||||||
) {
|
|
||||||
LazyLibraryGrid(
|
|
||||||
modifier = Modifier.fillMaxSize(),
|
|
||||||
columns = columns,
|
|
||||||
contentPadding = contentPadding,
|
|
||||||
) {
|
|
||||||
globalSearchItem(searchQuery, onGlobalSearchClicked)
|
|
||||||
|
|
||||||
items(
|
|
||||||
items = items,
|
|
||||||
contentType = { "library_only_cover_grid_item" },
|
|
||||||
) { libraryItem ->
|
|
||||||
LibraryCoverOnlyGridItem(
|
|
||||||
item = libraryItem,
|
|
||||||
showDownloadBadge = showDownloadBadges,
|
|
||||||
showUnreadBadge = showUnreadBadges,
|
|
||||||
showLocalBadge = showLocalBadges,
|
|
||||||
showLanguageBadge = showLanguageBadges,
|
|
||||||
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryCoverOnlyGridItem(
|
|
||||||
item: LibraryItem,
|
|
||||||
showDownloadBadge: Boolean,
|
|
||||||
showUnreadBadge: Boolean,
|
|
||||||
showLocalBadge: Boolean,
|
|
||||||
showLanguageBadge: Boolean,
|
|
||||||
isSelected: Boolean,
|
|
||||||
onClick: (LibraryManga) -> Unit,
|
|
||||||
onLongClick: (LibraryManga) -> Unit,
|
|
||||||
) {
|
|
||||||
val libraryManga = item.libraryManga
|
|
||||||
val manga = libraryManga.manga
|
|
||||||
LibraryGridCover(
|
|
||||||
modifier = Modifier
|
|
||||||
.selectedOutline(isSelected)
|
|
||||||
.combinedClickable(
|
|
||||||
onClick = {
|
|
||||||
onClick(libraryManga)
|
|
||||||
},
|
|
||||||
onLongClick = {
|
|
||||||
onLongClick(libraryManga)
|
|
||||||
},
|
|
||||||
),
|
|
||||||
mangaCover = eu.kanade.domain.manga.model.MangaCover(
|
|
||||||
manga.id,
|
|
||||||
manga.source,
|
|
||||||
manga.favorite,
|
|
||||||
manga.thumbnailUrl,
|
|
||||||
manga.coverLastModified,
|
|
||||||
),
|
|
||||||
item = item,
|
|
||||||
showDownloadBadge = showDownloadBadge,
|
|
||||||
showUnreadBadge = showUnreadBadge,
|
|
||||||
showLocalBadge = showLocalBadge,
|
|
||||||
showLanguageBadge = showLanguageBadge,
|
|
||||||
)
|
|
||||||
}
|
|
@ -1,80 +0,0 @@
|
|||||||
package eu.kanade.presentation.library.components
|
|
||||||
|
|
||||||
import androidx.compose.foundation.layout.Box
|
|
||||||
import androidx.compose.foundation.layout.BoxScope
|
|
||||||
import androidx.compose.foundation.layout.RowScope
|
|
||||||
import androidx.compose.foundation.layout.aspectRatio
|
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.runtime.Composable
|
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
import eu.kanade.presentation.components.BadgeGroup
|
|
||||||
import eu.kanade.presentation.components.MangaCover
|
|
||||||
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun MangaGridCover(
|
|
||||||
modifier: Modifier = Modifier,
|
|
||||||
cover: @Composable BoxScope.() -> Unit = {},
|
|
||||||
badgesStart: (@Composable RowScope.() -> Unit)? = null,
|
|
||||||
badgesEnd: (@Composable RowScope.() -> Unit)? = null,
|
|
||||||
content: @Composable BoxScope.() -> Unit = {},
|
|
||||||
) {
|
|
||||||
Box(
|
|
||||||
modifier = modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.aspectRatio(MangaCover.Book.ratio),
|
|
||||||
) {
|
|
||||||
cover()
|
|
||||||
content()
|
|
||||||
if (badgesStart != null) {
|
|
||||||
BadgeGroup(
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(4.dp)
|
|
||||||
.align(Alignment.TopStart),
|
|
||||||
content = badgesStart,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (badgesEnd != null) {
|
|
||||||
BadgeGroup(
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(4.dp)
|
|
||||||
.align(Alignment.TopEnd),
|
|
||||||
content = badgesEnd,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryGridCover(
|
|
||||||
modifier: Modifier = Modifier,
|
|
||||||
mangaCover: eu.kanade.domain.manga.model.MangaCover,
|
|
||||||
item: LibraryItem,
|
|
||||||
showDownloadBadge: Boolean,
|
|
||||||
showUnreadBadge: Boolean,
|
|
||||||
showLocalBadge: Boolean,
|
|
||||||
showLanguageBadge: Boolean,
|
|
||||||
content: @Composable BoxScope.() -> Unit = {},
|
|
||||||
) {
|
|
||||||
MangaGridCover(
|
|
||||||
modifier = modifier,
|
|
||||||
cover = {
|
|
||||||
MangaCover.Book(
|
|
||||||
modifier = Modifier.fillMaxWidth(),
|
|
||||||
data = mangaCover,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
badgesStart = {
|
|
||||||
DownloadsBadge(enabled = showDownloadBadge, item = item)
|
|
||||||
UnreadBadge(enabled = showUnreadBadge, item = item)
|
|
||||||
},
|
|
||||||
badgesEnd = {
|
|
||||||
LanguageBadge(showLanguage = showLanguageBadge, showLocal = showLocalBadge, item = item)
|
|
||||||
},
|
|
||||||
content = content,
|
|
||||||
)
|
|
||||||
}
|
|
@ -1,46 +0,0 @@
|
|||||||
package eu.kanade.presentation.library.components
|
|
||||||
|
|
||||||
import androidx.compose.foundation.layout.Box
|
|
||||||
import androidx.compose.material3.LocalContentColor
|
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.runtime.Composable
|
|
||||||
import androidx.compose.runtime.CompositionLocalProvider
|
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.composed
|
|
||||||
import androidx.compose.ui.draw.drawBehind
|
|
||||||
import androidx.compose.ui.geometry.CornerRadius
|
|
||||||
import androidx.compose.ui.geometry.Offset
|
|
||||||
import androidx.compose.ui.geometry.Size
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
|
|
||||||
fun Modifier.selectedOutline(isSelected: Boolean) = composed {
|
|
||||||
val secondary = MaterialTheme.colorScheme.secondary
|
|
||||||
if (isSelected) {
|
|
||||||
drawBehind {
|
|
||||||
val additional = 24.dp.value
|
|
||||||
val offset = additional / 2
|
|
||||||
val height = size.height + additional
|
|
||||||
val width = size.width + additional
|
|
||||||
drawRoundRect(
|
|
||||||
color = secondary,
|
|
||||||
topLeft = Offset(-offset, -offset),
|
|
||||||
size = Size(width, height),
|
|
||||||
cornerRadius = CornerRadius(offset),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
this
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryGridItemSelectable(
|
|
||||||
isSelected: Boolean,
|
|
||||||
content: @Composable () -> Unit,
|
|
||||||
) {
|
|
||||||
Box(Modifier.selectedOutline(isSelected)) {
|
|
||||||
CompositionLocalProvider(LocalContentColor provides if (isSelected) MaterialTheme.colorScheme.onSecondary else MaterialTheme.colorScheme.onBackground) {
|
|
||||||
content()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +1,21 @@
|
|||||||
package eu.kanade.presentation.library.components
|
package eu.kanade.presentation.library.components
|
||||||
|
|
||||||
import androidx.compose.foundation.combinedClickable
|
|
||||||
import androidx.compose.foundation.layout.PaddingValues
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
import androidx.compose.foundation.layout.Row
|
|
||||||
import androidx.compose.foundation.layout.RowScope
|
|
||||||
import androidx.compose.foundation.layout.fillMaxHeight
|
|
||||||
import androidx.compose.foundation.layout.fillMaxSize
|
import androidx.compose.foundation.layout.fillMaxSize
|
||||||
import androidx.compose.foundation.layout.height
|
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.foundation.lazy.items
|
import androidx.compose.foundation.lazy.items
|
||||||
import androidx.compose.material3.MaterialTheme
|
|
||||||
import androidx.compose.material3.Text
|
import androidx.compose.material3.Text
|
||||||
import androidx.compose.material3.TextButton
|
import androidx.compose.material3.TextButton
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.text.style.TextOverflow
|
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.compose.ui.util.fastAny
|
import androidx.compose.ui.util.fastAny
|
||||||
import androidx.compose.ui.zIndex
|
import androidx.compose.ui.zIndex
|
||||||
import eu.kanade.domain.library.model.LibraryManga
|
import eu.kanade.domain.library.model.LibraryManga
|
||||||
import eu.kanade.domain.manga.model.MangaCover
|
import eu.kanade.domain.manga.model.MangaCover
|
||||||
import eu.kanade.presentation.components.BadgeGroup
|
|
||||||
import eu.kanade.presentation.components.FastScrollLazyColumn
|
import eu.kanade.presentation.components.FastScrollLazyColumn
|
||||||
import eu.kanade.presentation.components.MangaCover.Square
|
import eu.kanade.presentation.components.MangaListItem
|
||||||
import eu.kanade.presentation.util.horizontalPadding
|
import eu.kanade.presentation.util.plus
|
||||||
import eu.kanade.presentation.util.selectedBackground
|
|
||||||
import eu.kanade.presentation.util.verticalPadding
|
|
||||||
import eu.kanade.tachiyomi.R
|
import eu.kanade.tachiyomi.R
|
||||||
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
import eu.kanade.tachiyomi.ui.library.LibraryItem
|
||||||
|
|
||||||
@ -47,7 +35,7 @@ fun LibraryList(
|
|||||||
) {
|
) {
|
||||||
FastScrollLazyColumn(
|
FastScrollLazyColumn(
|
||||||
modifier = Modifier.fillMaxSize(),
|
modifier = Modifier.fillMaxSize(),
|
||||||
contentPadding = contentPadding,
|
contentPadding = contentPadding + PaddingValues(vertical = 8.dp),
|
||||||
) {
|
) {
|
||||||
item {
|
item {
|
||||||
if (searchQuery.isNullOrEmpty().not()) {
|
if (searchQuery.isNullOrEmpty().not()) {
|
||||||
@ -64,116 +52,25 @@ fun LibraryList(
|
|||||||
items = items,
|
items = items,
|
||||||
contentType = { "library_list_item" },
|
contentType = { "library_list_item" },
|
||||||
) { libraryItem ->
|
) { libraryItem ->
|
||||||
LibraryListItem(
|
val manga = libraryItem.libraryManga.manga
|
||||||
item = libraryItem,
|
MangaListItem(
|
||||||
showDownloadBadge = showDownloadBadges,
|
|
||||||
showUnreadBadge = showUnreadBadges,
|
|
||||||
showLocalBadge = showLocalBadges,
|
|
||||||
showLanguageBadge = showLanguageBadges,
|
|
||||||
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
isSelected = selection.fastAny { it.id == libraryItem.libraryManga.id },
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun LibraryListItem(
|
|
||||||
item: LibraryItem,
|
|
||||||
showDownloadBadge: Boolean,
|
|
||||||
showUnreadBadge: Boolean,
|
|
||||||
showLocalBadge: Boolean,
|
|
||||||
showLanguageBadge: Boolean,
|
|
||||||
isSelected: Boolean,
|
|
||||||
onClick: (LibraryManga) -> Unit,
|
|
||||||
onLongClick: (LibraryManga) -> Unit,
|
|
||||||
) {
|
|
||||||
val libraryManga = item.libraryManga
|
|
||||||
val manga = libraryManga.manga
|
|
||||||
MangaListItem(
|
|
||||||
modifier = Modifier.selectedBackground(isSelected),
|
|
||||||
title = manga.title,
|
title = manga.title,
|
||||||
cover = MangaCover(
|
coverData = MangaCover(
|
||||||
manga.id,
|
mangaId = manga.id,
|
||||||
manga.source,
|
sourceId = manga.source,
|
||||||
manga.favorite,
|
isMangaFavorite = manga.favorite,
|
||||||
manga.thumbnailUrl,
|
url = manga.thumbnailUrl,
|
||||||
manga.coverLastModified,
|
lastModified = manga.coverLastModified,
|
||||||
),
|
),
|
||||||
onClick = { onClick(libraryManga) },
|
badge = {
|
||||||
onLongClick = { onLongClick(libraryManga) },
|
DownloadsBadge(enabled = showDownloadBadges, item = libraryItem)
|
||||||
) {
|
UnreadBadge(enabled = showUnreadBadges, item = libraryItem)
|
||||||
DownloadsBadge(enabled = showDownloadBadge, item = item)
|
LanguageBadge(showLanguage = showLanguageBadges, showLocal = showLocalBadges, item = libraryItem)
|
||||||
UnreadBadge(enabled = showUnreadBadge, item = item)
|
},
|
||||||
LanguageBadge(showLanguage = showLanguageBadge, showLocal = showLocalBadge, item = item)
|
onLongClick = { onLongClick(libraryItem.libraryManga) },
|
||||||
|
onClick = { onClick(libraryItem.libraryManga) },
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun MangaListItem(
|
|
||||||
modifier: Modifier = Modifier,
|
|
||||||
title: String,
|
|
||||||
cover: MangaCover,
|
|
||||||
onClick: () -> Unit,
|
|
||||||
onLongClick: () -> Unit = onClick,
|
|
||||||
badges: @Composable RowScope.() -> Unit,
|
|
||||||
) {
|
|
||||||
MangaListItem(
|
|
||||||
modifier = modifier,
|
|
||||||
coverContent = {
|
|
||||||
Square(
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(vertical = verticalPadding)
|
|
||||||
.fillMaxHeight(),
|
|
||||||
data = cover,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
badges = badges,
|
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
content = {
|
|
||||||
MangaListItemContent(title)
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun MangaListItem(
|
|
||||||
modifier: Modifier = Modifier,
|
|
||||||
coverContent: @Composable RowScope.() -> Unit,
|
|
||||||
badges: @Composable RowScope.() -> Unit,
|
|
||||||
onClick: () -> Unit,
|
|
||||||
onLongClick: () -> Unit,
|
|
||||||
content: @Composable RowScope.() -> Unit,
|
|
||||||
) {
|
|
||||||
Row(
|
|
||||||
modifier = modifier
|
|
||||||
.height(56.dp)
|
|
||||||
.combinedClickable(
|
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
)
|
|
||||||
.padding(horizontal = horizontalPadding),
|
|
||||||
verticalAlignment = Alignment.CenterVertically,
|
|
||||||
) {
|
|
||||||
coverContent()
|
|
||||||
content()
|
|
||||||
BadgeGroup(content = badges)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun RowScope.MangaListItemContent(
|
|
||||||
text: String,
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
text = text,
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(horizontal = horizontalPadding)
|
|
||||||
.weight(1f),
|
|
||||||
maxLines = 2,
|
|
||||||
overflow = TextOverflow.Ellipsis,
|
|
||||||
style = MaterialTheme.typography.bodyMedium,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
@ -72,9 +72,10 @@ fun LibraryPager(
|
|||||||
onGlobalSearchClicked = onGlobalSearchClicked,
|
onGlobalSearchClicked = onGlobalSearchClicked,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
LibraryDisplayMode.CompactGrid -> {
|
LibraryDisplayMode.CompactGrid, LibraryDisplayMode.CoverOnlyGrid -> {
|
||||||
LibraryCompactGrid(
|
LibraryCompactGrid(
|
||||||
items = library,
|
items = library,
|
||||||
|
showTitle = displayMode is LibraryDisplayMode.CompactGrid,
|
||||||
showDownloadBadges = showDownloadBadges,
|
showDownloadBadges = showDownloadBadges,
|
||||||
showUnreadBadges = showUnreadBadges,
|
showUnreadBadges = showUnreadBadges,
|
||||||
showLocalBadges = showLocalBadges,
|
showLocalBadges = showLocalBadges,
|
||||||
@ -104,22 +105,6 @@ fun LibraryPager(
|
|||||||
onGlobalSearchClicked = onGlobalSearchClicked,
|
onGlobalSearchClicked = onGlobalSearchClicked,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
LibraryDisplayMode.CoverOnlyGrid -> {
|
|
||||||
LibraryCoverOnlyGrid(
|
|
||||||
items = library,
|
|
||||||
showDownloadBadges = showDownloadBadges,
|
|
||||||
showUnreadBadges = showUnreadBadges,
|
|
||||||
showLocalBadges = showLocalBadges,
|
|
||||||
showLanguageBadges = showLanguageBadges,
|
|
||||||
columns = columns,
|
|
||||||
contentPadding = contentPadding,
|
|
||||||
selection = selectedManga,
|
|
||||||
onClick = onClickManga,
|
|
||||||
onLongClick = onLongClickManga,
|
|
||||||
searchQuery = searchQuery,
|
|
||||||
onGlobalSearchClicked = onGlobalSearchClicked,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user