blob: 2d32d91eb98fe749ae262ba06a1a399813aa7bab [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_BROWSER_RENDERER_HOST_RENDER_WIDGET_HOST_IMPL_H_
#define CONTENT_BROWSER_RENDERER_HOST_RENDER_WIDGET_HOST_IMPL_H_
#include <stddef.h>
#include <stdint.h>
#include <map>
#include <memory>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include "base/callback_list.h"
#include "base/containers/flat_map.h"
#include "base/containers/flat_set.h"
#include "base/functional/callback.h"
#include "base/gtest_prod_util.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/raw_ref.h"
#include "base/memory/safe_ref.h"
#include "base/memory/weak_ptr.h"
#include "base/observer_list.h"
#include "base/process/kill.h"
#include "base/scoped_observation_traits.h"
#include "base/time/time.h"
#include "base/timer/timer.h"
#include "base/types/pass_key.h"
#include "build/build_config.h"
#include "cc/mojom/render_frame_metadata.mojom.h"
#include "components/input/event_with_latency_info.h"
#include "components/input/input_disposition_handler.h"
#include "components/input/input_router_impl.h"
#include "components/input/render_input_router.h"
#include "components/input/render_input_router_client.h"
#include "components/input/render_input_router_delegate.h"
#include "components/input/touch_emulator_client.h"
#include "components/viz/common/surfaces/frame_sink_id.h"
#include "content/browser/renderer_host/agent_scheduling_group_host.h"
#include "content/browser/renderer_host/frame_token_message_queue.h"
#include "content/browser/renderer_host/input/touch_emulator_impl.h"
#include "content/browser/renderer_host/mojo_render_input_router_delegate_impl.h"
#include "content/browser/renderer_host/render_frame_metadata_provider_impl.h"
#include "content/browser/renderer_host/render_widget_host_delegate.h"
#include "content/browser/renderer_host/render_widget_host_view_base.h"
#include "content/common/content_export.h"
#include "content/common/frame.mojom-forward.h"
#include "content/common/input/synthetic_gesture.h"
#include "content/common/input/synthetic_gesture_controller.h"
#include "content/public/browser/render_process_host_observer.h"
#include "content/public/browser/render_process_host_priority_client.h"
#include "content/public/browser/render_widget_host.h"
#include "mojo/public/cpp/bindings/associated_remote.h"
#include "mojo/public/cpp/bindings/pending_receiver.h"
#include "mojo/public/cpp/bindings/pending_remote.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "services/viz/public/mojom/compositing/compositor_frame_sink.mojom-forward.h"
#include "third_party/blink/public/mojom/input/input_event_result.mojom-shared.h"
#include "third_party/blink/public/mojom/input/input_handler.mojom.h"
#include "third_party/blink/public/mojom/input/pointer_lock_context.mojom.h"
#include "third_party/blink/public/mojom/keyboard_lock/keyboard_lock.mojom.h"
#include "third_party/blink/public/mojom/manifest/display_mode.mojom.h"
#include "third_party/blink/public/mojom/page/widget.mojom.h"
#include "third_party/blink/public/mojom/widget/platform_widget.mojom.h"
#include "third_party/blink/public/mojom/widget/record_content_to_visible_time_request.mojom-forward.h"
#include "ui/base/dragdrop/mojom/drag_drop_types.mojom-forward.h"
#include "ui/base/ime/text_input_mode.h"
#include "ui/base/ime/text_input_type.h"
#include "ui/base/mojom/menu_source_type.mojom-forward.h"
#include "ui/gfx/native_ui_types.h"
#include "ui/latency/latency_info.h"
#include "url/origin.h"
#if BUILDFLAG(IS_ANDROID)
#include "content/public/browser/android/child_process_importance.h"
#endif
#if BUILDFLAG(IS_MAC)
#include "services/device/public/mojom/wake_lock.mojom.h"
#endif
class SkBitmap;
namespace blink {
class WebInputEvent;
class WebMouseEvent;
} // namespace blink
namespace cc {
struct BrowserControlsOffsetTagModifications;
} // namespace cc
namespace gfx {
class Image;
class Range;
class Vector2dF;
} // namespace gfx
namespace input {
class InputRouter;
class TimeoutMonitor;
class FlingSchedulerBase;
} // namespace input
namespace ui {
enum class DomCode : uint32_t;
}
namespace viz {
struct CopyOutputBitmapWithMetadata;
} // namespace viz
namespace content {
class FrameTree;
class MockRenderWidgetHost;
class MockRenderWidgetHostImpl;
class RenderWidgetHostOwnerDelegate;
class RenderWidgetHostFactory;
class SiteInstanceGroup;
class SyntheticGestureController;
class VisibleTimeRequestTrigger;
// This implements the RenderWidgetHost interface that is exposed to
// embedders of content, and adds things only visible to content.
//
// Several core rendering primitives are mirrored between the browser and
// renderer. These are `blink::WidgetBase`, `RenderFrame` and `blink::WebView`.
// Their browser counterparts are `RenderWidgetHost`, `RenderFrameHost` and
// `RenderViewHost`.
//
// For simplicity and clarity, we want the object ownership graph in the
// renderer to mirror the object ownership graph in the browser. The IPC message
// that tears down the renderer object graph should be targeted at the root
// object, and should be sent by the destructor/finalizer of the root object in
// the browser.
//
// Note: We must tear down the renderer object graph with a single IPC to avoid
// inconsistencies in renderer state.
//
// RenderWidget represents a surface that can paint and receive input. It is
// used in four contexts:
// * Main frame for webpage (root is `blink::WebView`)
// * Child frame for webpage (root is RenderFrame)
// * Popups (root is RenderWidget)
//
// Destruction of the RenderWidgetHost will trigger destruction of the
// RenderWidget iff RenderWidget is the root of the renderer object graph.
//
// Note: We want to converge on RenderFrame always being the root.
class CONTENT_EXPORT RenderWidgetHostImpl
: public RenderWidgetHost,
public FrameTokenMessageQueue::Client,
public RenderProcessHostObserver,
public RenderProcessHostPriorityClient,
public SyntheticGestureController::Delegate,
public RenderFrameMetadataProvider::Observer,
public blink::mojom::FrameWidgetHost,
public blink::mojom::PopupWidgetHost,
public blink::mojom::WidgetHost,
public blink::mojom::PointerLockContext,
public input::RenderInputRouterDelegate,
public input::RenderInputRouterClient {
public:
// See the constructor for documentation.
//
// This static factory method is restricted to being called from the factory,
// to ensure all RenderWidgetHostImpl creation can be hooked for tests.
static std::unique_ptr<RenderWidgetHostImpl> Create(
base::PassKey<RenderWidgetHostFactory>,
FrameTree* frame_tree,
RenderWidgetHostDelegate* delegate,
viz::FrameSinkId frame_sink_id,
base::SafeRef<SiteInstanceGroup> site_instance_group,
int32_t routing_id,
bool hidden,
bool renderer_initiated_creation);
// Similar to `Create()`, but creates a self-owned `RenderWidgetHostImpl`. The
// returned widget deletes itself when either:
// - ShutdownAndDestroyWidget(also_delete = true) is called.
// - its RenderProcess exits.
static RenderWidgetHostImpl* CreateSelfOwned(
base::PassKey<RenderWidgetHostFactory>,
FrameTree* frame_tree,
RenderWidgetHostDelegate* delegate,
base::SafeRef<SiteInstanceGroup> site_instance_group,
int32_t routing_id,
bool hidden);
RenderWidgetHostImpl(const RenderWidgetHostImpl&) = delete;
RenderWidgetHostImpl& operator=(const RenderWidgetHostImpl&) = delete;
~RenderWidgetHostImpl() override;
// Similar to RenderWidgetHost::FromID, but returning the Impl object.
static RenderWidgetHostImpl* FromID(int32_t process_id, int32_t routing_id);
// Returns all RenderWidgetHosts including swapped out ones for
// internal use. The public interface
// RenderWidgetHost::GetRenderWidgetHosts only returns active ones.
static std::unique_ptr<RenderWidgetHostIterator> GetAllRenderWidgetHosts();
// Use RenderWidgetHostImpl::From(rwh) to downcast a RenderWidgetHost to a
// RenderWidgetHostImpl.
static RenderWidgetHostImpl* From(RenderWidgetHost* rwh);
// Generates the FrameSinkId to use for a RenderWidgetHost allocated with
// `group` and `routing_id`.
static viz::FrameSinkId DefaultFrameSinkId(const SiteInstanceGroup& group,
int routing_id);
// TODO(crbug.com/40169570): FrameTree and FrameTreeNode will not be const as
// with prerenderer activation the page needs to move between FrameTreeNodes
// and FrameTrees. As it's hard to make sure that all places handle this
// transition correctly, MPArch will remove references from this class to
// FrameTree/FrameTreeNode.
FrameTree* frame_tree() const { return frame_tree_; }
void SetFrameTree(FrameTree& frame_tree) { frame_tree_ = &frame_tree; }
void set_owner_delegate(RenderWidgetHostOwnerDelegate* owner_delegate) {
owner_delegate_ = owner_delegate;
}
RenderWidgetHostOwnerDelegate* owner_delegate() { return owner_delegate_; }
AgentSchedulingGroupHost& agent_scheduling_group() {
return *agent_scheduling_group_;
}
// Returns the object that tracks the start of content to visible events for
// the WebContents.
VisibleTimeRequestTrigger& GetVisibleTimeRequestTrigger();
// RenderWidgetHost implementation.
const viz::FrameSinkId& GetFrameSinkId() override;
void UpdateTextDirection(base::i18n::TextDirection direction) override;
void NotifyTextDirection() override;
void Focus() override;
void Blur() override;
void FlushForTesting() override;
void SetActive(bool active) override;
void ForwardMouseEvent(const blink::WebMouseEvent& mouse_event) override;
void ForwardWheelEvent(const blink::WebMouseWheelEvent& wheel_event) override;
void ForwardKeyboardEvent(
const input::NativeWebKeyboardEvent& key_event) override;
void ForwardGestureEvent(
const blink::WebGestureEvent& gesture_event) override;
RenderProcessHost* GetProcess() override;
int GetRoutingID() final;
RenderWidgetHostViewBase* GetView() override;
const RenderWidgetHostViewBase* GetView() const override;
bool IsCurrentlyUnresponsive() override;
bool SynchronizeVisualProperties() override;
void AddKeyPressEventCallback(const KeyPressEventCallback& callback) override;
void RemoveKeyPressEventCallback(
const KeyPressEventCallback& callback) override;
void AddMouseEventCallback(const MouseEventCallback& callback) override;
void RemoveMouseEventCallback(const MouseEventCallback& callback) override;
void AddSuppressShowingImeCallback(
const SuppressShowingImeCallback& callback) override;
void RemoveSuppressShowingImeCallback(
const SuppressShowingImeCallback& callback,
bool trigger_ime) override;
void AddInputEventObserver(
RenderWidgetHost::InputEventObserver* observer) override;
void RemoveInputEventObserver(
RenderWidgetHost::InputEventObserver* observer) override;
void AddObserver(RenderWidgetHostObserver* observer) override;
void RemoveObserver(RenderWidgetHostObserver* observer) override;
display::ScreenInfo GetScreenInfo() const override;
display::ScreenInfos GetScreenInfos() const override;
float GetDeviceScaleFactor() override;
std::optional<cc::TouchAction> GetAllowedTouchAction() override;
void WriteIntoTrace(perfetto::TracedValue context) override;
// |drop_data| must have been filtered. The embedder should call
// FilterDropData before passing the drop data to RWHI.
void DragTargetDragEnter(const DropData& drop_data,
const gfx::PointF& client_pt,
const gfx::PointF& screen_pt,
blink::DragOperationsMask operations_allowed,
int key_modifiers,
DragOperationCallback callback) override;
void DragTargetDragEnterWithMetaData(
const std::vector<DropData::Metadata>& metadata,
const gfx::PointF& client_pt,
const gfx::PointF& screen_pt,
blink::DragOperationsMask operations_allowed,
int key_modifiers,
DragOperationCallback callback) override;
void DragTargetDragOver(const gfx::PointF& client_point,
const gfx::PointF& screen_point,
blink::DragOperationsMask operations_allowed,
int key_modifiers,
DragOperationCallback callback) override;
void DragTargetDragLeave(const gfx::PointF& client_point,
const gfx::PointF& screen_point) override;
// |drop_data| must have been filtered. The embedder should call
// FilterDropData before passing the drop data to RWHI.
void DragTargetDrop(const DropData& drop_data,
const gfx::PointF& client_point,
const gfx::PointF& screen_point,
int key_modifiers,
base::OnceClosure callback) override;
void DragSourceEndedAt(const gfx::PointF& client_pt,
const gfx::PointF& screen_pt,
ui::mojom::DragOperation operation,
base::OnceClosure callback) override;
void DragSourceSystemDragEnded() override;
void FilterDropData(DropData* drop_data) override;
void SetCursor(const ui::Cursor& cursor) override;
void ShowContextMenuAtPoint(
const gfx::Point& point,
const ui::mojom::MenuSourceType source_type) override;
void InsertVisualStateCallback(VisualStateCallback callback) override;
// RenderProcessHostPriorityClient implementation.
RenderProcessHostPriorityClient::Priority GetPriority() override;
// RenderProcessHostObserver implementation.
void RenderProcessExited(RenderProcessHost* host,
const ChildProcessTerminationInfo& info) override;
// blink::mojom::WidgetHost implementation.
void UpdateTooltipUnderCursor(
const std::u16string& tooltip_text,
base::i18n::TextDirection text_direction_hint) override;
void UpdateTooltipFromKeyboard(const std::u16string& tooltip_text,
base::i18n::TextDirection text_direction_hint,
const gfx::Rect& bounds) override;
void ClearKeyboardTriggeredTooltip() override;
void TextInputStateChanged(ui::mojom::TextInputStatePtr state) override;
void SelectionBoundsChanged(const gfx::Rect& anchor_rect,
base::i18n::TextDirection anchor_dir,
const gfx::Rect& focus_rect,
base::i18n::TextDirection focus_dir,
const gfx::Rect& bounding_box,
bool is_anchor_first) override;
void CreateFrameSink(
mojo::PendingReceiver<viz::mojom::CompositorFrameSink>
compositor_frame_sink_receiver,
mojo::PendingRemote<viz::mojom::CompositorFrameSinkClient>
compositor_frame_sink_client,
mojo::PendingRemote<blink::mojom::RenderInputRouterClient>
viz_rir_client_remote) override;
void RegisterRenderFrameMetadataObserver(
mojo::PendingReceiver<cc::mojom::RenderFrameMetadataObserverClient>
render_frame_metadata_observer_client_receiver,
mojo::PendingRemote<cc::mojom::RenderFrameMetadataObserver>
render_frame_metadata_observer) override;
// blink::mojom::PopupWidgetHost implementation.
void RequestClosePopup() override;
void ShowPopup(const gfx::Rect& initial_screen_rect,
const gfx::Rect& anchor_screen_rect,
ShowPopupCallback callback) override;
void SetPopupBounds(const gfx::Rect& bounds,
SetPopupBoundsCallback callback) override;
// RenderInputRouterDelegate implementation.
input::RenderWidgetHostViewInput* GetPointerLockView() override;
std::optional<bool> IsDelegatedInkHovering() override;
std::unique_ptr<input::RenderInputRouterIterator>
GetEmbeddedRenderInputRouters() override;
input::RenderWidgetHostInputEventRouter* GetInputEventRouter() override;
void ForwardDelegatedInkPoint(gfx::DelegatedInkPoint& delegated_ink_point,
bool& ended_delegated_ink_trail) override;
void ResetDelegatedInkPointPrediction(
bool& ended_delegated_ink_trail) override;
void NotifyObserversOfInputEvent(const blink::WebInputEvent& event,
bool dispatched_to_renderer) override;
void NotifyObserversOfInputEventAcks(
blink::mojom::InputEventResultSource ack_source,
blink::mojom::InputEventResultState ack_result,
const blink::WebInputEvent& event) override;
bool PreHandleGestureEvent(const blink::WebGestureEvent& event) override;
TouchEmulatorImpl* GetTouchEmulator(bool create_if_necessary) override;
std::unique_ptr<viz::PeakGpuMemoryTracker> MakePeakGpuMemoryTracker(
viz::PeakGpuMemoryTracker::Usage usage) override;
void OnWheelEventAck(const input::MouseWheelEventWithLatencyInfo& event,
blink::mojom::InputEventResultSource ack_source,
blink::mojom::InputEventResultState ack_result) override;
bool IsInitializedAndNotDead() override;
void OnInputEventPreDispatch(const blink::WebInputEvent& event) override;
void OnInvalidInputEventSource() override;
void OnInputIgnored(const blink::WebInputEvent& event) override;
input::StylusInterface* GetStylusInterface() override;
void OnInputEventAckTimeout(base::TimeTicks ack_timeout_ts) override;
void RendererIsResponsive() override;
void DidOverscroll(blink::mojom::DidOverscrollParamsPtr params) override;
// Update the stored set of visual properties for the renderer. If 'propagate'
// is true, the new properties will be sent to the renderer process.
bool UpdateVisualProperties(bool propagate);
// Notification that the screen info has changed.
void NotifyScreenInfoChanged();
// Forces redraw in the renderer and when the update reaches the browser.
// grabs snapshot from the compositor.
// If |from_surface| is false, it will obtain the snapshot directly from the
// view (On MacOS, the snapshot is taken from the Cocoa view for end-to-end
// testing purposes).
// Otherwise, the snapshot is obtained from the view's surface, with no bounds
// defined.
// Returns a gfx::Image that is backed by an NSImage on MacOS or by an
// SkBitmap otherwise. The gfx::Image may be empty if the snapshot failed.
using GetSnapshotFromBrowserCallback =
base::OnceCallback<void(const gfx::Image&)>;
void GetSnapshotFromBrowser(GetSnapshotFromBrowserCallback callback,
bool from_surface);
// Sets the View of this RenderWidgetHost. It is called just once when a new
// RenderWidgetHostView for a RWH is created. If the corresponding renderer
// process crashes and then be recreated, SetView is called first with a
// nullptr and then with the new RenderWidgetHostView.
void SetView(RenderWidgetHostViewBase* view);
RenderWidgetHostDelegate* delegate() const { return delegate_; }
MojoRenderInputRouterDelegateImpl* mojo_rir_delegate() {
return &mojo_rir_delegate_impl_;
}
// Bind the provided widget interfaces.
void BindWidgetInterfaces(
mojo::PendingAssociatedReceiver<blink::mojom::WidgetHost> widget_host,
mojo::PendingAssociatedRemote<blink::mojom::Widget> widget);
// Bind the provided popup widget interface.
void BindPopupWidgetInterface(
mojo::PendingAssociatedReceiver<blink::mojom::PopupWidgetHost>
popup_widget_host);
// Bind the provided frame widget interfaces.
void BindFrameWidgetInterfaces(
mojo::PendingAssociatedReceiver<blink::mojom::FrameWidgetHost>
frame_widget_host,
mojo::PendingAssociatedRemote<blink::mojom::FrameWidget> frame_widget);
// The Bind*Interfaces() methods are called before creating the renderer-side
// Widget object, and RendererWidgetCreated() is called afterward. At that
// point the bound mojo interfaces are connected to the renderer Widget. The
// `for_frame_widget` informs if this widget should enable frame-specific
// behaviour and mojo connections.
void RendererWidgetCreated(bool for_frame_widget);
// Renderer-created top-level widgets (either for a main frame or for a popup)
// wait to be shown until the renderer requests it. When that condition is
// satisfied we are notified through Init(). This will always happen after
// RendererWidgetCreated().
void Init();
// Returns true if the frame content needs be stored before being evicted.
bool ShouldShowStaleContentOnEviction();
void SetFrameDepth(unsigned int depth);
void SetIntersectsViewport(bool intersects);
void UpdatePriority();
void SetShouldContributePriorityToProcess(
bool should_contribute_priority_to_process);
// Tells the renderer to die and optionally delete |this|.
void ShutdownAndDestroyWidget(bool also_delete);
// Indicates if the page has finished loading.
void SetIsLoading(bool is_loading);
// Called to notify the RenderWidget that it has been hidden or restored from
// having been hidden.
void WasHidden();
void WasShown(blink::mojom::RecordContentToVisibleTimeRequestPtr
record_tab_switch_time_request);
// Called to request the presentation time for the next frame or cancel any
// requests when the RenderWidget's visibility state is not changing. If the
// visibility state is changing call WasHidden or WasShown instead.
void RequestSuccessfulPresentationTimeForNextFrame(
blink::mojom::RecordContentToVisibleTimeRequestPtr visible_time_request);
void CancelSuccessfulPresentationTimeRequest();
#if BUILDFLAG(IS_ANDROID)
// Set the importance of widget. The importance is passed onto
// RenderProcessHost which aggregates importance of all of its widgets.
void SetImportance(ChildProcessImportance importance);
ChildProcessImportance importance() const { return importance_; }
void AddImeInputEventObserver(
RenderWidgetHost::InputEventObserver* observer) override;
void RemoveImeInputEventObserver(
RenderWidgetHost::InputEventObserver* observer) override;
#endif
// Called to notify the RenderWidget that its associated native window
// got/lost focused.
void GotFocus();
void LostFocus();
void LostCapture();
// Used by the RenderFrameHost to help with verifying changes in focus. Tells
// whether LostFocus() was called after any frame on this page was focused.
bool HasLostFocus() const { return has_lost_focus_; }
void ResetLostFocus() { has_lost_focus_ = false; }
// Indicates whether the RenderWidgetHost thinks it is focused.
// This is different from RenderWidgetHostView::HasFocus() in the sense that
// it reflects what the renderer process knows: it saves the state that is
// sent/received.
// RenderWidgetHostView::HasFocus() is checking whether the view is focused so
// it is possible in some edge cases that a view was requested to be focused
// but it failed, thus HasFocus() returns false.
bool is_focused() const { return is_focused_; }
// Support for focus tracking on multi-FrameTree cases. This will notify all
// descendants (including nested FrameTrees) to distribute a "page focus"
// update. Users other than WebContents and RenderWidgetHost should use
// Focus()/Blur().
void SetPageFocus(bool focused);
// Returns true if the RenderWidgetHost thinks it is active. This
// is different than `is_focused` but must always be true if `is_focused`
// is true. All RenderWidgetHosts in an active tab are considered active,
// but only one FrameTree can have page focus (e.g., an inner frame
// tree (fenced frame) will not have focus if the primary frame tree has
// focus. See
// https://www.chromium.org/developers/design-documents/aura/focus-and-activation.
bool is_active() const { return is_active_; }
// Called to notify the RenderWidget that it has lost the pointer lock.
void LostPointerLock();
// Notifies the RenderWidget that it lost the pointer lock.
void SendPointerLockLost();
bool is_last_unlocked_by_target() const {
return is_last_unlocked_by_target_;
}
// Notifies the RenderWidget of the current mouse cursor visibility state.
void OnCursorVisibilityStateChanged(bool is_visible);
// Notifies the RenderWidgetHost that the View was destroyed.
void ViewDestroyed();
bool visual_properties_ack_pending_for_testing() {
return visual_properties_ack_pending_;
}
// Requests the generation of a new CompositorFrame from the renderer
// by forcing a new surface id.
// It will return false if the renderer is not ready (e.g. there's an
// in flight change).
bool RequestRepaintOnNewSurface();
// Called after every cross-document navigation. Note that for prerender
// navigations, this is called before the renderer is shown.
void DidNavigate();
// Called after every cross-document navigation. When `active` is true, the
// displayed graphics of the renderer is cleared after a certain timeout if it
// does not produce a new CompositorFrame after navigation.
//
// This is called after either navigation (for non-prerender pages) or
// activation (for prerender pages).
// TODO(mustaq@chromium.org): Is this still correct for prerendered pages?
void InitializePaintHolding(bool active);
// Customize the value of `new_content_rendering_delay_` for testing.
void SetNewContentRenderingTimeoutForTesting(base::TimeDelta timeout);
// Forwards the keyboard event with optional commands to the renderer. If
// |key_event| is not forwarded for any reason, then |commands| are ignored.
// |update_event| (if non-null) is set to indicate whether the underlying
// event in |key_event| should be updated. |update_event| is only used on
// aura.
void ForwardKeyboardEventWithCommands(
const input::NativeWebKeyboardEvent& key_event,
const ui::LatencyInfo& latency,
std::vector<blink::mojom::EditCommandPtr> commands,
bool* update_event = nullptr);
// Forwards the given message to the renderer. These are called by the view
// when it has received a message.
void ForwardKeyboardEventWithLatencyInfo(
const input::NativeWebKeyboardEvent& key_event,
const ui::LatencyInfo& latency) override;
void ForwardMouseEventWithLatencyInfo(const blink::WebMouseEvent& mouse_event,
const ui::LatencyInfo& latency);
void ForwardWheelEventWithLatencyInfo(
const blink::WebMouseWheelEvent& wheel_event,
const ui::LatencyInfo& latency) override;
// Resolves the given callback once all effects of prior input have been
// fully realized.
void WaitForInputProcessed(SyntheticGestureParams::GestureType type,
content::mojom::GestureSourceType source,
base::OnceClosure callback);
// Resolves the given callback once all effects of previously forwarded input
// have been fully realized (i.e. resulting compositor frame has been drawn,
// swapped, and presented).
void WaitForInputProcessed(base::OnceClosure callback);
// Queues a synthetic gesture for testing purposes. Invokes the on_complete
// callback when the gesture is finished running.
void QueueSyntheticGesture(
std::unique_ptr<SyntheticGesture> synthetic_gesture,
base::OnceCallback<void(SyntheticGesture::Result)> on_complete);
void QueueSyntheticGestureCompleteImmediately(
std::unique_ptr<SyntheticGesture> synthetic_gesture);
// Ensures the renderer is in a state ready to receive synthetic input. The
// SyntheticGestureController will normally ensure this before sending the
// first gesture; however, in some tests that may be a bad time (e.g. the
// gesture is sent while the main thread is blocked) so this allows the
// caller to do so manually.
void EnsureReadyForSyntheticGestures(base::OnceClosure on_ready);
// Update the composition node of the renderer (or WebKit).
// WebKit has a special node (a composition node) for input method to change
// its text without affecting any other DOM nodes. When the input method
// (attached to the browser) updates its text, the browser sends IPC messages
// to update the composition node of the renderer.
// (Read the comments of each function for its detail.)
// Sets the text of the composition node.
// This function can also update the cursor position and mark the specified
// range in the composition node.
// A browser should call this function:
// * when it receives a WM_IME_COMPOSITION message with a GCS_COMPSTR flag
// (on Windows);
// * when it receives a "preedit_changed" signal of GtkIMContext (on Linux);
// * when markedText of NSTextInput is called (on Mac).
void ImeSetComposition(const std::u16string& text,
const std::vector<ui::ImeTextSpan>& ime_text_spans,
const gfx::Range& replacement_range,
int selection_start,
int selection_end);
// Deletes the ongoing composition if any, inserts the specified text, and
// moves the cursor.
// A browser should call this function or ImeFinishComposingText:
// * when it receives a WM_IME_COMPOSITION message with a GCS_RESULTSTR flag
// (on Windows);
// * when it receives a "commit" signal of GtkIMContext (on Linux);
// * when insertText of NSTextInput is called (on Mac).
void ImeCommitText(const std::u16string& text,
const std::vector<ui::ImeTextSpan>& ime_text_spans,
const gfx::Range& replacement_range,
int relative_cursor_pos);
// Finishes an ongoing composition.
// A browser should call this function or ImeCommitText:
// * when it receives a WM_IME_COMPOSITION message with a GCS_RESULTSTR flag
// (on Windows);
// * when it receives a "commit" signal of GtkIMContext (on Linux);
// * when insertText of NSTextInput is called (on Mac).
void ImeFinishComposingText(bool keep_selection);
// Cancels an ongoing composition.
void ImeCancelComposition();
// Whether forwarded WebInputEvents or other events are being ignored.
bool IsIgnoringWebInputEvents(
const blink::WebInputEvent& event) const override;
bool IsIgnoringInputEvents() const;
// Called when the response to a pending pointer lock request has arrived.
// Returns true if |allowed| is true and the pointer has been successfully
// locked.
bool GotResponseToPointerLockRequest(blink::mojom::PointerLockResult result);
// Called when the response to a pending keyboard lock request has arrived.
// |allowed| should be true if the current tab is in tab initiated fullscreen
// mode.
void GotResponseToKeyboardLockRequest(bool allowed);
// Called when the response to an earlier WidgetMsg_ForceRedraw message has
// arrived. The reply includes the snapshot-id from the request.
void GotResponseToForceRedraw(int snapshot_id);
// When the WebContents (which acts as the Delegate) is destroyed, this object
// may still outlive it while the renderer is shutting down. In that case the
// delegate pointer is removed (since it would be a UAF).
void DetachDelegate();
// Update the renderer's cache of the screen rect of the view and window.
void SendScreenRects();
// Indicates whether the renderer drives the RenderWidgetHosts's size or the
// other way around.
bool auto_resize_enabled() { return auto_resize_enabled_; }
// The minimum size of this renderer when auto-resize is enabled.
const gfx::Size& min_size_for_auto_resize() const {
return min_size_for_auto_resize_;
}
// The maximum size of this renderer when auto-resize is enabled.
const gfx::Size& max_size_for_auto_resize() const {
return max_size_for_auto_resize_;
}
// Don't check whether we expected a resize ack during web tests.
static void DisableResizeAckCheckForTesting();
// TODO(mustaq@chromium.org): Fix the odd name, should be capitalized!
input::InputRouter* input_router();
void SetForceEnableZoom(bool);
// Get the BrowserAccessibilityManager for the root of the frame tree,
ui::BrowserAccessibilityManager* GetRootBrowserAccessibilityManager();
// Get the BrowserAccessibilityManager for the root of the frame tree,
// or create it if it doesn't already exist.
ui::BrowserAccessibilityManager* GetOrCreateRootBrowserAccessibilityManager();
// Virtual for testing.
virtual void RejectPointerLockOrUnlockIfNecessary(
blink::mojom::PointerLockResult reason);
// Store values received in a child frame RenderWidgetHost from a parent
// RenderWidget, in order to pass them to the renderer and continue their
// propagation down the RenderWidget tree.
void SetVisualPropertiesFromParentFrame(
float page_scale_factor,
float compositing_scale_factor,
bool is_pinch_gesture_active,
const gfx::Size& visible_viewport_size,
const gfx::Rect& compositor_viewport,
std::vector<gfx::Rect> root_widget_viewport_segments);
// Indicates if the render widget host should track the render widget's size
// as opposed to visa versa.
// In main frame RenderWidgetHosts this controls the value for the frame tree.
// In child frame RenderWidgetHosts this value comes from the parent
// RenderWidget and should be propagated down the RenderWidgetTree.
void SetAutoResize(bool enable,
const gfx::Size& min_size,
const gfx::Size& max_size);
// Generates a filled in VisualProperties struct representing the current
// properties of this widget.
blink::VisualProperties GetVisualProperties();
// Returns the result of GetVisualProperties(), resetting and storing that
// value as what has been sent to the renderer. This should be called when
// getting VisualProperties that will be sent in order to create a
// RenderWidget, since the creation acts as the initial
// SynchronizeVisualProperties().
//
// This has the side effect of resetting state that should match a newly
// created RenderWidget in the renderer.
//
// TODO(dcheng): Tests call this directly but shouldn't have to. Investigate
// getting rid of this.
blink::VisualProperties GetInitialVisualProperties();
// Clears the state of the VisualProperties of this widget.
void ClearVisualProperties();
// Pushes updated visual properties to the renderer as well as whether the
// focused node should be scrolled into view.
bool SynchronizeVisualProperties(bool scroll_focused_node_into_view,
bool propagate = true);
// Similar to SynchronizeVisualProperties(), but performed even if
// |visual_properties_ack_pending_| is set. Used to guarantee that the
// latest visual properties are sent to the renderer before another IPC.
bool SynchronizeVisualPropertiesIgnoringPendingAck();
// Called when we receive a notification indicating that the renderer process
// is gone. This will reset our state so that our state will be consistent if
// a new renderer is created.
void RendererExited();
// Called from a RenderFrameHost when the text selection has changed.
void SelectionChanged(const std::u16string& text,
uint32_t offset,
const gfx::Range& range);
bool renderer_initialized() const { return renderer_widget_created_; }
base::WeakPtr<RenderWidgetHostImpl> GetWeakPtr() {
return weak_factory_.GetWeakPtr();
}
// Request composition updates from RenderWidget. If |immediate_request| is
// true, RenderWidget will respond immediately. If |monitor_updates| is true,
// then RenderWidget sends updates for each compositor frame when there are
// changes, or when the text selection changes inside a frame. If both fields
// are false, RenderWidget will not send any updates. To avoid sending
// unnecessary IPCs to RenderWidget (e.g., asking for monitor updates while
// we are already receiving updates), when
// |monitoring_composition_info_| == |monitor_updates| no IPC is sent to the
// renderer unless it is for an immediate request.
void RequestCompositionUpdates(bool immediate_request, bool monitor_updates);
RenderFrameMetadataProviderImpl* render_frame_metadata_provider() {
return &render_frame_metadata_provider_;
}
// SyntheticGestureController::Delegate overrides.
bool HasGestureStopped() override;
// SyntheticGestureController::Delegate, RenderInputRouterDelegate overrides.
// Both of these classes declare this as pure virtual, ensuring that any class
// inheriting from both interfaces must provide a concrete implementation,
// resolving any ambiguity that could arise from multiple inheritance.
bool IsHidden() const override;
// Signals that a frame with token |frame_token| was finished processing. If
// there are any queued messages belonging to it, they will be processed.
void DidProcessFrame(uint32_t frame_token, base::TimeTicks activation_time);
// virtual for testing.
virtual blink::mojom::WidgetInputHandler* GetWidgetInputHandler();
// RenderInputRouterClient overrides.
void OnImeCompositionRangeChanged(
const gfx::Range& range,
const std::optional<std::vector<gfx::Rect>>& character_bounds) override;
void OnImeCancelComposition() override;
void OnStartStylusWriting() override;
void UpdateElementFocusForStylusWriting(
#if BUILDFLAG(IS_WIN)
const gfx::Rect& focus_widget_rect_in_dips
#endif // BUILDFLAG(IS_WIN)
) override;
bool IsAutoscrollInProgress() override;
void SetMouseCapture(bool capture) override;
void SetAutoscrollSelectionActiveInMainFrame(
bool autoscroll_selection) override;
void RequestMouseLock(
bool from_user_gesture,
bool unadjusted_movement,
input::InputRouterImpl::RequestMouseLockCallback response) override;
// PointerLockContext overrides
void RequestMouseLockChange(
bool unadjusted_movement,
PointerLockContext::RequestMouseLockChangeCallback response) override;
// FrameTokenMessageQueue::Client:
void OnInvalidFrameToken(uint32_t frame_token) override;
// FrameTokenMessageQueue::Client:
std::string GetMainFrameLastCommittedURLSpec() override;
void ProgressFlingIfNeeded(base::TimeTicks current_time);
void StopFling();
RenderWidgetHostViewBase* GetRenderWidgetHostViewBase();
// The RenderWidgetHostImpl will keep showing the old page (for a while) after
// navigation until the first frame of the new page arrives. This reduces
// flicker. However, if for some reason it is known that the frames won't be
// arriving, this call can be used for force a timeout, to avoid showing the
// content of the old page under UI from the new page.
void ForceFirstFrameAfterNavigationTimeout();
void SetScreenOrientationForTesting(uint16_t angle,
display::mojom::ScreenOrientation type);
// Requests keyboard lock. If `codes` has no value then all keys will be
// locked, otherwise only the keys specified will be intercepted and routed to
// the web page. `request_keyboard_lock_callback` gets called with the result
// of the request, possibly before the lock actually takes effect.
void RequestKeyboardLock(
std::optional<base::flat_set<ui::DomCode>> codes,
base::OnceCallback<void(blink::mojom::KeyboardLockRequestResult)>
request_keyboard_lock_callback);
// Cancels a previous keyboard lock request.
void CancelKeyboardLock();
// Indicates whether keyboard lock is active.
bool IsKeyboardLocked() const;
// Returns the keyboard layout mapping.
base::flat_map<std::string, std::string> GetKeyboardLayoutMap();
void RequestForceRedraw(int snapshot_id);
bool IsContentRenderingTimeoutRunning() const;
enum class RendererIsUnresponsiveReason {
kOnInputEventAckTimeout = 0,
kNavigationRequestCommitTimeout = 1,
kRendererCancellationThrottleTimeout = 2,
kMaxValue = kRendererCancellationThrottleTimeout,
};
// Called on delayed response from the renderer by either
// 1) |hang_monitor_timeout_| (slow to ack input events) or
// 2) NavigationHandle::OnCommitTimeout (slow to commit) or
// 3) RendererCancellationThrottle::OnTimeout (slow cancelling navigation).
void RendererIsUnresponsive(
RendererIsUnresponsiveReason reason,
base::RepeatingClosure restart_hang_monitor_timeout);
// Called during frame eviction to return all SurfaceIds in the frame tree.
// Marks all views in the frame tree as evicted.
std::vector<viz::SurfaceId> CollectSurfaceIdsForEviction();
const mojo::AssociatedRemote<blink::mojom::FrameWidget>&
GetAssociatedFrameWidget();
blink::mojom::FrameWidgetInputHandler* GetFrameWidgetInputHandler();
// Exposed so that tests can swap the implementation and intercept calls.
mojo::AssociatedReceiver<blink::mojom::FrameWidgetHost>&
frame_widget_host_receiver_for_testing() {
return blink_frame_widget_host_receiver_;
}
// Exposed so that tests can swap the implementation and intercept calls.
mojo::AssociatedReceiver<blink::mojom::PopupWidgetHost>&
popup_widget_host_receiver_for_testing() {
return blink_popup_widget_host_receiver_;
}
// Exposed so that tests can swap the implementation and intercept calls.
mojo::AssociatedReceiver<blink::mojom::WidgetHost>&
widget_host_receiver_for_testing() {
return blink_widget_host_receiver_;
}
std::optional<blink::VisualProperties> LastComputedVisualProperties() const;
// Generates widget creation params that will be passed to the renderer to
// create a new widget. As a side effect, this resets various widget and frame
// widget Mojo interfaces and rebinds them, passing the new endpoints in the
// returned params.
mojom::CreateFrameWidgetParamsPtr BindAndGenerateCreateFrameWidgetParams();
// RenderFrameMetadataProvider::Observer implementation.
void OnRenderFrameMetadataChangedBeforeActivation(
const cc::RenderFrameMetadata& metadata) override;
void OnRenderFrameMetadataChangedAfterActivation(
base::TimeTicks activation_time) override;
void OnRenderFrameSubmission() override;
void OnLocalSurfaceIdChanged(
const cc::RenderFrameMetadata& metadata) override;
SiteInstanceGroup* GetSiteInstanceGroup();
void PassImeRenderWidgetHost(
mojo::PendingRemote<blink::mojom::ImeRenderWidgetHost> pending_remote);
// Updates the browser controls by directly IPCing onto the compositor thread.
void UpdateBrowserControlsState(
cc::BrowserControlsState constraints,
cc::BrowserControlsState current,
bool animate,
const std::optional<cc::BrowserControlsOffsetTagModifications>&
offset_tag_modifications);
void StartDragging(blink::mojom::DragDataPtr drag_data,
const url::Origin& source_origin,
blink::DragOperationsMask drag_operations_mask,
const SkBitmap& unsafe_bitmap,
const gfx::Vector2d& cursor_offset_in_dip,
const gfx::Rect& drag_obj_rect_in_dip,
blink::mojom::DragEventSourceInfoPtr event_info);
// Notifies the widget that the viz::FrameSinkId assigned to it is now bound
// to its renderer side widget. If the renderer issued a FrameSink request
// before this handoff, the request is buffered and will be issued here.
void SetViewIsFrameSinkIdOwner(bool is_owner);
bool view_is_frame_sink_id_owner() const {
return view_is_frame_sink_id_owner_;
}
// Helper class to log navigation-related compositor metrics. Keeps track of
// the timestamp when navigation commit/RFH swap/frame sink request happened
// for the first navigation that uses this RenderWidgetHost.
class CompositorMetricRecorder {
public:
CompositorMetricRecorder(RenderWidgetHostImpl* owner);
~CompositorMetricRecorder() = default;
// The functions below are called when the first navigation that uses this
// RenderWidgetHost commits/swaps in the RenderFrameHost/requested frame
// sink creation respectively.
void DidStartNavigationCommit();
void DidSwap();
void DidRequestFrameSink();
base::TimeTicks CommitNavigationTime();
private:
void TryToRecordMetrics();
// The timestamp of the last call to
// `MaybeDispatchBufferedFrameSinkRequest()` where we run
// `create_frame_sink_callback_`.
base::TimeTicks create_frame_sink_timestamp_;
// The timestamp of when the navigation that created this RenderWidgetHost
// committed/swapped in the RenderFrameHost.
base::TimeTicks commit_nav_timestamp_;
base::TimeTicks swap_rfh_timestamp_;
const raw_ptr<RenderWidgetHostImpl> owner_;
};
CompositorMetricRecorder* compositor_metric_recorder() const {
return compositor_metric_recorder_.get();
}
// Disables recording metrics through CompositorMetricRecorder by resetting
// the owned `compositor_metric_recorder_`.
void DisableCompositorMetricRecording();
virtual input::RenderInputRouter*
GetRenderInputRouter(); // virtual for testing.
// Requests a commit and forced redraw in the renderer compositor.
void ForceRedrawForTesting();
// Indicates the page is discarding. The renderer process will get
// cpu-priority boosted to run discard logic.
void SetIsDiscarding(bool is_discarding);
SyntheticGestureController* SyntheticGestureControllerForTesting() {
return synthetic_gesture_controller_.get();
}
protected:
// |routing_id| must not be IPC::mojom::kRoutingIdNone.
// If this object outlives |delegate|, DetachDelegate() must be called when
// |delegate| goes away. |site_instance_group| will outlive this
// widget but we store it via a `base::SafeRef` instead of a scoped_refptr to
// not create a cycle and keep alive the `SiteInstanceGroup`.
RenderWidgetHostImpl(FrameTree* frame_tree,
bool self_owned,
viz::FrameSinkId frame_sink_id,
RenderWidgetHostDelegate* delegate,
base::SafeRef<SiteInstanceGroup> site_instance_group,
int32_t routing_id,
bool hidden,
bool renderer_initiated_creation);
// ---------------------------------------------------------------------------
// The following method is overridden by RenderViewHost to send upwards to
// its delegate.
// Callback for notification that we failed to receive any rendered graphics
// from a newly loaded page. Used for testing.
virtual void NotifyNewContentRenderingTimeoutForTesting() {}
// virtual for testing.
virtual void OnMouseEventAck(const input::MouseEventWithLatencyInfo& event,
blink::mojom::InputEventResultSource ack_source,
blink::mojom::InputEventResultState ack_result);
// ---------------------------------------------------------------------------
bool IsPointerLocked() const;
std::unique_ptr<input::FlingSchedulerBase> MakeFlingScheduler();
private:
FRIEND_TEST_ALL_PREFIXES(FullscreenDetectionTest,
EncompassingDivNotFullscreen);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
DoNotAcceptPopupBoundsUntilScreenRectsAcked);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
DontPostponeInputEventAckTimeout);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest, RendererExitedNoDrag);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
StopAndStartInputEventAckTimeout);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
ShorterDelayInputEventAckTimeout);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
AddAndRemoveInputEventObserver);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
ScopedObservationWithInputEventObserver);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
AddAndRemoveImeInputEventObserver);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest,
InputRouterReceivesHasTouchEventHandlers);
FRIEND_TEST_ALL_PREFIXES(OOPIFScrollBubblingTest,
ScrollBubblingWithTouchMoveInjection);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest, EventDispatchPostDetach);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostTest, InputEventRWHLatencyComponent);
FRIEND_TEST_ALL_PREFIXES(DevToolsAgentHostImplTest,
NoUnresponsiveDialogInInspectedContents);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewMacTest,
ConflictingAllocationsResolve);
FRIEND_TEST_ALL_PREFIXES(SitePerProcessBrowserTest,
ResizeAndCrossProcessPostMessagePreserveOrder);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostInputEventRouterTest,
EnsureRendererDestroyedHandlesUnAckedTouchEvents);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewAuraTest, TouchEventState);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewAuraTest, TouchEventSyncAsync);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewAuraOverscrollTest,
OverscrollWithTouchEvents);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewAuraOverscrollTest,
TouchGestureEndDispatchedAfterOverscrollComplete);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostViewAuraTest,
InvalidEventsHaveSyncHandlingDisabled);
FRIEND_TEST_ALL_PREFIXES(RenderWidgetHostInputEventRouterTest,
EnsureRendererDestroyedHandlesUnAckedTouchEvents);
friend class MockRenderWidgetHost;
friend class MockRenderWidgetHostImpl;
friend class OverscrollNavigationOverlayTest;
friend class RenderViewHostTester;
friend class TestRenderViewHost;
// Tell this object to destroy itself. If |also_delete| is specified, the
// destructor is called as well.
void Destroy(bool also_delete);
// Called by |new_content_rendering_timeout_| if a renderer has loaded new
// content but failed to produce a compositor frame in a defined time.
void ClearDisplayedGraphics();
// InputRouter::SendKeyboardEvent() callbacks to this. This may be called
// synchronously.
void OnKeyboardEventAck(
const input::NativeWebKeyboardEventWithLatencyInfo& event,
blink::mojom::InputEventResultSource ack_source,
blink::mojom::InputEventResultState ack_result);
// Release the pointer lock
void UnlockPointer();
// IPC message handlers
void OnClose();
void OnUpdateScreenRectsAck();
void OnUpdateDragOperation(DragOperationCallback callback,
ui::mojom::DragOperation current_op,
bool document_is_handling_drag);
// blink::mojom::FrameWidgetHost overrides.
void AnimateDoubleTapZoomInMainFrame(const gfx::Point& tap_point,
const gfx::Rect& rect_to_zoom) override;
void ZoomToFindInPageRectInMainFrame(const gfx::Rect& rect_to_zoom) override;
void SetHasTouchEventConsumers(
blink::mojom::TouchEventConsumersPtr consumers) override;
void IntrinsicSizingInfoChanged(
blink::mojom::IntrinsicSizingInfoPtr sizing_info) override;
void AutoscrollStart(const gfx::PointF& position) override;
void AutoscrollFling(const gfx::Vector2dF& velocity) override;
void AutoscrollEnd() override;
// When the RenderWidget is destroyed and recreated, this resets states in the
// browser to match the clean start for the renderer side.
void ResetStateForCreatedRenderWidget(
const blink::VisualProperties& initial_props);
// Returns true if the |new_visual_properties| differs from
// |old_page_visual_properties| in a way that indicates a size changed.
static bool DidVisualPropertiesSizeChange(
const blink::VisualProperties& old_visual_properties,
const blink::VisualProperties& new_visual_properties);
// Returns true if the new visual properties requires an ack from a
// synchronization message.
static bool DoesVisualPropertiesNeedAck(
const std::unique_ptr<blink::VisualProperties>& old_visual_properties,
const blink::VisualProperties& new_visual_properties);
// Returns true if |old_visual_properties| is out of sync with
// |new_visual_properties|.
static bool StoredVisualPropertiesNeedsUpdate(
const std::unique_ptr<blink::VisualProperties>& old_visual_properties,
const blink::VisualProperties& new_visual_properties);
// Give key press listeners a chance to handle this key press. This allow
// widgets that don't have focus to still handle key presses.
bool KeyPressListenersHandleEvent(const input::NativeWebKeyboardEvent& event);
void WindowSnapshotReachedScreen(int snapshot_id);
void OnSnapshotFromSurfaceReceived(
int snapshot_id,
int retry_count,
const viz::CopyOutputBitmapWithMetadata& result);
void OnSnapshotReceived(int snapshot_id, gfx::Image image);
// This is called after the renderer attempts to focus content eligible for
// handwriting via mojom::blink::FrameWidget::OnStartStylusWriting. If content
// eligible for stylus handwriting has focus, then `focus_result` will be set,
// otherwise it will be nullptr.
void OnUpdateElementFocusForStylusWritingHandled(
blink::mojom::StylusWritingFocusResultPtr focus_result);
// Called by the RenderProcessHost to handle the case when the process
// changed its state of being blocked.
void RenderProcessBlockedStateChanged(bool blocked);
void NotifyVizOfPageVisibilityUpdates();
// 1. Grants permissions to URL (if any)
// 2. Grants permissions to filenames
// 3. Grants permissions to file system files.
// 4. Register the files with the IsolatedContext.
void GrantFileAccessFromDropData(DropData* drop_data);
// Implementation of |hang_monitor_restarter| callback passed to
// RenderWidgetHostDelegate::RendererUnresponsive if the unresponsiveness
// was noticed because of input event ack timeout.
void RestartRenderInputRouterInputEventAckTimeout();
void SetupRenderInputRouter();
void SetupInputRouter();
// Start intercepting system keyboard events.
void LockKeyboard();
// Stop intercepting system keyboard events.
void UnlockKeyboard();
#if BUILDFLAG(IS_MAC)
device::mojom::WakeLock* GetWakeLock();
#endif
void CreateSyntheticGestureControllerIfNecessary();
// Converts the |window_point| from the coordinates in native window in DIP
// to Blink's Viewport coordinates. They're identical in traditional world,
// but will differ when use-zoom-for-dsf feature is enabled.
// TODO(oshima): Update the comment when the migration is completed.
gfx::PointF ConvertWindowPointToViewport(const gfx::PointF& window_point);
// Dispatch any buffered FrameSink requests from the renderer if the widget
// has a view and is the owner for the FrameSinkId assigned to it.
void MaybeDispatchBufferedFrameSinkRequest();
raw_ptr<FrameTree> frame_tree_;
// RenderWidgetHost are either:
// - Owned by RenderViewHostImpl.
// - Owned by RenderFrameHost, for local root iframes.
// - Self owned. Lifetime is managed from the renderer process, via Mojo IPC;
// This is used to implement:
// - Color Chooser popup.
// - Date/Time chooser popup.
// - Internal popup. Essentially, the <select> element popup.
//
// self_owned RenderWidgetHost are expected to be deleted using:
// ShutdownAndDestroyWidget(true /* also_delete */);
bool self_owned_;
// True while there is an established connection to a renderer-side Widget
// object.
bool renderer_widget_created_ = false;
// When the renderer widget is created, if created by the renderer, it may
// request to avoid showing the widget until requested. In that case, this
// value is set to true, and we defer WasShown() events until the request
// arrives which is signaled by Init().
bool waiting_for_init_;
// True if |Destroy()| has been called.
bool destroyed_ = false;
// Handles mojo connections for RenderInputRouterDelegate[Client] interface to
// allow sycing information between the Browser and the GPU process for input
// handling with InputVizard.
MojoRenderInputRouterDelegateImpl mojo_rir_delegate_impl_{this};
// Our delegate, which wants to know mainly about keyboard events.
// It will remain non-null until DetachDelegate() is called.
raw_ptr<RenderWidgetHostDelegate, FlakyDanglingUntriaged> delegate_;
// The delegate of the owner of this object.
// This member is non-null if and only if this RenderWidgetHost is associated
// with a main frame RenderWidget.
raw_ptr<RenderWidgetHostOwnerDelegate> owner_delegate_ = nullptr;
// AgentSchedulingGroupHost to be used for IPC with the corresponding
// (renderer-side) AgentSchedulingGroup. Its channel may be nullptr if the
// renderer crashed. We store it here as a separate reference instead of
// dynamically fetching it from `site_instance_group_` since its
// value gets cleared early in `SiteInstanceGroup` via
// RenderProcessHostDestroyed before this object is destroyed.
const raw_ref<AgentSchedulingGroupHost> agent_scheduling_group_;
// The SiteInstanceGroup this RenderWidgetHost belongs to.
// TODO(crbug.com/40258727) Turn this into base::SafeRef
base::WeakPtr<SiteInstanceGroup> site_instance_group_;
// The ID of the corresponding object in the Renderer Instance.
const int routing_id_;
// Indicates whether the page is hidden or not. Need to call
// process_->UpdateClientPriority when this value changes.
bool is_hidden_ = true;
// Indicates whether the widget is ever shown to the user so far. This state
// remains `false` until `is_hidden_` becomes `false` for the first time.
bool was_ever_shown_ = false;
// Records the time when `was_ever_shown_` above becomes `true` for the first
// time.
base::TimeTicks first_shown_time_;
// Records the latest time when |this| widget's visibility state changes from
// hidden to shown.
base::TimeTicks latest_shown_time_;
// Indicates whether the renderer host has received the first metadata signal
// implying the renderer has pushed content to cc.
bool first_content_metadata_received_ = false;
// Records the time when `first_content_metadata_received_` above becomes
// `true` for the first time.
base::TimeTicks first_content_metadata_time_;
// True if both of the following conditions are met:
// - This widget is for the main frame of the outermost frame tree.
// - A RenderWidgetHostView has been attached to this widget using `SetView`.
bool is_topmost_frame_widget_with_view_ = false;
// For a widget that does not have an associated RenderFrame/View, assume it
// is depth 1, ie just below the root widget.
unsigned int frame_depth_ = 1u;
// Indicates that widget has a frame that intersects with the viewport. Note
// this is independent of |is_hidden_|. For widgets not associated with
// RenderFrame/View, assume false.
bool intersects_viewport_ = false;
// While the main frame is being discarded, the renderer process need to be
// foreground.
// This is only effective when WebContentsDiscard feature is enabled.
bool is_discarding_ = false;
// Indicates whether this widget contributes to the priority of the process.
bool should_contribute_priority_to_process_ = true;
// Determines whether the page is mobile optimized or not.
bool is_mobile_optimized_ = false;
// One side of a pipe that is held open while the pointer is locked.
// The other side is held be the renderer.
mojo::Receiver<blink::mojom::PointerLockContext> pointer_lock_context_{this};
// Tracks if LostFocus() has been called on this RenderWidgetHost since the
// previous change in focus. This tracks behaviors like a user clicking out of
// the page and into a UI element when verifying if a change in focus is
// allowed. The value will be reset after a RFHI gets focus. The RFHI will
// then keep track of this value to handle passing focus to other frames.
bool has_lost_focus_ = false;
#if BUILDFLAG(IS_ANDROID)
// Tracks the current importance of widget.
ChildProcessImportance importance_ = ChildProcessImportance::NORMAL;
#endif
// True when waiting for visual_properties_ack.
bool visual_properties_ack_pending_ = false;
// Visual properties that were most recently sent to the renderer.
std::unique_ptr<blink::VisualProperties> old_visual_properties_;
// True if the render widget host should track the render widget's size as
// opposed to visa versa.
bool auto_resize_enabled_ = false;
// The minimum size for the render widget if auto-resize is enabled.
gfx::Size min_size_for_auto_resize_;
// The maximum size for the render widget if auto-resize is enabled.
gfx::Size max_size_for_auto_resize_;
// These properties are propagated down the RenderWidget tree from the main
// frame to a child frame RenderWidgetHost. They are not used on a top-level
// RenderWidgetHost. The child frame RenderWidgetHost stores these values to
// pass them to the renderer, instead of computing them for itself. It
// collects them and passes them though
// blink::mojom::Widget::UpdateVisualProperties so that the renderer receives
// updates in an atomic fashion along with a synchronization token for the
// compositor in a LocalSurfaceId.
struct MainFramePropagationProperties {
MainFramePropagationProperties();
~MainFramePropagationProperties();
// The page-scale factor of the main-frame.
float page_scale_factor = 1.f;
// This represents the child frame's raster scale factor which takes into
// account the transform from child frame space to main frame space.
float compositing_scale_factor = 1.f;
// True when the renderer is currently undergoing a pinch-zoom gesture.
bool is_pinch_gesture_active = false;
// The size of the main frame's widget in device px.
gfx::Size visible_viewport_size;
gfx::Rect compositor_viewport;
// The logical segments of the root widget, in DIPs relative to the root
// RenderWidgetHost.
std::vector<gfx::Rect> root_widget_viewport_segments;
} properties_from_parent_local_root_;
bool waiting_for_screen_rects_ack_ = false;
gfx::Rect last_view_screen_rect_;
gfx::Rect last_window_screen_rect_;
// Keyboard event listeners.
std::vector<KeyPressEventCallback> key_press_event_callbacks_;
// Mouse event callbacks.
std::vector<MouseEventCallback> mouse_event_callbacks_;
// Suppress showing keyboard callbacks.
std::vector<SuppressShowingImeCallback> suppress_showing_ime_callbacks_;
// Input event callbacks.
base::ObserverList<RenderWidgetHost::InputEventObserver>::Unchecked
input_event_observers_;
#if BUILDFLAG(IS_ANDROID)
// Ime input event callbacks. This is separated from input_event_observers_,
// because not all text events are triggered by input events on Android.
base::ObserverList<RenderWidgetHost::InputEventObserver>::Unchecked
ime_input_event_observers_;
#endif
// The observers watching us.
base::ObserverList<RenderWidgetHostObserver> observers_;
// This is true if the renderer is currently unresponsive.
bool is_unresponsive_ = false;
// Set when we update the text direction of the selected input element.
bool text_direction_updated_ = false;
base::i18n::TextDirection text_direction_ = base::i18n::LEFT_TO_RIGHT;
// Indicates if Char and KeyUp events should be suppressed or not. Usually all
// events are sent to the renderer directly in sequence. However, if a
// RawKeyDown event was handled by PreHandleKeyboardEvent() or
// KeyPressListenersHandleEvent(), e.g. as an accelerator key, then the
// RawKeyDown event is not sent to the renderer, and the following sequence of
// Char and KeyUp events should also not be sent. Otherwise the renderer will
// see only the Char and KeyUp events and cause unexpected behavior. For
// example, pressing alt-2 may let the browser switch to the second tab, but
// the Char event generated by alt-2 may also activate a HTML element if its
// accesskey happens to be "2", then the user may get confused when switching
// back to the original tab, because the content may already have changed.
bool suppress_events_until_keydown_ = false;
bool pending_pointer_lock_request_ = false;
bool pointer_lock_raw_movement_ = false;
// Stores the keyboard keys to lock while waiting for a pending lock request.
std::optional<base::flat_set<ui::DomCode>> keyboard_keys_to_lock_;
bool keyboard_lock_allowed_ = false;
base::OnceCallback<void(blink::mojom::KeyboardLockRequestResult)>
keyboard_lock_request_callback_;
// Used when locking to indicate when a target application has voluntarily
// unlocked and desires to relock the mouse. If the mouse is unlocked due
// to ESC being pressed by the user, this will be false.
bool is_last_unlocked_by_target_ = false;
// True when the cursor has entered the autoscroll mode. A GSB is not
// necessarily sent yet.
bool autoscroll_in_progress_ = false;
// TODO(crbug.com/40263900): The gesture controller can cause synchronous
// destruction of the page (sending a click to the tab close button). Since
// that'll destroy the RenderWidgetHostImpl, having it own the controller is
// awkward.
std::unique_ptr<SyntheticGestureController> synthetic_gesture_controller_;
// The View associated with the RenderWidgetHost. The lifetime of this object
// is associated with the lifetime of the Render process. If the Renderer
// crashes, its View is destroyed and this pointer becomes NULL, even though
// render_view_host_ lives on to load another URL (creating a new View while
// doing so).
base::WeakPtr<RenderWidgetHostViewBase> view_;
// Receives and handles input events.
std::unique_ptr<input::RenderInputRouter> render_input_router_;
base::CallbackListSubscription
render_process_blocked_state_changed_subscription_;
std::unique_ptr<input::TimeoutMonitor> new_content_rendering_timeout_;
bool paint_holding_activated_ = false;
int next_browser_snapshot_id_ = 1;
using PendingSnapshotMap = std::map<int, GetSnapshotFromBrowserCallback>;
PendingSnapshotMap pending_browser_snapshots_;
PendingSnapshotMap pending_surface_browser_snapshots_;
// Indicates whether this RenderWidgetHost thinks is focused. This is trying
// to match what the renderer process knows. It is different from
// RenderWidgetHostView::HasFocus in that in that the focus request may fail,
// causing HasFocus to return false when is_focused_ is true.
bool is_focused_ = false;
// Indicates whether what the last focus active state that was sent to the
// renderer.
bool is_active_ = false;
// This value indicates how long to wait for a new compositor frame from a
// renderer process before clearing any previously displayed content.
base::TimeDelta new_content_rendering_delay_;
// When true, the RenderWidget is regularly sending updates regarding
// composition info. It should only be true when there is a focused editable
// node.
bool monitoring_composition_info_ = false;
#if BUILDFLAG(IS_MAC)
mojo::Remote<device::mojom::WakeLock> wake_lock_;
#endif
// Stash a request to create a CompositorFrameSink if it arrives before we
// have a view.
base::OnceCallback<void(base::UnguessableToken, const viz::FrameSinkId&)>
create_frame_sink_callback_;
std::optional<uint16_t> screen_orientation_angle_for_testing_;
std::optional<display::mojom::ScreenOrientation>
screen_orientation_type_for_testing_;
RenderFrameMetadataProviderImpl render_frame_metadata_provider_;
bool surface_id_allocation_suppressed_ = false;
const viz::FrameSinkId frame_sink_id_;
// Used to avoid unnecessary IPC calls when ForwardDelegatedInkPoint receives
// the same point twice.
std::optional<gfx::DelegatedInkPoint> last_delegated_ink_point_sent_;
bool sent_autoscroll_scroll_begin_ = false;
gfx::PointF autoscroll_start_position_;
input::InputRouterImpl::RequestMouseLockCallback request_pointer_lock_callback_;
ui::mojom::TextInputStatePtr saved_text_input_state_for_suppression_;
// Parameters to pass to blink::mojom::Widget::WasShown after
// `waiting_for_init_` becomes true. These are stored in a struct instead of
// storing a callback so that they can be updated if
// RequestSuccessfulPresentationTimeForNextFrame is called while waiting.
struct PendingShowParams {
PendingShowParams(bool is_evicted,
blink::mojom::RecordContentToVisibleTimeRequestPtr
visible_time_request);
~PendingShowParams();
PendingShowParams(const PendingShowParams&) = delete;
PendingShowParams& operator=(const PendingShowParams&) = delete;
bool is_evicted;
blink::mojom::RecordContentToVisibleTimeRequestPtr visible_time_request;
};
std::optional<PendingShowParams> pending_show_params_;
// If this is initialized with a frame this member will be valid and
// can be used to send messages directly to blink.
mojo::AssociatedReceiver<blink::mojom::FrameWidgetHost>
blink_frame_widget_host_receiver_{this};
mojo::AssociatedRemote<blink::mojom::FrameWidget> blink_frame_widget_;
// If this is initialized with a popup this member will be valid and
// manages the lifecycle of the popup in blink.
mojo::AssociatedReceiver<blink::mojom::PopupWidgetHost>
blink_popup_widget_host_receiver_{this};
mojo::AssociatedReceiver<blink::mojom::WidgetHost>
blink_widget_host_receiver_{this};
mojo::AssociatedRemote<blink::mojom::Widget> blink_widget_;
mojo::Remote<blink::mojom::WidgetCompositor> widget_compositor_;
// Same-process cross-RenderFrameHost navigations may reuse the compositor
// from the previous RenderFrameHost. While the speculative RenderWidgetHost
// is created early, ownership of the FrameSinkId is transferred at commit.
// This bit is used to track which view owns the FrameSinkId.
//
// The renderer side WebFrameWidget's compositor can create a FrameSink and
// produce frames associated with `frame_sink_id_` only when it owns that
// FrameSinkId.
bool view_is_frame_sink_id_owner_{false};
std::unique_ptr<CompositorMetricRecorder> compositor_metric_recorder_;
base::WeakPtrFactory<RenderWidgetHostImpl> weak_factory_{this};
};
} // namespace content
namespace base {
template <>
struct ScopedObservationTraits<content::RenderWidgetHostImpl,
content::RenderWidgetHost::InputEventObserver> {
static void AddObserver(
content::RenderWidgetHostImpl* source,
content::RenderWidgetHost::InputEventObserver* observer) {
source->AddInputEventObserver(observer);
}
static void RemoveObserver(
content::RenderWidgetHostImpl* source,
content::RenderWidgetHost::InputEventObserver* observer) {
source->RemoveInputEventObserver(observer);
}
};
} // namespace base
#endif // CONTENT_BROWSER_RENDERER_HOST_RENDER_WIDGET_HOST_IMPL_H_