Android 混合开发实战:统一 View 与 Compose 的浅色/深色主题方案

整个应用(包括 View 和 Compose 部分)的浅色/深色模式保持一致。以下是完整的解决方案:

全局配置方案

1. 基础主题设置

res/values/themes.xmlres/values-night/themes.xml 中定义统一的主题

xml 复制代码
<!-- values/themes.xml -->
<style name="Theme.MyApp" parent="Theme.MaterialComponents.DayNight">
    <!-- 浅色主题颜色 -->
    <item name="colorPrimary">@color/purple_500</item>
    <item name="colorPrimaryVariant">@color/purple_700</item>
    <item name="colorSecondary">@color/teal_200</item>
</style>

<!-- values-night/themes.xml -->
<style name="Theme.MyApp" parent="Theme.MaterialComponents.DayNight">
    <!-- 深色主题颜色 -->
    <item name="colorPrimary">@color/purple_200</item>
    <item name="colorPrimaryVariant">@color/purple_500</item>
    <item name="colorSecondary">@color/teal_200</item>
</style>

2. Compose 主题适配器

创建 Compose 主题与 XML 主题的桥梁:

kotlin 复制代码
// Theme.kt
@Composable
fun MyAppTheme(
    darkTheme: Boolean = isSystemInDarkTheme(),
    content: @Composable () -> Unit
) {
    val context = LocalContext.current
    val colors = if (darkTheme) {
        // 从 XML 资源获取颜色
        val primaryColor = context.getColor(R.color.purple_200)
        val primaryVariantColor = context.getColor(R.color.purple_500)
        val secondaryColor = context.getColor(R.color.teal_200)
        
        darkColors(
            primary = Color(primaryColor),
            primaryVariant = Color(primaryVariantColor),
            secondary = Color(secondaryColor)
        )
    } else {
        val primaryColor = context.getColor(R.color.purple_500)
        val primaryVariantColor = context.getColor(R.color.purple_700)
        val secondaryColor = context.getColor(R.color.teal_200)
        
        lightColors(
            primary = Color(primaryColor),
            primaryVariant = Color(primaryVariantColor),
            secondary = Color(secondaryColor)
        )
    }
    
    MaterialTheme(
        colors = colors,
        typography = Typography,
        shapes = Shapes,
        content = content
    )
}

3. 全局主题切换控制

在 Application 类中统一管理主题:

kotlin 复制代码
class MyApp : Application() {
    override fun onCreate() {
        super.onCreate()
        // 设置默认主题模式
        AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM)
    }
    
    companion object {
        fun setDarkMode(enabled: Boolean) {
            AppCompatDelegate.setDefaultNightMode(
                if (enabled) AppCompatDelegate.MODE_NIGHT_YES
                else AppCompatDelegate.MODE_NIGHT_NO
            )
        }
    }
}

实际使用示例

传统 View Activity

kotlin 复制代码
class LegacyActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // 会自动应用 themes.xml 中定义的主题
        setContentView(R.layout.activity_legacy)
    }
}

Compose Activity

kotlin 复制代码
class ComposeActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        
        setContent {
            MyAppTheme {
                // 你的 Compose 内容
                Surface {
                    Text("这是 Compose 界面")
                }
            }
        }
    }
}

迁移过程中的注意事项

  1. 混合 Activity 处理

    • 对于同时包含 View 和 Compose 的 Activity,使用 ComposeView 桥接
    xml 复制代码
    <!-- activity_mixed.xml -->
    <LinearLayout>
        <TextView android:text="传统 View"/>
        <androidx.compose.ui.platform.ComposeView
            android:id="@+id/compose_view"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>
    </LinearLayout>
  2. 颜色资源统一

    • 所有颜色定义在 res/values/colors.xmlres/values-night/colors.xml
    • Compose 通过 colorResource(id = R.color.your_color) 使用
  3. 主题切换同步

    • 使用 AppCompatDelegate.setDefaultNightMode() 切换时,所有 Activity 都会自动重建并应用新主题

通过这种方式,你可以确保项目中前期使用 View 的 Activity 和后期使用 Compose 的 Activity 在浅色/深色模式下保持完全一致的视觉风格。

相关推荐
阿巴斯甜10 小时前
Android 报错:Zip file '/Users/lyy/develop/repoAndroidLapp/l-app-android-ble/app/bu
android
Kapaseker11 小时前
实战 Compose 中的 IntrinsicSize
android·kotlin
xq952712 小时前
Andorid Google 登录接入文档
android
黄林晴13 小时前
告别 Modifier 地狱,Compose 样式系统要变天了
android·android jetpack
冬奇Lab1 天前
Android触摸事件分发、手势识别与输入优化实战
android·源码阅读
城东米粉儿1 天前
Android MediaPlayer 笔记
android
Jony_1 天前
Android 启动优化方案
android
阿巴斯甜1 天前
Android studio 报错:Cause: error=86, Bad CPU type in executable
android
张小潇1 天前
AOSP15 Input专题InputReader源码分析
android
_小马快跑_1 天前
Kotlin | 协程调度器选择:何时用CoroutineScope配置,何时用launch指定?
android