265 lines
8.1 KiB
C++
265 lines
8.1 KiB
C++
// Copyright 2024 The Chromium Embedded Framework 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 "libcef/browser/views/widget_impl.h"
|
|
|
|
#include "libcef/browser/thread_util.h"
|
|
#include "libcef/browser/views/window_view.h"
|
|
#include "libcef/features/runtime.h"
|
|
|
|
#include "chrome/browser/themes/custom_theme_supplier.h"
|
|
#include "chrome/browser/themes/theme_service.h"
|
|
#include "chrome/browser/themes/theme_service_factory.h"
|
|
|
|
#if BUILDFLAG(IS_LINUX)
|
|
#include "ui/linux/linux_ui.h"
|
|
#endif
|
|
|
|
namespace {
|
|
|
|
ui::ColorProviderKey::SchemeVariant GetSchemeVariant(
|
|
ui::mojom::BrowserColorVariant color_variant) {
|
|
using BCV = ui::mojom::BrowserColorVariant;
|
|
using SV = ui::ColorProviderKey::SchemeVariant;
|
|
static constexpr auto kSchemeVariantMap = base::MakeFixedFlatMap<BCV, SV>({
|
|
{BCV::kTonalSpot, SV::kTonalSpot},
|
|
{BCV::kNeutral, SV::kNeutral},
|
|
{BCV::kVibrant, SV::kVibrant},
|
|
{BCV::kExpressive, SV::kExpressive},
|
|
});
|
|
return kSchemeVariantMap.at(color_variant);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
CefWidgetImpl::CefWidgetImpl(CefWindowView* window_view)
|
|
: window_view_(window_view) {}
|
|
|
|
CefWidgetImpl::~CefWidgetImpl() {
|
|
DCHECK(associated_profiles_.empty());
|
|
}
|
|
|
|
void CefWidgetImpl::Initialized() {
|
|
initialized_ = true;
|
|
|
|
// Based on BrowserFrame::InitBrowserFrame.
|
|
// This is the first call that will trigger theme-related client callbacks.
|
|
#if BUILDFLAG(IS_LINUX)
|
|
// Calls ThemeChanged() or OnNativeThemeUpdated().
|
|
SelectNativeTheme();
|
|
#else
|
|
// Calls ThemeChanged().
|
|
SetNativeTheme(ui::NativeTheme::GetInstanceForNativeUi());
|
|
#endif
|
|
}
|
|
|
|
void CefWidgetImpl::AddAssociatedProfile(Profile* profile) {
|
|
DCHECK(profile);
|
|
ProfileMap::iterator it = associated_profiles_.find(profile);
|
|
if (it != associated_profiles_.end()) {
|
|
// Another instance of a known Profile.
|
|
(it->second)++;
|
|
return;
|
|
}
|
|
|
|
auto* current_profile = GetThemeProfile();
|
|
|
|
associated_profiles_.insert(std::make_pair(profile, 1));
|
|
|
|
if (auto* theme_service = ThemeServiceFactory::GetForProfile(profile)) {
|
|
theme_service->AddObserver(this);
|
|
}
|
|
|
|
auto* new_profile = GetThemeProfile();
|
|
if (new_profile != current_profile) {
|
|
// Switching to a different theme.
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/!!new_profile,
|
|
/*call_theme_changed=*/true);
|
|
}
|
|
}
|
|
|
|
void CefWidgetImpl::RemoveAssociatedProfile(Profile* profile) {
|
|
DCHECK(profile);
|
|
ProfileMap::iterator it = associated_profiles_.find(profile);
|
|
if (it == associated_profiles_.end()) {
|
|
DCHECK(false); // Not reached.
|
|
return;
|
|
}
|
|
if (--(it->second) > 0) {
|
|
// More instances of the Profile exist.
|
|
return;
|
|
}
|
|
|
|
auto* current_profile = GetThemeProfile();
|
|
|
|
associated_profiles_.erase(it);
|
|
|
|
if (auto* theme_service = ThemeServiceFactory::GetForProfile(profile)) {
|
|
theme_service->RemoveObserver(this);
|
|
}
|
|
|
|
auto* new_profile = GetThemeProfile();
|
|
if (new_profile != current_profile) {
|
|
// Switching to a different theme.
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/!!new_profile,
|
|
/*call_theme_changed=*/true);
|
|
}
|
|
}
|
|
|
|
Profile* CefWidgetImpl::GetThemeProfile() const {
|
|
if (!associated_profiles_.empty()) {
|
|
return associated_profiles_.begin()->first;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
const ui::ThemeProvider* CefWidgetImpl::GetThemeProvider() const {
|
|
auto* profile = GetThemeProfile();
|
|
if (!profile) {
|
|
return Widget::GetThemeProvider();
|
|
}
|
|
|
|
// Based on BrowserFrame::GetThemeProvider.
|
|
return &ThemeService::GetThemeProviderForProfile(profile);
|
|
}
|
|
|
|
ui::ColorProviderKey::ThemeInitializerSupplier* CefWidgetImpl::GetCustomTheme()
|
|
const {
|
|
auto* profile = GetThemeProfile();
|
|
if (!profile) {
|
|
return Widget::GetCustomTheme();
|
|
}
|
|
|
|
// Based on BrowserFrame::GetCustomTheme.
|
|
auto* theme_service = ThemeServiceFactory::GetForProfile(profile);
|
|
return theme_service->UsingDeviceTheme() ? nullptr
|
|
: theme_service->GetThemeSupplier();
|
|
}
|
|
|
|
void CefWidgetImpl::OnNativeWidgetDestroyed() {
|
|
window_view_ = nullptr;
|
|
views::Widget::OnNativeWidgetDestroyed();
|
|
}
|
|
|
|
void CefWidgetImpl::OnNativeThemeUpdated(ui::NativeTheme* observed_theme) {
|
|
// TODO: Reduce the frequency of this callback on Windows/Linux.
|
|
// See https://issues.chromium.org/issues/40280130#comment7
|
|
|
|
color_provider_tracker_.OnNativeThemeUpdated();
|
|
|
|
// Native/OS theme changed.
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/false,
|
|
/*call_theme_changed=*/false);
|
|
|
|
// Calls ThemeChanged().
|
|
Widget::OnNativeThemeUpdated(observed_theme);
|
|
}
|
|
|
|
ui::ColorProviderKey CefWidgetImpl::GetColorProviderKey() const {
|
|
auto* profile = GetThemeProfile();
|
|
if (!profile) {
|
|
return Widget::GetColorProviderKey();
|
|
}
|
|
|
|
// Based on BrowserFrame::GetColorProviderKey.
|
|
auto key = Widget::GetColorProviderKey();
|
|
|
|
const auto* theme_service = ThemeServiceFactory::GetForProfile(profile);
|
|
CHECK(theme_service);
|
|
|
|
// color_mode.
|
|
[&key, theme_service]() {
|
|
const auto browser_color_scheme = theme_service->GetBrowserColorScheme();
|
|
if (browser_color_scheme != ThemeService::BrowserColorScheme::kSystem) {
|
|
key.color_mode =
|
|
browser_color_scheme == ThemeService::BrowserColorScheme::kLight
|
|
? ui::ColorProviderKey::ColorMode::kLight
|
|
: ui::ColorProviderKey::ColorMode::kDark;
|
|
}
|
|
}();
|
|
|
|
// user_color.
|
|
// Device theme retains the user_color from `Widget`.
|
|
if (!theme_service->UsingDeviceTheme()) {
|
|
if (theme_service->UsingAutogeneratedTheme()) {
|
|
key.user_color = theme_service->GetAutogeneratedThemeColor();
|
|
} else if (auto user_color = theme_service->GetUserColor()) {
|
|
key.user_color = user_color;
|
|
}
|
|
}
|
|
|
|
// user_color_source.
|
|
if (theme_service->UsingDeviceTheme()) {
|
|
key.user_color_source = ui::ColorProviderKey::UserColorSource::kAccent;
|
|
} else if (theme_service->GetIsGrayscale()) {
|
|
key.user_color_source = ui::ColorProviderKey::UserColorSource::kGrayscale;
|
|
} else if (theme_service->GetIsBaseline()) {
|
|
key.user_color_source = ui::ColorProviderKey::UserColorSource::kBaseline;
|
|
} else {
|
|
CHECK(key.user_color.has_value());
|
|
key.user_color_source = ui::ColorProviderKey::UserColorSource::kAccent;
|
|
}
|
|
|
|
// scheme_variant.
|
|
ui::mojom::BrowserColorVariant color_variant =
|
|
theme_service->GetBrowserColorVariant();
|
|
if (!theme_service->UsingDeviceTheme() &&
|
|
color_variant != ui::mojom::BrowserColorVariant::kSystem) {
|
|
key.scheme_variant = GetSchemeVariant(color_variant);
|
|
}
|
|
|
|
// frame_type.
|
|
key.frame_type = ui::ColorProviderKey::FrameType::kNative;
|
|
|
|
return key;
|
|
}
|
|
|
|
void CefWidgetImpl::OnThemeChanged() {
|
|
// When the Chrome theme changes, the NativeTheme may also change.
|
|
SelectNativeTheme();
|
|
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/true, /*call_theme_changed=*/true);
|
|
}
|
|
|
|
void CefWidgetImpl::NotifyThemeColorsChanged(bool chrome_theme,
|
|
bool call_theme_changed) {
|
|
if (window_view_) {
|
|
window_view_->OnThemeColorsChanged(chrome_theme);
|
|
if (call_theme_changed) {
|
|
// Call ThemeChanged() asynchronously to avoid possible reentrancy.
|
|
CEF_POST_TASK(TID_UI, base::BindOnce(&CefWidgetImpl::ThemeChanged,
|
|
weak_ptr_factory_.GetWeakPtr()));
|
|
}
|
|
}
|
|
}
|
|
|
|
void CefWidgetImpl::SelectNativeTheme() {
|
|
// Based on BrowserFrame::SelectNativeTheme.
|
|
#if BUILDFLAG(IS_LINUX)
|
|
ui::NativeTheme* native_theme = ui::NativeTheme::GetInstanceForNativeUi();
|
|
|
|
// Always use the NativeTheme for forced color modes.
|
|
if (ui::NativeTheme::IsForcedDarkMode() ||
|
|
ui::NativeTheme::IsForcedLightMode()) {
|
|
SetNativeTheme(native_theme);
|
|
return;
|
|
}
|
|
|
|
const auto* linux_ui_theme =
|
|
ui::LinuxUiTheme::GetForWindow(GetNativeWindow());
|
|
SetNativeTheme(linux_ui_theme ? linux_ui_theme->GetNativeTheme()
|
|
: native_theme);
|
|
#endif
|
|
}
|
|
|
|
void CefWidgetImpl::OnColorProviderCacheResetMissed() {
|
|
// Ignore calls during Widget::Init().
|
|
if (!initialized_) {
|
|
return;
|
|
}
|
|
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/false,
|
|
/*call_theme_changed=*/true);
|
|
}
|