UpdatesUiItem.kt 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. package eu.kanade.presentation.updates
  2. import android.text.format.DateUtils
  3. import androidx.compose.foundation.background
  4. import androidx.compose.foundation.combinedClickable
  5. import androidx.compose.foundation.layout.Box
  6. import androidx.compose.foundation.layout.Column
  7. import androidx.compose.foundation.layout.Row
  8. import androidx.compose.foundation.layout.Spacer
  9. import androidx.compose.foundation.layout.fillMaxHeight
  10. import androidx.compose.foundation.layout.height
  11. import androidx.compose.foundation.layout.padding
  12. import androidx.compose.foundation.layout.sizeIn
  13. import androidx.compose.foundation.layout.width
  14. import androidx.compose.foundation.lazy.LazyListScope
  15. import androidx.compose.foundation.lazy.items
  16. import androidx.compose.material.icons.Icons
  17. import androidx.compose.material.icons.filled.Bookmark
  18. import androidx.compose.material3.Icon
  19. import androidx.compose.material3.LocalTextStyle
  20. import androidx.compose.material3.MaterialTheme
  21. import androidx.compose.material3.Text
  22. import androidx.compose.runtime.Composable
  23. import androidx.compose.runtime.getValue
  24. import androidx.compose.runtime.mutableStateOf
  25. import androidx.compose.runtime.remember
  26. import androidx.compose.runtime.setValue
  27. import androidx.compose.ui.Alignment
  28. import androidx.compose.ui.Modifier
  29. import androidx.compose.ui.draw.alpha
  30. import androidx.compose.ui.graphics.Color
  31. import androidx.compose.ui.hapticfeedback.HapticFeedbackType
  32. import androidx.compose.ui.platform.LocalDensity
  33. import androidx.compose.ui.platform.LocalHapticFeedback
  34. import androidx.compose.ui.res.stringResource
  35. import androidx.compose.ui.text.font.FontStyle
  36. import androidx.compose.ui.text.style.TextOverflow
  37. import androidx.compose.ui.unit.dp
  38. import eu.kanade.domain.updates.model.UpdatesWithRelations
  39. import eu.kanade.presentation.components.ChapterDownloadAction
  40. import eu.kanade.presentation.components.ChapterDownloadIndicator
  41. import eu.kanade.presentation.components.MangaCover
  42. import eu.kanade.presentation.components.RelativeDateHeader
  43. import eu.kanade.presentation.util.ReadItemAlpha
  44. import eu.kanade.presentation.util.horizontalPadding
  45. import eu.kanade.tachiyomi.R
  46. import eu.kanade.tachiyomi.data.download.model.Download
  47. import eu.kanade.tachiyomi.ui.recent.updates.UpdatesItem
  48. import java.text.DateFormat
  49. import java.util.Date
  50. import kotlin.time.Duration.Companion.minutes
  51. fun LazyListScope.updatesLastUpdatedItem(
  52. lastUpdated: Long,
  53. ) {
  54. item(key = "updates-lastUpdated") {
  55. val time = remember(lastUpdated) {
  56. val now = Date().time
  57. if (now - lastUpdated < 1.minutes.inWholeMilliseconds) {
  58. null
  59. } else {
  60. DateUtils.getRelativeTimeSpanString(lastUpdated, now, DateUtils.MINUTE_IN_MILLIS)
  61. }
  62. }
  63. Box(
  64. modifier = Modifier
  65. .animateItemPlacement()
  66. .padding(horizontal = horizontalPadding, vertical = 8.dp),
  67. ) {
  68. Text(
  69. text = if (time.isNullOrEmpty()) {
  70. stringResource(R.string.updates_last_update_info, stringResource(R.string.updates_last_update_info_just_now))
  71. } else {
  72. stringResource(R.string.updates_last_update_info, time)
  73. },
  74. style = LocalTextStyle.current.copy(
  75. fontStyle = FontStyle.Italic,
  76. ),
  77. )
  78. }
  79. }
  80. }
  81. fun LazyListScope.updatesUiItems(
  82. uiModels: List<UpdatesUiModel>,
  83. selectionMode: Boolean,
  84. onUpdateSelected: (UpdatesItem, Boolean, Boolean, Boolean) -> Unit,
  85. onClickCover: (UpdatesItem) -> Unit,
  86. onClickUpdate: (UpdatesItem) -> Unit,
  87. onDownloadChapter: (List<UpdatesItem>, ChapterDownloadAction) -> Unit,
  88. relativeTime: Int,
  89. dateFormat: DateFormat,
  90. ) {
  91. items(
  92. items = uiModels,
  93. contentType = {
  94. when (it) {
  95. is UpdatesUiModel.Header -> "header"
  96. is UpdatesUiModel.Item -> "item"
  97. }
  98. },
  99. key = {
  100. when (it) {
  101. is UpdatesUiModel.Header -> "updatesHeader-${it.hashCode()}"
  102. is UpdatesUiModel.Item -> "updates-${it.item.update.mangaId}-${it.item.update.chapterId}"
  103. }
  104. },
  105. ) { item ->
  106. when (item) {
  107. is UpdatesUiModel.Header -> {
  108. RelativeDateHeader(
  109. modifier = Modifier.animateItemPlacement(),
  110. date = item.date,
  111. relativeTime = relativeTime,
  112. dateFormat = dateFormat,
  113. )
  114. }
  115. is UpdatesUiModel.Item -> {
  116. val updatesItem = item.item
  117. UpdatesUiItem(
  118. modifier = Modifier.animateItemPlacement(),
  119. update = updatesItem.update,
  120. selected = updatesItem.selected,
  121. onLongClick = {
  122. onUpdateSelected(updatesItem, !updatesItem.selected, true, true)
  123. },
  124. onClick = {
  125. when {
  126. selectionMode -> onUpdateSelected(updatesItem, !updatesItem.selected, true, false)
  127. else -> onClickUpdate(updatesItem)
  128. }
  129. },
  130. onClickCover = { if (selectionMode.not()) onClickCover(updatesItem) },
  131. onDownloadChapter = {
  132. if (selectionMode.not()) onDownloadChapter(listOf(updatesItem), it)
  133. },
  134. downloadStateProvider = updatesItem.downloadStateProvider,
  135. downloadProgressProvider = updatesItem.downloadProgressProvider,
  136. )
  137. }
  138. }
  139. }
  140. }
  141. @Composable
  142. fun UpdatesUiItem(
  143. modifier: Modifier,
  144. update: UpdatesWithRelations,
  145. selected: Boolean,
  146. onClick: () -> Unit,
  147. onLongClick: () -> Unit,
  148. onClickCover: () -> Unit,
  149. onDownloadChapter: (ChapterDownloadAction) -> Unit,
  150. // Download Indicator
  151. downloadStateProvider: () -> Download.State,
  152. downloadProgressProvider: () -> Int,
  153. ) {
  154. val haptic = LocalHapticFeedback.current
  155. Row(
  156. modifier = modifier
  157. .background(if (selected) MaterialTheme.colorScheme.surfaceVariant else Color.Transparent)
  158. .combinedClickable(
  159. onClick = onClick,
  160. onLongClick = {
  161. onLongClick()
  162. haptic.performHapticFeedback(HapticFeedbackType.LongPress)
  163. },
  164. )
  165. .height(56.dp)
  166. .padding(horizontal = horizontalPadding),
  167. verticalAlignment = Alignment.CenterVertically,
  168. ) {
  169. MangaCover.Square(
  170. modifier = Modifier
  171. .padding(vertical = 6.dp)
  172. .fillMaxHeight(),
  173. data = update.coverData,
  174. onClick = onClickCover,
  175. )
  176. Column(
  177. modifier = Modifier
  178. .padding(horizontal = horizontalPadding)
  179. .weight(1f),
  180. ) {
  181. val bookmark = remember(update.bookmark) { update.bookmark }
  182. val read = remember(update.read) { update.read }
  183. val textAlpha = remember(read) { if (read) ReadItemAlpha else 1f }
  184. val secondaryTextColor = if (bookmark && !read) {
  185. MaterialTheme.colorScheme.primary
  186. } else {
  187. MaterialTheme.colorScheme.onSurface
  188. }
  189. Text(
  190. text = update.mangaTitle,
  191. maxLines = 1,
  192. style = MaterialTheme.typography.bodyMedium,
  193. overflow = TextOverflow.Ellipsis,
  194. modifier = Modifier.alpha(textAlpha),
  195. )
  196. Row(verticalAlignment = Alignment.CenterVertically) {
  197. var textHeight by remember { mutableStateOf(0) }
  198. if (bookmark) {
  199. Icon(
  200. imageVector = Icons.Default.Bookmark,
  201. contentDescription = stringResource(R.string.action_filter_bookmarked),
  202. modifier = Modifier
  203. .sizeIn(maxHeight = with(LocalDensity.current) { textHeight.toDp() - 2.dp }),
  204. tint = MaterialTheme.colorScheme.primary,
  205. )
  206. Spacer(modifier = Modifier.width(2.dp))
  207. }
  208. Text(
  209. text = update.chapterName,
  210. maxLines = 1,
  211. style = MaterialTheme.typography.bodySmall
  212. .copy(color = secondaryTextColor),
  213. overflow = TextOverflow.Ellipsis,
  214. onTextLayout = { textHeight = it.size.height },
  215. modifier = Modifier.alpha(textAlpha),
  216. )
  217. }
  218. }
  219. ChapterDownloadIndicator(
  220. modifier = Modifier.padding(start = 4.dp),
  221. downloadStateProvider = downloadStateProvider,
  222. downloadProgressProvider = downloadProgressProvider,
  223. onClick = onDownloadChapter,
  224. )
  225. }
  226. }