blob: 820ed44c81841422e7941c0448509ccd15ef5384 [file] [log] [blame]
// 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