#ifndef ANDROID_WEBVIEW_BROWSER_GFX_BROWSER_VIEW_RENDERER_H_
#define ANDROID_WEBVIEW_BROWSER_GFX_BROWSER_VIEW_RENDERER_H_
#include <stddef.h>
#include <map>
#include <optional>
#include <set>
#include "android_webview/browser/gfx/begin_frame_source_webview.h"
#include "android_webview/browser/gfx/child_frame.h"
#include "android_webview/browser/gfx/compositor_frame_producer.h"
#include "android_webview/browser/gfx/parent_compositor_draw_constraints.h"
#include "android_webview/browser/gfx/root_frame_sink_proxy.h"
#include "base/cancelable_callback.h"
#include "base/functional/callback.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/scoped_refptr.h"
#include "base/task/single_thread_task_runner.h"
#include "base/trace_event/trace_event.h"
#include "components/viz/common/surfaces/frame_sink_id.h"
#include "content/public/browser/android/synchronous_compositor.h"
#include "content/public/browser/android/synchronous_compositor_client.h"
#include "third_party/skia/include/core/SkRefCnt.h"
#include "ui/gfx/geometry/point_f.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/geometry/size_f.h"
#include "ui/gfx/geometry/vector2d_f.h"
class SkCanvas;
class SkPicture;
namespace content {
class WebContents;
}
namespace android_webview {
class BrowserViewRendererClient;
class ChildFrame;
class CompositorFrameConsumer;
class RootFrameSinkProxy;
class BrowserViewRenderer : public content::SynchronousCompositorClient,
public CompositorFrameProducer,
public RootFrameSinkProxyClient {
public:
static BrowserViewRenderer* FromWebContents(
content::WebContents* web_contents);
BrowserViewRenderer(
BrowserViewRendererClient* client,
const scoped_refptr<base::SingleThreadTaskRunner>& ui_task_runner,
const scoped_refptr<base::SingleThreadTaskRunner>& io_task_runner);
BrowserViewRenderer(const BrowserViewRenderer&) = delete;
BrowserViewRenderer& operator=(const BrowserViewRenderer&) = delete;
~BrowserViewRenderer() override;
void RegisterWithWebContents(content::WebContents* web_contents);
void SetCurrentCompositorFrameConsumer(
CompositorFrameConsumer* compositor_frame_consumer);
void PrepareToDraw(const gfx::Point& scroll,
const gfx::Rect& global_visible_rect);
bool OnDrawHardware();
bool OnDrawSoftware(SkCanvas* canvas);
float GetVelocityInPixelsPerSecond();
bool NeedToDrawBackgroundColor();
sk_sp<SkPicture> CapturePicture(int width, int height);
void EnableOnNewPicture(bool enabled);
void ClearView();
void SetOffscreenPreRaster(bool enabled);
void SetIsPaused(bool paused);
void SetViewVisibility(bool visible);
void SetWindowVisibility(bool visible);
void OnSizeChanged(int width, int height);
void OnAttachedToWindow(int width, int height);
void OnDetachedFromWindow();
void ZoomBy(float delta);
void OnComputeScroll(base::TimeTicks animation_time);
void SetDipScale(float dip_scale);
float dip_scale() const { return dip_scale_; }
float page_scale_factor() const { return page_scale_factor_; }
void ScrollTo(const gfx::Point& new_value);
void RestoreScrollAfterTransition(const gfx::Point& new_value);
bool IsVisible() const;
gfx::Rect GetScreenRect() const;
bool view_visible() const { return view_visible_; }
bool window_visible() const { return window_visible_; }
bool attached_to_window() const { return attached_to_window_; }
bool was_attached() const { return was_attached_; }
gfx::Size size() const { return size_; }
bool IsClientVisible() const;
void TrimMemory();
void DidInitializeCompositor(content::SynchronousCompositor* compositor,
const viz::FrameSinkId& frame_sink_id) override;
void DidDestroyCompositor(content::SynchronousCompositor* compositor,
const viz::FrameSinkId& frame_sink_id) override;
void PostInvalidate(content::SynchronousCompositor* compositor) override;
void DidUpdateContent(content::SynchronousCompositor* compositor) override;
void OnInputEvent();
void UpdateRootLayerState(content::SynchronousCompositor* compositor,
const gfx::PointF& total_scroll_offset,
const gfx::PointF& total_max_scroll_offset,
const gfx::SizeF& scrollable_size,
float page_scale_factor,
float min_page_scale_factor,
float max_page_scale_factor) override;
void DidOverscroll(content::SynchronousCompositor* compositor,
const gfx::Vector2dF& accumulated_overscroll,
const gfx::Vector2dF& latest_overscroll_delta,
const gfx::Vector2dF& current_fling_velocity) override;
ui::TouchHandleDrawable* CreateDrawable() override;
void CopyOutput(
content::SynchronousCompositor* compositor,
std::unique_ptr<viz::CopyOutputRequest> copy_request) override;
void AddBeginFrameCompletionCallback(base::OnceClosure callback) override;
void SetThreads(const std::vector<viz::Thread>& threads) override;
base::WeakPtr<CompositorFrameProducer> GetWeakPtr() override;
void RemoveCompositorFrameConsumer(
CompositorFrameConsumer* consumer) override;
void ReturnUsedResources(std::vector<viz::ReturnedResource> resources,
const viz::FrameSinkId& frame_sink_id,
uint32_t layer_tree_frame_sink_id) override;
void OnParentDrawDataUpdated(
CompositorFrameConsumer* compositor_frame_consumer) override;
void OnViewTreeForceDarkStateChanged(
bool view_tree_force_dark_state) override;
void ChildSurfaceWasEvicted() override;
void SetActiveFrameSinkId(const viz::FrameSinkId& frame_sink_id);
void Invalidate() override;
void ReturnResourcesFromViz(
viz::FrameSinkId frame_sink_id,
uint32_t layer_tree_frame_sink_id,
std::vector<viz::ReturnedResource> resources) override;
void OnCompositorFrameTransitionDirectiveProcessed(
viz::FrameSinkId frame_sink_id,
uint32_t layer_tree_frame_sink_id,
uint32_t sequence_id) override;
content::SynchronousCompositor* GetActiveCompositorForTesting() const {
return compositor_;
}
bool window_visible_for_tests() const { return window_visible_; }
private:
void SetActiveCompositor(content::SynchronousCompositor* compositor);
void SetTotalRootLayerScrollOffset(const gfx::PointF& new_value_dip);
bool CanOnDraw();
bool CompositeSW(SkCanvas* canvas, bool software_canvas);
std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
RootLayerStateAsValue(const gfx::PointF& total_scroll_offset_dip,
const gfx::SizeF& scrollable_size_dip);
void ReturnUncommittedFrames(ChildFrameQueue frame);
void ReturnUnusedResource(std::unique_ptr<ChildFrame> frame);
void ReturnResourceFromParent(
CompositorFrameConsumer* compositor_frame_consumer);
void ReleaseHardware();
bool DoUpdateParentDrawData();
void UpdateBeginFrameSource();
void UpdateForegroundForGpuResources();
gfx::Point max_scroll_offset() const;
gfx::Rect ComputeTileRectAndUpdateMemoryPolicy();
content::SynchronousCompositor* FindCompositor(
const viz::FrameSinkId& frame_sink_id) const;
std::string ToString() const;
void SetBrowserIOThreadId(base::PlatformThreadId thread_id);
const raw_ptr<BrowserViewRendererClient> client_;
const scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner_;
raw_ptr<CompositorFrameConsumer> current_compositor_frame_consumer_;
std::unique_ptr<RootFrameSinkProxy> root_frame_sink_proxy_;
raw_ptr<content::SynchronousCompositor> compositor_;
viz::FrameSinkId frame_sink_id_;
std::map<viz::FrameSinkId,
raw_ptr<content::SynchronousCompositor, CtnExperimental>>
compositor_map_;
bool is_paused_;
bool view_visible_;
bool window_visible_;
bool attached_to_window_;
bool was_attached_;
bool hardware_enabled_;
float dip_scale_;
float page_scale_factor_;
float min_page_scale_factor_;
float max_page_scale_factor_;
bool on_new_picture_enable_;
bool clear_view_;
bool foreground_for_gpu_resources_ = false;
bool did_invalidate_since_last_draw_ = false;
bool has_rendered_frame_ = false;
bool offscreen_pre_raster_;
CopyOutputRequestQueue copy_requests_;
gfx::Point last_on_draw_scroll_offset_;
gfx::Rect last_on_draw_global_visible_rect_;
gfx::Size size_;
gfx::SizeF scrollable_size_dip_;
gfx::PointF scroll_offset_unscaled_;
gfx::PointF max_scroll_offset_unscaled_;
gfx::Vector2dF overscroll_rounding_error_;
std::optional<gfx::Point> scroll_on_scroll_state_update_;
ParentCompositorDrawConstraints external_draw_constraints_;
std::unique_ptr<BeginFrameSourceWebView> begin_frame_source_;
std::vector<viz::Thread> renderer_threads_;
base::PlatformThreadId browser_io_thread_id_ = base::kInvalidThreadId;
base::WeakPtrFactory<BrowserViewRenderer> weak_ptr_factory_{this};
};
}
#endif