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

在 Jetpack Compose 中实现 iOS 风格输入框

在 Jetpack Compose 中实现 iOS 风格输入框

要在 Android Compose 中模仿 iOS 风格的输入框,我们需要关注几个关键特点:简洁的边框、圆角设计、特定的交互效果等。以下是完整的实现方案:

1. 基础 iOS 风格输入框

@Composable
fun IosStyleTextField(value: String,onValueChange: (String) -> Unit,modifier: Modifier = Modifier,label: String? = null,placeholder: String? = null
) {Column(modifier = modifier) {label?.let {Text(text = it,style = MaterialTheme.typography.bodyMedium,color = Color.Gray,modifier = Modifier.padding(bottom = 4.dp))}Box(modifier = Modifier.fillMaxWidth().background(color = Color.White,shape = RoundedCornerShape(10.dp)).border(width = 1.dp,color = Color(0xFFC7C7CC),shape = RoundedCornerShape(10.dp)).padding(horizontal = 16.dp, vertical = 12.dp)) {BasicTextField(value = value,onValueChange = onValueChange,modifier = Modifier.fillMaxWidth(),textStyle = MaterialTheme.typography.bodyMedium.copy(color = Color.Black),singleLine = true,decorationBox = { innerTextField ->if (value.isEmpty() && placeholder != null) {Text(text = placeholder,style = MaterialTheme.typography.bodyMedium.copy(color = Color(0xFFC7C7CC)))}innerTextField()})}}
}// 使用示例
var text by remember { mutableStateOf("") }
IosStyleTextField(value = text,onValueChange = { text = it },label = "用户名",placeholder = "请输入用户名",modifier = Modifier.padding(16.dp)
)

2. 带清除按钮的 iOS 风格输入框

@Composable
fun IosStyleTextFieldWithClear(value: String,onValueChange: (String) -> Unit,modifier: Modifier = Modifier,label: String? = null,placeholder: String? = null
) {Column(modifier = modifier) {label?.let {Text(text = it,style = MaterialTheme.typography.bodyMedium,color = Color.Gray,modifier = Modifier.padding(bottom = 4.dp))}Box(modifier = Modifier.fillMaxWidth().background(color = Color.White,shape = RoundedCornerShape(10.dp)).border(width = 1.dp,color = if (value.isNotEmpty()) Color(0xFF007AFF) else Color(0xFFC7C7CC),shape = RoundedCornerShape(10.dp)).padding(horizontal = 16.dp, vertical = 12.dp)) {Row(verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth()) {BasicTextField(value = value,onValueChange = onValueChange,modifier = Modifier.weight(1f),textStyle = MaterialTheme.typography.bodyMedium.copy(color = Color.Black),singleLine = true,decorationBox = { innerTextField ->if (value.isEmpty() && placeholder != null) {Text(text = placeholder,style = MaterialTheme.typography.bodyMedium.copy(color = Color(0xFFC7C7CC)))}innerTextField()})if (value.isNotEmpty()) {IconButton(onClick = { onValueChange("") },modifier = Modifier.size(20.dp)) {Icon(imageVector = Icons.Default.Close,contentDescription = "清除",tint = Color(0xFFC7C7CC),modifier = Modifier.size(16.dp))}}}}}
}

3. 搜索框样式

@Composable
fun IosStyleSearchField(value: String,onValueChange: (String) -> Unit,modifier: Modifier = Modifier,placeholder: String = "搜索"
) {Box(modifier = modifier.fillMaxWidth().height(36.dp).background(color = Color(0xFFF2F2F7),shape = RoundedCornerShape(10.dp)).padding(horizontal = 8.dp),contentAlignment = Alignment.CenterStart) {Row(verticalAlignment = Alignment.CenterVertically,modifier = Modifier.fillMaxWidth()) {Icon(imageVector = Icons.Default.Search,contentDescription = "搜索",tint = Color(0xFF8E8E93),modifier = Modifier.size(16.dp).padding(end = 4.dp))BasicTextField(value = value,onValueChange = onValueChange,modifier = Modifier.weight(1f),textStyle = MaterialTheme.typography.bodyMedium.copy(color = Color.Black,fontSize = 14.sp),singleLine = true,decorationBox = { innerTextField ->if (value.isEmpty()) {Text(text = placeholder,style = MaterialTheme.typography.bodyMedium.copy(color = Color(0xFF8E8E93),fontSize = 14.sp))}innerTextField()})if (value.isNotEmpty()) {IconButton(onClick = { onValueChange("") },modifier = Modifier.size(20.dp)) {Icon(imageVector = Icons.Default.Close,contentDescription = "清除",tint = Color(0xFF8E8E93),modifier = Modifier.size(16.dp))}}}}
}

4. 带分段的 iOS 风格表单

@Composable
fun IosStyleForm(items: List<Pair<String, @Composable () -> Unit>>,modifier: Modifier = Modifier
) {Column(modifier = modifier.background(Color.White).clip(RoundedCornerShape(10.dp))) {items.forEachIndexed { index, (title, content) ->Column {if (index > 0) {Divider(color = Color(0xFFC6C6C8),thickness = 0.5.dp,modifier = Modifier.padding(horizontal = 16.dp))}Row(modifier = Modifier.fillMaxWidth().padding(horizontal = 16.dp, vertical = 12.dp),verticalAlignment = Alignment.CenterVertically,horizontalArrangement = Arrangement.SpaceBetween) {Text(text = title,style = MaterialTheme.typography.bodyMedium.copy(color = Color.Black),modifier = Modifier.weight(1f))Box(modifier = Modifier.weight(1f)) {content()}}}}}
}// 使用示例
var username by remember { mutableStateOf("") }
var password by remember { mutableStateOf("") }IosStyleForm(items = listOf("账号" to {BasicTextField(value = username,onValueChange = { username = it },modifier = Modifier.fillMaxWidth(),textStyle = MaterialTheme.typography.bodyMedium.copy(textAlign = TextAlign.End))},"密码" to {BasicTextField(value = password,onValueChange = { password = it },modifier = Modifier.fillMaxWidth(),textStyle = MaterialTheme.typography.bodyMedium.copy(textAlign = TextAlign.End),visualTransformation = PasswordVisualTransformation())}),modifier = Modifier.padding(16.dp)
)

5. 动画效果增强

@Composable
fun AnimatedIosTextField(value: String,onValueChange: (String) -> Unit,modifier: Modifier = Modifier,label: String,placeholder: String
) {val animatedBorderColor by animateColorAsState(targetValue = if (value.isNotEmpty()) Color(0xFF007AFF) else Color(0xFFC7C7CC),label = "borderColorAnimation")Column(modifier = modifier) {Text(text = label,style = MaterialTheme.typography.bodyMedium,color = Color.Gray,modifier = Modifier.padding(bottom = 4.dp))Box(modifier = Modifier.fillMaxWidth().background(color = Color.White,shape = RoundedCornerShape(10.dp)).border(width = 1.dp,color = animatedBorderColor,shape = RoundedCornerShape(10.dp)).padding(horizontal = 16.dp, vertical = 12.dp)) {BasicTextField(value = value,onValueChange = onValueChange,modifier = Modifier.fillMaxWidth(),textStyle = MaterialTheme.typography.bodyMedium.copy(color = Color.Black),singleLine = true,decorationBox = { innerTextField ->Box {if (value.isEmpty()) {Text(text = placeholder,style = MaterialTheme.typography.bodyMedium.copy(color = Color(0xFFC7C7CC)))}innerTextField()}})}}
}

关键 iOS 风格特点实现

  1. 圆角设计

    shape = RoundedCornerShape(10.dp)
    
  2. 边框颜色

    borderColor = if (focused) Color(0xFF007AFF) else Color(0xFFC7C7CC)
    
  3. 占位符样式

    color = Color(0xFFC7C7CC)
    
  4. 清除按钮

    Icon(Icons.Default.Close, contentDescription = "清除")
    
  5. 分段列表样式

    Divider(color = Color(0xFFC6C6C8), thickness = 0.5.dp)
    

这些组件可以根据你的具体需求进一步定制,调整颜色、尺寸和动画效果,以达到更接近 iOS 设计风格的效果。

相关文章:

  • Zenodo上传文件流程
  • 安卓手机万能遥控器APP推荐
  • Git 解决“Filename too long”问题
  • Java学习笔记--多态:多态的介绍,多态的基本使用,多态的条件下成员的访问特点,多态的好处
  • 【科研绘图系列】R语言绘制多个气泡图组合图(bubble plot)
  • k8s 调整Node节点 Max_Pods
  • Linux网络编程 深入解析TFTP协议:基于UDP的文件传输实战
  • 三大等待和三大切换
  • PP-OCR的安卓端部署
  • Google Colab测试部署Qwen大模型,实现PDF转MD场景OCR 识别(支持单机环境)
  • CSS3笔记
  • 设计模式 --- 外观模式
  • 基于FPGA的AES加解密系统verilog实现,包含testbench和开发板硬件测试
  • 【ESP32-IDF笔记】06-触摸传感IO配置
  • 基于尚硅谷FreeRTOS视频笔记——6—滴答时钟—上下文切换
  • OpenAI重返巅峰:o3与o4-mini引领AI推理新时代
  • Qt QThread 两种线程管理方法
  • 【 解决Cline插件无法激活及DeepSeek模型请求卡顿或者无法加载问题】
  • C++ `unique_ptr` 多线程使用
  • SpringAI+DeepSeek大模型应用开发——5 ChatPDF
  • 《“四有”好老师系列丛书》发布,由顾明远总主编
  • 为什么要研制大型水陆两栖飞机?AG600总设计师给出答案
  • 美伊第二轮核问题间接谈判结束,伊方称“结果是建设性的”
  • 撤销逾千名留学生签证,特朗普政府面临集体诉讼
  • 华夏幸福:累计未能如期偿还债务金额合计为227.91亿元
  • 价格周报|本周生猪均价环比上涨,交易均重继续上升