MangaBottomActionMenu.kt 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. package eu.kanade.presentation.components
  2. import androidx.compose.animation.AnimatedVisibility
  3. import androidx.compose.animation.core.animateFloatAsState
  4. import androidx.compose.animation.expandVertically
  5. import androidx.compose.animation.fadeIn
  6. import androidx.compose.animation.fadeOut
  7. import androidx.compose.animation.shrinkVertically
  8. import androidx.compose.foundation.combinedClickable
  9. import androidx.compose.foundation.interaction.MutableInteractionSource
  10. import androidx.compose.foundation.layout.Arrangement
  11. import androidx.compose.foundation.layout.Column
  12. import androidx.compose.foundation.layout.Row
  13. import androidx.compose.foundation.layout.RowScope
  14. import androidx.compose.foundation.layout.WindowInsets
  15. import androidx.compose.foundation.layout.WindowInsetsSides
  16. import androidx.compose.foundation.layout.asPaddingValues
  17. import androidx.compose.foundation.layout.navigationBars
  18. import androidx.compose.foundation.layout.navigationBarsPadding
  19. import androidx.compose.foundation.layout.only
  20. import androidx.compose.foundation.layout.padding
  21. import androidx.compose.foundation.layout.size
  22. import androidx.compose.foundation.shape.ZeroCornerSize
  23. import androidx.compose.material.icons.Icons
  24. import androidx.compose.material.icons.filled.BookmarkAdd
  25. import androidx.compose.material.icons.filled.BookmarkRemove
  26. import androidx.compose.material.icons.filled.DoneAll
  27. import androidx.compose.material.icons.filled.RemoveDone
  28. import androidx.compose.material.icons.outlined.Delete
  29. import androidx.compose.material.icons.outlined.Download
  30. import androidx.compose.material.icons.outlined.Label
  31. import androidx.compose.material.ripple.rememberRipple
  32. import androidx.compose.material3.Icon
  33. import androidx.compose.material3.MaterialTheme
  34. import androidx.compose.material3.Surface
  35. import androidx.compose.material3.Text
  36. import androidx.compose.runtime.Composable
  37. import androidx.compose.runtime.getValue
  38. import androidx.compose.runtime.mutableStateListOf
  39. import androidx.compose.runtime.remember
  40. import androidx.compose.runtime.rememberCoroutineScope
  41. import androidx.compose.ui.Alignment
  42. import androidx.compose.ui.Modifier
  43. import androidx.compose.ui.graphics.vector.ImageVector
  44. import androidx.compose.ui.hapticfeedback.HapticFeedbackType
  45. import androidx.compose.ui.platform.LocalHapticFeedback
  46. import androidx.compose.ui.res.stringResource
  47. import androidx.compose.ui.res.vectorResource
  48. import androidx.compose.ui.text.style.TextOverflow
  49. import androidx.compose.ui.unit.dp
  50. import eu.kanade.tachiyomi.R
  51. import kotlinx.coroutines.Job
  52. import kotlinx.coroutines.delay
  53. import kotlinx.coroutines.isActive
  54. import kotlinx.coroutines.launch
  55. @Composable
  56. fun MangaBottomActionMenu(
  57. visible: Boolean,
  58. modifier: Modifier = Modifier,
  59. onBookmarkClicked: (() -> Unit)? = null,
  60. onRemoveBookmarkClicked: (() -> Unit)? = null,
  61. onMarkAsReadClicked: (() -> Unit)? = null,
  62. onMarkAsUnreadClicked: (() -> Unit)? = null,
  63. onMarkPreviousAsReadClicked: (() -> Unit)? = null,
  64. onDownloadClicked: (() -> Unit)? = null,
  65. onDeleteClicked: (() -> Unit)? = null,
  66. ) {
  67. AnimatedVisibility(
  68. visible = visible,
  69. enter = expandVertically(expandFrom = Alignment.Bottom),
  70. exit = shrinkVertically(shrinkTowards = Alignment.Bottom),
  71. ) {
  72. val scope = rememberCoroutineScope()
  73. Surface(
  74. modifier = modifier,
  75. shape = MaterialTheme.shapes.large.copy(bottomEnd = ZeroCornerSize, bottomStart = ZeroCornerSize),
  76. tonalElevation = 3.dp,
  77. ) {
  78. val haptic = LocalHapticFeedback.current
  79. val confirm = remember { mutableStateListOf(false, false, false, false, false, false, false) }
  80. var resetJob: Job? = remember { null }
  81. val onLongClickItem: (Int) -> Unit = { toConfirmIndex ->
  82. haptic.performHapticFeedback(HapticFeedbackType.LongPress)
  83. (0 until 7).forEach { i -> confirm[i] = i == toConfirmIndex }
  84. resetJob?.cancel()
  85. resetJob = scope.launch {
  86. delay(1000)
  87. if (isActive) confirm[toConfirmIndex] = false
  88. }
  89. }
  90. Row(
  91. modifier = Modifier
  92. .padding(WindowInsets.navigationBars.only(WindowInsetsSides.Bottom).asPaddingValues())
  93. .padding(horizontal = 8.dp, vertical = 12.dp),
  94. ) {
  95. if (onBookmarkClicked != null) {
  96. Button(
  97. title = stringResource(R.string.action_bookmark),
  98. icon = Icons.Default.BookmarkAdd,
  99. toConfirm = confirm[0],
  100. onLongClick = { onLongClickItem(0) },
  101. onClick = onBookmarkClicked,
  102. )
  103. }
  104. if (onRemoveBookmarkClicked != null) {
  105. Button(
  106. title = stringResource(R.string.action_remove_bookmark),
  107. icon = Icons.Default.BookmarkRemove,
  108. toConfirm = confirm[1],
  109. onLongClick = { onLongClickItem(1) },
  110. onClick = onRemoveBookmarkClicked,
  111. )
  112. }
  113. if (onMarkAsReadClicked != null) {
  114. Button(
  115. title = stringResource(R.string.action_mark_as_read),
  116. icon = Icons.Default.DoneAll,
  117. toConfirm = confirm[2],
  118. onLongClick = { onLongClickItem(2) },
  119. onClick = onMarkAsReadClicked,
  120. )
  121. }
  122. if (onMarkAsUnreadClicked != null) {
  123. Button(
  124. title = stringResource(R.string.action_mark_as_unread),
  125. icon = Icons.Default.RemoveDone,
  126. toConfirm = confirm[3],
  127. onLongClick = { onLongClickItem(3) },
  128. onClick = onMarkAsUnreadClicked,
  129. )
  130. }
  131. if (onMarkPreviousAsReadClicked != null) {
  132. Button(
  133. title = stringResource(R.string.action_mark_previous_as_read),
  134. icon = ImageVector.vectorResource(id = R.drawable.ic_done_prev_24dp),
  135. toConfirm = confirm[4],
  136. onLongClick = { onLongClickItem(4) },
  137. onClick = onMarkPreviousAsReadClicked,
  138. )
  139. }
  140. if (onDownloadClicked != null) {
  141. Button(
  142. title = stringResource(R.string.action_download),
  143. icon = Icons.Outlined.Download,
  144. toConfirm = confirm[5],
  145. onLongClick = { onLongClickItem(5) },
  146. onClick = onDownloadClicked,
  147. )
  148. }
  149. if (onDeleteClicked != null) {
  150. Button(
  151. title = stringResource(R.string.action_delete),
  152. icon = Icons.Outlined.Delete,
  153. toConfirm = confirm[6],
  154. onLongClick = { onLongClickItem(6) },
  155. onClick = onDeleteClicked,
  156. )
  157. }
  158. }
  159. }
  160. }
  161. }
  162. @Composable
  163. private fun RowScope.Button(
  164. title: String,
  165. icon: ImageVector,
  166. toConfirm: Boolean,
  167. onLongClick: () -> Unit,
  168. onClick: () -> Unit,
  169. ) {
  170. val animatedWeight by animateFloatAsState(if (toConfirm) 2f else 1f)
  171. Column(
  172. modifier = Modifier
  173. .size(48.dp)
  174. .weight(animatedWeight)
  175. .combinedClickable(
  176. interactionSource = remember { MutableInteractionSource() },
  177. indication = rememberRipple(bounded = false),
  178. onLongClick = onLongClick,
  179. onClick = onClick,
  180. ),
  181. verticalArrangement = Arrangement.Center,
  182. horizontalAlignment = Alignment.CenterHorizontally,
  183. ) {
  184. Icon(
  185. imageVector = icon,
  186. contentDescription = title,
  187. )
  188. AnimatedVisibility(
  189. visible = toConfirm,
  190. enter = expandVertically(expandFrom = Alignment.Top) + fadeIn(),
  191. exit = shrinkVertically(shrinkTowards = Alignment.Top) + fadeOut(),
  192. ) {
  193. Text(
  194. text = title,
  195. overflow = TextOverflow.Visible,
  196. maxLines = 1,
  197. style = MaterialTheme.typography.labelSmall,
  198. )
  199. }
  200. }
  201. }
  202. @Composable
  203. fun LibraryBottomActionMenu(
  204. visible: Boolean,
  205. modifier: Modifier = Modifier,
  206. onChangeCategoryClicked: (() -> Unit)?,
  207. onMarkAsReadClicked: (() -> Unit)?,
  208. onMarkAsUnreadClicked: (() -> Unit)?,
  209. onDownloadClicked: (() -> Unit)?,
  210. onDeleteClicked: (() -> Unit)?,
  211. ) {
  212. AnimatedVisibility(
  213. visible = visible,
  214. enter = expandVertically(expandFrom = Alignment.Bottom),
  215. exit = shrinkVertically(shrinkTowards = Alignment.Bottom),
  216. ) {
  217. val scope = rememberCoroutineScope()
  218. Surface(
  219. modifier = modifier,
  220. shape = MaterialTheme.shapes.large.copy(bottomEnd = ZeroCornerSize, bottomStart = ZeroCornerSize),
  221. tonalElevation = 3.dp,
  222. ) {
  223. val haptic = LocalHapticFeedback.current
  224. val confirm = remember { mutableStateListOf(false, false, false, false, false) }
  225. var resetJob: Job? = remember { null }
  226. val onLongClickItem: (Int) -> Unit = { toConfirmIndex ->
  227. haptic.performHapticFeedback(HapticFeedbackType.LongPress)
  228. (0 until 5).forEach { i -> confirm[i] = i == toConfirmIndex }
  229. resetJob?.cancel()
  230. resetJob = scope.launch {
  231. delay(1000)
  232. if (isActive) confirm[toConfirmIndex] = false
  233. }
  234. }
  235. Row(
  236. modifier = Modifier
  237. .navigationBarsPadding()
  238. .padding(horizontal = 8.dp, vertical = 12.dp),
  239. ) {
  240. if (onChangeCategoryClicked != null) {
  241. Button(
  242. title = stringResource(R.string.action_move_category),
  243. icon = Icons.Outlined.Label,
  244. toConfirm = confirm[0],
  245. onLongClick = { onLongClickItem(0) },
  246. onClick = onChangeCategoryClicked,
  247. )
  248. }
  249. if (onMarkAsReadClicked != null) {
  250. Button(
  251. title = stringResource(R.string.action_mark_as_read),
  252. icon = Icons.Default.DoneAll,
  253. toConfirm = confirm[1],
  254. onLongClick = { onLongClickItem(1) },
  255. onClick = onMarkAsReadClicked,
  256. )
  257. }
  258. if (onMarkAsUnreadClicked != null) {
  259. Button(
  260. title = stringResource(R.string.action_mark_as_unread),
  261. icon = Icons.Default.RemoveDone,
  262. toConfirm = confirm[2],
  263. onLongClick = { onLongClickItem(2) },
  264. onClick = onMarkAsUnreadClicked,
  265. )
  266. }
  267. if (onDownloadClicked != null) {
  268. Button(
  269. title = stringResource(R.string.action_download),
  270. icon = Icons.Outlined.Download,
  271. toConfirm = confirm[3],
  272. onLongClick = { onLongClickItem(3) },
  273. onClick = onDownloadClicked,
  274. )
  275. }
  276. if (onDeleteClicked != null) {
  277. Button(
  278. title = stringResource(R.string.action_delete),
  279. icon = Icons.Outlined.Delete,
  280. toConfirm = confirm[4],
  281. onLongClick = { onLongClickItem(4) },
  282. onClick = onDeleteClicked,
  283. )
  284. }
  285. }
  286. }
  287. }
  288. }