mirror of
				https://bitbucket.org/chromiumembedded/cef
				synced 2025-06-05 21:39:12 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			1458 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1458 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (c) 2012 The Chromium Embedded Framework Authors.
 | |
| // Portions copyright (c) 2011 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 "cef/libcef/browser/alloy/alloy_browser_host_impl.h"
 | |
| 
 | |
| #include <memory>
 | |
| #include <string>
 | |
| #include <utility>
 | |
| 
 | |
| #include "base/command_line.h"
 | |
| #include "base/functional/bind.h"
 | |
| #include "base/functional/callback_helpers.h"
 | |
| #include "cef/libcef/browser/alloy/browser_platform_delegate_alloy.h"
 | |
| #include "cef/libcef/browser/audio_capturer.h"
 | |
| #include "cef/libcef/browser/browser_context.h"
 | |
| #include "cef/libcef/browser/browser_guest_util.h"
 | |
| #include "cef/libcef/browser/browser_info.h"
 | |
| #include "cef/libcef/browser/browser_info_manager.h"
 | |
| #include "cef/libcef/browser/browser_platform_delegate.h"
 | |
| #include "cef/libcef/browser/context.h"
 | |
| #include "cef/libcef/browser/hang_monitor.h"
 | |
| #include "cef/libcef/browser/media_access_query.h"
 | |
| #include "cef/libcef/browser/osr/osr_util.h"
 | |
| #include "cef/libcef/browser/request_context_impl.h"
 | |
| #include "cef/libcef/browser/thread_util.h"
 | |
| #include "cef/libcef/common/cef_switches.h"
 | |
| #include "cef/libcef/common/drag_data_impl.h"
 | |
| #include "cef/libcef/common/frame_util.h"
 | |
| #include "cef/libcef/common/net/url_util.h"
 | |
| #include "cef/libcef/common/request_impl.h"
 | |
| #include "cef/libcef/common/values_impl.h"
 | |
| #include "chrome/browser/file_select_helper.h"
 | |
| #include "chrome/browser/picture_in_picture/picture_in_picture_window_manager.h"
 | |
| #include "chrome/common/webui_url_constants.h"
 | |
| #include "components/input/native_web_keyboard_event.h"
 | |
| #include "components/zoom/page_zoom.h"
 | |
| #include "content/browser/gpu/compositor_util.h"
 | |
| #include "content/public/browser/desktop_media_id.h"
 | |
| #include "content/public/browser/file_select_listener.h"
 | |
| #include "content/public/browser/keyboard_event_processing_result.h"
 | |
| #include "content/public/browser/navigation_controller.h"
 | |
| #include "content/public/browser/navigation_handle.h"
 | |
| #include "content/public/browser/render_frame_host.h"
 | |
| #include "content/public/browser/render_process_host.h"
 | |
| #include "content/public/browser/render_view_host.h"
 | |
| #include "content/public/browser/render_widget_host.h"
 | |
| #include "content/public/browser/web_contents.h"
 | |
| #include "content/public/common/url_constants.h"
 | |
| #include "extensions/common/constants.h"
 | |
| #include "net/base/net_errors.h"
 | |
| #include "ui/events/base_event_utils.h"
 | |
| 
 | |
| using content::KeyboardEventProcessingResult;
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| static constexpr base::TimeDelta kRecentlyAudibleTimeout = base::Seconds(2);
 | |
| 
 | |
| // List of WebUI hosts that have been tested to work in Alloy-style browsers.
 | |
| // Do not add new hosts to this list without also manually testing all related
 | |
| // functionality in CEF.
 | |
| const char* kAllowedWebUIHosts[] = {
 | |
|     chrome::kChromeUIAccessibilityHost,
 | |
|     content::kChromeUIBlobInternalsHost,
 | |
|     chrome::kChromeUIChromeURLsHost,
 | |
|     chrome::kChromeUICreditsHost,
 | |
|     content::kChromeUIGpuHost,
 | |
|     content::kChromeUIHistogramHost,
 | |
|     content::kChromeUIIndexedDBInternalsHost,
 | |
|     chrome::kChromeUILicenseHost,
 | |
|     content::kChromeUIMediaInternalsHost,
 | |
|     chrome::kChromeUINetExportHost,
 | |
|     chrome::kChromeUINetInternalsHost,
 | |
|     content::kChromeUINetworkErrorHost,
 | |
|     content::kChromeUINetworkErrorsListingHost,
 | |
|     chrome::kChromeUIPrintHost,
 | |
|     content::kChromeUIProcessInternalsHost,
 | |
|     content::kChromeUIResourcesHost,
 | |
| #if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_LINUX)
 | |
|     chrome::kChromeUISandboxHost,
 | |
| #endif
 | |
|     content::kChromeUIServiceWorkerInternalsHost,
 | |
|     chrome::kChromeUISystemInfoHost,
 | |
|     chrome::kChromeUITermsHost,
 | |
|     chrome::kChromeUIThemeHost,
 | |
|     content::kChromeUITracingHost,
 | |
|     chrome::kChromeUIVersionHost,
 | |
|     content::kChromeUIWebRTCInternalsHost,
 | |
| };
 | |
| 
 | |
| bool IsAllowedWebUIHost(const std::string_view& host) {
 | |
|   for (auto& allowed_host : kAllowedWebUIHosts) {
 | |
|     if (base::EqualsCaseInsensitiveASCII(allowed_host, host)) {
 | |
|       return true;
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| }  // namespace
 | |
| 
 | |
| // AlloyBrowserHostImpl static methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::Create(
 | |
|     CefBrowserCreateParams& create_params) {
 | |
|   std::unique_ptr<CefBrowserPlatformDelegate> platform_delegate =
 | |
|       CefBrowserPlatformDelegate::Create(create_params);
 | |
|   CHECK(platform_delegate);
 | |
| 
 | |
|   // Expect runtime style to match.
 | |
|   CHECK(platform_delegate->IsAlloyStyle());
 | |
| 
 | |
|   scoped_refptr<CefBrowserInfo> info =
 | |
|       CefBrowserInfoManager::GetInstance()->CreateBrowserInfo(
 | |
|           /*is_devtools_popup=*/false, platform_delegate->IsWindowless(),
 | |
|           platform_delegate->IsPrintPreviewSupported(),
 | |
|           create_params.extra_info);
 | |
| 
 | |
|   bool own_web_contents = false;
 | |
| 
 | |
|   // This call may modify |create_params|.
 | |
|   auto web_contents =
 | |
|       platform_delegate->CreateWebContents(create_params, own_web_contents);
 | |
| 
 | |
|   auto request_context_impl =
 | |
|       static_cast<CefRequestContextImpl*>(create_params.request_context.get());
 | |
| 
 | |
|   CefRefPtr<AlloyBrowserHostImpl> browser =
 | |
|       CreateInternal(create_params.settings, create_params.client, web_contents,
 | |
|                      own_web_contents, info,
 | |
|                      /*opener=*/nullptr, /*is_devtools_popup=*/false,
 | |
|                      request_context_impl, std::move(platform_delegate));
 | |
|   if (!browser) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   GURL url = url_util::MakeGURL(create_params.url, /*fixup=*/true);
 | |
| 
 | |
|   if (!url.is_empty()) {
 | |
|     content::OpenURLParams params(url, content::Referrer(),
 | |
|                                   WindowOpenDisposition::CURRENT_TAB,
 | |
|                                   CefFrameHostImpl::kPageTransitionExplicit,
 | |
|                                   /*is_renderer_initiated=*/false);
 | |
|     browser->LoadMainFrameURL(params);
 | |
|   }
 | |
| 
 | |
|   return browser.get();
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::CreateInternal(
 | |
|     const CefBrowserSettings& settings,
 | |
|     CefRefPtr<CefClient> client,
 | |
|     content::WebContents* web_contents,
 | |
|     bool own_web_contents,
 | |
|     scoped_refptr<CefBrowserInfo> browser_info,
 | |
|     CefRefPtr<AlloyBrowserHostImpl> opener,
 | |
|     bool is_devtools_popup,
 | |
|     CefRefPtr<CefRequestContextImpl> request_context,
 | |
|     std::unique_ptr<CefBrowserPlatformDelegate> platform_delegate) {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   DCHECK(web_contents);
 | |
|   DCHECK(browser_info);
 | |
|   DCHECK(request_context);
 | |
|   DCHECK(platform_delegate);
 | |
| 
 | |
|   // If |opener| is non-NULL it must be a popup window.
 | |
|   DCHECK(!opener.get() || browser_info->is_popup());
 | |
| 
 | |
|   if (opener) {
 | |
|     if (!opener->platform_delegate_) {
 | |
|       // The opener window is being destroyed. Cancel the popup.
 | |
|       if (own_web_contents) {
 | |
|         delete web_contents;
 | |
|       }
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     // Give the opener browser's platform delegate an opportunity to modify the
 | |
|     // new browser's platform delegate.
 | |
|     opener->platform_delegate_->PopupWebContentsCreated(
 | |
|         settings, client, web_contents, platform_delegate.get(),
 | |
|         is_devtools_popup);
 | |
|   }
 | |
| 
 | |
|   // Take ownership of |web_contents| if |own_web_contents| is true.
 | |
|   platform_delegate->WebContentsCreated(web_contents, own_web_contents);
 | |
| 
 | |
|   CefRefPtr<AlloyBrowserHostImpl> browser = new AlloyBrowserHostImpl(
 | |
|       settings, client, web_contents, browser_info, opener, request_context,
 | |
|       std::move(platform_delegate));
 | |
|   browser->InitializeBrowser();
 | |
| 
 | |
|   if (!browser->CreateHostWindow()) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   // Notify that the browser has been created. These must be delivered in the
 | |
|   // expected order.
 | |
| 
 | |
|   if (opener && opener->platform_delegate_) {
 | |
|     // 1. Notify the opener browser's platform delegate. With Views this will
 | |
|     // result in a call to CefBrowserViewDelegate::OnPopupBrowserViewCreated().
 | |
|     // Do this first for consistency with Chrome style.
 | |
|     opener->platform_delegate_->PopupBrowserCreated(
 | |
|         browser->platform_delegate(), browser.get(), is_devtools_popup);
 | |
|   }
 | |
| 
 | |
|   // 2. Notify the browser's LifeSpanHandler. This must always be the first
 | |
|   // notification for the browser. Block navigation to avoid issues with focus
 | |
|   // changes being sent to an unbound interface.
 | |
|   {
 | |
|     auto navigation_lock = browser_info->CreateNavigationLock();
 | |
|     browser->OnAfterCreated();
 | |
|   }
 | |
| 
 | |
|   // 3. Notify the platform delegate. With Views this will result in a call to
 | |
|   // CefBrowserViewDelegate::OnBrowserCreated().
 | |
|   browser->platform_delegate_->NotifyBrowserCreated();
 | |
| 
 | |
|   return browser;
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::FromBaseChecked(
 | |
|     CefRefPtr<CefBrowserHostBase> host_base) {
 | |
|   if (!host_base) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   CHECK(host_base->IsAlloyStyle());
 | |
|   return static_cast<AlloyBrowserHostImpl*>(host_base.get());
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::GetBrowserForHost(
 | |
|     const content::RenderViewHost* host) {
 | |
|   return FromBaseChecked(CefBrowserHostBase::GetBrowserForHost(host));
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::GetBrowserForHost(
 | |
|     const content::RenderFrameHost* host) {
 | |
|   return FromBaseChecked(CefBrowserHostBase::GetBrowserForHost(host));
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::GetBrowserForContents(
 | |
|     const content::WebContents* contents) {
 | |
|   return FromBaseChecked(CefBrowserHostBase::GetBrowserForContents(contents));
 | |
| }
 | |
| 
 | |
| // static
 | |
| CefRefPtr<AlloyBrowserHostImpl> AlloyBrowserHostImpl::GetBrowserForGlobalId(
 | |
|     const content::GlobalRenderFrameHostId& global_id) {
 | |
|   return FromBaseChecked(CefBrowserHostBase::GetBrowserForGlobalId(global_id));
 | |
| }
 | |
| 
 | |
| // AlloyBrowserHostImpl methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| AlloyBrowserHostImpl::~AlloyBrowserHostImpl() = default;
 | |
| 
 | |
| void AlloyBrowserHostImpl::CloseBrowser(bool force_close) {
 | |
|   if (CEF_CURRENTLY_ON_UIT()) {
 | |
|     // Exit early if a close attempt is already pending and this method is
 | |
|     // called again from somewhere other than WindowDestroyed().
 | |
|     if (destruction_state_ >= DESTRUCTION_STATE_PENDING &&
 | |
|         (IsWindowless() || !window_destroyed_)) {
 | |
|       if (force_close && destruction_state_ == DESTRUCTION_STATE_PENDING) {
 | |
|         // Upgrade the destruction state.
 | |
|         destruction_state_ = DESTRUCTION_STATE_ACCEPTED;
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (destruction_state_ < DESTRUCTION_STATE_ACCEPTED) {
 | |
|       destruction_state_ = (force_close ? DESTRUCTION_STATE_ACCEPTED
 | |
|                                         : DESTRUCTION_STATE_PENDING);
 | |
|     }
 | |
| 
 | |
|     content::WebContents* contents = web_contents();
 | |
|     if (contents && contents->NeedToFireBeforeUnloadOrUnloadEvents()) {
 | |
|       // Will result in a call to BeforeUnloadFired() and, if the close isn't
 | |
|       // canceled, CloseContents().
 | |
|       contents->DispatchBeforeUnload(false /* auto_cancel */);
 | |
|     } else {
 | |
|       CloseContents(contents);
 | |
|     }
 | |
|   } else {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::CloseBrowser,
 | |
|                                           this, force_close));
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::TryCloseBrowser() {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     DCHECK(false) << "called on invalid thread";
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Protect against multiple requests to close while the close is pending.
 | |
|   if (destruction_state_ <= DESTRUCTION_STATE_PENDING) {
 | |
|     if (destruction_state_ == DESTRUCTION_STATE_NONE) {
 | |
|       // Request that the browser close.
 | |
|       CloseBrowser(false);
 | |
|     }
 | |
| 
 | |
|     // Cancel the close.
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Allow the close.
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| CefWindowHandle AlloyBrowserHostImpl::GetWindowHandle() {
 | |
|   if (is_views_hosted_ && CEF_CURRENTLY_ON_UIT()) {
 | |
|     // Always return the most up-to-date window handle for a views-hosted
 | |
|     // browser since it may change if the view is re-parented.
 | |
|     if (platform_delegate_) {
 | |
|       return platform_delegate_->GetHostWindowHandle();
 | |
|     }
 | |
|   }
 | |
|   return host_window_handle_;
 | |
| }
 | |
| 
 | |
| CefWindowHandle AlloyBrowserHostImpl::GetOpenerWindowHandle() {
 | |
|   return opener_;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::Find(const CefString& searchText,
 | |
|                                 bool forward,
 | |
|                                 bool matchCase,
 | |
|                                 bool findNext) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::Find, this, searchText,
 | |
|                                  forward, matchCase, findNext));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->Find(searchText, forward, matchCase, findNext);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::StopFinding(bool clearSelection) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::StopFinding,
 | |
|                                           this, clearSelection));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->StopFinding(clearSelection);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SetAutoResizeEnabled(bool enabled,
 | |
|                                                 const CefSize& min_size,
 | |
|                                                 const CefSize& max_size) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::SetAutoResizeEnabled,
 | |
|                                  this, enabled, min_size, max_size));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SetAutoResizeEnabled(enabled, min_size, max_size);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::CanExecuteChromeCommand(int command_id) {
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ExecuteChromeCommand(
 | |
|     int command_id,
 | |
|     cef_window_open_disposition_t disposition) {
 | |
|   NOTIMPLEMENTED();
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsWindowRenderingDisabled() {
 | |
|   return IsWindowless();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::WasResized() {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::WasResized, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->WasResized();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::WasHidden(bool hidden) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&CefBrowserHost::WasHidden, this, hidden));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->WasHidden(hidden);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::NotifyScreenInfoChanged() {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::NotifyScreenInfoChanged, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->NotifyScreenInfoChanged();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::Invalidate(PaintElementType type) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::Invalidate, this, type));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->Invalidate(type);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SendExternalBeginFrame() {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::SendExternalBeginFrame, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SendExternalBeginFrame();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SendTouchEvent(const CefTouchEvent& event) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::SendTouchEvent,
 | |
|                                           this, event));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SendTouchEvent(event);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SendCaptureLostEvent() {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::SendCaptureLostEvent, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SendCaptureLostEvent();
 | |
|   }
 | |
| }
 | |
| 
 | |
| int AlloyBrowserHostImpl::GetWindowlessFrameRate() {
 | |
|   // Verify that this method is being called on the UI thread.
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     DCHECK(false) << "called on invalid thread";
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return osr_util::ClampFrameRate(settings_.windowless_frame_rate);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SetWindowlessFrameRate(int frame_rate) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::SetWindowlessFrameRate,
 | |
|                                  this, frame_rate));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   settings_.windowless_frame_rate = frame_rate;
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SetWindowlessFrameRate(frame_rate);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // AlloyBrowserHostImpl public methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsWindowless() const {
 | |
|   return is_windowless_;
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsVisible() const {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   if (IsWindowless() && platform_delegate_) {
 | |
|     return !platform_delegate_->IsHidden();
 | |
|   }
 | |
|   return CefBrowserHostBase::IsVisible();
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsPictureInPictureSupported() const {
 | |
|   // Not currently supported with OSR.
 | |
|   return !IsWindowless();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::WindowDestroyed() {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   DCHECK(!window_destroyed_);
 | |
|   window_destroyed_ = true;
 | |
|   CloseBrowser(true);
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::WillBeDestroyed() const {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   return destruction_state_ >= DESTRUCTION_STATE_ACCEPTED;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DestroyBrowser() {
 | |
|   CEF_REQUIRE_UIT();
 | |
| 
 | |
|   destruction_state_ = DESTRUCTION_STATE_COMPLETED;
 | |
| 
 | |
|   // Destroy any platform constructs first.
 | |
|   if (javascript_dialog_manager_.get()) {
 | |
|     javascript_dialog_manager_->Destroy();
 | |
|   }
 | |
|   if (menu_manager_.get()) {
 | |
|     menu_manager_->Destroy();
 | |
|   }
 | |
| 
 | |
|   // Disassociate the platform delegate from this browser. This will trigger
 | |
|   // WebContents destruction in most cases.
 | |
|   platform_delegate_->BrowserDestroyed(this);
 | |
| 
 | |
|   // Delete objects created by the platform delegate that may be referenced by
 | |
|   // the WebContents.
 | |
|   file_dialog_manager_.reset(nullptr);
 | |
|   javascript_dialog_manager_.reset(nullptr);
 | |
|   menu_manager_.reset(nullptr);
 | |
| 
 | |
|   // Delete the audio capturer
 | |
|   if (recently_audible_timer_) {
 | |
|     recently_audible_timer_->Stop();
 | |
|     recently_audible_timer_.reset();
 | |
|   }
 | |
|   audio_capturer_.reset(nullptr);
 | |
| 
 | |
|   CefBrowserHostBase::DestroyBrowser();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::CancelContextMenu() {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   if (menu_manager_) {
 | |
|     menu_manager_->CancelContextMenu();
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::MaybeAllowNavigation(
 | |
|     content::RenderFrameHost* opener,
 | |
|     const content::OpenURLParams& params) {
 | |
|   const bool is_guest_view =
 | |
|       IsBrowserPluginGuest(content::WebContents::FromRenderFrameHost(opener));
 | |
|   if (is_guest_view && !params.is_pdf &&
 | |
|       !params.url.SchemeIs(extensions::kExtensionScheme) &&
 | |
|       !params.url.SchemeIs(content::kChromeUIScheme)) {
 | |
|     // The PDF viewer will load the PDF extension in the guest view, and print
 | |
|     // preview will load chrome://print in the guest view. The PDF renderer
 | |
|     // used with PdfUnseasoned will set |params.is_pdf| when loading the PDF
 | |
|     // stream (see PdfNavigationThrottle::WillStartRequest). All other guest
 | |
|     // view navigations are passed to the owner browser.
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(
 | |
|                       base::IgnoreResult(&AlloyBrowserHostImpl::OpenURLFromTab),
 | |
|                       this, nullptr, params, base::NullCallback()));
 | |
| 
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (!is_guest_view && params.url.SchemeIs(content::kChromeUIScheme) &&
 | |
|       !IsAllowedWebUIHost(params.url.host_piece())) {
 | |
|     // Block navigation to non-allowlisted WebUI pages.
 | |
|     LOG(WARNING) << "Navigation to " << params.url.spec()
 | |
|                  << " is blocked in Alloy-style browser.";
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::OnSetFocus(cef_focus_source_t source) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::OnSetFocus,
 | |
|                                           this, source));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (contents_delegate_.OnSetFocus(source)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SetFocus(true);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::EnterFullscreenModeForTab(
 | |
|     content::RenderFrameHost* requesting_frame,
 | |
|     const blink::mojom::FullscreenOptions& options) {
 | |
|   contents_delegate_.EnterFullscreenModeForTab(requesting_frame, options);
 | |
|   WasResized();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ExitFullscreenModeForTab(
 | |
|     content::WebContents* web_contents) {
 | |
|   contents_delegate_.ExitFullscreenModeForTab(web_contents);
 | |
|   WasResized();
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsFullscreenForTabOrPending(
 | |
|     const content::WebContents* web_contents) {
 | |
|   return is_fullscreen_;
 | |
| }
 | |
| 
 | |
| blink::mojom::DisplayMode AlloyBrowserHostImpl::GetDisplayMode(
 | |
|     const content::WebContents* web_contents) {
 | |
|   return is_fullscreen_ ? blink::mojom::DisplayMode::kFullscreen
 | |
|                         : blink::mojom::DisplayMode::kBrowser;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::FindReply(content::WebContents* web_contents,
 | |
|                                      int request_id,
 | |
|                                      int number_of_matches,
 | |
|                                      const gfx::Rect& selection_rect,
 | |
|                                      int active_match_ordinal,
 | |
|                                      bool final_update) {
 | |
|   auto alloy_delegate =
 | |
|       static_cast<CefBrowserPlatformDelegateAlloy*>(platform_delegate());
 | |
|   if (alloy_delegate->HandleFindReply(request_id, number_of_matches,
 | |
|                                       selection_rect, active_match_ordinal,
 | |
|                                       final_update)) {
 | |
|     if (client_) {
 | |
|       if (auto handler = client_->GetFindHandler()) {
 | |
|         const auto& details = alloy_delegate->last_search_result();
 | |
|         CefRect rect(details.selection_rect().x(), details.selection_rect().y(),
 | |
|                      details.selection_rect().width(),
 | |
|                      details.selection_rect().height());
 | |
|         handler->OnFindResult(
 | |
|             this, details.request_id(), details.number_of_matches(), rect,
 | |
|             details.active_match_ordinal(), details.final_update());
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ImeSetComposition(
 | |
|     const CefString& text,
 | |
|     const std::vector<CefCompositionUnderline>& underlines,
 | |
|     const CefRange& replacement_range,
 | |
|     const CefRange& selection_range) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::ImeSetComposition, this, text,
 | |
|                        underlines, replacement_range, selection_range));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->ImeSetComposition(text, underlines, replacement_range,
 | |
|                                           selection_range);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ImeCommitText(const CefString& text,
 | |
|                                          const CefRange& replacement_range,
 | |
|                                          int relative_cursor_pos) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::ImeCommitText, this,
 | |
|                                  text, replacement_range, relative_cursor_pos));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->ImeCommitText(text, replacement_range,
 | |
|                                       relative_cursor_pos);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ImeFinishComposingText(bool keep_selection) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::ImeFinishComposingText,
 | |
|                                  this, keep_selection));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->ImeFinishComposingText(keep_selection);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ImeCancelComposition() {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::ImeCancelComposition, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->ImeCancelComposition();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragTargetDragEnter(
 | |
|     CefRefPtr<CefDragData> drag_data,
 | |
|     const CefMouseEvent& event,
 | |
|     CefBrowserHost::DragOperationsMask allowed_ops) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::DragTargetDragEnter,
 | |
|                                  this, drag_data, event, allowed_ops));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!drag_data.get()) {
 | |
|     DCHECK(false);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragTargetDragEnter(drag_data, event, allowed_ops);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragTargetDragOver(
 | |
|     const CefMouseEvent& event,
 | |
|     CefBrowserHost::DragOperationsMask allowed_ops) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::DragTargetDragOver, this,
 | |
|                                 event, allowed_ops));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragTargetDragOver(event, allowed_ops);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragTargetDragLeave() {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::DragTargetDragLeave, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragTargetDragLeave();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragTargetDrop(const CefMouseEvent& event) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::DragTargetDrop,
 | |
|                                           this, event));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragTargetDrop(event);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragSourceSystemDragEnded() {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(
 | |
|         CEF_UIT,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::DragSourceSystemDragEnded, this));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragSourceSystemDragEnded();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DragSourceEndedAt(
 | |
|     int x,
 | |
|     int y,
 | |
|     CefBrowserHost::DragOperationsMask op) {
 | |
|   if (!IsWindowless()) {
 | |
|     DCHECK(false) << "Window rendering is not disabled";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT,
 | |
|                   base::BindOnce(&AlloyBrowserHostImpl::DragSourceEndedAt, this,
 | |
|                                  x, y, op));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->DragSourceEndedAt(x, y, op);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::SetAudioMuted(bool mute) {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::SetAudioMuted,
 | |
|                                           this, mute));
 | |
|     return;
 | |
|   }
 | |
|   if (!web_contents()) {
 | |
|     return;
 | |
|   }
 | |
|   web_contents()->SetAudioMuted(mute);
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsAudioMuted() {
 | |
|   if (!CEF_CURRENTLY_ON_UIT()) {
 | |
|     DCHECK(false) << "called on invalid thread";
 | |
|     return false;
 | |
|   }
 | |
|   if (!web_contents()) {
 | |
|     return false;
 | |
|   }
 | |
|   return web_contents()->IsAudioMuted();
 | |
| }
 | |
| 
 | |
| // content::WebContentsDelegate methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| content::WebContents* AlloyBrowserHostImpl::OpenURLFromTab(
 | |
|     content::WebContents* source,
 | |
|     const content::OpenURLParams& params,
 | |
|     base::OnceCallback<void(content::NavigationHandle&)>
 | |
|         navigation_handle_callback) {
 | |
|   auto target_contents = contents_delegate_.OpenURLFromTabEx(
 | |
|       source, params, navigation_handle_callback);
 | |
|   if (target_contents) {
 | |
|     // Start a navigation in the current browser that will result in the
 | |
|     // creation of a new render process.
 | |
|     LoadMainFrameURL(params);
 | |
|     return target_contents;
 | |
|   }
 | |
| 
 | |
|   // Cancel the navigation.
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::AddNewContents(
 | |
|     content::WebContents* source,
 | |
|     std::unique_ptr<content::WebContents> new_contents,
 | |
|     const GURL& target_url,
 | |
|     WindowOpenDisposition disposition,
 | |
|     const blink::mojom::WindowFeatures& window_features,
 | |
|     bool user_gesture,
 | |
|     bool* was_blocked) {
 | |
|   platform_delegate_->AddNewContents(source, std::move(new_contents),
 | |
|                                      target_url, disposition, window_features,
 | |
|                                      user_gesture, was_blocked);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::LoadingStateChanged(content::WebContents* source,
 | |
|                                                bool should_show_loading_ui) {
 | |
|   contents_delegate_.LoadingStateChanged(source, should_show_loading_ui);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::CloseContents(content::WebContents* source) {
 | |
|   CEF_REQUIRE_UIT();
 | |
| 
 | |
|   if (destruction_state_ == DESTRUCTION_STATE_COMPLETED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   bool close_browser = true;
 | |
| 
 | |
|   // If this method is called in response to something other than
 | |
|   // WindowDestroyed() ask the user if the browser should close.
 | |
|   if (client_.get() && (IsWindowless() || !window_destroyed_)) {
 | |
|     CefRefPtr<CefLifeSpanHandler> handler = client_->GetLifeSpanHandler();
 | |
|     if (handler.get()) {
 | |
|       close_browser = !handler->DoClose(this);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (close_browser) {
 | |
|     if (destruction_state_ != DESTRUCTION_STATE_ACCEPTED) {
 | |
|       destruction_state_ = DESTRUCTION_STATE_ACCEPTED;
 | |
|     }
 | |
| 
 | |
|     if (!IsWindowless() && !window_destroyed_) {
 | |
|       // A window exists so try to close it using the platform method. Will
 | |
|       // result in a call to WindowDestroyed() if/when the window is destroyed
 | |
|       // via the platform window destruction mechanism.
 | |
|       platform_delegate_->CloseHostWindow();
 | |
|     } else {
 | |
|       // Keep a reference to the browser while it's in the process of being
 | |
|       // destroyed.
 | |
|       CefRefPtr<AlloyBrowserHostImpl> browser(this);
 | |
| 
 | |
|       if (source) {
 | |
|         // Try to fast shutdown the associated process.
 | |
|         source->GetPrimaryMainFrame()->GetProcess()->FastShutdownIfPossible(
 | |
|             1, false);
 | |
|       }
 | |
| 
 | |
|       // No window exists. Destroy the browser immediately. Don't call other
 | |
|       // browser methods after calling DestroyBrowser().
 | |
|       DestroyBrowser();
 | |
|     }
 | |
|   } else if (destruction_state_ != DESTRUCTION_STATE_NONE) {
 | |
|     destruction_state_ = DESTRUCTION_STATE_NONE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::UpdateTargetURL(content::WebContents* source,
 | |
|                                            const GURL& url) {
 | |
|   contents_delegate_.UpdateTargetURL(source, url);
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::DidAddMessageToConsole(
 | |
|     content::WebContents* source,
 | |
|     blink::mojom::ConsoleMessageLevel level,
 | |
|     const std::u16string& message,
 | |
|     int32_t line_no,
 | |
|     const std::u16string& source_id) {
 | |
|   return contents_delegate_.DidAddMessageToConsole(source, level, message,
 | |
|                                                    line_no, source_id);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ContentsZoomChange(bool zoom_in) {
 | |
|   zoom::PageZoom::Zoom(
 | |
|       web_contents(), zoom_in ? content::PAGE_ZOOM_IN : content::PAGE_ZOOM_OUT);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::BeforeUnloadFired(content::WebContents* source,
 | |
|                                              bool proceed,
 | |
|                                              bool* proceed_to_fire_unload) {
 | |
|   if (destruction_state_ == DESTRUCTION_STATE_ACCEPTED || proceed) {
 | |
|     *proceed_to_fire_unload = true;
 | |
|   } else if (!proceed) {
 | |
|     *proceed_to_fire_unload = false;
 | |
|     destruction_state_ = DESTRUCTION_STATE_NONE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::TakeFocus(content::WebContents* source,
 | |
|                                      bool reverse) {
 | |
|   if (client_.get()) {
 | |
|     CefRefPtr<CefFocusHandler> handler = client_->GetFocusHandler();
 | |
|     if (handler.get()) {
 | |
|       handler->OnTakeFocus(this, !reverse);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::CanDownload(
 | |
|     const GURL& url,
 | |
|     const std::string& request_method,
 | |
|     base::OnceCallback<void(bool)> callback) {
 | |
|   contents_delegate_.CanDownload(url, request_method, std::move(callback));
 | |
| }
 | |
| 
 | |
| KeyboardEventProcessingResult AlloyBrowserHostImpl::PreHandleKeyboardEvent(
 | |
|     content::WebContents* source,
 | |
|     const input::NativeWebKeyboardEvent& event) {
 | |
|   return contents_delegate_.PreHandleKeyboardEvent(source, event);
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::HandleKeyboardEvent(
 | |
|     content::WebContents* source,
 | |
|     const input::NativeWebKeyboardEvent& event) {
 | |
|   // Check to see if event should be ignored.
 | |
|   if (event.skip_if_unhandled) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (contents_delegate_.HandleKeyboardEvent(source, event)) {
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   if (platform_delegate_) {
 | |
|     return platform_delegate_->HandleKeyboardEvent(event);
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::CanDragEnter(content::WebContents* source,
 | |
|                                         const content::DropData& data,
 | |
|                                         blink::DragOperationsMask mask) {
 | |
|   CefRefPtr<CefDragHandler> handler;
 | |
|   if (client_) {
 | |
|     handler = client_->GetDragHandler();
 | |
|   }
 | |
|   if (handler) {
 | |
|     CefRefPtr<CefDragDataImpl> drag_data(new CefDragDataImpl(data));
 | |
|     drag_data->SetReadOnly(true);
 | |
|     if (handler->OnDragEnter(
 | |
|             this, drag_data.get(),
 | |
|             static_cast<CefDragHandler::DragOperationsMask>(mask))) {
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::GetCustomWebContentsView(
 | |
|     content::WebContents* web_contents,
 | |
|     const GURL& target_url,
 | |
|     int opener_render_process_id,
 | |
|     int opener_render_frame_id,
 | |
|     raw_ptr<content::WebContentsView>* view,
 | |
|     raw_ptr<content::RenderViewHostDelegateView>* delegate_view) {
 | |
|   CefBrowserInfoManager::GetInstance()->GetCustomWebContentsView(
 | |
|       target_url,
 | |
|       frame_util::MakeGlobalId(opener_render_process_id,
 | |
|                                opener_render_frame_id),
 | |
|       view, delegate_view);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::WebContentsCreated(
 | |
|     content::WebContents* source_contents,
 | |
|     int opener_render_process_id,
 | |
|     int opener_render_frame_id,
 | |
|     const std::string& frame_name,
 | |
|     const GURL& target_url,
 | |
|     content::WebContents* new_contents) {
 | |
|   CefBrowserSettings settings;
 | |
|   CefRefPtr<CefClient> client;
 | |
|   std::unique_ptr<CefBrowserPlatformDelegate> platform_delegate;
 | |
|   CefRefPtr<CefDictionaryValue> extra_info;
 | |
| 
 | |
|   CefBrowserInfoManager::GetInstance()->WebContentsCreated(
 | |
|       target_url,
 | |
|       frame_util::MakeGlobalId(opener_render_process_id,
 | |
|                                opener_render_frame_id),
 | |
|       settings, client, platform_delegate, extra_info, new_contents);
 | |
| 
 | |
|   scoped_refptr<CefBrowserInfo> info =
 | |
|       CefBrowserInfoManager::GetInstance()->CreatePopupBrowserInfo(
 | |
|           new_contents, platform_delegate->IsWindowless(),
 | |
|           platform_delegate->IsPrintPreviewSupported(), extra_info);
 | |
|   CHECK(info.get());
 | |
|   CHECK(info->is_popup());
 | |
| 
 | |
|   CefRefPtr<AlloyBrowserHostImpl> opener =
 | |
|       GetBrowserForContents(source_contents);
 | |
|   if (!opener) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Popups must share the same RequestContext as the parent.
 | |
|   CefRefPtr<CefRequestContextImpl> request_context = opener->request_context();
 | |
|   CHECK(request_context);
 | |
| 
 | |
|   // We don't officially own |new_contents| until AddNewContents() is called.
 | |
|   // However, we need to install observers/delegates here.
 | |
|   CefRefPtr<AlloyBrowserHostImpl> browser = CreateInternal(
 | |
|       settings, client, new_contents, /*own_web_contents=*/false, info, opener,
 | |
|       /*is_devtools_popup=*/false, request_context,
 | |
|       std::move(platform_delegate));
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::RendererUnresponsive(
 | |
|     content::WebContents* source,
 | |
|     content::RenderWidgetHost* render_widget_host,
 | |
|     base::RepeatingClosure hang_monitor_restarter) {
 | |
|   hang_monitor::RendererUnresponsive(this, render_widget_host,
 | |
|                                      hang_monitor_restarter);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::RendererResponsive(
 | |
|     content::WebContents* source,
 | |
|     content::RenderWidgetHost* render_widget_host) {
 | |
|   hang_monitor::RendererResponsive(this, render_widget_host);
 | |
| }
 | |
| 
 | |
| content::JavaScriptDialogManager*
 | |
| AlloyBrowserHostImpl::GetJavaScriptDialogManager(content::WebContents* source) {
 | |
|   return CefBrowserHostBase::GetJavaScriptDialogManager();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::RunFileChooser(
 | |
|     content::RenderFrameHost* render_frame_host,
 | |
|     scoped_refptr<content::FileSelectListener> listener,
 | |
|     const blink::mojom::FileChooserParams& params) {
 | |
|   // This will eventually call into CefFileDialogManager.
 | |
|   FileSelectHelper::RunFileChooser(render_frame_host, std::move(listener),
 | |
|                                    params);
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::ShowContextMenu(
 | |
|     const content::ContextMenuParams& params) {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   if (!menu_manager_.get() && platform_delegate_) {
 | |
|     menu_manager_ = std::make_unique<CefMenuManager>(
 | |
|         this, platform_delegate_->CreateMenuRunner());
 | |
|   }
 | |
|   return menu_manager_->CreateContextMenu(params);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::UpdatePreferredSize(content::WebContents* source,
 | |
|                                                const gfx::Size& pref_size) {
 | |
| #if BUILDFLAG(IS_WIN) || (BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_MAC))
 | |
|   CEF_REQUIRE_UIT();
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->SizeTo(pref_size.width(), pref_size.height());
 | |
|   }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ResizeDueToAutoResize(content::WebContents* source,
 | |
|                                                  const gfx::Size& new_size) {
 | |
|   CEF_REQUIRE_UIT();
 | |
| 
 | |
|   if (client_) {
 | |
|     CefRefPtr<CefDisplayHandler> handler = client_->GetDisplayHandler();
 | |
|     if (handler && handler->OnAutoResize(
 | |
|                        this, CefSize(new_size.width(), new_size.height()))) {
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UpdatePreferredSize(source, new_size);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::RequestMediaAccessPermission(
 | |
|     content::WebContents* web_contents,
 | |
|     const content::MediaStreamRequest& request,
 | |
|     content::MediaResponseCallback callback) {
 | |
|   auto returned_callback = media_access_query::RequestMediaAccessPermission(
 | |
|       this, request, std::move(callback), /*default_disallow=*/true);
 | |
|   // Callback should not be returned.
 | |
|   DCHECK(returned_callback.is_null());
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::CheckMediaAccessPermission(
 | |
|     content::RenderFrameHost* render_frame_host,
 | |
|     const url::Origin& security_origin,
 | |
|     blink::mojom::MediaStreamType type) {
 | |
|   return media_access_query::CheckMediaAccessPermission(this, render_frame_host,
 | |
|                                                         security_origin, type);
 | |
| }
 | |
| 
 | |
| content::PictureInPictureResult AlloyBrowserHostImpl::EnterPictureInPicture(
 | |
|     content::WebContents* web_contents) {
 | |
|   if (!IsPictureInPictureSupported()) {
 | |
|     return content::PictureInPictureResult::kNotSupported;
 | |
|   }
 | |
| 
 | |
|   return PictureInPictureWindowManager::GetInstance()
 | |
|       ->EnterVideoPictureInPicture(web_contents);
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::ExitPictureInPicture() {
 | |
|   DCHECK(IsPictureInPictureSupported());
 | |
|   PictureInPictureWindowManager::GetInstance()->ExitPictureInPicture();
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::IsBackForwardCacheSupported(
 | |
|     content::WebContents& web_contents) {
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| content::PreloadingEligibility AlloyBrowserHostImpl::IsPrerender2Supported(
 | |
|     content::WebContents& web_contents) {
 | |
|   return content::PreloadingEligibility::kEligible;
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::DraggableRegionsChanged(
 | |
|     const std::vector<blink::mojom::DraggableRegionPtr>& regions,
 | |
|     content::WebContents* contents) {
 | |
|   contents_delegate_.DraggableRegionsChanged(regions, contents);
 | |
| }
 | |
| 
 | |
| // content::WebContentsObserver methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| void AlloyBrowserHostImpl::DidFinishNavigation(
 | |
|     content::NavigationHandle* navigation_handle) {
 | |
|   if (web_contents()) {
 | |
|     auto cef_browser_context = CefBrowserContext::FromBrowserContext(
 | |
|         web_contents()->GetBrowserContext());
 | |
|     if (cef_browser_context) {
 | |
|       cef_browser_context->AddVisitedURLs(
 | |
|           navigation_handle->GetURL(), navigation_handle->GetRedirectChain(),
 | |
|           navigation_handle->GetPageTransition());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::OnAudioStateChanged(bool audible) {
 | |
|   if (audible) {
 | |
|     if (recently_audible_timer_) {
 | |
|       recently_audible_timer_->Stop();
 | |
|     }
 | |
| 
 | |
|     StartAudioCapturer();
 | |
|   } else if (audio_capturer_) {
 | |
|     if (!recently_audible_timer_) {
 | |
|       recently_audible_timer_ = std::make_unique<base::OneShotTimer>();
 | |
|     }
 | |
| 
 | |
|     // If you have a media playing that has a short quiet moment, web_contents
 | |
|     // will immediately switch to non-audible state. We don't want to stop
 | |
|     // audio stream so quickly, let's give the stream some time to resume
 | |
|     // playing.
 | |
|     recently_audible_timer_->Start(
 | |
|         FROM_HERE, kRecentlyAudibleTimeout,
 | |
|         base::BindOnce(&AlloyBrowserHostImpl::OnRecentlyAudibleTimerFired,
 | |
|                        this));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::OnRecentlyAudibleTimerFired() {
 | |
|   audio_capturer_.reset();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::AccessibilityEventReceived(
 | |
|     const ui::AXUpdatesAndEvents& details) {
 | |
|   // Only needed in windowless mode.
 | |
|   if (IsWindowless()) {
 | |
|     if (!web_contents() || !platform_delegate_) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     platform_delegate_->AccessibilityEventReceived(details);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::AccessibilityLocationChangesReceived(
 | |
|     const std::vector<ui::AXLocationChanges>& details) {
 | |
|   // Only needed in windowless mode.
 | |
|   if (IsWindowless()) {
 | |
|     if (!web_contents() || !platform_delegate_) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     platform_delegate_->AccessibilityLocationChangesReceived(details);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::WebContentsDestroyed() {
 | |
|   // In case we're notified before the CefBrowserContentsDelegate,
 | |
|   // reset it first for consistent state in DestroyWebContents.
 | |
|   if (GetWebContents()) {
 | |
|     contents_delegate_.WebContentsDestroyed();
 | |
|   }
 | |
| 
 | |
|   auto wc = web_contents();
 | |
|   content::WebContentsObserver::Observe(nullptr);
 | |
|   DestroyWebContents(wc);
 | |
| 
 | |
|   if (destruction_state_ < DESTRUCTION_STATE_COMPLETED) {
 | |
|     // We were not called via DestroyBrowser. This can occur when (for example)
 | |
|     // a pending popup WebContents is destroyed during parent WebContents
 | |
|     // destruction. Try to close the associated browser now.
 | |
|     CEF_POST_TASK(CEF_UIT, base::BindOnce(&AlloyBrowserHostImpl::CloseBrowser,
 | |
|                                           this, /*force_close=*/true));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::StartAudioCapturer() {
 | |
|   if (!client_.get() || audio_capturer_) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   CefRefPtr<CefAudioHandler> audio_handler = client_->GetAudioHandler();
 | |
|   if (!audio_handler.get()) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   CefAudioParameters params;
 | |
|   params.channel_layout = CEF_CHANNEL_LAYOUT_STEREO;
 | |
|   params.sample_rate = media::AudioParameters::kAudioCDSampleRate;
 | |
|   params.frames_per_buffer = 1024;
 | |
| 
 | |
|   if (!audio_handler->GetAudioParameters(this, params)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   audio_capturer_ =
 | |
|       std::make_unique<CefAudioCapturer>(params, this, audio_handler);
 | |
| }
 | |
| 
 | |
| // AlloyBrowserHostImpl private methods.
 | |
| // -----------------------------------------------------------------------------
 | |
| 
 | |
| AlloyBrowserHostImpl::AlloyBrowserHostImpl(
 | |
|     const CefBrowserSettings& settings,
 | |
|     CefRefPtr<CefClient> client,
 | |
|     content::WebContents* web_contents,
 | |
|     scoped_refptr<CefBrowserInfo> browser_info,
 | |
|     CefRefPtr<AlloyBrowserHostImpl> opener,
 | |
|     CefRefPtr<CefRequestContextImpl> request_context,
 | |
|     std::unique_ptr<CefBrowserPlatformDelegate> platform_delegate)
 | |
|     : CefBrowserHostBase(settings,
 | |
|                          client,
 | |
|                          std::move(platform_delegate),
 | |
|                          browser_info,
 | |
|                          request_context),
 | |
|       content::WebContentsObserver(web_contents),
 | |
|       opener_(kNullWindowHandle),
 | |
|       is_windowless_(platform_delegate_->IsWindowless()) {
 | |
|   contents_delegate_.ObserveWebContents(web_contents);
 | |
| 
 | |
|   if (opener.get() && !is_views_hosted_) {
 | |
|     // GetOpenerWindowHandle() only returns a value for non-views-hosted
 | |
|     // popup browsers.
 | |
|     opener_ = opener->GetWindowHandle();
 | |
|   }
 | |
| 
 | |
|   // Associate the platform delegate with this browser.
 | |
|   platform_delegate_->BrowserCreated(this);
 | |
| 
 | |
|   // Make sure RenderFrameCreated is called at least one time.
 | |
|   RenderFrameCreated(web_contents->GetPrimaryMainFrame());
 | |
| }
 | |
| 
 | |
| bool AlloyBrowserHostImpl::CreateHostWindow() {
 | |
|   // |host_window_handle_| will not change after initial host creation for
 | |
|   // non-views-hosted browsers.
 | |
|   bool success = true;
 | |
|   if (!IsWindowless()) {
 | |
|     success = platform_delegate_->CreateHostWindow();
 | |
|   }
 | |
|   if (success && !is_views_hosted_) {
 | |
|     host_window_handle_ = platform_delegate_->GetHostWindowHandle();
 | |
|   }
 | |
|   return success;
 | |
| }
 | |
| 
 | |
| gfx::Point AlloyBrowserHostImpl::GetScreenPoint(const gfx::Point& view,
 | |
|                                                 bool want_dip_coords) const {
 | |
|   CEF_REQUIRE_UIT();
 | |
|   if (platform_delegate_) {
 | |
|     return platform_delegate_->GetScreenPoint(view, want_dip_coords);
 | |
|   }
 | |
|   return gfx::Point();
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::StartDragging(
 | |
|     const content::DropData& drop_data,
 | |
|     blink::DragOperationsMask allowed_ops,
 | |
|     const gfx::ImageSkia& image,
 | |
|     const gfx::Vector2d& image_offset,
 | |
|     const blink::mojom::DragEventSourceInfo& event_info,
 | |
|     content::RenderWidgetHostImpl* source_rwh) {
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->StartDragging(drop_data, allowed_ops, image,
 | |
|                                       image_offset, event_info, source_rwh);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AlloyBrowserHostImpl::UpdateDragOperation(
 | |
|     ui::mojom::DragOperation operation,
 | |
|     bool document_is_handling_drag) {
 | |
|   if (platform_delegate_) {
 | |
|     platform_delegate_->UpdateDragOperation(operation,
 | |
|                                             document_is_handling_drag);
 | |
|   }
 | |
| }
 |