HorizontalPager
Common
Component in Compose Foundation
A Pager that scrolls horizontally. Pages are lazily placed in accordance to the available viewport size. By definition, pages in a [Pager] have the same size, defined by [pageSize] and use a snap animation (provided by [flingBehavior] to scroll pages into a specific position). You can use [beyondViewportPageCount] to place more pages before and after the visible pages.
If you need snapping with pages of different size, you can use a [snapFlingBehavior] with a [SnapLayoutInfoProvider] adapted to a LazyList.
Last updated:
Installation
dependencies {
implementation("androidx.compose.foundation:foundation:1.8.0-alpha01")
}
Overloads
@Composable
fun HorizontalPager(
state: PagerState,
modifier: Modifier = Modifier,
contentPadding: PaddingValues = PaddingValues(0.dp),
pageSize: PageSize = PageSize.Fill,
beyondViewportPageCount: Int = PagerDefaults.BeyondViewportPageCount,
pageSpacing: Dp = 0.dp,
verticalAlignment: Alignment.Vertical = Alignment.CenterVertically,
flingBehavior: TargetedFlingBehavior = PagerDefaults.flingBehavior(state = state),
userScrollEnabled: Boolean = true,
reverseLayout: Boolean = false,
key: ((index: Int) -> Any)? = null,
pageNestedScrollConnection: NestedScrollConnection =
PagerDefaults.pageNestedScrollConnection(state, Orientation.Horizontal),
snapPosition: SnapPosition = SnapPosition.Start,
pageContent: @Composable PagerScope.(page: Int) -> Unit
)
Parameters
name | description |
---|---|
state | The state to control this pager |
modifier | A modifier instance to be applied to this Pager outer layout |
contentPadding | a padding around the whole content. This will add padding for the content after it has been clipped, which is not possible via [modifier] param. You can use it to add a padding before the first page or after the last one. Use [pageSpacing] to add spacing between the pages. |
pageSize | Use this to change how the pages will look like inside this pager. |
beyondViewportPageCount | Pages to compose and layout before and after the list of visible pages. Note: Be aware that using a large value for [beyondViewportPageCount] will cause a lot of pages to be composed, measured and placed which will defeat the purpose of using lazy loading. This should be used as an optimization to pre-load a couple of pages before and after the visible ones. This does not include the pages automatically composed and laid out by the pre-fetcher in the direction of the scroll during scroll events. |
pageSpacing | The amount of space to be used to separate the pages in this Pager |
verticalAlignment | How pages are aligned vertically in this Pager. |
flingBehavior | The [TargetedFlingBehavior] to be used for post scroll gestures. |
userScrollEnabled | whether the scrolling via the user gestures or accessibility actions is allowed. You can still scroll programmatically using [PagerState.scroll] even when it is disabled. |
reverseLayout | reverse the direction of scrolling and layout. |
key | a stable and unique key representing the item. When you specify the key the scroll position will be maintained based on the key, which means if you add/remove items before the current visible item the item with the given key will be kept as the first visible one. If null is passed the position in the list will represent the key. |
pageNestedScrollConnection | A [NestedScrollConnection] that dictates how this [Pager] behaves with nested lists. The default behavior will see [Pager] to consume all nested deltas. |
snapPosition | The calculation of how this Pager will perform snapping of pages. Use this to provide different settling to different positions in the layout. This is used by [Pager] as a way to calculate [PagerState.currentPage], currentPage is the page closest to the snap position in the layout (e.g. if the snap position is the start of the layout, then currentPage will be the page closest to that). |
pageContent | This Pager's page Composable. |
Code Examples
SimpleHorizontalPagerSample
@Composable
fun SimpleHorizontalPagerSample() {
// Creates a 1-pager/viewport horizontal pager with single page snapping
val state = rememberPagerState { 10 }
HorizontalPager(
state = state,
modifier = Modifier.fillMaxSize(),
) { page ->
Box(
modifier =
Modifier.padding(10.dp).background(Color.Blue).fillMaxWidth().aspectRatio(1f),
contentAlignment = Alignment.Center
) {
Text(text = page.toString(), fontSize = 32.sp)
}
}
}
HorizontalPagerWithScrollableContent
@Composable
fun HorizontalPagerWithScrollableContent() {
// This is a sample using NestedScroll and Pager.
// We use the toolbar offset changing example from
// androidx.compose.ui.samples.NestedScrollConnectionSample
val pagerState = rememberPagerState { 10 }
val toolbarHeight = 48.dp
val toolbarHeightPx = with(LocalDensity.current) { toolbarHeight.roundToPx().toFloat() }
val toolbarOffsetHeightPx = remember { mutableStateOf(0f) }
val nestedScrollConnection = remember {
object : NestedScrollConnection {
override fun onPreScroll(available: Offset, source: NestedScrollSource): Offset {
val delta = available.y
val newOffset = toolbarOffsetHeightPx.value + delta
toolbarOffsetHeightPx.value = newOffset.coerceIn(-toolbarHeightPx, 0f)
return Offset.Zero
}
}
}
Box(modifier = Modifier.fillMaxSize().nestedScroll(nestedScrollConnection)) {
TopAppBar(
modifier =
Modifier.height(toolbarHeight).offset {
IntOffset(x = 0, y = toolbarOffsetHeightPx.value.roundToInt())
},
title = { Text("Toolbar offset is ${toolbarOffsetHeightPx.value}") }
)
val paddingOffset =
toolbarHeight + with(LocalDensity.current) { toolbarOffsetHeightPx.value.toDp() }
HorizontalPager(
modifier = Modifier.fillMaxSize(),
state = pagerState,
contentPadding = PaddingValues(top = paddingOffset)
) {
Column(modifier = Modifier.fillMaxWidth().verticalScroll(rememberScrollState())) {
repeat(20) {
Box(
modifier =
Modifier.fillMaxWidth()
.height(64.dp)
.padding(4.dp)
.background(if (it % 2 == 0) Color.Black else Color.Yellow),
contentAlignment = Alignment.Center
) {
Text(
text = it.toString(),
color = if (it % 2 != 0) Color.Black else Color.Yellow
)
}
}
}
}
}
}