mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-20 21:44:06 +01:00
6354d8daf1
- Use raw_ptr in class container fields. - Use defined lifespan for StreamReaderURLLoader. - Fix lifespan assumptions for WebContents/RFH usage.
252 lines
7.6 KiB
C++
252 lines
7.6 KiB
C++
// Copyright 2021 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 "cef/libcef/browser/chrome/views/chrome_browser_frame.h"
|
|
|
|
#include "cef/libcef/browser/chrome/chrome_browser_host_impl.h"
|
|
#include "cef/libcef/browser/thread_util.h"
|
|
#include "cef/libcef/browser/views/window_view.h"
|
|
#include "chrome/browser/themes/theme_service.h"
|
|
#include "chrome/browser/themes/theme_service_factory.h"
|
|
#include "chrome/browser/ui/browser.h"
|
|
#include "chrome/browser/ui/browser_commands.h"
|
|
#include "chrome/browser/ui/views/frame/browser_view.h"
|
|
|
|
#if BUILDFLAG(IS_MAC)
|
|
#include "cef/libcef/browser/views/native_widget_mac.h"
|
|
#include "cef/libcef/browser/views/view_util.h"
|
|
#include "ui/views/widget/native_widget_private.h"
|
|
#endif
|
|
|
|
ChromeBrowserFrame::ChromeBrowserFrame(CefWindowView* window_view)
|
|
: window_view_(window_view) {}
|
|
|
|
ChromeBrowserFrame::~ChromeBrowserFrame() {
|
|
DCHECK(associated_profiles_.empty());
|
|
}
|
|
|
|
void ChromeBrowserFrame::Init(BrowserView* browser_view,
|
|
std::unique_ptr<Browser> browser) {
|
|
DCHECK(browser_view);
|
|
DCHECK(browser);
|
|
|
|
DCHECK(!BrowserFrame::browser_view());
|
|
|
|
// Initialize BrowserFrame state.
|
|
SetBrowserView(browser_view);
|
|
|
|
// Initialize BrowserView state.
|
|
browser_view->InitBrowser(std::move(browser));
|
|
|
|
#if BUILDFLAG(IS_MAC)
|
|
// Initialize native window state.
|
|
if (auto native_window = view_util::GetNativeWindow(this)) {
|
|
if (auto* native_widget_private = views::internal::NativeWidgetPrivate::
|
|
GetNativeWidgetForNativeWindow(native_window)) {
|
|
auto* native_widget_mac =
|
|
static_cast<CefNativeWidgetMac*>(native_widget_private);
|
|
native_widget_mac->SetBrowserView(browser_view);
|
|
native_widget_mac->OnWindowInitialized();
|
|
}
|
|
}
|
|
#endif // BUILDFLAG(IS_MAC)
|
|
}
|
|
|
|
void ChromeBrowserFrame::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 ChromeBrowserFrame::AddAssociatedProfile(Profile* profile) {
|
|
DCHECK(profile);
|
|
|
|
// Always call ThemeChanged() when the Chrome style BrowserView is added.
|
|
bool call_theme_changed =
|
|
browser_view() && browser_view()->GetProfile() == profile;
|
|
|
|
ProfileMap::iterator it = associated_profiles_.find(profile);
|
|
if (it != associated_profiles_.end()) {
|
|
// Another instance of a known Profile.
|
|
(it->second)++;
|
|
} else {
|
|
auto* current_profile = GetThemeProfile();
|
|
|
|
associated_profiles_.insert(std::make_pair(profile, 1));
|
|
|
|
if (auto* theme_service = ThemeServiceFactory::GetForProfile(profile)) {
|
|
theme_service->AddObserver(this);
|
|
}
|
|
|
|
// Potentially switching to a different theme.
|
|
call_theme_changed |= GetThemeProfile() != current_profile;
|
|
}
|
|
|
|
if (call_theme_changed) {
|
|
// Calls ThemeChanged().
|
|
UserChangedTheme(BrowserThemeChangeType::kBrowserTheme);
|
|
}
|
|
}
|
|
|
|
void ChromeBrowserFrame::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);
|
|
}
|
|
}
|
|
|
|
Profile* ChromeBrowserFrame::GetThemeProfile() const {
|
|
// Always prefer the Browser Profile, if any.
|
|
if (browser_view()) {
|
|
return browser_view()->GetProfile();
|
|
}
|
|
if (!associated_profiles_.empty()) {
|
|
return associated_profiles_.begin()->first;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool ChromeBrowserFrame::ToggleFullscreenMode() {
|
|
if (browser_view()) {
|
|
// Toggle fullscreen mode via the Chrome command for consistent behavior.
|
|
chrome::ToggleFullscreenMode(browser_view()->browser());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void ChromeBrowserFrame::UserChangedTheme(
|
|
BrowserThemeChangeType theme_change_type) {
|
|
// Callback from Browser::OnThemeChanged() and OnNativeThemeUpdated().
|
|
|
|
// Calls ThemeChanged() and possibly SelectNativeTheme().
|
|
BrowserFrame::UserChangedTheme(theme_change_type);
|
|
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/!native_theme_change_);
|
|
}
|
|
|
|
views::internal::RootView* ChromeBrowserFrame::CreateRootView() {
|
|
// Bypass the BrowserFrame implementation.
|
|
return views::Widget::CreateRootView();
|
|
}
|
|
|
|
std::unique_ptr<views::NonClientFrameView>
|
|
ChromeBrowserFrame::CreateNonClientFrameView() {
|
|
// Bypass the BrowserFrame implementation.
|
|
return views::Widget::CreateNonClientFrameView();
|
|
}
|
|
|
|
void ChromeBrowserFrame::Activate() {
|
|
if (browser_view() && browser_view()->browser() &&
|
|
browser_view()->browser()->is_type_devtools()) {
|
|
if (auto browser_host = ChromeBrowserHostImpl::GetBrowserForBrowser(
|
|
browser_view()->browser())) {
|
|
if (browser_host->platform_delegate()->HasExternalParent()) {
|
|
// Handle activation of DevTools with external parent via the platform
|
|
// delegate. On Windows the default platform implementation
|
|
// (HWNDMessageHandler::Activate) will call SetForegroundWindow but that
|
|
// doesn't seem to work for DevTools windows when activated via the
|
|
// right-click context menu.
|
|
browser_host->SetFocus(true);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Proceed with default handling.
|
|
BrowserFrame::Activate();
|
|
}
|
|
|
|
void ChromeBrowserFrame::OnNativeWidgetDestroyed() {
|
|
window_view_ = nullptr;
|
|
SetBrowserView(nullptr);
|
|
BrowserFrame::OnNativeWidgetDestroyed();
|
|
}
|
|
|
|
void ChromeBrowserFrame::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_theme_change_ = true;
|
|
|
|
// Calls UserChangedTheme().
|
|
BrowserFrame::OnNativeThemeUpdated(observed_theme);
|
|
|
|
native_theme_change_ = false;
|
|
}
|
|
|
|
ui::ColorProviderKey ChromeBrowserFrame::GetColorProviderKey() const {
|
|
if (browser_view()) {
|
|
// Use the default Browser implementation.
|
|
return BrowserFrame::GetColorProviderKey();
|
|
}
|
|
|
|
const auto& widget_key = Widget::GetColorProviderKey();
|
|
if (auto* profile = GetThemeProfile()) {
|
|
return CefWidget::GetColorProviderKey(widget_key, profile);
|
|
}
|
|
return widget_key;
|
|
}
|
|
|
|
void ChromeBrowserFrame::OnThemeChanged() {
|
|
if (browser_view()) {
|
|
// Ignore these notifications if we have a Browser.
|
|
return;
|
|
}
|
|
|
|
// When the Chrome theme changes, the NativeTheme may also change.
|
|
SelectNativeTheme();
|
|
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/true);
|
|
}
|
|
|
|
void ChromeBrowserFrame::OnColorProviderCacheResetMissed() {
|
|
// Ignore calls during Widget::Init().
|
|
if (!initialized_) {
|
|
return;
|
|
}
|
|
|
|
NotifyThemeColorsChanged(/*chrome_theme=*/false);
|
|
}
|
|
|
|
void ChromeBrowserFrame::NotifyThemeColorsChanged(bool chrome_theme) {
|
|
if (window_view_) {
|
|
window_view_->OnThemeColorsChanged(chrome_theme);
|
|
|
|
// Call ThemeChanged() asynchronously to avoid possible reentrancy.
|
|
CEF_POST_TASK(TID_UI, base::BindOnce(&ChromeBrowserFrame::ThemeChanged,
|
|
weak_ptr_factory_.GetWeakPtr()));
|
|
}
|
|
}
|