blob: 61f97e2c7cf4cb7adafd7ec5eae23a1673b01337 [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_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;
bool IsIndependentZoomFrameTreeNode(FrameTreeNodeId ftn_id) const;
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_