Android Compose——ScrollableTabRow和LazyColumn同步滑动

Android Compose——ScrollableTabRow和LazyColumn同步滑动

  • 效果
  • 数据
  • 实现
    • Tab
    • List列表
  • 如何同步实现?
    • 监听列表滑动变化
    • 计算列表子项索引位置
    • Tab滑动

效果

Demo简述:此Demo所实现的效果为当滑动List列表时,所对应的Tab相对应进行滑动切换,为了模拟复杂数据环境,通过不同类别的数据进行操作,通过sealed或者enum结合when 进行区分不同的类。具体效果如下视频所示。

数据

下列通过AnimalVegetableFruitVehicle四个不同的类用来模拟数据环境

data class Animal(val content:String)
data class Vegetable(val content:String)
data class Fruit(val content:String)
data class Vehicle(val content:String)data class Type(val animals:List<Animal>,val vegetables:List<Vegetable>,val fruits:List<Fruit>,val vehicles:List<Vehicle>,val categories:List<String>
)

其中categories记录的是所有存在的类别列表标题,其余四个分别为对应的Tab的数据列表

val type = Type(animals = listOf(Animal("子鼠"),Animal("丑牛"),Animal("寅虎"),Animal("卯兔"),Animal("辰龙"),Animal("已蛇"),Animal("午马"),Animal("未羊"),Animal("申猴"),Animal("酉鸡"),Animal("戌狗"),Animal("亥猪")),vegetables = listOf(Vegetable("白萝卜"),Vegetable("红萝卜"),Vegetable("黄萝卜"),Vegetable("绿萝卜"),Vegetable("紫萝卜"),Vegetable("橙萝卜"),Vegetable("黑萝卜"),Vegetable("粉红萝卜"),Vegetable("蓝萝卜"),Vegetable("青萝卜"),Vegetable("灰萝卜"),Vegetable("棕萝卜"),Vegetable("朱砂萝卜"),Vegetable("胭脂萝卜"),),fruits = listOf(Fruit("苹果"),Fruit("香蕉"),Fruit("海棠"),Fruit("樱桃"),Fruit("枇杷"),Fruit("山楂"),Fruit("梨"),Fruit("李子"),Fruit("蓝莓"),Fruit("黑莓"),Fruit("西瓜"),Fruit("火龙果"),Fruit("榴莲"),),vehicles = listOf(Vehicle("飞机"),Vehicle("火箭"),Vehicle("坦克"),Vehicle("共享单车"),Vehicle("汽车"),Vehicle("摩托车"),Vehicle("三轮车"),Vehicle("自行车"),Vehicle("电动车"),Vehicle("高铁"),Vehicle("马车"),Vehicle("驴车"),Vehicle("出租车"),Vehicle("地铁"),),categories = listOf("Animals","Vegetables","Fruits","Vehicles",)
)

实现

lazyListTabSync是一个封装的组合函数,其中传入的参数是一个列表,上述我们建立了四个类别数据,则此参数传入应为mutableListOf(0,1,2,3),与下列所传入的参数效果一致

 val (selectedTabIndex, setSelectedTabIndex, listState) = lazyListTabSync(type.categories.indices.toList())

除了上述用法之外,还可以像下面一样使用,但是所传入的tabsCount个数不能小于种类个数(mutableListOf(0,1,2,3)的个数)

val (selectedTabIndex, setSelectedTabIndex, listState) = tabSyncMediator(mutableListOf(0, 2, 4), tabsCount = 3, lazyListState = rememberLazyListState(), smoothScroll = true, )

Tab

Tab的实现主要在于当前Tab的位置和Tab的点击事件

@Composable
fun MyTabBar(type: Type,selectedTabIndex: Int,onTabClicked: (index: Int, type: String) -> Unit
) {ScrollableTabRow(selectedTabIndex = selectedTabIndex,edgePadding = 0.dp) {type.categories.forEachIndexed { index, category ->Tab(selected = index == selectedTabIndex,onClick = { onTabClicked(index, category) },text = { Text(category) })}}
}

List列表

LazyListState是用来同步滑动状态,下列通过enum对四个类别名称进行封装,然后通过when进行区分,最后在分别实现不同类别的数据效果

@Composable
fun MyLazyList(type: Type,listState: LazyListState = rememberLazyListState(),
) {LazyColumn(state = listState,verticalArrangement = Arrangement.spacedBy(16.dp)) {itemsIndexed(type.categories) { tabIndex, tab ->Column(modifier = Modifier.fillMaxWidth()){Text(text = tab, fontSize = 18.sp)Spacer(modifier = Modifier.height(10.dp))when (tab) {Category.Vegetables.title -> {type.vegetables.forEachIndexed { index, vegetable ->Row(horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth().height(40.dp).background(Color.Gray)) {Text(vegetable.content,textAlign = TextAlign.Center,color = Color.White)}if (index < type.vegetables.size - 1) Spacer(modifier = Modifier.height(10.dp))}}Category.Vehicles.title -> {type.vehicles.forEachIndexed { index, vehicle ->Row(horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth().height(40.dp).background(Color.Gray)) {Text(vehicle.content,textAlign = TextAlign.Center,color = Color.White)}if (index < type.vehicles.size - 1) Spacer(modifier = Modifier.height(10.dp))}}Category.Animals.title -> {type.animals.forEachIndexed { index, animal ->Row(horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth().height(40.dp).background(Color.Gray)) {Text(animal.content,textAlign = TextAlign.Center,color = Color.White)}if (index < type.animals.size - 1) Spacer(modifier = Modifier.height(10.dp))}}Category.Fruits.title -> {type.fruits.forEachIndexed { index, fruit ->Row(horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth().height(40.dp).background(Color.Gray)) {Text(fruit.content,textAlign = TextAlign.Center,color = Color.White)}if (index < type.fruits.size - 1) Spacer(modifier = Modifier.height(10.dp))}}}if (tabIndex < type.categories.size - 1) Spacer(modifier = Modifier.height(20.dp))}}}
}

如何同步实现?

我们需要一种方法来反映一个状态与另一个状态的状态,这意味着无论当前所选索引的值是多少,它都应该反映列表状态中的正确位置,反之亦然,无论列表状态的当前位置是什么,它都反映正确的索引。

监听列表滑动变化

通过LazyListState来了解列表的滑动状态,每次滑动都会都会重新计算列表滑动位置和Tab对应的滑动关系。通过查找第一个完全或部分可见的列表子项以及最后一个完全可见的列表子项来最终确定将所要选择的索引。如果发生变化,则返回变化结果,然后随即变化Tab状态

@Composable
fun lazyListTabSync(syncedIndices: List<Int>,lazyListState: LazyListState = rememberLazyListState(),tabsCount: Int? = null,smoothScroll: Boolean = true
): TabSyncState {require(syncedIndices.isNotEmpty()) {"You can't use the mediator without providing at least one index in the syncedIndices array"}if (tabsCount != null) {require(tabsCount <= syncedIndices.size) {"The tabs count is out of the bounds of the syncedIndices list provided. " +"Either add an index to syncedIndices that corresponds to an item to your lazy list, " +"or remove your excessive tab"}}var selectedTabIndex by remember { mutableStateOf(0) }LaunchedEffect(lazyListState) {snapshotFlow { lazyListState.layoutInfo }.collect {var itemPosition = lazyListState.findFirstFullyVisibleItemIndex()if (itemPosition == -1) {itemPosition = lazyListState.firstVisibleItemIndex}if (itemPosition == -1) {return@collect}if (lazyListState.findLastFullyVisibleItemIndex() == syncedIndices.last()) {itemPosition = syncedIndices.last()}if (syncedIndices.contains(itemPosition) && itemPosition != syncedIndices[selectedTabIndex]) {selectedTabIndex = syncedIndices.indexOf(itemPosition)}}}return TabSyncState(selectedTabIndex,lazyListState,rememberCoroutineScope(),syncedIndices,smoothScroll)
}

计算列表子项索引位置

由上述可知,我们一共建立了四个数据类别,则共有四个子项,每一个子项又是一个列表,此处我们计算的是单个数据类别位置,通过计算其可见子项的偏移量判断是否在对应范围内,从而返回对应的Tab子项下标

fun LazyListState.findFirstFullyVisibleItemIndex(): Int = findFullyVisibleItemIndex(reversed = false)fun LazyListState.findLastFullyVisibleItemIndex(): Int = findFullyVisibleItemIndex(reversed = true)fun LazyListState.findFullyVisibleItemIndex(reversed: Boolean): Int {layoutInfo.visibleItemsInfo.run { if (reversed) reversed() else this }.forEach { itemInfo ->val itemStartOffset = itemInfo.offsetval itemEndOffset = itemInfo.offset + itemInfo.sizeval viewportStartOffset = layoutInfo.viewportStartOffsetval viewportEndOffset = layoutInfo.viewportEndOffsetif (itemStartOffset >= viewportStartOffset && itemEndOffset <= viewportEndOffset) {return itemInfo.index //返回当前滑动列表的子项所属Tab的下标}}return -1
}

Tab滑动

下面定义了三个解构声明语句,其中其中的 component1()component2() component3() 函数是在 Kotlin 中广泛使用的约定原则。下列实现的功能为通过启动一个协程作用域让Tab对应的列表滚动到相应的位置

@Stable
class TabSyncState(var selectedTabIndex: Int,var lazyListState: LazyListState,private var coroutineScope: CoroutineScope,private var syncedIndices: List<Int>,private var smoothScroll: Boolean,
) {operator fun component1(): Int = selectedTabIndexoperator fun component2(): (Int) -> Unit = {require(it <= syncedIndices.size - 1) {"The selected tab's index is out of the bounds of the syncedIndices list provided. " +"Either add an index to syncedIndices that corresponds to an item to your lazy list, " +"or remove your excessive tab"}selectedTabIndex = itcoroutineScope.launch {if (smoothScroll) {lazyListState.animateScrollToItem(syncedIndices[selectedTabIndex])} else {lazyListState.scrollToItem(syncedIndices[selectedTabIndex])}}}operator fun component3(): LazyListState = lazyListState
}

此Demo改编至一个国外博主,其原作者Github链接地址如下所示
原作者Github链接

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/602125.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux 目录结构及其说明

Linux 操作系统遵循一种标准的目录结构&#xff0c;称为 Filesystem Hierarchy Standard&#xff08;文件系统层次结构标准&#xff09;&#xff0c;其定义了不同目录的用途和内容。 浅蓝色文字 /&#xff08;根目录&#xff09;&#xff1a; /根目录是整个文件系统的起点&…

第二百五十回

文章目录 1. 概念介绍2. 使用方法2.1 简单用法2.2 自定义用法 3. 示例代码4. 内容总结 我们在上一章回中介绍了"三方包open_settings"相关的内容&#xff0c;本章回中将介绍另外一个三方包&#xff1a;bluetooth_enable_fork.闲话休提&#xff0c;让我们一起Talk Flu…

力扣2807.在链表中插入最大公约数

思路&#xff1a;遍历链表&#xff0c;对于每一个结点求出它与下一个结点的最大公约数并插入到俩个结点之间 代码&#xff1a; /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullptr) {}…

【数据库系统概念】第7-14章集合

文章目录 第七章 数据库设计和E-R模型&#xff08;重点&#xff01;&#xff01;&#xff01;&#xff09;~~7.1 设计过程概览&#xff08;了解&#xff09;~~7.1.1 设计阶段7.1.2 设计选择 7.2 实体-联系模型&#xff08;重点掌握&#xff09;7.2.1 实体集7.2.2 联系集联系集的…

ubuntu 22.04 安装r-base时缺少r-recommended

sudo apt-get install r-base时报错&#xff1a; 下列软件包有未满足的依赖关系&#xff1a; r-base : 依赖: r-recommended ( 4.3.2-1.2004.0) 但无法安装它 E: 无法修正错误&#xff0c;因为您要求某些软件包保持现状&#xff0c;就是它们破坏了软件包间的依赖关系。 解决方…

BLE Mesh蓝牙组网技术详细解析之Model Layer模型层(八)

目录 一、什么是BLE Mesh Model Layer模型层&#xff1f; 二、SIG Model 2.1 模型概念 2.2 消息格式 2.3 开关模型 四、资料获取 一、什么是BLE Mesh Model Layer模型层&#xff1f; Models Layer的作用是定义了一些通用的或特定的模型&#xff0c;用于实现网络节点设备…

Beauty algorithm(四)眼影

一、skills 前瞻 略 二、目标区域定位 1、 眼影区域 1、眼部关键点 左侧:36,37,38,39,40,41 右侧:42,43,44,45,46,47 2、计算roi区域的w,h,center 目的调整mask的比列。 FaceRegion left_es, right_es; left_es.w = landmarks.at(39).x - landmarks.at(36).x; left_es.…

书生·浦语大模型实战营第一次课堂笔记

书生浦语大模型全链路开源体系。大模型是发展通用人工智能的重要途径,是人工通用人工智能的一个重要途径。书生浦语大模型覆盖轻量级、重量级、重量级的三种不同大小模型,可用于智能客服、个人助手等领域。还介绍了书生浦语大模型的性能在多个数据集上全面超过了相似量级或相近…

并发(3)

目录 11.Synchronized本质上是通过什么保证线程安全的&#xff1f; 12.Synchronized使得同时只有一个线程可以执行&#xff0c;性能比较差&#xff0c;有什么提升的方法&#xff1f; 13.Synchronized由什么样的缺陷&#xff1f;Java Lock是怎么弥补这些缺陷的&#xff1f; 1…

《微信小程序开发从入门到实战》学习七十三

6.7数据缓存API 6.7.2 获取数据API 使用wx.getStorageSync和wx.getStorage接口可从本地缓存读取指定key中的数据。使用方式如下&#xff1a; // 异步接口&#xff0c;可以使用三回调函数 wx.getStorage({ key: key, success(res) { console.log(res.data) // 读取的数据保存到…

牛客网编程题——“求IBSN码”

这是我今天在牛客网上面刷题看到的一道相对而言比较有价值的题&#xff0c;个人非常的喜欢昂&#xff08;因为我没有做起...&#xff09;&#xff0c;先看题目&#xff1a; 每一本正式出版的图书都有一个ISBN号码与之对应&#xff0c;ISBN码包括9位数字、1位识别码和3位分隔符&…

看了致远OA的表单设计后的思考

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a; https://gitee.com/nbacheng/n…

通信原理期末复习——计算大题(一)

个人名片&#xff1a; &#x1f981;作者简介&#xff1a;一名喜欢分享和记录学习的在校大学生 &#x1f42f;个人主页&#xff1a;妄北y &#x1f427;个人QQ&#xff1a;2061314755 &#x1f43b;个人邮箱&#xff1a;2061314755qq.com &#x1f989;个人WeChat&#xff1a;V…

Linux操作系统基础(13):文件管理-文件目录命令

1. 文件操作 1.1. 创建文件和目录 touch命令用于创建新的空文件或更新现有文件的时间戳。 #1.创建一个test.txt文件 touch test.txt#2.创建多个文件&#xff08;test1.txt、test2.txt、test3.txt&#xff09; touch test{1..3}.txtmkdir命令用于创建一个新的目录&#xff08…

最小覆盖子串【子串】【滑动窗口】【哈希】

Problem: 76. 最小覆盖子串 文章目录 思路 & 解题方法复杂度Code 思路 & 解题方法 窗口左右边界为i和j&#xff0c;初始值都为0&#xff0c;j一直往右搜索&#xff0c;然后记录一下窗口内的字符是否达到了全部覆盖&#xff0c;如果达到了&#xff0c;那么就开始i往右搜…

交换机04_远程连接

通过远程管理方式连接交换机 1、telnet简介 telnet 是应用层协议 基于传输层TCP协议的&#xff0c;默认端口&#xff1a;23 采用的是明文密码方式 不是很安全&#xff0c;一般用于内网管理。 2、ssh协议简介 ssh 是应用层的协议&#xff0c;基于传输层的TCP协议&#x…

打造清晰的日志管理策略:如何在 NestJS 中集成 winston 高级日志系统

前言 在Web应用程序的开发过程中&#xff0c;日志管理是不可或缺的一部分。日志可以帮助我们了解应用程序的运行状态&#xff0c;监控系统行为&#xff0c;以及在出现问题时快速定位和解决问题。 对于使用NestJS框架的项目来说&#xff0c;集成一个高效、可扩展的日志系统尤为…

面试官:说说接口和抽象类有什么区别

程序员的公众号&#xff1a;源1024&#xff0c;获取更多资料&#xff0c;无加密无套路&#xff01; 最近整理了一波电子书籍资料&#xff0c;包含《Effective Java中文版 第2版》《深入JAVA虚拟机》&#xff0c;《重构改善既有代码设计》&#xff0c;《MySQL高性能-第3版》&…

高密集型数据服务--第2章 数据模型与查询语言

一、引言 数据模型可能是开发软件最重要的部分,而且还对如何思考待解决的问题都有深远的影响。 大多数应用程序是通过一层一层叠加数据模型来构建的。每一层都面临的关键问题是&#xff1a;如何将其用下一层来表示&#xff1f; 1.作为一名应用程序开发人员&#xff0c;观测现实…