83

我正在使用内置日/夜模式功能的 Android,我想在我的应用程序中添加一个选项AppCompatDelegate.MODE_NIGHT_AUTO

我遇到了一个问题,因为我的应用程序需要以编程方式对某些内容进行着色,而且我不知道如何检查应用程序是否认为自己处于夜间或白天模式。没有它,我无法设置标志来选择正确的颜色。

调用AppCompatDelegate.getDefaultNightMode()只是返回 AppCompatDelegate.MODE_NIGHT_AUTO 这是没用的。

我没有看到任何其他可以告诉我的信息,但一定有什么?

4

8 回答 8

138
int nightModeFlags =
    getContext().getResources().getConfiguration().uiMode &
    Configuration.UI_MODE_NIGHT_MASK;
switch (nightModeFlags) {
    case Configuration.UI_MODE_NIGHT_YES:
         doStuff();
         break;

    case Configuration.UI_MODE_NIGHT_NO:
         doStuff();
         break;

    case Configuration.UI_MODE_NIGHT_UNDEFINED:
         doStuff();
         break;
}
于 2017-05-24T23:49:33.753 回答
41

如果你是 kotlin 开发者,那么你可以使用下面的代码来判断暗模式。

when (context.resources?.configuration?.uiMode?.and(Configuration.UI_MODE_NIGHT_MASK)) {
    Configuration.UI_MODE_NIGHT_YES -> {}
    Configuration.UI_MODE_NIGHT_NO -> {}
    Configuration.UI_MODE_NIGHT_UNDEFINED -> {}
}

有关深色主题模式的更多信息

https://github.com/android/user-interface-samples/tree/main/DarkTheme

于 2019-06-25T08:42:39.200 回答
12

一个非常简单的解决方案是添加这样的字符串资源。

res/values/strings.xml

<string name="mode">Day</string>

res/values-night/strings.xml

<string name="mode">Night</string>

然后在需要检查的地方:

if (resources.getString(R.string.mode) == "Day") {
    // Do Day stuff here
} else {
    // Do night stuff here
}

但是您不能super.onCreate()在活动的生命周期之前调用它。它总是会返回 "Day" before onCreate

于 2020-04-08T16:05:27.490 回答
10

Java 中的位运算符(在 @ephemient 的答案中使用)在 kotlin 中是不同的,因此对于初学者来说,代码可能不容易转换。这是 kotlin 版本,以防万一有人需要它:

    private fun isUsingNightModeResources(): Boolean {
        return when (resources.configuration.uiMode and 
Configuration.UI_MODE_NIGHT_MASK) {
            Configuration.UI_MODE_NIGHT_YES -> true
            Configuration.UI_MODE_NIGHT_NO -> false
            Configuration.UI_MODE_NIGHT_UNDEFINED -> false
            else -> false
    }
}
于 2019-07-25T03:46:36.870 回答
1

对于 Xamarin.Android

if (Build.VERSION.SdkInt >= BuildVersionCodes.Froyo)
{
    var uiModeFlags = Application.Context.Resources.Configuration.UiMode & UiMode.NightMask;
    switch (uiModeFlags)
    {
        case UiMode.NightYes:
            // dark mode
            break;
        case UiMode.NightNo:
            // default mode
            break;
        default:
            // default mode
            break;
    }
}
else
{
// default mode
}
于 2020-09-02T13:28:08.833 回答
1

也许只是一个属性,它会告诉你是否有暗模式?

val Context.isDarkMode
    get() = if (getDefaultNightMode() == MODE_NIGHT_FOLLOW_SYSTEM)
        resources.configuration.uiMode and UI_MODE_NIGHT_MASK == UI_MODE_NIGHT_YES
    else getDefaultNightMode() == MODE_NIGHT_YES
于 2021-09-04T05:50:47.063 回答
0

当我尝试在每个答案代码中获取当前主题时,我的 UI 测试出现问题。它每次都返回相同的值。

所以我在应用程序类中创建静态变量并获取 UI 测试的主题抛出它:

@Theme
fun Context.getAppTheme(): Int? {
    val theme = when (resources.configuration.uiMode and Configuration.UI_MODE_NIGHT_MASK) {
        Configuration.UI_MODE_NIGHT_NO -> Theme.LIGHT
        Configuration.UI_MODE_NIGHT_YES -> Theme.DARK
        else -> null
    }

    Application.theme = theme

    return theme
}

在应用程序类中:

companion object {
    @Theme
    @RunNone var theme: Int? = null
}

可能对某人有用!

于 2021-02-09T19:28:36.870 回答
-1

在将应用程序更改为深色主题后,我发现没有任何效果

AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);

用这种方法更改应用程序后有什么方法可以知道系统设置?
谢谢,
FF

于 2021-03-30T22:01:04.097 回答