让 Android 上的 WebView 使用 prefers-color-scheme: dark

新手上路,请多包涵

我有一个使用 webview 的 Android 应用程序,最近我试图弄清楚如何使用新的 @media (prefers-color-scheme: dark) CSS 语法来添加深色主题。我在我的页面上写了正确的 CSS,如果我在 Chrome 中打开它并打开 Chrome 的暗模式,它就可以工作。我也有我的 AppTheme 继承 Theme.AppCompat.DayNight ,当我为设备上的整个操作系统打开暗模式时,我的应用程序显示暗加载对话框等。甚至 <input> 元素的自动完成选项也会变暗。但是,我的 webview 加载的网页仍然没有变暗。根据 这个页面,webviews 应该支持这个特性,但我就是不能让它工作。我在这里想念什么?

我还刚刚发现在 API 29 中有这个 WebSettings.setForceDark() 方法;会不会是我要找的东西?我希望找到一个适用于较低 API 级别的解决方案。

顺便说一句,我目前的解决方法是注入这样的 JavaScript 接口:

 webView.addJavascriptInterface(new JSInterface(), "jsInterface");

...

public class JSInterface {
    @JavascriptInterface
    public boolean isNightMode() {
        int nightModeFlags = getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;
        return nightModeFlags == Configuration.UI_MODE_NIGHT_YES;
    }
}

然后在我的网页中,调用 jsInterface.isNightMode() 方法并根据结果动态加载不同的 CSS 文件。它确实可以根据需要对全局暗模式设置做出响应,但我仍然想知道是否可以使 prefers-color-scheme 工作。

原文由 Mu-Tsun Tsai 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 2.3k
2 个回答

Android Webview 处理日/夜模式与其他视图略有不同。将您的主题设置为深色会将 WebView 组件(滚动条、缩放按钮等)更改为深色模式版本,但不会更改它加载的内容。

要更改内容,您需要使用 webview 设置的 setForceDark 方法使其也更改其内容。此方法的兼容版本可以在 AndroidX webkit 包中找到。

将以下依赖项添加到您的 gradle 构建中:

 implementation 'androidx.webkit:webkit:1.3.0'

(1.3.0 是此软件包的最低要求版本。但更高版本也应该可以工作。)

并将以下代码行添加到您的 webview 初始化中:

 if(WebViewFeature.isFeatureSupported(WebViewFeature.FORCE_DARK)) {
    WebSettingsCompat.setForceDark(myWebView.getSettings(), WebSettingsCompat.FORCE_DARK_ON);
}

isFeatureSupported 检查以确保用户在其设备上安装的 Android 系统 WebView 版本支持暗模式(因为这可以通过 Google Play 独立于 Android 版本进行更新或降级)。

注意: setForceDark 功能需要在运行的设备上安装 Android System WebView v76 或更高版本。

webview 内容的 force dark 功能有两个所谓的策略:

  • 用户代理变暗: webview 将通过自动反转或变暗其内容的颜色将其内容设置为暗模式。

  • 基于主题的变暗: webview 将根据内容的主题(包括 @media (prefers-color-scheme: dark) 查询)更改为暗模式。

要设置 webview 应该使用哪种策略来应用强制黑暗,您可以使用以下代码:

 if(WebViewFeature.isFeatureSupported(WebViewFeature.FORCE_DARK_STRATEGY)) {
    WebSettingsCompat.setForceDarkStrategy(myWebView.getSettings(), WebSettingsCompat.DARK_STRATEGY_WEB_THEME_DARKENING_ONLY);
}

注意: 策略选择需要在运行设备上安装 Android System WebView v83 或更高版本。支持 setForceDark 但不支持策略选择(v76 到 v81)的 WebView 版本将使用用户代理变暗

支持的策略选项有:

  • DARK_STRATEGY_USER_AGENT_DARKENING_ONLY :仅使用用户代理变暗并忽略内容中的任何主题(默认)
  • DARK_STRATEGY_WEB_THEME_DARKENING_ONLY :仅使用内容本身的深色主题将页面设置为深色模式。如果内容没有深色主题,则 webview 不会应用任何变暗并“按原样”显示。
  • DARK_STRATEGY_PREFER_WEB_THEME_OVER_USER_AGENT_DARKENING :使用内容本身的深色主题将页面设置为深色模式。如果内容没有深色主题,请使用用户代理变暗。

Javascript 检查如何对变暗的 webviews 起作用?

JavaScript 调用 window.matchMedia('(prefers-color-scheme: dark)') 将匹配用户代理变暗和 Web 主题变暗策略。

我将 webview 设置为 FORCE_DARK_AUTO 并且我的应用程序在昼夜主题中运行,但不知何故,我的 webview 不会根据我的应用程序主题自动应用暗模式。为什么会这样?

这是因为 FORCE_DARK_AUTO webview 的设置值不适用于主题(如 文档中所述)。它检查 Android 10 强制黑暗 功能(应用程序的“快速修复”黑暗模式功能。它的名称相似,但与 WebView 强制黑暗没有直接关系)。

如果您没有使用强制黑暗,而是使用应用程序主题来处理黑暗模式(如推荐的那样),您必须自己检查何时应用 webview 的强制黑暗功能。使用 DayNight 主题的示例:

 int nightModeFlags = getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;
if (nightModeFlags == Configuration.UI_MODE_NIGHT_YES) {
   //Code to enable force dark using FORCE_DARK_ON and select force dark strategy
}

原文由 Leon Lucardie 发布,翻译遵循 CC BY-SA 4.0 许可协议

这里有两个相关的文档:

https://developer.android.com/guide/topics/ui/look-and-feel/darktheme#force_dark

https://developer.android.com/reference/android/view/View.html#setForceDarkAllowed(boolean)

关键点是

  1. WebView 必须允许强制变暗,它的所有父项也必须允许。
  2. 主题必须标记为浅色。

这意味着让 FORCE_DARK_AUTO 行为在 WebView 中起作用有点复杂。将以下内容添加到 AppCompat.DayNight 主题确实有效,但可能不是最佳解决方案,因为它可能会将 Android 强制黑暗应用于 WebView 以外的视图。

 <item name="android:forceDarkAllowed">true</item>
<item name="android:isLightTheme">true</item>

另一种选择是手动处理此问题,方法是根据相关配置更改设置 FORCE_DARK_ON/OFF。

目前 WebView 不支持 prefers-color-scheme,部分原因是在 prefers-color-scheme 标准化之前实现了强制暗,部分原因是无法将其明智地(没有样式闪烁)与强制暗集成。目的是提供通过 androidx.webkit 选择强制黑暗或偏好颜色方案的能力。

原文由 Tobias Sargeant 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题