Scrollbar.kt 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. package eu.kanade.presentation.util
  2. /*
  3. * MIT License
  4. *
  5. * Copyright (c) 2022 Albert Chang
  6. *
  7. * Permission is hereby granted, free of charge, to any person obtaining a copy
  8. * of this software and associated documentation files (the "Software"), to deal
  9. * in the Software without restriction, including without limitation the rights
  10. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  11. * copies of the Software, and to permit persons to whom the Software is
  12. * furnished to do so, subject to the following conditions:
  13. *
  14. * The above copyright notice and this permission notice shall be included in all
  15. * copies or substantial portions of the Software.
  16. *
  17. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  18. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  19. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  20. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  21. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  22. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  23. * SOFTWARE.
  24. */
  25. /**
  26. * Code taken from https://gist.github.com/mxalbert1996/33a360fcab2105a31e5355af98216f5a
  27. * with some modifications to handle contentPadding.
  28. *
  29. * Modifiers for regular scrollable list is omitted.
  30. */
  31. import android.view.ViewConfiguration
  32. import androidx.compose.animation.core.Animatable
  33. import androidx.compose.animation.core.tween
  34. import androidx.compose.foundation.gestures.Orientation
  35. import androidx.compose.foundation.layout.fillMaxWidth
  36. import androidx.compose.foundation.layout.padding
  37. import androidx.compose.foundation.lazy.LazyColumn
  38. import androidx.compose.foundation.lazy.LazyListState
  39. import androidx.compose.foundation.lazy.LazyRow
  40. import androidx.compose.foundation.lazy.rememberLazyListState
  41. import androidx.compose.material3.MaterialTheme
  42. import androidx.compose.material3.Text
  43. import androidx.compose.runtime.Composable
  44. import androidx.compose.runtime.LaunchedEffect
  45. import androidx.compose.runtime.remember
  46. import androidx.compose.ui.Modifier
  47. import androidx.compose.ui.composed
  48. import androidx.compose.ui.draw.drawWithContent
  49. import androidx.compose.ui.geometry.Offset
  50. import androidx.compose.ui.geometry.Size
  51. import androidx.compose.ui.graphics.Color
  52. import androidx.compose.ui.graphics.drawscope.ContentDrawScope
  53. import androidx.compose.ui.graphics.drawscope.DrawScope
  54. import androidx.compose.ui.input.nestedscroll.NestedScrollConnection
  55. import androidx.compose.ui.input.nestedscroll.NestedScrollSource
  56. import androidx.compose.ui.input.nestedscroll.nestedScroll
  57. import androidx.compose.ui.platform.LocalContext
  58. import androidx.compose.ui.platform.LocalLayoutDirection
  59. import androidx.compose.ui.tooling.preview.Preview
  60. import androidx.compose.ui.unit.LayoutDirection
  61. import androidx.compose.ui.unit.dp
  62. import androidx.compose.ui.util.fastFirstOrNull
  63. import androidx.compose.ui.util.fastSumBy
  64. import eu.kanade.presentation.components.Scroller.STICKY_HEADER_KEY_PREFIX
  65. import kotlinx.coroutines.channels.BufferOverflow
  66. import kotlinx.coroutines.flow.MutableSharedFlow
  67. import kotlinx.coroutines.flow.collectLatest
  68. /**
  69. * Draws horizontal scrollbar to a LazyList.
  70. *
  71. * Set key with [STICKY_HEADER_KEY_PREFIX] prefix to any sticky header item in the list.
  72. */
  73. fun Modifier.drawHorizontalScrollbar(
  74. state: LazyListState,
  75. reverseScrolling: Boolean = false,
  76. // The amount of offset the scrollbar position towards the top of the layout
  77. positionOffsetPx: Float = 0f,
  78. ): Modifier = drawScrollbar(state, Orientation.Horizontal, reverseScrolling, positionOffsetPx)
  79. /**
  80. * Draws vertical scrollbar to a LazyList.
  81. *
  82. * Set key with [STICKY_HEADER_KEY_PREFIX] prefix to any sticky header item in the list.
  83. */
  84. fun Modifier.drawVerticalScrollbar(
  85. state: LazyListState,
  86. reverseScrolling: Boolean = false,
  87. // The amount of offset the scrollbar position towards the start of the layout
  88. positionOffsetPx: Float = 0f,
  89. ): Modifier = drawScrollbar(state, Orientation.Vertical, reverseScrolling, positionOffsetPx)
  90. private fun Modifier.drawScrollbar(
  91. state: LazyListState,
  92. orientation: Orientation,
  93. reverseScrolling: Boolean,
  94. positionOffset: Float,
  95. ): Modifier = drawScrollbar(
  96. orientation,
  97. reverseScrolling,
  98. ) { reverseDirection, atEnd, thickness, color, alpha ->
  99. val layoutInfo = state.layoutInfo
  100. val viewportSize = if (orientation == Orientation.Horizontal) {
  101. layoutInfo.viewportSize.width
  102. } else {
  103. layoutInfo.viewportSize.height
  104. } - layoutInfo.beforeContentPadding - layoutInfo.afterContentPadding
  105. val items = layoutInfo.visibleItemsInfo
  106. val itemsSize = items.fastSumBy { it.size }
  107. val showScrollbar = items.size < layoutInfo.totalItemsCount || itemsSize > viewportSize
  108. val estimatedItemSize = if (items.isEmpty()) 0f else itemsSize.toFloat() / items.size
  109. val totalSize = estimatedItemSize * layoutInfo.totalItemsCount
  110. val thumbSize = viewportSize / totalSize * viewportSize
  111. val startOffset = if (items.isEmpty()) {
  112. 0f
  113. } else {
  114. items
  115. .fastFirstOrNull { (it.key as? String)?.startsWith(STICKY_HEADER_KEY_PREFIX)?.not() ?: true }!!
  116. .run {
  117. val startPadding = if (reverseDirection) layoutInfo.afterContentPadding else layoutInfo.beforeContentPadding
  118. startPadding + ((estimatedItemSize * index - offset) / totalSize * viewportSize)
  119. }
  120. }
  121. val drawScrollbar = onDrawScrollbar(
  122. orientation, reverseDirection, atEnd, showScrollbar,
  123. thickness, color, alpha, thumbSize, startOffset, positionOffset,
  124. )
  125. drawContent()
  126. drawScrollbar()
  127. }
  128. private fun ContentDrawScope.onDrawScrollbar(
  129. orientation: Orientation,
  130. reverseDirection: Boolean,
  131. atEnd: Boolean,
  132. showScrollbar: Boolean,
  133. thickness: Float,
  134. color: Color,
  135. alpha: () -> Float,
  136. thumbSize: Float,
  137. scrollOffset: Float,
  138. positionOffset: Float,
  139. ): DrawScope.() -> Unit {
  140. val topLeft = if (orientation == Orientation.Horizontal) {
  141. Offset(
  142. if (reverseDirection) size.width - scrollOffset - thumbSize else scrollOffset,
  143. if (atEnd) size.height - positionOffset - thickness else positionOffset,
  144. )
  145. } else {
  146. Offset(
  147. if (atEnd) size.width - positionOffset - thickness else positionOffset,
  148. if (reverseDirection) size.height - scrollOffset - thumbSize else scrollOffset,
  149. )
  150. }
  151. val size = if (orientation == Orientation.Horizontal) {
  152. Size(thumbSize, thickness)
  153. } else {
  154. Size(thickness, thumbSize)
  155. }
  156. return {
  157. if (showScrollbar) {
  158. drawRect(
  159. color = color,
  160. topLeft = topLeft,
  161. size = size,
  162. alpha = alpha(),
  163. )
  164. }
  165. }
  166. }
  167. private fun Modifier.drawScrollbar(
  168. orientation: Orientation,
  169. reverseScrolling: Boolean,
  170. onDraw: ContentDrawScope.(
  171. reverseDirection: Boolean,
  172. atEnd: Boolean,
  173. thickness: Float,
  174. color: Color,
  175. alpha: () -> Float,
  176. ) -> Unit,
  177. ): Modifier = composed {
  178. val scrolled = remember {
  179. MutableSharedFlow<Unit>(
  180. extraBufferCapacity = 1,
  181. onBufferOverflow = BufferOverflow.DROP_OLDEST,
  182. )
  183. }
  184. val nestedScrollConnection = remember(orientation, scrolled) {
  185. object : NestedScrollConnection {
  186. override fun onPostScroll(
  187. consumed: Offset,
  188. available: Offset,
  189. source: NestedScrollSource,
  190. ): Offset {
  191. val delta = if (orientation == Orientation.Horizontal) consumed.x else consumed.y
  192. if (delta != 0f) scrolled.tryEmit(Unit)
  193. return Offset.Zero
  194. }
  195. }
  196. }
  197. val alpha = remember { Animatable(0f) }
  198. LaunchedEffect(scrolled, alpha) {
  199. scrolled.collectLatest {
  200. alpha.snapTo(1f)
  201. alpha.animateTo(0f, animationSpec = FadeOutAnimationSpec)
  202. }
  203. }
  204. val isLtr = LocalLayoutDirection.current == LayoutDirection.Ltr
  205. val reverseDirection = if (orientation == Orientation.Horizontal) {
  206. if (isLtr) reverseScrolling else !reverseScrolling
  207. } else {
  208. reverseScrolling
  209. }
  210. val atEnd = if (orientation == Orientation.Vertical) isLtr else true
  211. val context = LocalContext.current
  212. val thickness = remember { ViewConfiguration.get(context).scaledScrollBarSize.toFloat() }
  213. val color = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.364f)
  214. Modifier
  215. .nestedScroll(nestedScrollConnection)
  216. .drawWithContent {
  217. onDraw(reverseDirection, atEnd, thickness, color, alpha::value)
  218. }
  219. }
  220. private val FadeOutAnimationSpec = tween<Float>(
  221. durationMillis = ViewConfiguration.getScrollBarFadeDuration(),
  222. delayMillis = ViewConfiguration.getScrollDefaultDelay(),
  223. )
  224. @Preview(widthDp = 400, heightDp = 400, showBackground = true)
  225. @Composable
  226. fun LazyListScrollbarPreview() {
  227. val state = rememberLazyListState()
  228. LazyColumn(
  229. modifier = Modifier.drawVerticalScrollbar(state),
  230. state = state,
  231. ) {
  232. items(50) {
  233. Text(
  234. text = "Item ${it + 1}",
  235. modifier = Modifier
  236. .fillMaxWidth()
  237. .padding(16.dp),
  238. )
  239. }
  240. }
  241. }
  242. @Preview(widthDp = 400, showBackground = true)
  243. @Composable
  244. fun LazyListHorizontalScrollbarPreview() {
  245. val state = rememberLazyListState()
  246. LazyRow(
  247. modifier = Modifier.drawHorizontalScrollbar(state),
  248. state = state,
  249. ) {
  250. items(50) {
  251. Text(
  252. text = (it + 1).toString(),
  253. modifier = Modifier
  254. .padding(horizontal = 8.dp, vertical = 16.dp),
  255. )
  256. }
  257. }
  258. }