当前位置: 首页 > article >正文

Jetpack Compose 自定义组件完全指南

Jetpack Compose 自定义组件完全指南

Compose 的声明式 UI 范式为创建自定义组件提供了前所未有的灵活性。本指南将带你从基础到高级全面掌握 Compose 自定义组件的开发技巧。

一、自定义组件基础

1.1 基本结构

一个最简单的自定义组件:

@Composable
fun Greeting(name: String) {Text(text = "Hello, $name!")
}

1.2 组件参数设计原则

  • 明确职责:每个组件应只做一件事
  • 合理的默认值:为非必要参数提供默认值
  • 命名规范:使用描述性名称,遵循 Kotlin 惯例
@Composable
fun CustomButton(text: String,onClick: () -> Unit,modifier: Modifier = Modifier,enabled: Boolean = true,icon: @Composable (() -> Unit)? = null
) {// 实现
}

二、布局与样式

2.1 自定义布局

使用 Layout 可创建完全自定义的布局:

@Composable
fun CustomLayout(modifier: Modifier = Modifier,content: @Composable () -> Unit
) {Layout(modifier = modifier,content = content) { measurables, constraints ->// 测量和布局逻辑val placeables = measurables.map { it.measure(constraints) }layout(constraints.maxWidth, constraints.maxHeight) {var yPos = 0placeables.forEach { placeable ->placeable.placeRelative(x = 0, y = yPos)yPos += placeable.height}}}
}

2.2 样式与主题

使用 MaterialTheme 实现主题化:
@Composable
fun ThemedComponent() {val colors = MaterialTheme.colorSchemeBox(modifier = Modifier.background(colors.primary).padding(16.dp)) {Text("主题化组件", color = colors.onPrimary)}
}
创建自定义主题系统:
object CustomTheme {val colors: CustomColors@Composableget() = LocalCustomColors.current
}@Composable
fun CustomTheme(colors: CustomColors = lightCustomColors(),content: @Composable () -> Unit
) {CompositionLocalProvider(LocalCustomColors provides colors,content = content)
}

三、状态管理

3.1 组件内部状态

@Composable
fun Counter() {var count by remember { mutableStateOf(0) }Button(onClick = { count++ }) {Text("Clicked $count times")}
}

3.2 状态提升

@Composable
fun Counter(count: Int,onIncrement: () -> Unit,modifier: Modifier = Modifier
) {Button(onClick = onIncrement,modifier = modifier) {Text("Clicked $count times")}
}

3.3 高级状态管理

@Stable
class CustomState(initialValue: Int
) {var value by mutableStateOf(initialValue)fun increment() { value++ }
}@Composable
fun rememberCustomState(initial: Int = 0) = remember {CustomState(initial)
}

四、交互与动画

4.1 手势交互

@Composable
fun DraggableBox() {var offsetX by remember { mutableStateOf(0f) }Box(modifier = Modifier.offset { IntOffset(offsetX.roundToInt(), 0) }.pointerInput(Unit) {detectDragGestures { change, dragAmount ->offsetX += dragAmount.x}}.size(100.dp).background(Color.Blue))
}

4.2 动画效果

@Composable
fun AnimatedComponent(visible: Boolean) {val alpha by animateFloatAsState(targetValue = if (visible) 1f else 0f,animationSpec = tween(durationMillis = 300))Box(modifier = Modifier.alpha(alpha).size(100.dp).background(Color.Red))
}

五、高级技巧

5.1 性能优化

@Composable
fun OptimizedList(items: List<String>) {LazyColumn {items(items, key = { it }) { item ->Text(item)}}
}

5.2 跨平台适配

@Composable
fun PlatformSpecificComponent() {val configuration = LocalConfiguration.currentval isWideScreen = configuration.screenWidthDp >= 600if (isWideScreen) {WideScreenLayout()} else {MobileLayout()}
}

5.3 测试策略

@Test
fun testCustomComponent() {composeTestRule.setContent {CustomButton(text = "Test", onClick = {})}composeTestRule.onNodeWithText("Test").assertIsDisplayed()
}

六、实战案例

6.1 自定义下拉刷新

@Composable
fun PullToRefreshLayout(refreshing: Boolean,onRefresh: () -> Unit,content: @Composable () -> Unit
) {Box(modifier = Modifier.pointerInput(Unit) {// 手势检测逻辑}) {content()if (refreshing) {CircularProgressIndicator()}}
}

6.2 复杂图表组件

@Composable
fun LineChart(dataPoints: List<Float>,modifier: Modifier = Modifier
) {Canvas(modifier = modifier) {// 自定义绘制逻辑}
}

七、最佳实践

  1. 组合优于继承:通过组合现有组件构建新组件
  2. 单向数据流:状态提升到可管理的层级
  3. 考虑重组:避免在组合函数中执行耗时操作
  4. 无障碍支持:添加适当的内容描述
  5. 主题适配:尊重用户的主题偏好

八、调试与优化

  1. 使用 Modifier.debugInspectorInfo 调试布局问题
  2. 通过 AndroidViewBinding 集成现有视图
  3. 使用 rememberSaveable 处理配置变更
  4. 利用 DerivedState 优化计算密集型操作

结语

掌握 Compose 自定义组件开发可以极大提升 UI 开发的灵活性和效率。从简单的布局组合到复杂的自定义绘制和动画,Compose 提供了完整的工具链来满足各种需求。记住在实践中不断尝试和优化,你会发现 Compose 的强大之处。

相关文章:

Jetpack Compose 自定义组件完全指南

Jetpack Compose 自定义组件完全指南 Compose 的声明式 UI 范式为创建自定义组件提供了前所未有的灵活性。本指南将带你从基础到高级全面掌握 Compose 自定义组件的开发技巧。 一、自定义组件基础 1.1 基本结构 一个最简单的自定义组件&#xff1a; Composable fun Greeti…...

java后端对时间进行格式处理

时间格式处理 通过java后端&#xff0c;使用jackson库的注解JsonFormat(pattern "yyyy-MM-dd HH:mm:ss")进行格式化 package com.weiyu.pojo;import com.fasterxml.jackson.annotation.JsonFormat; import lombok.AllArgsConstructor; import lombok.Data; import …...

汽车BMS技术分享及其HIL测试方案

一、BMS技术简介 在全球碳中和目标的战略驱动下&#xff0c;新能源汽车产业正以指数级速度重塑交通出行格局。动力电池作为电动汽车的"心脏"&#xff0c;其性能与安全性不仅直接决定了车辆的续航里程、使用寿命等关键指标&#xff0c;更深刻影响着消费者对电动汽车的…...

【Code】《代码整洁之道》笔记-Chapter3-函数

第3章 函数 在编程的早期岁月&#xff0c;系统由程序和子程序组成。后来&#xff0c;到Fortran和PL/1的年代&#xff0c;系统由程序、子程序和函数组成。如今&#xff0c;只有函数存活下来。函数是所有程序中的第一组代码。本章将讨论如何写好函数。 请看代码清单3-1。在Fit…...

【TI MSPM0】CMSIS-DSP库学习

一、什么是CMSIS-DSP库 基于Cortex微控制器软件接口标准的数字信号处理的函数库 二、页面概览 这个用户手册用来描述CMSIS-DSP软件的函数库&#xff0c;有通用的计算处理函数给Cortex-M和Cortex-A的处理器使用 三、工程学习 1.导入工程 2.样例介绍 在Q15的格式下&#xff0c…...

Vue3:初识Vue,Vite服务器别名及其代理配置

一、创建一个Vue3项目 创建Vue3项目默认使用Vite作为现代的构建工具&#xff0c;以下指令本质也是通过下载create-vue来构建项目。 基于NodeJs版本大于等于18.3&#xff0c;使用命令行进行操作。 1、命令执行 npm create vuelatest输入项目名称 2、选择附加功能 选择要包含的功…...

音频接口格式与通道

IEC 60958-3 文档结构概览&#xff08;通俗版&#xff09; 对于初学者来说&#xff0c;IEC 60958-3 的文档就像一个“数字音频传输的说明书”。它告诉设备如何把声音变成一堆0和1&#xff0c;再通过这些0和1的排列规则&#xff0c;让接收设备准确还原声音。 1. 接口格式&#…...

JS中的WeakMap

WeakMap weakmap是一种类似map的类型&#xff0c;但它的key是弱引用&#xff0c;并且key只能是对象。 weakmap和map的区别 weakmap的key只能是对象&#xff0c;value可以是任何值 const weakMap new WeakMap(); // 创建WeakMap实例const key1 "key1"; // 字符…...

Go语言类型捕获及内存大小判断

代码如下&#xff1a; 类型捕获可使用&#xff1a;reflect.TypeOf()&#xff0c;fmt.Printf在的%T。 内存大小判断&#xff1a;len()&#xff0c;unsafe.Sizeof。 package mainimport ("fmt""unsafe""reflect" )func main(){var i , j 1, 2f…...

学透Spring Boot — 017. 处理静态文件

这是我的《学透Spring Boot》专栏的第17篇文章&#xff0c;了解更多内容请移步我的专栏&#xff1a; Postnull CSDN 学透 Spring Boot 目录 静态文件 静态文件的默认位置 通过配置文件配置路径 通过代码配置路径 静态文件的自动配置 总结 静态文件 以前的传统MVC的项目…...

ARK no NIGHTS

《昨夜圆车》(ARK no NIGHTS) 于 2025 年 4 月 1 日 16:00 开服。在这款游戏中&#xff0c;玩家将扮演博士&#xff0c;带领整合运动击败罗德岛。FloorWinter 非常期待第一时间体验这款船新的游戏&#xff0c;于是他决定昏睡过去&#xff0c;直到游戏开服再醒来。 游戏开服的时…...

CMake实战指南一:add_custom_command

CMake 进阶&#xff1a;add_custom_command 用法详解与实战指南 在 CMake 构建系统中&#xff0c;add_custom_command 是一个灵活且强大的工具&#xff0c;允许开发者在构建流程中插入自定义操作。无论是生成中间文件、执行预处理脚本&#xff0c;还是在目标构建前后触发额外逻…...

指纹浏览器技术架构解析:高并发批量注册业务的工程化实践——基于分布式指纹引擎与防关联策略的深度实现

一、技术背景与行业痛点 在跨境电商、广告投放、问卷调查等场景中&#xff0c;批量注册与多账号矩阵运营已成为刚需。然而&#xff0c;主流平台&#xff08;如亚马逊、Facebook、Google&#xff09;的风控系统通过浏览器指纹追踪&#xff08;Canvas/WebGL/WebRTC等&#xff09…...

懂x帝二手车数据爬虫-涉及简单的字体加密,爬虫中遇到“口”问题的解决

#脚本如下 import requests import pprint import timeurl https://www.dongchedi.com/motor/pc/sh/sh_sku_list?aid1839&app_nameauto_web_pc headers {User-Agent: Mozilla/5.0 }font_map {58425: 0, 58700: 1, 58467: 2, 58525: 3,58397: 4, 58385: 5, 58676: 6, 58…...

4.7学习总结 java集合进阶

集合进阶 泛型 //没有泛型的时候&#xff0c;集合如何存储数据 //结论: //如果我们没有给集合指定类型&#xff0c;默认认为所有的数据类型都是object类型 //此时可以往集合添加任意的数据类型。 //带来一个坏处:我们在获取数据的时候&#xff0c;无法使用他的特有行为。 //此…...

Python高阶函数-eval深入解析

1. eval() 函数概述 eval() 是 Python 内置的一个强大但需要谨慎使用的高阶函数&#xff0c;它能够将字符串作为 Python 表达式进行解析并执行。 基本语法 eval(expression, globalsNone, localsNone)expression&#xff1a;字符串形式的 Python 表达式globals&#xff1a;可…...

LLM面试题八

推荐算法工程师面试题 二分类的分类损失函数&#xff1f; 二分类的分类损失函数一般采用交叉熵(Cross Entropy)损失函数&#xff0c;即CE损失函数。二分类问题的CE损失函数可以写成&#xff1a;其中&#xff0c;y是真实标签&#xff0c;p是预测标签&#xff0c;取值为0或1。 …...

【团体程序涉及天梯赛】L1~L2实战反思合集(C++)

实战反思汇总记录 仔细审题&#xff0c;想好再写 L1-104 九宫格 - 团体程序设计天梯赛-练习集 易忽略的错误&#xff1a;开始习惯性地看到n就以为是n*n数组了&#xff0c;实际上应该是9*9的固定大小数组&#xff0c;查了半天没查出来 L1-101 别再来这么多猫娘了&#xff01…...

Linux Terminal Mode | canonical / nocanonical / cbreak / raw

注&#xff1a;本文为 “Linux 终端模式” 相关文章合辑。 略作重排&#xff0c;如有内容异常&#xff0c;请看原文。 终端输入输出的三种模式 guidao 1 前言 在进行项目开发时&#xff0c;需要实时读取终端输入&#xff08;无需按下 Enter 键即可读取&#xff09;。然而&a…...

预测分析(二):基于机器学习的数值预测

文章目录 基于机器学习的数值预测机器学习简介监督学习的任务创建第一个机器学习模型机器学习的目标——泛化过拟合现象评价函数与最优化 建模前的数据处理进一步特征变换 多元线性回归模型LASSO回归kNN算法原理算法步骤k值的选择 基于机器学习的数值预测 机器学习是人工智能的…...

JavaScript双问号操作符(??)详解,解决使用 || 时因类型转换带来的问题

目录 JavaScript双问号操作符&#xff08;??&#xff09;详解&#xff0c;解决使用||时因类型转换带来的问题 一、双问号操作符??的基础用法 1、传统方式的痛点 2、双问号操作符??的精确判断 3、双问号操作符??与逻辑或操作符||的对比 二、复杂场景下的空值处理 …...

蓝桥杯 web 展开你的扇子(css3)

普通答案&#xff1a; #box:hover #item1{transform: rotate(-60deg); } #box:hover #item2{transform: rotate(-50deg); } #box:hover #item3{transform: rotate(-40deg); } #box:hover #item4{transform: rotate(-30deg); } #box:hover #item5{transform: rotate(-20deg); }…...

聚焦楼宇自控:优化建筑性能,引领智能化管控与舒适环境

在当今建筑行业蓬勃发展的浪潮中&#xff0c;人们对建筑的要求早已超越了传统的遮风避雨功能&#xff0c;而是更加注重建筑性能的优化、智能化的管控以及舒适环境的营造。楼宇自控系统作为现代建筑技术的核心力量&#xff0c;正凭借其卓越的功能和先进的技术&#xff0c;在这几…...

前端视频流技术深度解析

一、视频流技术体系架构 1.1 现代视频流技术栈 1.1.1 核心协议对比 协议传输方式延迟适用场景浏览器支持HLSHTTP分片6-30s点播、直播回看全平台DASHHTTP动态适配3-15s多码率自适应Chrome/FirefoxWebRTCP2P/UDP<500ms实时通信、直播现代浏览器RTMPTCP长连接1-3s传统直播推…...

k8s核心资源对象一(入门到精通)

本文将深入探讨Kubernetes中的核心资源对象&#xff0c;包括Pod、Deployment、Service、Ingress、ConfigMap和Secret&#xff0c;详细解析其概念、功能以及实际应用场景&#xff0c;帮助读者全面掌握这些关键组件的使用方法。 一、pod 1 pod概念 k8s最小调度单元&#xff0c;…...

Ubuntu16.04配置远程连接

配置静态IP Ubuntu16.04 修改超管账户默认密码 # 修改root账户默认密码 sudo passwd Ubuntu16.04安装SSH # 安装ssh服务&#xff1a; sudo apt-get install ssh# 启动SSH服务&#xff1a; sudo /etc/init.d/ssh start # 开机自启 sudo systemctl enable ssh# 如无法连接&…...

基于springboot微信小程序课堂签到及提问系统(源码+lw+部署文档+讲解),源码可白嫖!

摘要 随着信息时代的来临&#xff0c;过去的课堂签到及提问管理方式的缺点逐渐暴露&#xff0c;本次对过去的课堂签到及提问管理方式的缺点进行分析&#xff0c;采取计算机方式构建基于微信小程序的课堂签到及提问系统。本文通过阅读相关文献&#xff0c;研究国内外相关技术&a…...

互联网三高-高性能之JVM调优

1 运行时数据区 JVM运行时数据区是Java虚拟机管理的内存核心模块&#xff0c;主要分为线程共享和线程私有两部分。 &#xff08;1&#xff09;线程私有 ① 程序计数器&#xff1a;存储当前线程执行字节码指令的地址&#xff0c;用于分支、循环、异常处理等流程控制‌ ② 虚拟机…...

数据操作语言

一、DML的核心操作类型 1.添加数据(INSERT) (1)手动插入:逐行插入数据,适用于少量数据。 INSERT INTO 表名 (字段1, 字段2) VALUES (值1, 值2);(2)批量导入:通过外部文件导入数据,适用于大数据场景...

智谛达科技:以创新为翼,翱翔AI人形机器人蓝海

在科技创新的浩瀚星空中,智谛达科技集团犹如一颗璀璨的明星,以其独特的创新光芒,照亮了AI人形机器人的广阔蓝海。这家在AI领域深耕多年的企业,始终秉持着创新为翼的发展理念,不断突破技术瓶颈,拓展应用场景,以卓越的实力和前瞻性的思维,引领着人形机器人行业的未来发展。 智谛达…...