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

Android Material Design 3 主题配色终极指南:XML 与 Compose 全解析

最小必要颜色配置

<!-- res/values/themes.xml -->
<style name="Theme.MyApp" parent="Theme.Material3.DayNight">
    <!-- 基础三原色 -->
    <item name="colorPrimary">@color/purple_500</item>
    <item name="colorSecondary">@color/teal_200</item>
    <item name="colorTertiary">@color/orange_200</item>
    
    <!-- 背景+文字 -->
    <item name="colorSurface">@color/white</item>
    <item name="colorOnSurface">@color/black</item>
</style>

对应效果

  • 按钮/导航栏:紫色
  • 开关/复选框:水绿色
  • 悬浮按钮:橙色容器
  • 背景:白色
  • 文字:黑色

基础控件颜色映射

控件使用的颜色属性
按钮colorPrimary + colorOnPrimary
开关colorSecondary
卡片colorSurface
文本colorOnSurface

XML 和 Compose 双版本的完整代码注释,按功能模块划分:


一、基础颜色配置

1. XML 版本 (res/values/themes.xml)
<!-- 主品牌色:用于FAB、主要按钮等 -->
<item name="colorPrimary">@color/purple_500</item>

<!-- 主色上的内容色:确保与主色有足够对比度 -->
<item name="colorOnPrimary">@color/white</item>

<!-- 次要品牌色:用于开关、单选按钮等 -->
<item name="colorSecondary">@color/teal_200</item>

<!-- 背景系统:页面基础背景 -->
<item name="colorSurface">@color/white</item>

<!-- 背景上的内容色:普通文本颜色 -->
<item name="colorOnSurface">@color/black</item>

<!-- 错误色:错误提示文本/边框 -->
<item name="colorError">@color/red_500</item>
2. Compose 版本 (Theme.kt)
val LightColorScheme = lightColorScheme(
    // 主色容器:FAB背景色
    primary = Color(0xFF6750A4),
    // 主色上的内容:FAB图标颜色
    onPrimary = Color(0xFFFFFFFF),
    // 次要色:Switch滑块颜色
    secondary = Color(0xFF958DA5),
    // 背景色:页面底色
    surface = Color(0xFFFFFBFE),
    // 背景上的内容:普通文本
    onSurface = Color(0xFF1C1B1F),
    // 错误色
    error = Color(0xFFB3261E)
)

二、控件颜色映射

1. 按钮 (Button)
<!-- XML版 -->
<Button
    android:backgroundTint="?attr/colorPrimary"
    android:textColor="?attr/colorOnPrimary"/>
// Compose版
Button(
    colors = ButtonDefaults.buttonColors(
        containerColor = MaterialTheme.colorScheme.primary,
        contentColor = MaterialTheme.colorScheme.onPrimary,
        // 禁用状态颜色(自动降低透明度)
        disabledContainerColor = MaterialTheme.colorScheme.surfaceVariant,
        disabledContentColor = MaterialTheme.colorScheme.onSurfaceVariant
    )
) { Text("按钮") }
2. 卡片 (Card)
<!-- XML版 -->
<androidx.cardview.widget.CardView
    app:cardBackgroundColor="?attr/colorSurface"
    app:strokeColor="?attr/colorOutline">
// Compose版
Card(
    colors = CardDefaults.cardColors(
        containerColor = MaterialTheme.colorScheme.surface,
        contentColor = MaterialTheme.colorScheme.onSurface
    ),
    border = CardDefaults.outlinedCardBorder(
        borderColor = MaterialTheme.colorScheme.outline
    )
) { /* 内容 */ }
3. 开关 (Switch)
<!-- XML版 -->
<Switch
    android:trackTint="?attr/colorSecondaryContainer"
    android:thumbTint="?attr/colorOnSecondary"/>
// Compose版
Switch(
    colors = SwitchDefaults.colors(
        checkedThumbColor = MaterialTheme.colorScheme.onSecondary,
        checkedTrackColor = MaterialTheme.colorScheme.secondary,
        uncheckedThumbColor = MaterialTheme.colorScheme.surfaceVariant,
        uncheckedTrackColor = MaterialTheme.colorScheme.onSurfaceVariant
    )
)

三、深色模式适配

1. XML 配置 (res/values-night/themes.xml)
<style name="Theme.MyApp" parent="Theme.Material3.DayNight">
    <!-- 深色背景 -->
    <item name="colorSurface">@color/dark_surface</item>
    <!-- 深色文字 -->
    <item name="colorOnSurface">@color/dark_on_surface</item>
    <!-- 深色轮廓线 -->
    <item name="colorOutline">@color/dark_outline</item>
</style>
2. Compose 配置
val DarkColorScheme = darkColorScheme(
    surface = Color(0xFF1C1B1F),
    onSurface = Color(0xFFE6E1E5),
    outline = Color(0xFF938F99)
)

@Composable
fun MyAppTheme(
    darkTheme: Boolean = isSystemInDarkTheme(),
    content: @Composable () -> Unit
) {
    val colorScheme = if (darkTheme) DarkColorScheme else LightColorScheme
    MaterialTheme(
        colorScheme = colorScheme,
        content = content
    )
}

四、动态颜色(Android 12+)

1. XML 配置
<!-- res/values-v31/themes.xml -->
<style name="Theme.MyApp" parent="Theme.Material3.DayNight.DynamicColors">
    <item name="android:dynamicColorThemeOverlay">
        @style/ThemeOverlay.App.Dynamic
    </item>
</style>
2. Compose 集成
@Composable
fun DynamicColorWrapper(content: @Composable () -> Unit) {
    val context = LocalContext.current
    val dynamicColor = Build.VERSION.SDK_INT >= Build.VERSION_CODES.S
    
    if (dynamicColor) {
        DynamicColors.rememberDynamicColors().apply {
            if (isDynamicColorAvailable) {
                LaunchedEffect(key1 = colorScheme) {
                    // 动态颜色应用后的回调
                }
            }
        }
    }
    content()
}

五、完整主题构建示例

XML 全量配置
<style name="Theme.MyApp" parent="Theme.Material3.DayNight">
    <!-- 主色系 -->
    <item name="colorPrimary">@color/primary_500</item>
    <item name="colorOnPrimary">@color/white</item>
    <item name="colorPrimaryContainer">@color/primary_100</item>
    
    <!-- 文字系统 -->
    <item name="textColorPrimary">?attr/colorOnSurface</item>
    <item name="textColorSecondary">?attr/colorOnSurfaceVariant</item>
    
    <!-- 控件状态 -->
    <item name="colorControlNormal">?attr/colorOnSurface</item>
    <item name="colorControlActivated">?attr/colorPrimary</item>
</style>
Compose 全量主题
private val Typography = Typography(
    // 标题文字样式
    titleLarge = TextStyle(
        fontFamily = FontFamily.Default,
        fontWeight = FontWeight.Bold,
        fontSize = 22.sp,
        color = MaterialTheme.colorScheme.onSurface
    ),
    // 正文样式
    bodyMedium = TextStyle(
        fontFamily = FontFamily.Default,
        fontWeight = FontWeight.Normal,
        fontSize = 16.sp,
        color = MaterialTheme.colorScheme.onSurface
    )
)

@Composable
fun MyAppTheme(content: @Composable () -> Unit) {
    MaterialTheme(
        colorScheme = if (isDarkTheme()) DarkColorScheme else LightColorScheme,
        typography = Typography,
        shapes = Shapes(),
        content = content
    )
}

关键注意事项

  1. XML 与 Compose 混用场景

    // 在Compose中使用XML定义的颜色
    val color = Color(ContextCompat.getColor(context, R.color.primary_500))
    
  2. 颜色覆盖优先级

    控件直接设置 > 样式定义 > 主题默认值
    
  3. 调试命令

    # 查看当前主题属性
    adb shell dumpsys activity top | grep "mTheme"
    
http://www.dtcms.com/a/119450.html

相关文章:

  • C#调用Lua方法1+C#调用Lua方法2,3
  • 【深度学习:理论篇】--Pytorch基础入门
  • MCP+Deepseck王炸组合 | 附实战操作及其MCPserver | 可替代Manus,实现AGI
  • Vue3+Vite+TypeScript+Element Plus开发-08.登录设计
  • Linux 性能调优之CPU调优认知
  • TS知识补充第一篇 ✅
  • Uniapp 集成极光推送(JPush)完整指南
  • 医药流通行业批发公司IT运维转型:Prometheus+Grafana监控Spring Boot 3应用实践
  • RK 3562 转换模型
  • Python 序列构成的数组(序列的增量赋值)
  • Redis持久化 | RDB AOF | 常见问题
  • 计算机控制系统:arduino控制无源滤波器播放音乐
  • 机器学习 | 强化学习基本原理 | MDP | TD | PG | TRPO
  • 【Linux】进程基础入门指南(上)
  • 一体式胎心仪方案特点介绍----【其利天下技术】
  • Versatile-OCR-Program:可以从复杂的教育材料(如试卷)中提取结构化数据的开源多模态OCR工具
  • 进程的唤醒原语与挂起原语
  • 简单-快速-高效——模块化解析controlnet网络结构
  • 【xdoj离散数学上机】编程实现邮箱注册等价类划分(C++ 80/100分)
  • HTML5笔记: 什么是HTML
  • 十三届蓝桥杯省赛A组 扫描游戏
  • 如何挖掘客户的深层需求?需求挖掘指南
  • Java常见的23种设计模式
  • 入选ICLR‘25 Spotlight!深度强化学习(DRL)迎来新突破!
  • 残缺知识点
  • llama-factory微调qwen2.5-vl
  • 【开发经验】调试OpenBMC Redfish EventService功能
  • Open CASCADE学习|Convert BSpline Curve to Arc Spline
  • 优选算法系列(6.模拟)
  • 专业抑郁测试工具:让心理健康评估更简单