// Copyright 2020 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. #ifndef CEF_LIBCEF_BROWSER_BROWSER_HOST_BASE_H_ #define CEF_LIBCEF_BROWSER_BROWSER_HOST_BASE_H_ #pragma once #include "base/memory/raw_ptr.h" #include "base/observer_list.h" #include "base/synchronization/lock.h" #include "cef/include/cef_browser.h" #include "cef/include/cef_client.h" #include "cef/include/cef_unresponsive_process_callback.h" #include "cef/include/views/cef_browser_view.h" #include "cef/libcef/browser/browser_contents_delegate.h" #include "cef/libcef/browser/browser_info.h" #include "cef/libcef/browser/browser_platform_delegate.h" #include "cef/libcef/browser/devtools/devtools_protocol_manager.h" #include "cef/libcef/browser/devtools/devtools_window_runner.h" #include "cef/libcef/browser/file_dialog_manager.h" #include "cef/libcef/browser/frame_host_impl.h" #include "cef/libcef/browser/javascript_dialog_manager.h" #include "cef/libcef/browser/media_stream_registrar.h" #include "cef/libcef/browser/request_context_impl.h" class RenderViewContextMenuObserver; // Parameters that are passed to the runtime-specific Create methods. struct CefBrowserCreateParams { CefBrowserCreateParams() = default; // Copy constructor used with Chrome style only. CefBrowserCreateParams(const CefBrowserCreateParams& that) { operator=(that); } CefBrowserCreateParams& operator=(const CefBrowserCreateParams& that) { DCHECK(that.IsChromeStyle()); // Not all parameters can be copied. client = that.client; url = that.url; settings = that.settings; request_context = that.request_context; extra_info = that.extra_info; if (that.window_info) { MaybeSetWindowInfo(*that.window_info, /*allow_alloy_style=*/false, /*allow_chrome_style=*/true); } browser_view = that.browser_view; return *this; } // Initialize |window_info| with expected defaults before passing to a client // callback. |opener| will be non-nullptr for popups, DevTools windows, etc. static void InitWindowInfo(CefWindowInfo* window_info, CefBrowserHostBase* opener); // Set |window_info| if appropriate (see below). void MaybeSetWindowInfo(const CefWindowInfo& window_info, bool allow_alloy_style, bool allow_chrome_style); // Returns true if |window_info| indicates creation of a Chrome style window. static bool IsChromeStyle(const CefWindowInfo* window_info); bool IsChromeStyle() const; // Returns true if parameters indicate windowless (off-screen) rendering. bool IsWindowless() const; // Platform-specific window creation info. Will be nullptr for Views-hosted // browsers except when using Chrome style with a native parent handle. std::unique_ptr window_info; // The BrowserView that will own a Views-hosted browser. Will be nullptr for // popup browsers. CefRefPtr browser_view; // True if this browser is a popup and has a Views-hosted opener, in which // case the BrowserView for this browser will be created later (from // PopupWebContentsCreated). bool popup_with_views_hosted_opener = false; // True if this browser is a popup and has an Alloy style opener. Only used // with Chrome style. bool popup_with_alloy_style_opener = false; // Client implementation. May be nullptr. CefRefPtr client; // Initial URL to load. May be empty. If this is a valid extension URL then // the browser will be created as an app view extension host. CefString url; // Browser settings. CefBrowserSettings settings; // Request context to use when creating the browser. If nullptr the global // request context will be used. CefRefPtr request_context; // Extra information that will be passed to // CefRenderProcessHandler::OnBrowserCreated. CefRefPtr extra_info; }; // Base class for CefBrowserHost implementations. Includes functionality that is // shared by Alloy and Chrome styles. All methods are thread-safe unless // otherwise indicated. class CefBrowserHostBase : public CefBrowserHost, public CefBrowser, public CefBrowserContentsDelegate::Observer { public: // Interface to implement for observers that wish to be informed of changes // to the CefBrowserHostBase. All methods will be called on the UI thread. class Observer : public base::CheckedObserver { public: // Called before |browser| is destroyed. Any references to |browser| should // be cleared when this method is called. virtual void OnBrowserDestroyed(CefBrowserHostBase* browser) = 0; protected: ~Observer() override = default; }; // Create a new CefBrowserHost instance of the current runtime type with // owned WebContents. static CefRefPtr Create( CefBrowserCreateParams& create_params); // Safe conversion from CefBrowserHostBase to CefBrowserHostBase. // Use this method instead of static_cast. static CefRefPtr FromBrowser( CefRefPtr browser); // Returns the browser associated with the specified RenderViewHost. static CefRefPtr GetBrowserForHost( const content::RenderViewHost* host); // Returns the browser associated with the specified RenderFrameHost. static CefRefPtr GetBrowserForHost( const content::RenderFrameHost* host); // Returns the browser associated with the specified WebContents. static CefRefPtr GetBrowserForContents( const content::WebContents* contents); // Returns the browser associated with the specified global ID. static CefRefPtr GetBrowserForGlobalId( const content::GlobalRenderFrameHostId& global_id); // Returns the browser associated with the specified global token. static CefRefPtr GetBrowserForGlobalToken( const content::GlobalRenderFrameHostToken& global_token); // Returns the browser associated with the specified top-level window. static CefRefPtr GetBrowserForTopLevelNativeWindow( gfx::NativeWindow owning_window); // Returns the browser most likely to be focused. This may be somewhat iffy // with windowless browsers as there is no guarantee that the client has only // one browser focused at a time. static CefRefPtr GetLikelyFocusedBrowser(); CefBrowserHostBase( const CefBrowserSettings& settings, CefRefPtr client, std::unique_ptr platform_delegate, scoped_refptr browser_info, CefRefPtr request_context); CefBrowserHostBase(const CefBrowserHostBase&) = delete; CefBrowserHostBase& operator=(const CefBrowserHostBase&) = delete; // Called on the UI thread after the associated WebContents is created. virtual void InitializeBrowser(); // Called on the UI thread when the OS window hosting the browser is // destroyed. virtual void WindowDestroyed() = 0; // Returns true if the browser is in the process of being destroyed. Called on // the UI thread only. virtual bool WillBeDestroyed() const = 0; // Called on the UI thread to complete WebContents tear-down. In most cases // this will be called via WebContentsObserver::WebContentsDestroyed. Any // remaining objects that reference the WebContents (including RFH, etc) // should be cleared in this callback. virtual void DestroyWebContents(content::WebContents* web_contents); // Called on the UI thread to complete CefBrowserHost tear-down. // // With Chrome style the WebContents is owned by the Browser's TabStripModel // and will usually be destroyed first: CloseBrowser -> (async) DoCloseBrowser // -> [TabStripModel deletes the WebContents] -> OnWebContentsDestroyed -> // DestroyWebContents -> (async) DestroyBrowser. // // With Alloy style the WebContents is owned by the // CefBrowserPlatformDelegateAlloy and will usually be destroyed at the same // time: CloseBrowser -> [OS/platform logic] -> (async) DestroyBrowser -> // [CefBrowserPlatformDelegateAlloy deletes the WebContents] // -> WebContentsDestroyed -> DestoyWebContents. // // There are a few exceptions to the above rules: // 1. If the CefBrowserHost still exists at CefShutdown, in which case // DestroyBrowser will be called first via // CefBrowserInfoManager::DestroyAllBrowsers. // 2. If a popup WebContents is still pending when the parent WebContents is // destroyed, in which case WebContentsDestroyed will be called first via // the parent WebContents destructor. virtual void DestroyBrowser(); // CefBrowserHost methods: CefRefPtr GetBrowser() override; CefRefPtr GetClient() override; CefRefPtr GetRequestContext() override; bool CanZoom(cef_zoom_command_t command) override; void Zoom(cef_zoom_command_t command) override; double GetDefaultZoomLevel() override; double GetZoomLevel() override; void SetZoomLevel(double zoomLevel) override; bool HasView() override; void SetFocus(bool focus) override; void RunFileDialog(FileDialogMode mode, const CefString& title, const CefString& default_file_path, const std::vector& accept_filters, CefRefPtr callback) override; void StartDownload(const CefString& url) override; void DownloadImage(const CefString& image_url, bool is_favicon, uint32_t max_image_size, bool bypass_cache, CefRefPtr callback) override; void Print() override; void PrintToPDF(const CefString& path, const CefPdfPrintSettings& settings, CefRefPtr callback) override; void ShowDevTools(const CefWindowInfo& windowInfo, CefRefPtr client, const CefBrowserSettings& settings, const CefPoint& inspect_element_at) override; void CloseDevTools() override; bool HasDevTools() override; void ReplaceMisspelling(const CefString& word) override; void AddWordToDictionary(const CefString& word) override; void SendKeyEvent(const CefKeyEvent& event) override; void SendMouseClickEvent(const CefMouseEvent& event, MouseButtonType type, bool mouseUp, int clickCount) override; void SendMouseMoveEvent(const CefMouseEvent& event, bool mouseLeave) override; void SendMouseWheelEvent(const CefMouseEvent& event, int deltaX, int deltaY) override; bool SendDevToolsMessage(const void* message, size_t message_size) override; int ExecuteDevToolsMethod(int message_id, const CefString& method, CefRefPtr params) override; CefRefPtr AddDevToolsMessageObserver( CefRefPtr observer) override; void GetNavigationEntries(CefRefPtr visitor, bool current_only) override; CefRefPtr GetVisibleNavigationEntry() override; void NotifyMoveOrResizeStarted() override; bool IsFullscreen() override; void ExitFullscreen(bool will_cause_resize) override; bool IsRenderProcessUnresponsive() override; cef_runtime_style_t GetRuntimeStyle() override; // CefBrowser methods: bool IsValid() override; CefRefPtr GetHost() override; bool CanGoBack() override; void GoBack() override; bool CanGoForward() override; void GoForward() override; bool IsLoading() override; void Reload() override; void ReloadIgnoreCache() override; void StopLoad() override; int GetIdentifier() override; bool IsSame(CefRefPtr that) override; bool HasDocument() override; bool IsPopup() override; CefRefPtr GetMainFrame() override; CefRefPtr GetFocusedFrame() override; CefRefPtr GetFrameByIdentifier( const CefString& identifier) override; CefRefPtr GetFrameByName(const CefString& name) override; size_t GetFrameCount() override; void GetFrameIdentifiers(std::vector& identifiers) override; void GetFrameNames(std::vector& names) override; void SetAccessibilityState(cef_state_t accessibility_state) override; // CefBrowserContentsDelegate::Observer methods: void OnStateChanged(CefBrowserContentsState state_changed) override; void OnWebContentsDestroyed(content::WebContents* web_contents) override; // Returns the frame object matching the specified |host| or nullptr if no // match is found. Must be called on the UI thread. CefRefPtr GetFrameForHost(const content::RenderFrameHost* host); // Returns the frame associated with the specified global ID/token. See // documentation on RenderFrameHost::GetFrameTreeNodeId/Token() for why the // global ID/token is preferred. CefRefPtr GetFrameForGlobalId( const content::GlobalRenderFrameHostId& global_id); CefRefPtr GetFrameForGlobalToken( const content::GlobalRenderFrameHostToken& global_token); // Manage observer objects. The observer must either outlive this object or // be removed before destruction. Must be called on the UI thread. void AddObserver(Observer* observer); void RemoveObserver(Observer* observer); bool HasObserver(Observer* observer) const; // Methods called from CefFrameHostImpl. void LoadMainFrameURL(const content::OpenURLParams& params); virtual void OnSetFocus(cef_focus_source_t source) = 0; void ViewText(const std::string& text); // Calls CefFileDialogManager methods. void RunFileChooserForBrowser( const blink::mojom::FileChooserParams& params, CefFileDialogManager::RunFileChooserCallback callback); void RunSelectFile(ui::SelectFileDialog::Listener* listener, std::unique_ptr policy, ui::SelectFileDialog::Type type, const std::u16string& title, const base::FilePath& default_path, const ui::SelectFileDialog::FileTypeInfo* file_types, int file_type_index, const base::FilePath::StringType& default_extension, gfx::NativeWindow owning_window, void* params); void SelectFileListenerDestroyed(ui::SelectFileDialog::Listener* listener); // Called from AlloyBrowserHostImpl::GetJavaScriptDialogManager and // ChromeBrowserDelegate::GetJavaScriptDialogManager. content::JavaScriptDialogManager* GetJavaScriptDialogManager(); // Called from CefBrowserInfoManager::MaybeAllowNavigation. virtual bool MaybeAllowNavigation(content::RenderFrameHost* opener, const content::OpenURLParams& params); // Helpers for executing client callbacks. Must be called on the UI thread. void OnAfterCreated(); void OnBeforeClose(); void OnBrowserDestroyed(); // Thread-safe accessors. const CefBrowserSettings& settings() const { return settings_; } CefRefPtr client() const { return client_; } scoped_refptr browser_info() const { return browser_info_; } int browser_id() const; CefRefPtr request_context() const { return request_context_; } bool is_views_hosted() const { return is_views_hosted_; } SkColor GetBackgroundColor() const; // Returns true if windowless rendering is enabled. virtual bool IsWindowless() const = 0; // Returns the runtime style of this browser. virtual bool IsAlloyStyle() const = 0; bool IsChromeStyle() const { return !IsAlloyStyle(); } // Accessors that must be called on the UI thread. content::WebContents* GetWebContents() const; content::BrowserContext* GetBrowserContext() const; CefBrowserPlatformDelegate* platform_delegate() const { return platform_delegate_.get(); } CefBrowserContentsDelegate* contents_delegate() const { return contents_delegate_.get(); } CefMediaStreamRegistrar* GetMediaStreamRegistrar(); CefDevToolsWindowRunner* GetDevToolsWindowRunner(); CefRefPtr unresponsive_process_callback() const { return unresponsive_process_callback_; } void set_unresponsive_process_callback( CefRefPtr callback) { unresponsive_process_callback_ = callback; } RenderViewContextMenuObserver* context_menu_observer() const { return context_menu_observer_; } void set_context_menu_observer(RenderViewContextMenuObserver* observer) { context_menu_observer_ = observer; } // Returns the Widget owner for the browser window. Only used with windowed // browsers. views::Widget* GetWindowWidget() const; // Returns the BrowserView associated with this browser. Only used with Views- // based browsers. CefRefPtr GetBrowserView() const; // Returns the top-level native window for this browser. With windowed // browsers this will be an aura::Window* on Aura platforms (Windows/Linux) // and an NSWindow wrapper object from native_widget_types.h on MacOS. With // windowless browsers this method will always return an empty value. gfx::NativeWindow GetTopLevelNativeWindow() const; // Returns true if this browser is currently focused. A browser is considered // focused when the top-level RenderFrameHost is in the parent chain of the // currently focused RFH within the frame tree. In addition, its associated // RenderWidgetHost must also be focused. With windowed browsers only one // browser should be focused at a time. With windowless browsers this relies // on the client to properly configure focus state. bool IsFocused() const; // Returns true if this browser is currently visible. virtual bool IsVisible() const; protected: bool EnsureDevToolsProtocolManager(); void InitializeDevToolsRegistrationOnUIThread( CefRefPtr registration); // Called from LoadMainFrameURL to perform the actual navigation. virtual bool Navigate(const content::OpenURLParams& params); // Called from ShowDevTools to perform the actual show. void ShowDevToolsOnUIThread(std::unique_ptr params); // Create the CefFileDialogManager if it doesn't already exist. bool EnsureFileDialogManager(); // Thread-safe members. CefBrowserSettings settings_; CefRefPtr client_; std::unique_ptr platform_delegate_; scoped_refptr browser_info_; CefRefPtr request_context_; const bool is_views_hosted_; // Only accessed on the UI thread. std::unique_ptr contents_delegate_; CefRefPtr unresponsive_process_callback_; raw_ptr context_menu_observer_ = nullptr; // Observers that want to be notified of changes to this object. // Only accessed on the UI thread. base::ObserverList observers_; // Used for creating and managing file dialogs. std::unique_ptr file_dialog_manager_; // Used for creating and managing JavaScript dialogs. std::unique_ptr javascript_dialog_manager_; // Volatile state accessed from multiple threads. All access must be protected // by |state_lock_|. base::Lock state_lock_; bool is_loading_ = false; bool can_go_back_ = false; bool can_go_forward_ = false; bool has_document_ = false; bool is_fullscreen_ = false; CefRefPtr focused_frame_; // Used for managing DevTools instances without a frontend. std::unique_ptr devtools_protocol_manager_; // Used for creating and running the DevTools window frontend. std::unique_ptr devtools_window_runner_; std::unique_ptr media_stream_registrar_; private: IMPLEMENT_REFCOUNTING(CefBrowserHostBase); }; #endif // CEF_LIBCEF_BROWSER_BROWSER_HOST_BASE_H_