Compose笔记(六十九)--Pager

这一节主要了解一下Compose中的Pager,在Jetpack Compose开发中,Pager是用于实现滑动页面集合的核心组件,支持水平(HorizontalPager)和垂直(VerticalPager)方向的滑动,能够灵活管理页面内容、状态和交互。简单总结:

API:

HorizontalPager/VerticalPager:核心组件,分别实现横向 / 纵向翻页

rememberPagerState:创建并记忆 Pager 状态,包含当前页码、滑动偏移等

PagerScope:页面作用域,提供 pageOffset 等页面滑动状态参数

PageSize:定义页面尺寸

PagerSnapDistance:定义滑动后吸附到页面的距离

场景:

1 图片轮播 在电商、新闻等应用中,顶部展示多张图片并自动轮播,用户可手动滑动切换。

2 引导页 应用首次启动时展示多页引导内容,用户滑动浏览后进入主界面。

3 分类标签页 顶部标签栏切换时,下方内容同步滑动

栗子:

Kotlin 复制代码
implementation("io.coil-kt:coil-compose:2.5.0")
Kotlin 复制代码
// 权限   
<uses-permission android:name="android.permission.INTERNET" />
Kotlin 复制代码
import androidx.compose.runtime.Composable
import androidx.compose.animation.core.animateFloatAsState
import androidx.compose.foundation.Image
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.pager.HorizontalPager
import androidx.compose.foundation.pager.PageSize
import androidx.compose.foundation.pager.rememberPagerState
import androidx.compose.foundation.shape.CircleShape
import androidx.compose.material3.Button
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.alpha
import androidx.compose.ui.draw.scale
import androidx.compose.ui.layout.ContentScale
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import coil.compose.rememberAsyncImagePainter
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch

data class PagerItem(
    val id: Int,
    val title: String,
    val imageUrl: String,
    val bgColor: androidx.compose.ui.graphics.Color
)
fun Float.format(digits: Int): String = "%.${digits}f".format(this)

@Composable
fun PagerDemo() {

     val pagerData = listOf(
        PagerItem(1, "商品A", "https://picsum.photos/800/400?random=1", androidx.compose.ui.graphics.Color(0xFF6495ED)),
        PagerItem(2, "商品B", "https://picsum.photos/800/400?random=2", androidx.compose.ui.graphics.Color(0xFF90EE90)),
        PagerItem(3, "商品C", "https://picsum.photos/800/400?random=3", androidx.compose.ui.graphics.Color(0xFFFFB6C1)),
        PagerItem(4, "商品D", "https://picsum.photos/800/400?random=4", androidx.compose.ui.graphics.Color(0xFFFFD700))
    )


    Surface(modifier = Modifier.fillMaxSize()) {
        val pagerState = rememberPagerState(
            initialPage = 0,
            pageCount = { pagerData.size }
        )
        val coroutineScope = rememberCoroutineScope()

        var slideOffset by remember { mutableStateOf(0.0f) }
        LaunchedEffect(pagerState.currentPage, pagerState.currentPageOffsetFraction) {
            slideOffset = pagerState.currentPage + pagerState.currentPageOffsetFraction
        }

        Column(
            modifier = Modifier.fillMaxSize(),
            horizontalAlignment = Alignment.CenterHorizontally
        ) {
            Text(
                text = "滑动进度:${slideOffset.format(2)}",
                fontSize = 16.sp,
                modifier = Modifier.padding(16.dp)
            )

            HorizontalPager(
                state = pagerState,
                modifier = Modifier
                    .weight(1f)
                    .padding(horizontal = 20.dp),
                pageSize = PageSize.Fill,
                pageSpacing = 16.dp,
                contentPadding = androidx.compose.foundation.layout.PaddingValues(horizontal = 20.dp)
            ) { pageIndex ->
                val item = pagerData[pageIndex]
                val pageOffset = pagerState.currentPageOffsetFraction
                val scale by animateFloatAsState(
                    targetValue = if (pageIndex == pagerState.currentPage) 1f else 0.9f,
                    label = "pageScale"
                )
                val alpha by animateFloatAsState(
                    targetValue = if (pageIndex == pagerState.currentPage) 1f else 0.7f,
                    label = "pageAlpha"
                )

                var isLoading by remember { mutableStateOf(true) }
                LaunchedEffect(pageIndex) {
                    delay(300)
                    isLoading = false
                }

                Box(
                    modifier = Modifier
                        .fillMaxSize()
                        .scale(scale)
                        .alpha(alpha)
                        .background(item.bgColor, MaterialTheme.shapes.medium)
                        .padding(16.dp),
                    contentAlignment = Alignment.Center
                ) {
                    if (isLoading) {
                        Text(text = "加载中...", color = androidx.compose.ui.graphics.Color.White)
                    } else {
                        Column(
                            horizontalAlignment = Alignment.CenterHorizontally
                        ) {
                            Image(
                                painter = rememberAsyncImagePainter(model = item.imageUrl),
                                contentDescription = item.title,
                                modifier = Modifier
                                    .size(200.dp, 150.dp)
                                    .padding(bottom = 16.dp),
                                contentScale = ContentScale.Crop
                            )
                            Text(
                                text = item.title,
                                fontSize = 20.sp,
                                color = androidx.compose.ui.graphics.Color.White
                            )
                        }
                    }
                }
            }

            Column(
                modifier = Modifier.padding(16.dp),
                horizontalAlignment = Alignment.CenterHorizontally
            ) {
                Row(
                    horizontalArrangement = Arrangement.spacedBy(8.dp),
                    modifier = Modifier.padding(bottom = 16.dp)
                ) {
                    pagerData.forEachIndexed { index, _ ->
                        val isSelected = index == pagerState.currentPage
                        Box(
                            modifier = Modifier
                                .size(if (isSelected) 12.dp else 8.dp)
                                .background(
                                    color = if (isSelected) androidx.compose.ui.graphics.Color.Blue else androidx.compose.ui.graphics.Color.Gray,
                                    shape = CircleShape
                                )
                        )
                    }
                }

                Row(
                    horizontalArrangement = Arrangement.spacedBy(16.dp)
                ) {
                    Button(
                        onClick = {
                            coroutineScope.launch {
                      
                                if (pagerState.currentPage > 0) {
                                    pagerState.animateScrollToPage(pagerState.currentPage - 1)
                                }
                            }
                        },
                        enabled = pagerState.currentPage > 0
                    ) {
                        Text(text = "上一页")
                    }

                    Button(
                        onClick = {
                            coroutineScope.launch {
 
                                if (pagerState.currentPage < pagerData.size - 1) {
                                    pagerState.animateScrollToPage(pagerState.currentPage + 1)
                                }
                            }
                        },
                        enabled = pagerState.currentPage < pagerData.size - 1
                    ) {
                        Text(text = "下一页")
                    }
                }
            }
        }
    }

}
Kotlin 复制代码
import androidx.compose.foundation.Image
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.pager.VerticalPager
import androidx.compose.foundation.pager.rememberPagerState
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.layout.ContentScale
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import coil.compose.rememberAsyncImagePainter

@Composable
fun PagerDemo() {
    val imageUrls = listOf(
        "https://picsum.photos/800/1200?random=1",
        "https://picsum.photos/800/1200?random=2",
        "https://picsum.photos/800/1200?random=3"
    )

    val pagerState = rememberPagerState(
        initialPage = 0,
        pageCount = { imageUrls.size }
    )
    Surface(modifier = Modifier.fillMaxSize()) {

        VerticalPager(
            state = pagerState,
            modifier = Modifier.fillMaxSize(),
            pageSize = androidx.compose.foundation.pager.PageSize.Fill
        ) { pageIndex ->
            Box(
                modifier = Modifier.fillMaxSize(),
                contentAlignment = Alignment.Center
            ) {
                Image(
                    painter = rememberAsyncImagePainter(model = imageUrls[pageIndex]),
                    contentDescription = "商品图片${pageIndex + 1}",
                    modifier = Modifier
                        .fillMaxSize()
                        .padding(20.dp),
                    contentScale = ContentScale.Crop
                )
                Text(
                    text = "${pageIndex + 1}/${imageUrls.size}",
                    fontSize = 20.sp,
                    color = Color.White,
                    modifier = Modifier
                        .background(Color.Black.copy(alpha = 0.5f))
                        .padding(8.dp)
                        .align(Alignment.BottomCenter)
                )
            }
        }
    }

}

注意:

1 页面预加载与优化 默认仅加载可见页面,通过beyondViewportPageCount设置屏幕外预加载页面数

2 状态管理与滚动监听 使用LaunchedEffect(pagerState.currentPage)监听页面变化,避免在pageContent中直接监听导致重复重组。

3 滚动控制 滚动需在协程中调用

相关推荐
shandianchengzi6 小时前
【小白向】错位排列|图文解释公考常见题目错位排列的递推式Dn=(n-1)(Dn-2+Dn-1)推导方式
笔记·算法·公考·递推·排列·考公
浅念-6 小时前
C语言编译与链接全流程:从源码到可执行程序的幕后之旅
c语言·开发语言·数据结构·经验分享·笔记·学习·算法
The森6 小时前
Linux IO 模型纵深解析 01:从 Unix 传统到 Linux 内核的 IO 第一性原理
linux·服务器·c语言·经验分享·笔记·unix
tq10867 小时前
Skills 的问题与解决方案
笔记
三水不滴7 小时前
有 HTTP 了为什么还要有 RPC?
经验分享·笔记·网络协议·计算机网络·http·rpc
三块可乐两块冰7 小时前
【第二十九周】机器学习笔记三十
笔记
听麟8 小时前
HarmonyOS 6.0+ 跨端智慧政务服务平台开发实战:多端协同办理与电子证照管理落地
笔记·华为·wpf·音视频·harmonyos·政务
risc1234569 小时前
认识一个事物,需要哪些基本能力与要素?
笔记
firewood20249 小时前
共射三极管放大电路相关情况分析
笔记·学习
Hello_Embed9 小时前
libmodbus STM32 主机实验(USB 串口版)
笔记·stm32·学习·嵌入式·freertos·modbus