AppBar.kt 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. package eu.kanade.presentation.components
  2. import androidx.compose.foundation.layout.Column
  3. import androidx.compose.foundation.layout.RowScope
  4. import androidx.compose.foundation.layout.WindowInsets
  5. import androidx.compose.foundation.layout.WindowInsetsSides
  6. import androidx.compose.foundation.layout.only
  7. import androidx.compose.foundation.layout.systemBars
  8. import androidx.compose.foundation.layout.windowInsetsPadding
  9. import androidx.compose.material.icons.Icons
  10. import androidx.compose.material.icons.filled.ArrowBack
  11. import androidx.compose.material.icons.filled.Close
  12. import androidx.compose.material.icons.filled.MoreVert
  13. import androidx.compose.material3.DropdownMenuItem
  14. import androidx.compose.material3.Icon
  15. import androidx.compose.material3.IconButton
  16. import androidx.compose.material3.MaterialTheme
  17. import androidx.compose.material3.SmallTopAppBar
  18. import androidx.compose.material3.Text
  19. import androidx.compose.material3.TopAppBarDefaults
  20. import androidx.compose.runtime.Composable
  21. import androidx.compose.runtime.derivedStateOf
  22. import androidx.compose.runtime.getValue
  23. import androidx.compose.runtime.mutableStateOf
  24. import androidx.compose.runtime.remember
  25. import androidx.compose.runtime.setValue
  26. import androidx.compose.ui.Modifier
  27. import androidx.compose.ui.draw.drawBehind
  28. import androidx.compose.ui.graphics.Color
  29. import androidx.compose.ui.graphics.vector.ImageVector
  30. import androidx.compose.ui.res.stringResource
  31. import androidx.compose.ui.text.font.FontWeight
  32. import androidx.compose.ui.text.style.TextOverflow
  33. import eu.kanade.tachiyomi.R
  34. @Composable
  35. fun AppBar(
  36. modifier: Modifier = Modifier,
  37. // Text
  38. title: String?,
  39. subtitle: String? = null,
  40. // Up button
  41. navigateUp: (() -> Unit)? = null,
  42. navigationIcon: ImageVector = Icons.Default.ArrowBack,
  43. // Menu
  44. actions: @Composable RowScope.() -> Unit = {},
  45. // Action mode
  46. actionModeCounter: Int = 0,
  47. onCancelActionMode: () -> Unit = {},
  48. actionModeActions: @Composable RowScope.() -> Unit = {},
  49. // Banners
  50. downloadedOnlyMode: Boolean = false,
  51. incognitoMode: Boolean = false,
  52. ) {
  53. val isActionMode by derivedStateOf { actionModeCounter > 0 }
  54. val backgroundColor = if (isActionMode) {
  55. TopAppBarDefaults.smallTopAppBarColors().containerColor(1f).value
  56. } else {
  57. MaterialTheme.colorScheme.surface
  58. }
  59. Column(
  60. modifier = modifier.drawBehind { drawRect(backgroundColor) },
  61. ) {
  62. SmallTopAppBar(
  63. modifier = Modifier.windowInsetsPadding(WindowInsets.systemBars.only(WindowInsetsSides.Top)),
  64. navigationIcon = {
  65. if (isActionMode) {
  66. IconButton(onClick = onCancelActionMode) {
  67. Icon(
  68. imageVector = Icons.Default.Close,
  69. contentDescription = stringResource(id = R.string.action_cancel),
  70. )
  71. }
  72. } else {
  73. navigateUp?.let {
  74. IconButton(onClick = it) {
  75. Icon(
  76. imageVector = navigationIcon,
  77. contentDescription = stringResource(R.string.abc_action_bar_up_description),
  78. )
  79. }
  80. }
  81. }
  82. },
  83. title = {
  84. if (isActionMode) {
  85. AppBarTitle(actionModeCounter.toString())
  86. } else {
  87. AppBarTitle(title, subtitle)
  88. }
  89. },
  90. actions = {
  91. if (isActionMode) {
  92. actionModeActions()
  93. } else {
  94. actions()
  95. }
  96. },
  97. // Background handled by parent
  98. colors = TopAppBarDefaults.smallTopAppBarColors(
  99. containerColor = Color.Transparent,
  100. scrolledContainerColor = Color.Transparent,
  101. ),
  102. )
  103. if (downloadedOnlyMode) {
  104. DownloadedOnlyModeBanner()
  105. }
  106. if (incognitoMode) {
  107. IncognitoModeBanner()
  108. }
  109. }
  110. }
  111. @Composable
  112. fun AppBarTitle(
  113. title: String?,
  114. subtitle: String? = null,
  115. ) {
  116. Column {
  117. title?.let {
  118. Text(
  119. text = it,
  120. maxLines = 1,
  121. overflow = TextOverflow.Ellipsis,
  122. )
  123. }
  124. subtitle?.let {
  125. Text(
  126. text = it,
  127. style = MaterialTheme.typography.bodyMedium,
  128. maxLines = 1,
  129. overflow = TextOverflow.Ellipsis,
  130. )
  131. }
  132. }
  133. }
  134. @Composable
  135. fun AppBarActions(
  136. actions: List<AppBar.AppBarAction>,
  137. ) {
  138. var showMenu by remember { mutableStateOf(false) }
  139. actions.filterIsInstance<AppBar.Action>().map {
  140. IconButton(
  141. onClick = it.onClick,
  142. enabled = it.enabled,
  143. ) {
  144. Icon(
  145. imageVector = it.icon,
  146. contentDescription = it.title,
  147. )
  148. }
  149. }
  150. val overflowActions = actions.filterIsInstance<AppBar.OverflowAction>()
  151. if (overflowActions.isNotEmpty()) {
  152. IconButton(onClick = { showMenu = !showMenu }) {
  153. Icon(Icons.Default.MoreVert, contentDescription = stringResource(R.string.label_more))
  154. }
  155. DropdownMenu(
  156. expanded = showMenu,
  157. onDismissRequest = { showMenu = false },
  158. ) {
  159. overflowActions.map {
  160. DropdownMenuItem(
  161. onClick = {
  162. it.onClick()
  163. showMenu = false
  164. },
  165. text = { Text(it.title, fontWeight = FontWeight.Normal) },
  166. )
  167. }
  168. }
  169. }
  170. }
  171. sealed interface AppBar {
  172. sealed interface AppBarAction
  173. data class Action(
  174. val title: String,
  175. val icon: ImageVector,
  176. val onClick: () -> Unit,
  177. val enabled: Boolean = true,
  178. ) : AppBarAction
  179. data class OverflowAction(
  180. val title: String,
  181. val onClick: () -> Unit,
  182. ) : AppBarAction
  183. }