| // Copyright 2021 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "android_webview/browser/aw_dark_mode.h" |
| |
| #include "android_webview/browser/aw_contents.h" |
| #include "android_webview/browser_jni_headers/AwDarkMode_jni.h" |
| #include "android_webview/common/aw_features.h" |
| #include "base/android/scoped_java_ref.h" |
| #include "base/memory/ptr_util.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "content/public/browser/web_contents.h" |
| #include "third_party/blink/public/common/web_preferences/web_preferences.h" |
| #include "third_party/blink/public/mojom/webpreferences/web_preferences.mojom.h" |
| |
| using base::android::JavaParamRef; |
| using base::android::ScopedJavaLocalRef; |
| |
| namespace android_webview { |
| namespace { |
| const void* const kAwDarkModeUserDataKey = &kAwDarkModeUserDataKey; |
| } |
| |
| // static |
| jlong JNI_AwDarkMode_Init(JNIEnv* env, |
| const JavaParamRef<jobject>& caller, |
| const JavaParamRef<jobject>& java_web_contents) { |
| content::WebContents* web_contents = |
| content::WebContents::FromJavaWebContents(java_web_contents); |
| DCHECK(web_contents); |
| return reinterpret_cast<intptr_t>(new AwDarkMode(env, caller, web_contents)); |
| } |
| |
| AwDarkMode* AwDarkMode::FromWebContents(content::WebContents* contents) { |
| return static_cast<AwDarkMode*>( |
| contents->GetUserData(kAwDarkModeUserDataKey)); |
| } |
| |
| AwDarkMode::AwDarkMode(JNIEnv* env, |
| jobject obj, |
| content::WebContents* web_contents) |
| : content::WebContentsObserver(web_contents), jobj_(env, obj) { |
| web_contents->SetUserData(kAwDarkModeUserDataKey, base::WrapUnique(this)); |
| } |
| |
| AwDarkMode::~AwDarkMode() { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jobject> scoped_obj = jobj_.get(env); |
| if (scoped_obj) |
| Java_AwDarkMode_onNativeObjectDestroyed(env, scoped_obj); |
| } |
| |
| void AwDarkMode::PopulateWebPreferences( |
| blink::web_pref::WebPreferences* web_prefs, |
| int force_dark_mode, |
| int force_dark_behavior) { |
| switch (force_dark_mode) { |
| case AwSettings::ForceDarkMode::FORCE_DARK_OFF: |
| is_dark_mode_ = false; |
| break; |
| case AwSettings::ForceDarkMode::FORCE_DARK_ON: |
| is_dark_mode_ = true; |
| break; |
| case AwSettings::ForceDarkMode::FORCE_DARK_AUTO: { |
| AwContents* contents = AwContents::FromWebContents(web_contents()); |
| is_dark_mode_ = contents && contents->GetViewTreeForceDarkState(); |
| break; |
| } |
| } |
| web_prefs->preferred_color_scheme = |
| is_dark_mode_ ? blink::mojom::PreferredColorScheme::kDark |
| : blink::mojom::PreferredColorScheme::kLight; |
| if (is_dark_mode_) { |
| switch (force_dark_behavior) { |
| case AwSettings::ForceDarkBehavior::FORCE_DARK_ONLY: { |
| web_prefs->preferred_color_scheme = |
| blink::mojom::PreferredColorScheme::kLight; |
| web_prefs->force_dark_mode_enabled = true; |
| break; |
| } |
| case AwSettings::ForceDarkBehavior::MEDIA_QUERY_ONLY: { |
| web_prefs->preferred_color_scheme = |
| blink::mojom::PreferredColorScheme::kDark; |
| web_prefs->force_dark_mode_enabled = false; |
| break; |
| } |
| // Blink's behavior is that if the preferred color scheme matches the |
| // supported color scheme, then force dark will be disabled, otherwise |
| // the preferred color scheme will be reset to 'light'. Therefore |
| // when enabling force dark, we also set the preferred color scheme to |
| // dark so that dark themed content will be preferred over force |
| // darkening. |
| case AwSettings::ForceDarkBehavior::PREFER_MEDIA_QUERY_OVER_FORCE_DARK: { |
| web_prefs->preferred_color_scheme = |
| blink::mojom::PreferredColorScheme::kDark; |
| web_prefs->force_dark_mode_enabled = true; |
| break; |
| } |
| } |
| } else { |
| web_prefs->preferred_color_scheme = |
| blink::mojom::PreferredColorScheme::kLight; |
| web_prefs->force_dark_mode_enabled = false; |
| } |
| } |
| |
| void AwDarkMode::DetachFromJavaObject(JNIEnv* env, |
| const JavaParamRef<jobject>& jcaller) { |
| jobj_.reset(); |
| } |
| |
| } // namespace android_webview |