summaryrefslogtreecommitdiffstats
path: root/layout/base/ZoomConstraintsClient.cpp
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--layout/base/ZoomConstraintsClient.cpp285
1 files changed, 285 insertions, 0 deletions
diff --git a/layout/base/ZoomConstraintsClient.cpp b/layout/base/ZoomConstraintsClient.cpp
new file mode 100644
index 0000000000..183f3eb22e
--- /dev/null
+++ b/layout/base/ZoomConstraintsClient.cpp
@@ -0,0 +1,285 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "ZoomConstraintsClient.h"
+
+#include <inttypes.h>
+#include "mozilla/layers/APZCCallbackHelper.h"
+#include "mozilla/layers/ScrollableLayerGuid.h"
+#include "mozilla/layers/ZoomConstraints.h"
+#include "mozilla/Preferences.h"
+#include "mozilla/PresShell.h"
+#include "mozilla/StaticPrefs_apz.h"
+#include "mozilla/dom/Document.h"
+#include "mozilla/dom/Event.h"
+#include "nsIFrame.h"
+#include "nsIScrollableFrame.h"
+#include "nsLayoutUtils.h"
+#include "nsPoint.h"
+#include "nsView.h"
+#include "nsViewportInfo.h"
+#include "Units.h"
+#include "UnitTransforms.h"
+
+static mozilla::LazyLogModule sApzZoomLog("apz.zoom");
+#define ZCC_LOG(...) MOZ_LOG(sApzZoomLog, LogLevel::Debug, (__VA_ARGS__))
+
+NS_IMPL_ISUPPORTS(ZoomConstraintsClient, nsIDOMEventListener, nsIObserver)
+
+#define DOM_META_ADDED u"DOMMetaAdded"_ns
+#define DOM_META_CHANGED u"DOMMetaChanged"_ns
+#define FULLSCREEN_CHANGED u"fullscreenchange"_ns
+#define BEFORE_FIRST_PAINT "before-first-paint"_ns
+#define COMPOSITOR_REINITIALIZED "compositor-reinitialized"_ns
+#define NS_PREF_CHANGED "nsPref:changed"_ns
+
+using namespace mozilla;
+using namespace mozilla::dom;
+using namespace mozilla::layers;
+
+ZoomConstraintsClient::ZoomConstraintsClient()
+ : mDocument(nullptr),
+ mPresShell(nullptr),
+ mZoomConstraints(false, false, CSSToParentLayerScale(1.f),
+ CSSToParentLayerScale(1.f)) {}
+
+ZoomConstraintsClient::~ZoomConstraintsClient() = default;
+
+static nsIWidget* GetWidget(PresShell* aPresShell) {
+ if (!aPresShell) {
+ return nullptr;
+ }
+ if (nsIFrame* rootFrame = aPresShell->GetRootFrame()) {
+#if defined(MOZ_WIDGET_ANDROID) || defined(MOZ_WIDGET_UIKIT)
+ return rootFrame->GetNearestWidget();
+#else
+ if (nsView* view = rootFrame->GetView()) {
+ return view->GetWidget();
+ }
+#endif
+ }
+ return nullptr;
+}
+
+void ZoomConstraintsClient::Destroy() {
+ if (!(mPresShell && mDocument)) {
+ return;
+ }
+
+ ZCC_LOG("Destroying %p\n", this);
+
+ if (mEventTarget) {
+ mEventTarget->RemoveEventListener(DOM_META_ADDED, this, false);
+ mEventTarget->RemoveEventListener(DOM_META_CHANGED, this, false);
+ mEventTarget->RemoveSystemEventListener(FULLSCREEN_CHANGED, this, false);
+ mEventTarget = nullptr;
+ }
+
+ nsCOMPtr<nsIObserverService> observerService =
+ mozilla::services::GetObserverService();
+ if (observerService) {
+ observerService->RemoveObserver(this, BEFORE_FIRST_PAINT.Data());
+ observerService->RemoveObserver(this, COMPOSITOR_REINITIALIZED.Data());
+ }
+
+ Preferences::RemoveObserver(this, "browser.ui.zoom.force-user-scalable");
+
+ if (mGuid) {
+ if (nsIWidget* widget = GetWidget(mPresShell)) {
+ ZCC_LOG("Sending null constraints in %p for { %u, %" PRIu64 " }\n", this,
+ mGuid->mPresShellId, mGuid->mScrollId);
+ widget->UpdateZoomConstraints(mGuid->mPresShellId, mGuid->mScrollId,
+ Nothing());
+ mGuid = Nothing();
+ }
+ }
+
+ mDocument = nullptr;
+ mPresShell = nullptr;
+}
+
+void ZoomConstraintsClient::Init(PresShell* aPresShell, Document* aDocument) {
+ if (!(aPresShell && aDocument)) {
+ return;
+ }
+
+ mPresShell = aPresShell;
+ mDocument = aDocument;
+
+ if (nsCOMPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow()) {
+ mEventTarget = window->GetParentTarget();
+ }
+ if (mEventTarget) {
+ mEventTarget->AddEventListener(DOM_META_ADDED, this, false);
+ mEventTarget->AddEventListener(DOM_META_CHANGED, this, false);
+ mEventTarget->AddSystemEventListener(FULLSCREEN_CHANGED, this, false);
+ }
+
+ nsCOMPtr<nsIObserverService> observerService =
+ mozilla::services::GetObserverService();
+ if (observerService) {
+ observerService->AddObserver(this, BEFORE_FIRST_PAINT.Data(), false);
+ observerService->AddObserver(this, COMPOSITOR_REINITIALIZED.Data(), false);
+ }
+
+ Preferences::AddStrongObserver(this, "browser.ui.zoom.force-user-scalable");
+}
+
+NS_IMETHODIMP
+ZoomConstraintsClient::HandleEvent(dom::Event* event) {
+ nsAutoString type;
+ event->GetType(type);
+
+ if (type.Equals(DOM_META_ADDED)) {
+ ZCC_LOG("Got a dom-meta-added event in %p\n", this);
+ RefreshZoomConstraints();
+ } else if (type.Equals(DOM_META_CHANGED)) {
+ ZCC_LOG("Got a dom-meta-changed event in %p\n", this);
+ RefreshZoomConstraints();
+ } else if (type.Equals(FULLSCREEN_CHANGED)) {
+ ZCC_LOG("Got a fullscreen-change event in %p\n", this);
+ RefreshZoomConstraints();
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+ZoomConstraintsClient::Observe(nsISupports* aSubject, const char* aTopic,
+ const char16_t* aData) {
+ if (SameCOMIdentity(aSubject, ToSupports(mDocument)) &&
+ BEFORE_FIRST_PAINT.EqualsASCII(aTopic)) {
+ ZCC_LOG("Got a before-first-paint event in %p\n", this);
+ RefreshZoomConstraints();
+ } else if (COMPOSITOR_REINITIALIZED.EqualsASCII(aTopic)) {
+ ZCC_LOG("Got a compositor-reinitialized notification in %p\n", this);
+ RefreshZoomConstraints();
+ } else if (NS_PREF_CHANGED.EqualsASCII(aTopic)) {
+ ZCC_LOG("Got a pref-change event in %p\n", this);
+ // We need to run this later because all the pref change listeners need
+ // to execute before we can be guaranteed that
+ // StaticPrefs::browser_ui_zoom_force_user_scalable() returns the updated
+ // value.
+
+ RefPtr<nsRunnableMethod<ZoomConstraintsClient>> event =
+ NewRunnableMethod("ZoomConstraintsClient::RefreshZoomConstraints", this,
+ &ZoomConstraintsClient::RefreshZoomConstraints);
+ mDocument->Dispatch(TaskCategory::Other, event.forget());
+ }
+ return NS_OK;
+}
+
+void ZoomConstraintsClient::ScreenSizeChanged() {
+ ZCC_LOG("Got a screen-size change notification in %p\n", this);
+ RefreshZoomConstraints();
+}
+
+static mozilla::layers::ZoomConstraints ComputeZoomConstraintsFromViewportInfo(
+ const nsViewportInfo& aViewportInfo, Document* aDocument) {
+ mozilla::layers::ZoomConstraints constraints;
+ constraints.mAllowZoom = aViewportInfo.IsZoomAllowed() &&
+ nsLayoutUtils::AllowZoomingForDocument(aDocument);
+ constraints.mAllowDoubleTapZoom =
+ constraints.mAllowZoom && StaticPrefs::apz_allow_double_tap_zooming();
+ if (constraints.mAllowZoom) {
+ constraints.mMinZoom.scale = aViewportInfo.GetMinZoom().scale;
+ constraints.mMaxZoom.scale = aViewportInfo.GetMaxZoom().scale;
+ } else {
+ constraints.mMinZoom.scale = aViewportInfo.GetDefaultZoom().scale;
+ constraints.mMaxZoom.scale = aViewportInfo.GetDefaultZoom().scale;
+ }
+ return constraints;
+}
+
+void ZoomConstraintsClient::RefreshZoomConstraints() {
+ mZoomConstraints = ZoomConstraints(false, false, CSSToParentLayerScale(1.f),
+ CSSToParentLayerScale(1.f));
+
+ nsIWidget* widget = GetWidget(mPresShell);
+ if (!widget) {
+ return;
+ }
+
+ uint32_t presShellId = 0;
+ ScrollableLayerGuid::ViewID viewId = ScrollableLayerGuid::NULL_SCROLL_ID;
+ bool scrollIdentifiersValid =
+ APZCCallbackHelper::GetOrCreateScrollIdentifiers(
+ mDocument->GetDocumentElement(), &presShellId, &viewId);
+ if (!scrollIdentifiersValid) {
+ return;
+ }
+
+ LayoutDeviceIntSize screenSize;
+ if (!nsLayoutUtils::GetContentViewerSize(mPresShell->GetPresContext(),
+ screenSize)) {
+ return;
+ }
+
+ nsViewportInfo viewportInfo = mDocument->GetViewportInfo(ViewAs<ScreenPixel>(
+ screenSize, PixelCastJustification::LayoutDeviceIsScreenForBounds));
+
+ mZoomConstraints =
+ ComputeZoomConstraintsFromViewportInfo(viewportInfo, mDocument);
+
+ if (mDocument->Fullscreen()) {
+ ZCC_LOG("%p is in fullscreen, disallowing zooming\n", this);
+ mZoomConstraints.mAllowZoom = false;
+ mZoomConstraints.mAllowDoubleTapZoom = false;
+ }
+
+ if (mDocument->IsStaticDocument()) {
+ ZCC_LOG("%p is in print or print preview, disallowing double tap zooming\n",
+ this);
+ mZoomConstraints.mAllowDoubleTapZoom = false;
+ }
+
+ if (nsContentUtils::IsPDFJS(mDocument->GetPrincipal())) {
+ ZCC_LOG("%p is pdf.js viewer, disallowing double tap zooming\n", this);
+ mZoomConstraints.mAllowDoubleTapZoom = false;
+ }
+
+ // On macOS the OS can send us a double tap zoom event from the touchpad and
+ // there are no touch screen macOS devices so we never wait to see if a second
+ // tap is coming so we can always allow double tap zooming on mac. We need
+ // this because otherwise the width check usually disables it.
+ bool allow_double_tap_always = false;
+#ifdef XP_MACOSX
+ allow_double_tap_always =
+ StaticPrefs::apz_mac_enable_double_tap_zoom_touchpad_gesture();
+#endif
+ if (!allow_double_tap_always && mZoomConstraints.mAllowDoubleTapZoom) {
+ // If the CSS viewport is narrower than the screen (i.e. width <=
+ // device-width) then we disable double-tap-to-zoom behaviour.
+ CSSToLayoutDeviceScale scale =
+ mPresShell->GetPresContext()->CSSToDevPixelScale();
+ if ((viewportInfo.GetSize() * scale).width <= screenSize.width) {
+ mZoomConstraints.mAllowDoubleTapZoom = false;
+ }
+ }
+
+ // We only ever create a ZoomConstraintsClient for an RCD, so the RSF of
+ // the presShell must be the RCD-RSF (if it exists).
+ MOZ_ASSERT(mPresShell->GetPresContext()->IsRootContentDocumentCrossProcess());
+ if (nsIScrollableFrame* rcdrsf =
+ mPresShell->GetRootScrollFrameAsScrollable()) {
+ ZCC_LOG("Notifying RCD-RSF that it is zoomable: %d\n",
+ mZoomConstraints.mAllowZoom);
+ rcdrsf->SetZoomableByAPZ(mZoomConstraints.mAllowZoom);
+ }
+
+ ScrollableLayerGuid newGuid(LayersId{0}, presShellId, viewId);
+ if (mGuid && mGuid.value() != newGuid) {
+ ZCC_LOG("Clearing old constraints in %p for { %u, %" PRIu64 " }\n", this,
+ mGuid->mPresShellId, mGuid->mScrollId);
+ // If the guid changes, send a message to clear the old one
+ widget->UpdateZoomConstraints(mGuid->mPresShellId, mGuid->mScrollId,
+ Nothing());
+ }
+ mGuid = Some(newGuid);
+ ZCC_LOG("Sending constraints %s in %p for { %u, %" PRIu64 " }\n",
+ ToString(mZoomConstraints).c_str(), this, presShellId, viewId);
+ widget->UpdateZoomConstraints(presShellId, viewId, Some(mZoomConstraints));
+}