0
Files
src/content/browser/host_zoom_map_impl.h
W. James MacLean 37af605ea5 Update HostZoomMapImpl, WebContentsImpl to allow independent subframe zoom.
This CL modifies HostZoomMapImpl and WebContentsImpl to allow
independent zoom for subframes. This is needed for the conversion to
MPArch Guestviews as they won't have their own WebContents.

In this CL that is limited to <webview> and GuestView
subframes, but it will be extended to work with the extensions zoom API
and PDFs rendered with OOPIFs instead of GuestViews.

Bug: 376084060
Change-Id: I52e049f2f7aa47d551e1e3aa9a7f85275db08d94
Reviewed-on: https://chromium-review.googlesource.com/c/chromium/src/+/6148714
Commit-Queue: James Maclean <wjmaclean@chromium.org>
Reviewed-by: Elly FJ <ellyjones@chromium.org>
Reviewed-by: Alex Moshchuk <alexmos@chromium.org>
Reviewed-by: Marc Treib <treib@chromium.org>
Reviewed-by: Kevin McNee <mcnee@chromium.org>
Cr-Commit-Position: refs/heads/main@{#1408705}
2025-01-20 08:59:38 -08:00

167 lines
7.0 KiB
C++

// 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_HOST_ZOOM_MAP_IMPL_H_
#define CONTENT_BROWSER_HOST_ZOOM_MAP_IMPL_H_
#include <map>
#include <set>
#include <string>
#include <tuple>
#include <vector>
#include "base/memory/raw_ptr.h"
#include "base/task/sequenced_task_runner_helpers.h"
#include "base/time/time.h"
#include "content/common/content_export.h"
#include "content/public/browser/global_routing_id.h"
#include "content/public/browser/host_zoom_map.h"
namespace content {
class RenderFrameHostImpl;
class WebContentsImpl;
// HostZoomMap lives on the UI thread.
class CONTENT_EXPORT HostZoomMapImpl : public HostZoomMap {
public:
HostZoomMapImpl();
HostZoomMapImpl(const HostZoomMapImpl&) = delete;
HostZoomMapImpl& operator=(const HostZoomMapImpl&) = delete;
~HostZoomMapImpl() override;
// HostZoomMap implementation:
void CopyFrom(HostZoomMap* copy) override;
double GetZoomLevelForHostAndScheme(const std::string& scheme,
const std::string& host) override;
// TODO(wjmaclean) Should we use a GURL here? crbug.com/384486
bool HasZoomLevel(const std::string& scheme,
const std::string& host) override;
ZoomLevelVector GetAllZoomLevels() override;
void SetZoomLevelForHost(const std::string& host, double level) override;
void InitializeZoomLevelForHost(const std::string& host,
double level,
base::Time last_modified) override;
void SetZoomLevelForHostAndScheme(const std::string& scheme,
const std::string& host,
double level) override;
bool UsesTemporaryZoomLevel(const GlobalRenderFrameHostId& rfh_id) override;
void SetNoLongerUsesTemporaryZoomLevel(const GlobalRenderFrameHostId& rfh_id);
void SetTemporaryZoomLevel(const GlobalRenderFrameHostId& rfh_id,
double level) override;
void ClearZoomLevels(base::Time delete_begin, base::Time delete_end) override;
void ClearTemporaryZoomLevel(const GlobalRenderFrameHostId& rfh_id) override;
double GetDefaultZoomLevel() override;
void SetDefaultZoomLevel(double level) override;
base::CallbackListSubscription AddZoomLevelChangedCallback(
ZoomLevelChangedCallback callback) override;
// Returns the current zoom level for the specified WebContents. This may
// be a temporary zoom level, depending on UsesTemporaryZoomLevel().
double GetZoomLevelForWebContents(WebContentsImpl* web_contents_impl);
double GetZoomLevelForWebContents(WebContentsImpl* web_contents_impl,
GlobalRenderFrameHostId rfh_id);
// Sets the zoom level for this WebContents. If this WebContents is using
// a temporary zoom level, then level is only applied to this WebContents.
// Otherwise, the level will be applied on a host level.
void SetZoomLevelForWebContents(WebContentsImpl* web_contents_impl,
double level);
void SetZoomLevelForWebContents(WebContentsImpl* web_contents_impl,
GlobalRenderFrameHostId rfh_id,
double level);
// Returns the temporary zoom level that's only valid for the lifetime of
// the given RenderFrameHost identified by `rfh_id` (i.e. isn't saved and
// doesn't affect other RenderFrameHosts) if it exists, the default zoom
// level otherwise.
double GetTemporaryZoomLevel(const GlobalRenderFrameHostId& rfh_id) const;
void SendErrorPageZoomLevelRefresh();
void SetClockForTesting(base::Clock* clock) override;
#if BUILDFLAG(IS_ANDROID)
void SetDefaultZoomLevelPrefCallback(
HostZoomMap::DefaultZoomChangedCallback callback) override;
HostZoomMap::DefaultZoomChangedCallback* GetDefaultZoomLevelPrefCallback();
double GetZoomLevelForHostAndSchemeAndroid(const std::string& scheme,
const std::string& host) override;
void SetSystemFontScaleForTesting(float scale);
void SetShouldAdjustForOSLevelForTesting(bool shouldAdjustForOSLevel);
#endif
double GetZoomLevelForPreviewAndHost(const std::string& host) override;
void SetZoomLevelForPreviewAndHost(const std::string& host,
double level) override;
void SetIndependentZoomForFrameTreeNode(WebContents* web_contents,
FrameTreeNodeId ftn_id) override;
void ClearIndependentZoomForFrameTreeNode(FrameTreeNodeId ftn_id) override;
private:
struct ZoomLevel {
double level;
base::Time last_modified;
};
typedef std::map<std::string, ZoomLevel> HostZoomLevels;
typedef std::map<std::string, HostZoomLevels> SchemeHostZoomLevels;
typedef std::map<GlobalRenderFrameHostId, double> TemporaryZoomLevels;
typedef std::set<FrameTreeNodeId> IndependentZoomFrameTreeNodes;
double GetZoomLevelForHost(const std::string& host) const;
// Set a zoom level for |host| and store the |last_modified| timestamp.
// Use only to explicitly set a timestamp.
void SetZoomLevelForHostInternal(const std::string& host,
double level,
base::Time last_modified);
// Internal helper for SetDefaultZoomLevel().
void SetDefaultZoomLevelInternal(double level,
WebContentsImpl* web_contents,
RenderFrameHostImpl* rfh);
// Notifies the renderers from this browser context to change the zoom level
// for the specified host and scheme.
// |zoom level| will be extracted from |host_zoom_levels_| when needed, so no
// need to pass them in.
// TODO(wjmaclean) Should we use a GURL here? crbug.com/384486
void SendZoomLevelChange(const std::string& scheme, const std::string& host);
// Callbacks called when zoom level changes.
base::RepeatingCallbackList<void(const ZoomLevelChange&)>
zoom_level_changed_callbacks_;
#if BUILDFLAG(IS_ANDROID)
// Callback called when Java-side UI updates the default zoom level.
HostZoomMap::DefaultZoomChangedCallback default_zoom_level_pref_callback_;
#endif
// Copy of the pref data.
HostZoomLevels host_zoom_levels_;
SchemeHostZoomLevels scheme_host_zoom_levels_;
double default_zoom_level_;
TemporaryZoomLevels temporary_zoom_levels_;
// Used to track which FrameTreeNodes have independent zoom. A FrameTreeNode
// can have a zoom level that is independent from the main frame when it is
// displaying content in a GuestView (or possibly a PDF in a OOPIF without a
// GuestView), and features::kGuestViewMPArch is enabled. When this feature is
// not enabled it means that GuestViews will have their own WebContents, and
// so the use of a single zoom level for an entire WebContents suffices.
IndependentZoomFrameTreeNodes independent_zoom_frame_tree_nodes_;
HostZoomLevels host_zoom_levels_for_preview_;
raw_ptr<base::Clock> clock_;
};
} // namespace content
#endif // CONTENT_BROWSER_HOST_ZOOM_MAP_IMPL_H_