forked from mirrors/gecko-dev
		
	See https://bugzilla.mozilla.org/show_bug.cgi?id=1879770#c2 for an analysis. Differential Revision: https://phabricator.services.mozilla.com/D201893
		
			
				
	
	
		
			8025 lines
		
	
	
	
		
			312 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			8025 lines
		
	
	
	
		
			312 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* -*- 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/. */
 | 
						|
 | 
						|
/* rendering object to wrap rendering objects that should be scrollable */
 | 
						|
 | 
						|
#include "nsGfxScrollFrame.h"
 | 
						|
 | 
						|
#include "ScrollPositionUpdate.h"
 | 
						|
#include "mozilla/layers/LayersTypes.h"
 | 
						|
#include "nsIXULRuntime.h"
 | 
						|
#include "base/compiler_specific.h"
 | 
						|
#include "DisplayItemClip.h"
 | 
						|
#include "nsCOMPtr.h"
 | 
						|
#include "nsIDocumentViewer.h"
 | 
						|
#include "nsPresContext.h"
 | 
						|
#include "nsView.h"
 | 
						|
#include "nsViewportInfo.h"
 | 
						|
#include "nsContainerFrame.h"
 | 
						|
#include "nsGkAtoms.h"
 | 
						|
#include "nsNameSpaceManager.h"
 | 
						|
#include "mozilla/intl/BidiEmbeddingLevel.h"
 | 
						|
#include "mozilla/dom/DocumentInlines.h"
 | 
						|
#include "mozilla/gfx/gfxVars.h"
 | 
						|
#include "nsFontMetrics.h"
 | 
						|
#include "mozilla/dom/NodeInfo.h"
 | 
						|
#include "nsScrollbarFrame.h"
 | 
						|
#include "nsINode.h"
 | 
						|
#include "nsIScrollbarMediator.h"
 | 
						|
#include "nsITextControlFrame.h"
 | 
						|
#include "nsILayoutHistoryState.h"
 | 
						|
#include "nsNodeInfoManager.h"
 | 
						|
#include "nsContentCreatorFunctions.h"
 | 
						|
#include "nsStyleTransformMatrix.h"
 | 
						|
#include "mozilla/PresState.h"
 | 
						|
#include "nsContentUtils.h"
 | 
						|
#include "nsDisplayList.h"
 | 
						|
#include "nsHTMLDocument.h"
 | 
						|
#include "nsLayoutUtils.h"
 | 
						|
#include "nsBidiPresUtils.h"
 | 
						|
#include "nsBidiUtils.h"
 | 
						|
#include "nsDocShell.h"
 | 
						|
#include "mozilla/ContentEvents.h"
 | 
						|
#include "mozilla/DisplayPortUtils.h"
 | 
						|
#include "mozilla/EventDispatcher.h"
 | 
						|
#include "mozilla/Preferences.h"
 | 
						|
#include "mozilla/PresShell.h"
 | 
						|
#include "mozilla/ScopeExit.h"
 | 
						|
#include "mozilla/ScrollbarPreferences.h"
 | 
						|
#include "mozilla/ScrollingMetrics.h"
 | 
						|
#include "mozilla/StaticPrefs_bidi.h"
 | 
						|
#include "mozilla/StaticPrefs_browser.h"
 | 
						|
#include "mozilla/StaticPrefs_toolkit.h"
 | 
						|
#include "mozilla/StaticPtr.h"
 | 
						|
#include "mozilla/SVGOuterSVGFrame.h"
 | 
						|
#include "mozilla/ViewportUtils.h"
 | 
						|
#include "mozilla/LookAndFeel.h"
 | 
						|
#include "mozilla/dom/Element.h"
 | 
						|
#include "mozilla/dom/Event.h"
 | 
						|
#include "mozilla/dom/HTMLMarqueeElement.h"
 | 
						|
#include "mozilla/dom/ScrollTimeline.h"
 | 
						|
#include "mozilla/dom/BrowserChild.h"
 | 
						|
#include <stdint.h>
 | 
						|
#include "mozilla/MathAlgorithms.h"
 | 
						|
#include "mozilla/Telemetry.h"
 | 
						|
#include "nsSubDocumentFrame.h"
 | 
						|
#include "mozilla/Attributes.h"
 | 
						|
#include "ScrollbarActivity.h"
 | 
						|
#include "nsRefreshDriver.h"
 | 
						|
#include "nsStyleConsts.h"
 | 
						|
#include "nsIScrollPositionListener.h"
 | 
						|
#include "StickyScrollContainer.h"
 | 
						|
#include "nsIFrameInlines.h"
 | 
						|
#include "gfxPlatform.h"
 | 
						|
#include "mozilla/StaticPrefs_apz.h"
 | 
						|
#include "mozilla/StaticPrefs_general.h"
 | 
						|
#include "mozilla/StaticPrefs_layers.h"
 | 
						|
#include "mozilla/StaticPrefs_layout.h"
 | 
						|
#include "mozilla/StaticPrefs_mousewheel.h"
 | 
						|
#include "mozilla/ToString.h"
 | 
						|
#include "ScrollAnimationPhysics.h"
 | 
						|
#include "ScrollAnimationBezierPhysics.h"
 | 
						|
#include "ScrollAnimationMSDPhysics.h"
 | 
						|
#include "ScrollSnap.h"
 | 
						|
#include "UnitTransforms.h"
 | 
						|
#include "nsSliderFrame.h"
 | 
						|
#include "ViewportFrame.h"
 | 
						|
#include "mozilla/gfx/gfxVars.h"
 | 
						|
#include "mozilla/layers/APZCCallbackHelper.h"
 | 
						|
#include "mozilla/layers/APZPublicUtils.h"
 | 
						|
#include "mozilla/layers/AxisPhysicsModel.h"
 | 
						|
#include "mozilla/layers/AxisPhysicsMSDModel.h"
 | 
						|
#include "mozilla/layers/ScrollingInteractionContext.h"
 | 
						|
#include "mozilla/layers/ScrollLinkedEffectDetector.h"
 | 
						|
#include "mozilla/Unused.h"
 | 
						|
#include "MobileViewportManager.h"
 | 
						|
#include "VisualViewport.h"
 | 
						|
#include "WindowRenderer.h"
 | 
						|
#include <algorithm>
 | 
						|
#include <cstdlib>  // for std::abs(int/long)
 | 
						|
#include <cmath>    // for std::abs(float/double)
 | 
						|
#include <tuple>    // for std::tie
 | 
						|
 | 
						|
static mozilla::LazyLogModule sApzPaintSkipLog("apz.paintskip");
 | 
						|
#define PAINT_SKIP_LOG(...) \
 | 
						|
  MOZ_LOG(sApzPaintSkipLog, LogLevel::Debug, (__VA_ARGS__))
 | 
						|
static mozilla::LazyLogModule sScrollRestoreLog("scrollrestore");
 | 
						|
#define SCROLLRESTORE_LOG(...) \
 | 
						|
  MOZ_LOG(sScrollRestoreLog, LogLevel::Debug, (__VA_ARGS__))
 | 
						|
static mozilla::LazyLogModule sRootScrollbarsLog("rootscrollbars");
 | 
						|
#define ROOT_SCROLLBAR_LOG(...)                                  \
 | 
						|
  if (mIsRoot) {                                                 \
 | 
						|
    MOZ_LOG(sRootScrollbarsLog, LogLevel::Debug, (__VA_ARGS__)); \
 | 
						|
  }
 | 
						|
static mozilla::LazyLogModule sDisplayportLog("apz.displayport");
 | 
						|
 | 
						|
using namespace mozilla;
 | 
						|
using namespace mozilla::dom;
 | 
						|
using namespace mozilla::gfx;
 | 
						|
using namespace mozilla::layers;
 | 
						|
using namespace mozilla::layout;
 | 
						|
using nsStyleTransformMatrix::TransformReferenceBox;
 | 
						|
 | 
						|
static ScrollDirections GetOverflowChange(const nsRect& aCurScrolledRect,
 | 
						|
                                          const nsRect& aPrevScrolledRect) {
 | 
						|
  ScrollDirections result;
 | 
						|
  if (aPrevScrolledRect.x != aCurScrolledRect.x ||
 | 
						|
      aPrevScrolledRect.width != aCurScrolledRect.width) {
 | 
						|
    result += ScrollDirection::eHorizontal;
 | 
						|
  }
 | 
						|
  if (aPrevScrolledRect.y != aCurScrolledRect.y ||
 | 
						|
      aPrevScrolledRect.height != aCurScrolledRect.height) {
 | 
						|
    result += ScrollDirection::eVertical;
 | 
						|
  }
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * This class handles the dispatching of scroll events to content.
 | 
						|
 *
 | 
						|
 * Scroll events are posted to the refresh driver via
 | 
						|
 * nsRefreshDriver::PostScrollEvent(), and they are fired during a refresh
 | 
						|
 * driver tick, after running requestAnimationFrame callbacks but before
 | 
						|
 * the style flush. This allows rAF callbacks to perform scrolling and have
 | 
						|
 * that scrolling be reflected on the same refresh driver tick, while at
 | 
						|
 * the same time allowing scroll event listeners to make style changes and
 | 
						|
 * have those style changes be reflected on the same refresh driver tick.
 | 
						|
 *
 | 
						|
 * ScrollEvents cannot be refresh observers, because none of the existing
 | 
						|
 * categories of refresh observers (FlushType::Style, FlushType::Layout,
 | 
						|
 * and FlushType::Display) are run at the desired time in a refresh driver
 | 
						|
 * tick. They behave similarly to refresh observers in that their presence
 | 
						|
 * causes the refresh driver to tick.
 | 
						|
 *
 | 
						|
 * ScrollEvents are one-shot runnables; the refresh driver drops them after
 | 
						|
 * running them.
 | 
						|
 */
 | 
						|
class nsHTMLScrollFrame::ScrollEvent : public Runnable {
 | 
						|
 public:
 | 
						|
  NS_DECL_NSIRUNNABLE
 | 
						|
  explicit ScrollEvent(nsHTMLScrollFrame* aHelper, bool aDelayed);
 | 
						|
  void Revoke() { mHelper = nullptr; }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mHelper;
 | 
						|
};
 | 
						|
 | 
						|
class nsHTMLScrollFrame::ScrollEndEvent : public Runnable {
 | 
						|
 public:
 | 
						|
  NS_DECL_NSIRUNNABLE
 | 
						|
  explicit ScrollEndEvent(nsHTMLScrollFrame* aHelper);
 | 
						|
  void Revoke() { mHelper = nullptr; }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mHelper;
 | 
						|
};
 | 
						|
 | 
						|
class nsHTMLScrollFrame::AsyncScrollPortEvent : public Runnable {
 | 
						|
 public:
 | 
						|
  NS_DECL_NSIRUNNABLE
 | 
						|
  explicit AsyncScrollPortEvent(nsHTMLScrollFrame* helper)
 | 
						|
      : Runnable("nsHTMLScrollFrame::AsyncScrollPortEvent"), mHelper(helper) {}
 | 
						|
  void Revoke() { mHelper = nullptr; }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mHelper;
 | 
						|
};
 | 
						|
 | 
						|
class nsHTMLScrollFrame::ScrolledAreaEvent : public Runnable {
 | 
						|
 public:
 | 
						|
  NS_DECL_NSIRUNNABLE
 | 
						|
  explicit ScrolledAreaEvent(nsHTMLScrollFrame* helper)
 | 
						|
      : Runnable("nsHTMLScrollFrame::ScrolledAreaEvent"), mHelper(helper) {}
 | 
						|
  void Revoke() { mHelper = nullptr; }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mHelper;
 | 
						|
};
 | 
						|
 | 
						|
//----------------------------------------------------------------------
 | 
						|
 | 
						|
//----------nsHTMLScrollFrame-------------------------------------------
 | 
						|
 | 
						|
class ScrollFrameActivityTracker final
 | 
						|
    : public nsExpirationTracker<nsHTMLScrollFrame, 4> {
 | 
						|
 public:
 | 
						|
  // Wait for 3-4s between scrolls before we remove our layers.
 | 
						|
  // That's 4 generations of 1s each.
 | 
						|
  enum { TIMEOUT_MS = 1000 };
 | 
						|
  explicit ScrollFrameActivityTracker(nsIEventTarget* aEventTarget)
 | 
						|
      : nsExpirationTracker<nsHTMLScrollFrame, 4>(
 | 
						|
            TIMEOUT_MS, "ScrollFrameActivityTracker", aEventTarget) {}
 | 
						|
  ~ScrollFrameActivityTracker() { AgeAllGenerations(); }
 | 
						|
 | 
						|
  virtual void NotifyExpired(nsHTMLScrollFrame* aObject) override {
 | 
						|
    RemoveObject(aObject);
 | 
						|
    aObject->MarkNotRecentlyScrolled();
 | 
						|
  }
 | 
						|
};
 | 
						|
static StaticAutoPtr<ScrollFrameActivityTracker> gScrollFrameActivityTracker;
 | 
						|
 | 
						|
nsHTMLScrollFrame* NS_NewHTMLScrollFrame(PresShell* aPresShell,
 | 
						|
                                         ComputedStyle* aStyle, bool aIsRoot) {
 | 
						|
  return new (aPresShell)
 | 
						|
      nsHTMLScrollFrame(aStyle, aPresShell->GetPresContext(), aIsRoot);
 | 
						|
}
 | 
						|
 | 
						|
NS_IMPL_FRAMEARENA_HELPERS(nsHTMLScrollFrame)
 | 
						|
 | 
						|
nsHTMLScrollFrame::nsHTMLScrollFrame(ComputedStyle* aStyle,
 | 
						|
                                     nsPresContext* aPresContext,
 | 
						|
                                     nsIFrame::ClassID aID, bool aIsRoot)
 | 
						|
    : nsContainerFrame(aStyle, aPresContext, aID),
 | 
						|
      mHScrollbarBox(nullptr),
 | 
						|
      mVScrollbarBox(nullptr),
 | 
						|
      mScrolledFrame(nullptr),
 | 
						|
      mScrollCornerBox(nullptr),
 | 
						|
      mResizerBox(nullptr),
 | 
						|
      mReferenceFrameDuringPainting(nullptr),
 | 
						|
      mAsyncScroll(nullptr),
 | 
						|
      mAsyncSmoothMSDScroll(nullptr),
 | 
						|
      mLastScrollOrigin(ScrollOrigin::None),
 | 
						|
      mDestination(0, 0),
 | 
						|
      mRestorePos(-1, -1),
 | 
						|
      mLastPos(-1, -1),
 | 
						|
      mApzScrollPos(0, 0),
 | 
						|
      mLastUpdateFramesPos(-1, -1),
 | 
						|
      mScrollParentID(mozilla::layers::ScrollableLayerGuid::NULL_SCROLL_ID),
 | 
						|
      mAnchor(this),
 | 
						|
      mCurrentAPZScrollAnimationType(APZScrollAnimationType::No),
 | 
						|
      mIsFirstScrollableFrameSequenceNumber(Nothing()),
 | 
						|
      mInScrollingGesture(InScrollingGesture::No),
 | 
						|
      mAllowScrollOriginDowngrade(false),
 | 
						|
      mHadDisplayPortAtLastFrameUpdate(false),
 | 
						|
      mHasVerticalScrollbar(false),
 | 
						|
      mHasHorizontalScrollbar(false),
 | 
						|
      mOnlyNeedVScrollbarToScrollVVInsideLV(false),
 | 
						|
      mOnlyNeedHScrollbarToScrollVVInsideLV(false),
 | 
						|
      mFrameIsUpdatingScrollbar(false),
 | 
						|
      mDidHistoryRestore(false),
 | 
						|
      mIsRoot(aIsRoot),
 | 
						|
      mSuppressScrollbarUpdate(false),
 | 
						|
      mSkippedScrollbarLayout(false),
 | 
						|
      mHadNonInitialReflow(false),
 | 
						|
      mFirstReflow(true),
 | 
						|
      mHorizontalOverflow(false),
 | 
						|
      mVerticalOverflow(false),
 | 
						|
      mPostedReflowCallback(false),
 | 
						|
      mMayHaveDirtyFixedChildren(false),
 | 
						|
      mUpdateScrollbarAttributes(false),
 | 
						|
      mHasBeenScrolledRecently(false),
 | 
						|
      mWillBuildScrollableLayer(false),
 | 
						|
      mIsParentToActiveScrollFrames(false),
 | 
						|
      mHasBeenScrolled(false),
 | 
						|
      mIgnoreMomentumScroll(false),
 | 
						|
      mTransformingByAPZ(false),
 | 
						|
      mScrollableByAPZ(false),
 | 
						|
      mZoomableByAPZ(false),
 | 
						|
      mHasOutOfFlowContentInsideFilter(false),
 | 
						|
      mSuppressScrollbarRepaints(false),
 | 
						|
      mIsUsingMinimumScaleSize(false),
 | 
						|
      mMinimumScaleSizeChanged(false),
 | 
						|
      mProcessingScrollEvent(false),
 | 
						|
      mApzAnimationRequested(false),
 | 
						|
      mApzAnimationTriggeredByScriptRequested(false),
 | 
						|
      mReclampVVOffsetInReflowFinished(false),
 | 
						|
      mMayScheduleScrollAnimations(false),
 | 
						|
#ifdef MOZ_WIDGET_ANDROID
 | 
						|
      mHasVerticalOverflowForDynamicToolbar(false),
 | 
						|
#endif
 | 
						|
      mVelocityQueue(PresContext()) {
 | 
						|
  AppendScrollUpdate(ScrollPositionUpdate::NewScrollframe(nsPoint()));
 | 
						|
 | 
						|
  if (UsesOverlayScrollbars()) {
 | 
						|
    mScrollbarActivity = new ScrollbarActivity(this);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    mZoomableByAPZ = PresShell()->GetZoomableByAPZ();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsHTMLScrollFrame::~nsHTMLScrollFrame() = default;
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollbarActivityStarted() const {
 | 
						|
  if (mScrollbarActivity) {
 | 
						|
    mScrollbarActivity->ActivityStarted();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollbarActivityStopped() const {
 | 
						|
  if (mScrollbarActivity) {
 | 
						|
    mScrollbarActivity->ActivityStopped();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::Destroy(DestroyContext& aContext) {
 | 
						|
  DestroyAbsoluteFrames(aContext);
 | 
						|
  if (mIsRoot) {
 | 
						|
    PresShell()->ResetVisualViewportOffset();
 | 
						|
  }
 | 
						|
 | 
						|
  mAnchor.Destroy();
 | 
						|
 | 
						|
  if (mScrollbarActivity) {
 | 
						|
    mScrollbarActivity->Destroy();
 | 
						|
    mScrollbarActivity = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  // Unbind the content created in CreateAnonymousContent later...
 | 
						|
  aContext.AddAnonymousContent(mHScrollbarContent.forget());
 | 
						|
  aContext.AddAnonymousContent(mVScrollbarContent.forget());
 | 
						|
  aContext.AddAnonymousContent(mScrollCornerContent.forget());
 | 
						|
  aContext.AddAnonymousContent(mResizerContent.forget());
 | 
						|
 | 
						|
  if (mPostedReflowCallback) {
 | 
						|
    PresShell()->CancelReflowCallback(this);
 | 
						|
    mPostedReflowCallback = false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mDisplayPortExpiryTimer) {
 | 
						|
    mDisplayPortExpiryTimer->Cancel();
 | 
						|
    mDisplayPortExpiryTimer = nullptr;
 | 
						|
  }
 | 
						|
  if (mActivityExpirationState.IsTracked()) {
 | 
						|
    gScrollFrameActivityTracker->RemoveObject(this);
 | 
						|
  }
 | 
						|
  if (gScrollFrameActivityTracker && gScrollFrameActivityTracker->IsEmpty()) {
 | 
						|
    gScrollFrameActivityTracker = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mScrollActivityTimer) {
 | 
						|
    mScrollActivityTimer->Cancel();
 | 
						|
    mScrollActivityTimer = nullptr;
 | 
						|
  }
 | 
						|
  RemoveObservers();
 | 
						|
  if (mScrollEvent) {
 | 
						|
    mScrollEvent->Revoke();
 | 
						|
  }
 | 
						|
  if (mScrollEndEvent) {
 | 
						|
    mScrollEndEvent->Revoke();
 | 
						|
  }
 | 
						|
  nsContainerFrame::Destroy(aContext);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetInitialChildList(ChildListID aListID,
 | 
						|
                                            nsFrameList&& aChildList) {
 | 
						|
  nsContainerFrame::SetInitialChildList(aListID, std::move(aChildList));
 | 
						|
  ReloadChildFrames();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AppendFrames(ChildListID aListID,
 | 
						|
                                     nsFrameList&& aFrameList) {
 | 
						|
  NS_ASSERTION(aListID == FrameChildListID::Principal,
 | 
						|
               "Only main list supported");
 | 
						|
  mFrames.AppendFrames(nullptr, std::move(aFrameList));
 | 
						|
  ReloadChildFrames();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::InsertFrames(ChildListID aListID, nsIFrame* aPrevFrame,
 | 
						|
                                     const nsLineList::iterator* aPrevFrameLine,
 | 
						|
                                     nsFrameList&& aFrameList) {
 | 
						|
  NS_ASSERTION(aListID == FrameChildListID::Principal,
 | 
						|
               "Only main list supported");
 | 
						|
  NS_ASSERTION(!aPrevFrame || aPrevFrame->GetParent() == this,
 | 
						|
               "inserting after sibling frame with different parent");
 | 
						|
  mFrames.InsertFrames(nullptr, aPrevFrame, std::move(aFrameList));
 | 
						|
  ReloadChildFrames();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::RemoveFrame(DestroyContext& aContext,
 | 
						|
                                    ChildListID aListID, nsIFrame* aOldFrame) {
 | 
						|
  NS_ASSERTION(aListID == FrameChildListID::Principal,
 | 
						|
               "Only main list supported");
 | 
						|
  mFrames.DestroyFrame(aContext, aOldFrame);
 | 
						|
  ReloadChildFrames();
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 HTML scrolling implementation
 | 
						|
 | 
						|
 All other things being equal, we prefer layouts with fewer scrollbars showing.
 | 
						|
*/
 | 
						|
 | 
						|
namespace mozilla {
 | 
						|
 | 
						|
enum class ShowScrollbar : uint8_t {
 | 
						|
  Auto,
 | 
						|
  Always,
 | 
						|
  // Never is a misnomer. We can still get a scrollbar if we need to scroll the
 | 
						|
  // visual viewport inside the layout viewport. Thus this enum is best thought
 | 
						|
  // of as value used by layout, which does not know about the visual viewport.
 | 
						|
  // The visual viewport does not affect any layout sizes, so this is sound.
 | 
						|
  Never,
 | 
						|
};
 | 
						|
 | 
						|
static ShowScrollbar ShouldShowScrollbar(StyleOverflow aOverflow) {
 | 
						|
  switch (aOverflow) {
 | 
						|
    case StyleOverflow::Scroll:
 | 
						|
      return ShowScrollbar::Always;
 | 
						|
    case StyleOverflow::Hidden:
 | 
						|
      return ShowScrollbar::Never;
 | 
						|
    default:
 | 
						|
    case StyleOverflow::Auto:
 | 
						|
      return ShowScrollbar::Auto;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
struct MOZ_STACK_CLASS ScrollReflowInput {
 | 
						|
  // === Filled in by the constructor. Members in this section shouldn't change
 | 
						|
  // their values after the constructor. ===
 | 
						|
  const ReflowInput& mReflowInput;
 | 
						|
  ShowScrollbar mHScrollbar;
 | 
						|
  // If the horizontal scrollbar is allowed (even if mHScrollbar ==
 | 
						|
  // ShowScrollbar::Never) provided that it is for scrolling the visual viewport
 | 
						|
  // inside the layout viewport only.
 | 
						|
  bool mHScrollbarAllowedForScrollingVVInsideLV = true;
 | 
						|
  ShowScrollbar mVScrollbar;
 | 
						|
  // If the vertical scrollbar is allowed (even if mVScrollbar ==
 | 
						|
  // ShowScrollbar::Never) provided that it is for scrolling the visual viewport
 | 
						|
  // inside the layout viewport only.
 | 
						|
  bool mVScrollbarAllowedForScrollingVVInsideLV = true;
 | 
						|
  nsMargin mComputedBorder;
 | 
						|
 | 
						|
  // === Filled in by ReflowScrolledFrame ===
 | 
						|
  OverflowAreas mContentsOverflowAreas;
 | 
						|
  // The scrollbar gutter sizes used in the most recent reflow of
 | 
						|
  // mScrolledFrame. The writing-mode is the same as the scroll
 | 
						|
  // container.
 | 
						|
  LogicalMargin mScrollbarGutterFromLastReflow;
 | 
						|
  // True if the most recent reflow of mScrolledFrame is with the
 | 
						|
  // horizontal scrollbar.
 | 
						|
  bool mReflowedContentsWithHScrollbar = false;
 | 
						|
  // True if the most recent reflow of mScrolledFrame is with the
 | 
						|
  // vertical scrollbar.
 | 
						|
  bool mReflowedContentsWithVScrollbar = false;
 | 
						|
 | 
						|
  // === Filled in when TryLayout succeeds ===
 | 
						|
  // The size of the inside-border area
 | 
						|
  nsSize mInsideBorderSize;
 | 
						|
  // Whether we decided to show the horizontal scrollbar in the most recent
 | 
						|
  // TryLayout.
 | 
						|
  bool mShowHScrollbar = false;
 | 
						|
  // Whether we decided to show the vertical scrollbar in the most recent
 | 
						|
  // TryLayout.
 | 
						|
  bool mShowVScrollbar = false;
 | 
						|
  // If mShow(H|V)Scrollbar is true then
 | 
						|
  // mOnlyNeed(V|H)ScrollbarToScrollVVInsideLV indicates if the only reason we
 | 
						|
  // need that scrollbar is to scroll the visual viewport inside the layout
 | 
						|
  // viewport. These scrollbars are special in that even if they are layout
 | 
						|
  // scrollbars they do not take up any layout space.
 | 
						|
  bool mOnlyNeedHScrollbarToScrollVVInsideLV = false;
 | 
						|
  bool mOnlyNeedVScrollbarToScrollVVInsideLV = false;
 | 
						|
 | 
						|
  ScrollReflowInput(nsHTMLScrollFrame* aFrame, const ReflowInput& aReflowInput);
 | 
						|
 | 
						|
  nscoord VScrollbarMinHeight() const { return mVScrollbarPrefSize.height; }
 | 
						|
  nscoord VScrollbarPrefWidth() const { return mVScrollbarPrefSize.width; }
 | 
						|
  nscoord HScrollbarMinWidth() const { return mHScrollbarPrefSize.width; }
 | 
						|
  nscoord HScrollbarPrefHeight() const { return mHScrollbarPrefSize.height; }
 | 
						|
 | 
						|
  // Returns the sizes occupied by the scrollbar gutters. If aShowVScroll or
 | 
						|
  // aShowHScroll is true, the sizes occupied by the scrollbars are also
 | 
						|
  // included.
 | 
						|
  nsMargin ScrollbarGutter(bool aShowVScrollbar, bool aShowHScrollbar,
 | 
						|
                           bool aScrollbarOnRight) const {
 | 
						|
    if (mOverlayScrollbars) {
 | 
						|
      return mScrollbarGutter;
 | 
						|
    }
 | 
						|
    nsMargin gutter = mScrollbarGutter;
 | 
						|
    if (aShowVScrollbar && gutter.right == 0 && gutter.left == 0) {
 | 
						|
      const nscoord w = VScrollbarPrefWidth();
 | 
						|
      if (aScrollbarOnRight) {
 | 
						|
        gutter.right = w;
 | 
						|
      } else {
 | 
						|
        gutter.left = w;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (aShowHScrollbar && gutter.bottom == 0) {
 | 
						|
      // The horizontal scrollbar is always at the bottom side.
 | 
						|
      gutter.bottom = HScrollbarPrefHeight();
 | 
						|
    }
 | 
						|
    return gutter;
 | 
						|
  }
 | 
						|
 | 
						|
  bool OverlayScrollbars() const { return mOverlayScrollbars; }
 | 
						|
 | 
						|
 private:
 | 
						|
  // Filled in by the constructor. Put variables here to keep them unchanged
 | 
						|
  // after initializing them in the constructor.
 | 
						|
  nsSize mVScrollbarPrefSize;
 | 
						|
  nsSize mHScrollbarPrefSize;
 | 
						|
  bool mOverlayScrollbars = false;
 | 
						|
  // The scrollbar gutter sizes resolved from the scrollbar-gutter and
 | 
						|
  // scrollbar-width property.
 | 
						|
  nsMargin mScrollbarGutter;
 | 
						|
};
 | 
						|
 | 
						|
ScrollReflowInput::ScrollReflowInput(nsHTMLScrollFrame* aFrame,
 | 
						|
                                     const ReflowInput& aReflowInput)
 | 
						|
    : mReflowInput(aReflowInput),
 | 
						|
      mComputedBorder(aReflowInput.ComputedPhysicalBorderPadding() -
 | 
						|
                      aReflowInput.ComputedPhysicalPadding()),
 | 
						|
      mScrollbarGutterFromLastReflow(aFrame->GetWritingMode()) {
 | 
						|
  ScrollStyles styles = aFrame->GetScrollStyles();
 | 
						|
  mHScrollbar = ShouldShowScrollbar(styles.mHorizontal);
 | 
						|
  mVScrollbar = ShouldShowScrollbar(styles.mVertical);
 | 
						|
  mOverlayScrollbars = aFrame->UsesOverlayScrollbars();
 | 
						|
 | 
						|
  if (nsScrollbarFrame* scrollbar = aFrame->GetScrollbarBox(false)) {
 | 
						|
    scrollbar->SetScrollbarMediatorContent(mReflowInput.mFrame->GetContent());
 | 
						|
    mHScrollbarPrefSize = scrollbar->ScrollbarMinSize();
 | 
						|
    // A zero minimum size is a bug with non-overlay scrollbars. That means
 | 
						|
    // we'll always try to place the scrollbar, even if it will ultimately not
 | 
						|
    // fit, see bug 1809630. XUL collapsing is the exception because the
 | 
						|
    // front-end uses it.
 | 
						|
    MOZ_ASSERT(mHScrollbarPrefSize.width && mHScrollbarPrefSize.height,
 | 
						|
               "Shouldn't have a zero horizontal scrollbar-size");
 | 
						|
  } else {
 | 
						|
    mHScrollbar = ShowScrollbar::Never;
 | 
						|
    mHScrollbarAllowedForScrollingVVInsideLV = false;
 | 
						|
  }
 | 
						|
  if (nsScrollbarFrame* scrollbar = aFrame->GetScrollbarBox(true)) {
 | 
						|
    scrollbar->SetScrollbarMediatorContent(mReflowInput.mFrame->GetContent());
 | 
						|
    mVScrollbarPrefSize = scrollbar->ScrollbarMinSize();
 | 
						|
    // See above.
 | 
						|
    MOZ_ASSERT(mVScrollbarPrefSize.width && mVScrollbarPrefSize.height,
 | 
						|
               "Shouldn't have a zero vertical scrollbar-size");
 | 
						|
  } else {
 | 
						|
    mVScrollbar = ShowScrollbar::Never;
 | 
						|
    mVScrollbarAllowedForScrollingVVInsideLV = false;
 | 
						|
  }
 | 
						|
 | 
						|
  const auto* scrollbarStyle =
 | 
						|
      nsLayoutUtils::StyleForScrollbar(mReflowInput.mFrame);
 | 
						|
  // Hide the scrollbar when the scrollbar-width is set to none.
 | 
						|
  //
 | 
						|
  // Note: In some cases this is unnecessary, because scrollbar-width:none
 | 
						|
  // makes us suppress scrollbars in CreateAnonymousContent. But if this frame
 | 
						|
  // initially had a non-'none' scrollbar-width and dynamically changed to
 | 
						|
  // 'none', then we'll need to handle it here.
 | 
						|
  const auto scrollbarWidth = scrollbarStyle->StyleUIReset()->ScrollbarWidth();
 | 
						|
  if (scrollbarWidth == StyleScrollbarWidth::None) {
 | 
						|
    mHScrollbar = ShowScrollbar::Never;
 | 
						|
    mHScrollbarAllowedForScrollingVVInsideLV = false;
 | 
						|
    mVScrollbar = ShowScrollbar::Never;
 | 
						|
    mVScrollbarAllowedForScrollingVVInsideLV = false;
 | 
						|
  }
 | 
						|
 | 
						|
  mScrollbarGutter = aFrame->ComputeStableScrollbarGutter(
 | 
						|
      scrollbarWidth, scrollbarStyle->StyleDisplay()->mScrollbarGutter);
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace mozilla
 | 
						|
 | 
						|
static nsSize ComputeInsideBorderSize(const ScrollReflowInput& aState,
 | 
						|
                                      const nsSize& aDesiredInsideBorderSize) {
 | 
						|
  // aDesiredInsideBorderSize is the frame size; i.e., it includes
 | 
						|
  // borders and padding (but the scrolled child doesn't have
 | 
						|
  // borders). The scrolled child has the same padding as us.
 | 
						|
  const WritingMode wm = aState.mReflowInput.GetWritingMode();
 | 
						|
  const LogicalSize desiredInsideBorderSize(wm, aDesiredInsideBorderSize);
 | 
						|
  LogicalSize contentSize = aState.mReflowInput.ComputedSize();
 | 
						|
  const LogicalMargin padding = aState.mReflowInput.ComputedLogicalPadding(wm);
 | 
						|
 | 
						|
  if (contentSize.ISize(wm) == NS_UNCONSTRAINEDSIZE) {
 | 
						|
    contentSize.ISize(wm) =
 | 
						|
        desiredInsideBorderSize.ISize(wm) - padding.IStartEnd(wm);
 | 
						|
  }
 | 
						|
  if (contentSize.BSize(wm) == NS_UNCONSTRAINEDSIZE) {
 | 
						|
    contentSize.BSize(wm) =
 | 
						|
        desiredInsideBorderSize.BSize(wm) - padding.BStartEnd(wm);
 | 
						|
  }
 | 
						|
 | 
						|
  contentSize.ISize(wm) =
 | 
						|
      aState.mReflowInput.ApplyMinMaxISize(contentSize.ISize(wm));
 | 
						|
  contentSize.BSize(wm) =
 | 
						|
      aState.mReflowInput.ApplyMinMaxBSize(contentSize.BSize(wm));
 | 
						|
 | 
						|
  return (contentSize + padding.Size(wm)).GetPhysicalSize(wm);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Assuming that we know the metrics for our wrapped frame and
 | 
						|
 * whether the horizontal and/or vertical scrollbars are present,
 | 
						|
 * compute the resulting layout and return true if the layout is
 | 
						|
 * consistent. If the layout is consistent then we fill in the
 | 
						|
 * computed fields of the ScrollReflowInput.
 | 
						|
 *
 | 
						|
 * The layout is consistent when both scrollbars are showing if and only
 | 
						|
 * if they should be showing. A horizontal scrollbar should be showing if all
 | 
						|
 * following conditions are met:
 | 
						|
 * 1) the style is not HIDDEN
 | 
						|
 * 2) our inside-border height is at least the scrollbar height (i.e., the
 | 
						|
 * scrollbar fits vertically)
 | 
						|
 * 3) the style is SCROLL, or the kid's overflow-area XMost is
 | 
						|
 * greater than the scrollport width
 | 
						|
 *
 | 
						|
 * @param aForce if true, then we just assume the layout is consistent.
 | 
						|
 */
 | 
						|
bool nsHTMLScrollFrame::TryLayout(ScrollReflowInput& aState,
 | 
						|
                                  ReflowOutput* aKidMetrics,
 | 
						|
                                  bool aAssumeHScroll, bool aAssumeVScroll,
 | 
						|
                                  bool aForce) {
 | 
						|
  if ((aState.mVScrollbar == ShowScrollbar::Never && aAssumeVScroll) ||
 | 
						|
      (aState.mHScrollbar == ShowScrollbar::Never && aAssumeHScroll)) {
 | 
						|
    NS_ASSERTION(!aForce, "Shouldn't be forcing a hidden scrollbar to show!");
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  const auto wm = GetWritingMode();
 | 
						|
  const nsMargin scrollbarGutter = aState.ScrollbarGutter(
 | 
						|
      aAssumeVScroll, aAssumeHScroll, IsScrollbarOnRight());
 | 
						|
  const LogicalMargin logicalScrollbarGutter(wm, scrollbarGutter);
 | 
						|
 | 
						|
  const bool inlineEndsGutterChanged =
 | 
						|
      aState.mScrollbarGutterFromLastReflow.IStartEnd(wm) !=
 | 
						|
      logicalScrollbarGutter.IStartEnd(wm);
 | 
						|
  const bool blockEndsGutterChanged =
 | 
						|
      aState.mScrollbarGutterFromLastReflow.BStartEnd(wm) !=
 | 
						|
      logicalScrollbarGutter.BStartEnd(wm);
 | 
						|
  const bool shouldReflowScrolledFrame =
 | 
						|
      inlineEndsGutterChanged ||
 | 
						|
      (blockEndsGutterChanged && ScrolledContentDependsOnBSize(aState));
 | 
						|
 | 
						|
  if (shouldReflowScrolledFrame) {
 | 
						|
    if (blockEndsGutterChanged) {
 | 
						|
      nsLayoutUtils::MarkIntrinsicISizesDirtyIfDependentOnBSize(mScrolledFrame);
 | 
						|
    }
 | 
						|
    aKidMetrics->mOverflowAreas.Clear();
 | 
						|
    ROOT_SCROLLBAR_LOG(
 | 
						|
        "TryLayout reflowing scrolled frame with scrollbars h=%d, v=%d\n",
 | 
						|
        aAssumeHScroll, aAssumeVScroll);
 | 
						|
    ReflowScrolledFrame(aState, aAssumeHScroll, aAssumeVScroll, aKidMetrics);
 | 
						|
  }
 | 
						|
 | 
						|
  const nsSize scrollbarGutterSize(scrollbarGutter.LeftRight(),
 | 
						|
                                   scrollbarGutter.TopBottom());
 | 
						|
 | 
						|
  // First, compute our inside-border size and scrollport size
 | 
						|
  nsSize kidSize = GetContainSizeAxes().ContainSize(
 | 
						|
      aKidMetrics->PhysicalSize(), *aState.mReflowInput.mFrame);
 | 
						|
  const nsSize desiredInsideBorderSize = kidSize + scrollbarGutterSize;
 | 
						|
  aState.mInsideBorderSize =
 | 
						|
      ComputeInsideBorderSize(aState, desiredInsideBorderSize);
 | 
						|
 | 
						|
  nsSize layoutSize =
 | 
						|
      mIsUsingMinimumScaleSize ? mMinimumScaleSize : aState.mInsideBorderSize;
 | 
						|
 | 
						|
  const nsSize scrollPortSize =
 | 
						|
      Max(nsSize(0, 0), layoutSize - scrollbarGutterSize);
 | 
						|
  if (mIsUsingMinimumScaleSize) {
 | 
						|
    mICBSize =
 | 
						|
        Max(nsSize(0, 0), aState.mInsideBorderSize - scrollbarGutterSize);
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize visualViewportSize = scrollPortSize;
 | 
						|
  ROOT_SCROLLBAR_LOG("TryLayout with VV %s\n",
 | 
						|
                     ToString(visualViewportSize).c_str());
 | 
						|
  mozilla::PresShell* presShell = PresShell();
 | 
						|
  // Note: we check for a non-null MobileViepwortManager here, but ideally we
 | 
						|
  // should be able to drop that clause as well. It's just that in some cases
 | 
						|
  // with extension popups the composition size comes back as stale, because
 | 
						|
  // the content viewer is only resized after the popup contents are reflowed.
 | 
						|
  // That case also happens to have no APZ and no MVM, so we use that as a
 | 
						|
  // way to detect the scenario. Bug 1648669 tracks removing this clause.
 | 
						|
  if (mIsRoot && presShell->GetMobileViewportManager()) {
 | 
						|
    visualViewportSize = nsLayoutUtils::CalculateCompositionSizeForFrame(
 | 
						|
        this, false, &layoutSize);
 | 
						|
    visualViewportSize =
 | 
						|
        Max(nsSize(0, 0), visualViewportSize - scrollbarGutterSize);
 | 
						|
 | 
						|
    float resolution = presShell->GetResolution();
 | 
						|
    visualViewportSize.width /= resolution;
 | 
						|
    visualViewportSize.height /= resolution;
 | 
						|
    ROOT_SCROLLBAR_LOG("TryLayout now with VV %s\n",
 | 
						|
                       ToString(visualViewportSize).c_str());
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect overflowRect = aState.mContentsOverflowAreas.ScrollableOverflow();
 | 
						|
  // If the content height expanded by the minimum-scale will be taller than
 | 
						|
  // the scrollable overflow area, we need to expand the area here to tell
 | 
						|
  // properly whether we need to render the overlay vertical scrollbar.
 | 
						|
  // NOTE: This expanded size should NOT be used for non-overley scrollbars
 | 
						|
  // cases since putting the vertical non-overlay scrollbar will make the
 | 
						|
  // content width narrow a little bit, which in turn the minimum scale value
 | 
						|
  // becomes a bit bigger than before, then the vertical scrollbar is no longer
 | 
						|
  // needed, which means the content width becomes the original width, then the
 | 
						|
  // minimum-scale is changed to the original one, and so forth.
 | 
						|
  if (UsesOverlayScrollbars() && mIsUsingMinimumScaleSize &&
 | 
						|
      mMinimumScaleSize.height > overflowRect.YMost()) {
 | 
						|
    overflowRect.height += mMinimumScaleSize.height - overflowRect.YMost();
 | 
						|
  }
 | 
						|
  nsRect scrolledRect =
 | 
						|
      GetUnsnappedScrolledRectInternal(overflowRect, scrollPortSize);
 | 
						|
  ROOT_SCROLLBAR_LOG(
 | 
						|
      "TryLayout scrolledRect:%s overflowRect:%s scrollportSize:%s\n",
 | 
						|
      ToString(scrolledRect).c_str(), ToString(overflowRect).c_str(),
 | 
						|
      ToString(scrollPortSize).c_str());
 | 
						|
  nscoord oneDevPixel = PresContext()->DevPixelsToAppUnits(1);
 | 
						|
 | 
						|
  bool showHScrollbar = aAssumeHScroll;
 | 
						|
  bool showVScrollbar = aAssumeVScroll;
 | 
						|
  if (!aForce) {
 | 
						|
    nsSize sizeToCompare = visualViewportSize;
 | 
						|
    if (gfxPlatform::UseDesktopZoomingScrollbars()) {
 | 
						|
      sizeToCompare = scrollPortSize;
 | 
						|
    }
 | 
						|
 | 
						|
    // No need to compute showHScrollbar if we got ShowScrollbar::Never.
 | 
						|
    if (aState.mHScrollbar != ShowScrollbar::Never) {
 | 
						|
      showHScrollbar =
 | 
						|
          aState.mHScrollbar == ShowScrollbar::Always ||
 | 
						|
          scrolledRect.XMost() >= sizeToCompare.width + oneDevPixel ||
 | 
						|
          scrolledRect.x <= -oneDevPixel;
 | 
						|
      // TODO(emilio): This should probably check this scrollbar's minimum size
 | 
						|
      // in both axes, for consistency?
 | 
						|
      if (aState.mHScrollbar == ShowScrollbar::Auto &&
 | 
						|
          scrollPortSize.width < aState.HScrollbarMinWidth()) {
 | 
						|
        showHScrollbar = false;
 | 
						|
      }
 | 
						|
      ROOT_SCROLLBAR_LOG("TryLayout wants H Scrollbar: %d =? %d\n",
 | 
						|
                         showHScrollbar, aAssumeHScroll);
 | 
						|
    }
 | 
						|
 | 
						|
    // No need to compute showVScrollbar if we got ShowScrollbar::Never.
 | 
						|
    if (aState.mVScrollbar != ShowScrollbar::Never) {
 | 
						|
      showVScrollbar =
 | 
						|
          aState.mVScrollbar == ShowScrollbar::Always ||
 | 
						|
          scrolledRect.YMost() >= sizeToCompare.height + oneDevPixel ||
 | 
						|
          scrolledRect.y <= -oneDevPixel;
 | 
						|
      // TODO(emilio): This should probably check this scrollbar's minimum size
 | 
						|
      // in both axes, for consistency?
 | 
						|
      if (aState.mVScrollbar == ShowScrollbar::Auto &&
 | 
						|
          scrollPortSize.height < aState.VScrollbarMinHeight()) {
 | 
						|
        showVScrollbar = false;
 | 
						|
      }
 | 
						|
      ROOT_SCROLLBAR_LOG("TryLayout wants V Scrollbar: %d =? %d\n",
 | 
						|
                         showVScrollbar, aAssumeVScroll);
 | 
						|
    }
 | 
						|
 | 
						|
    if (showHScrollbar != aAssumeHScroll || showVScrollbar != aAssumeVScroll) {
 | 
						|
      const nsMargin wantedScrollbarGutter = aState.ScrollbarGutter(
 | 
						|
          showVScrollbar, showHScrollbar, IsScrollbarOnRight());
 | 
						|
      // We report an inconsistent layout only when the desired visibility of
 | 
						|
      // the scrollbars can change the size of the scrollbar gutters.
 | 
						|
      if (scrollbarGutter != wantedScrollbarGutter) {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // If we reach here, the layout is consistent. Record the desired visibility
 | 
						|
  // of the scrollbars.
 | 
						|
  aState.mShowHScrollbar = showHScrollbar;
 | 
						|
  aState.mShowVScrollbar = showVScrollbar;
 | 
						|
  const nsPoint scrollPortOrigin(
 | 
						|
      aState.mComputedBorder.left + scrollbarGutter.left,
 | 
						|
      aState.mComputedBorder.top + scrollbarGutter.top);
 | 
						|
  SetScrollPort(nsRect(scrollPortOrigin, scrollPortSize));
 | 
						|
 | 
						|
  if (mIsRoot && gfxPlatform::UseDesktopZoomingScrollbars()) {
 | 
						|
    bool vvChanged = true;
 | 
						|
    const bool overlay = aState.OverlayScrollbars();
 | 
						|
    // This loop can run at most twice since we can only add a scrollbar once.
 | 
						|
    // At this point we've already decided that this layout is consistent so we
 | 
						|
    // will return true. Scrollbars added here never take up layout space even
 | 
						|
    // if they are layout scrollbars so any changes made here will not make us
 | 
						|
    // return false.
 | 
						|
    while (vvChanged) {
 | 
						|
      vvChanged = false;
 | 
						|
      if (!aState.mShowHScrollbar &&
 | 
						|
          aState.mHScrollbarAllowedForScrollingVVInsideLV) {
 | 
						|
        if (ScrollPort().width >= visualViewportSize.width + oneDevPixel &&
 | 
						|
            (overlay ||
 | 
						|
             visualViewportSize.width >= aState.HScrollbarMinWidth())) {
 | 
						|
          vvChanged = true;
 | 
						|
          if (!overlay) {
 | 
						|
            visualViewportSize.height -= aState.HScrollbarPrefHeight();
 | 
						|
          }
 | 
						|
          aState.mShowHScrollbar = true;
 | 
						|
          aState.mOnlyNeedHScrollbarToScrollVVInsideLV = true;
 | 
						|
          ROOT_SCROLLBAR_LOG("TryLayout added H scrollbar for VV, VV now %s\n",
 | 
						|
                             ToString(visualViewportSize).c_str());
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (!aState.mShowVScrollbar &&
 | 
						|
          aState.mVScrollbarAllowedForScrollingVVInsideLV) {
 | 
						|
        if (ScrollPort().height >= visualViewportSize.height + oneDevPixel &&
 | 
						|
            (overlay ||
 | 
						|
             visualViewportSize.height >= aState.VScrollbarMinHeight())) {
 | 
						|
          vvChanged = true;
 | 
						|
          if (!overlay) {
 | 
						|
            visualViewportSize.width -= aState.VScrollbarPrefWidth();
 | 
						|
          }
 | 
						|
          aState.mShowVScrollbar = true;
 | 
						|
          aState.mOnlyNeedVScrollbarToScrollVVInsideLV = true;
 | 
						|
          ROOT_SCROLLBAR_LOG("TryLayout added V scrollbar for VV, VV now %s\n",
 | 
						|
                             ToString(visualViewportSize).c_str());
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::ScrolledContentDependsOnBSize(
 | 
						|
    const ScrollReflowInput& aState) const {
 | 
						|
  return mScrolledFrame->HasAnyStateBits(
 | 
						|
             NS_FRAME_CONTAINS_RELATIVE_BSIZE |
 | 
						|
             NS_FRAME_DESCENDANT_INTRINSIC_ISIZE_DEPENDS_ON_BSIZE) ||
 | 
						|
         aState.mReflowInput.ComputedBSize() != NS_UNCONSTRAINEDSIZE ||
 | 
						|
         aState.mReflowInput.ComputedMinBSize() > 0 ||
 | 
						|
         aState.mReflowInput.ComputedMaxBSize() != NS_UNCONSTRAINEDSIZE;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ReflowScrolledFrame(ScrollReflowInput& aState,
 | 
						|
                                            bool aAssumeHScroll,
 | 
						|
                                            bool aAssumeVScroll,
 | 
						|
                                            ReflowOutput* aMetrics) {
 | 
						|
  const WritingMode wm = GetWritingMode();
 | 
						|
 | 
						|
  // these could be NS_UNCONSTRAINEDSIZE ... std::min arithmetic should
 | 
						|
  // be OK
 | 
						|
  LogicalMargin padding = aState.mReflowInput.ComputedLogicalPadding(wm);
 | 
						|
  nscoord availISize =
 | 
						|
      aState.mReflowInput.ComputedISize() + padding.IStartEnd(wm);
 | 
						|
 | 
						|
  nscoord computedBSize = aState.mReflowInput.ComputedBSize();
 | 
						|
  nscoord computedMinBSize = aState.mReflowInput.ComputedMinBSize();
 | 
						|
  nscoord computedMaxBSize = aState.mReflowInput.ComputedMaxBSize();
 | 
						|
  if (!ShouldPropagateComputedBSizeToScrolledContent()) {
 | 
						|
    computedBSize = NS_UNCONSTRAINEDSIZE;
 | 
						|
    computedMinBSize = 0;
 | 
						|
    computedMaxBSize = NS_UNCONSTRAINEDSIZE;
 | 
						|
  }
 | 
						|
 | 
						|
  const LogicalMargin scrollbarGutter(
 | 
						|
      wm, aState.ScrollbarGutter(aAssumeVScroll, aAssumeHScroll,
 | 
						|
                                 IsScrollbarOnRight()));
 | 
						|
  if (const nscoord inlineEndsGutter = scrollbarGutter.IStartEnd(wm);
 | 
						|
      inlineEndsGutter > 0) {
 | 
						|
    availISize = std::max(0, availISize - inlineEndsGutter);
 | 
						|
  }
 | 
						|
  if (const nscoord blockEndsGutter = scrollbarGutter.BStartEnd(wm);
 | 
						|
      blockEndsGutter > 0) {
 | 
						|
    if (computedBSize != NS_UNCONSTRAINEDSIZE) {
 | 
						|
      computedBSize = std::max(0, computedBSize - blockEndsGutter);
 | 
						|
    }
 | 
						|
    computedMinBSize = std::max(0, computedMinBSize - blockEndsGutter);
 | 
						|
    if (computedMaxBSize != NS_UNCONSTRAINEDSIZE) {
 | 
						|
      computedMaxBSize = std::max(0, computedMaxBSize - blockEndsGutter);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nsPresContext* presContext = PresContext();
 | 
						|
 | 
						|
  // Pass InitFlags::CallerWillInit so we can pass in the correct padding.
 | 
						|
  ReflowInput kidReflowInput(presContext, aState.mReflowInput, mScrolledFrame,
 | 
						|
                             LogicalSize(wm, availISize, NS_UNCONSTRAINEDSIZE),
 | 
						|
                             Nothing(), ReflowInput::InitFlag::CallerWillInit);
 | 
						|
  const WritingMode kidWM = kidReflowInput.GetWritingMode();
 | 
						|
  kidReflowInput.Init(presContext, Nothing(), Nothing(),
 | 
						|
                      Some(padding.ConvertTo(kidWM, wm)));
 | 
						|
  kidReflowInput.mFlags.mAssumingHScrollbar = aAssumeHScroll;
 | 
						|
  kidReflowInput.mFlags.mAssumingVScrollbar = aAssumeVScroll;
 | 
						|
  kidReflowInput.mFlags.mTreatBSizeAsIndefinite =
 | 
						|
      aState.mReflowInput.mFlags.mTreatBSizeAsIndefinite;
 | 
						|
  kidReflowInput.SetComputedBSize(computedBSize);
 | 
						|
  kidReflowInput.SetComputedMinBSize(computedMinBSize);
 | 
						|
  kidReflowInput.SetComputedMaxBSize(computedMaxBSize);
 | 
						|
  if (aState.mReflowInput.IsBResizeForWM(kidWM)) {
 | 
						|
    kidReflowInput.SetBResize(true);
 | 
						|
  }
 | 
						|
  if (aState.mReflowInput.IsBResizeForPercentagesForWM(kidWM)) {
 | 
						|
    kidReflowInput.mFlags.mIsBResizeForPercentages = true;
 | 
						|
  }
 | 
						|
 | 
						|
  // Temporarily set mHasHorizontalScrollbar/mHasVerticalScrollbar to
 | 
						|
  // reflect our assumptions while we reflow the child.
 | 
						|
  bool didHaveHorizontalScrollbar = mHasHorizontalScrollbar;
 | 
						|
  bool didHaveVerticalScrollbar = mHasVerticalScrollbar;
 | 
						|
  mHasHorizontalScrollbar = aAssumeHScroll;
 | 
						|
  mHasVerticalScrollbar = aAssumeVScroll;
 | 
						|
 | 
						|
  nsReflowStatus status;
 | 
						|
  // No need to pass a true container-size to ReflowChild or
 | 
						|
  // FinishReflowChild, because it's only used there when positioning
 | 
						|
  // the frame (i.e. if ReflowChildFlags::NoMoveFrame isn't set)
 | 
						|
  const nsSize dummyContainerSize;
 | 
						|
  ReflowChild(mScrolledFrame, presContext, *aMetrics, kidReflowInput, wm,
 | 
						|
              LogicalPoint(wm), dummyContainerSize,
 | 
						|
              ReflowChildFlags::NoMoveFrame, status);
 | 
						|
 | 
						|
  mHasHorizontalScrollbar = didHaveHorizontalScrollbar;
 | 
						|
  mHasVerticalScrollbar = didHaveVerticalScrollbar;
 | 
						|
 | 
						|
  // Don't resize or position the view (if any) because we're going to resize
 | 
						|
  // it to the correct size anyway in PlaceScrollArea. Allowing it to
 | 
						|
  // resize here would size it to the natural height of the frame,
 | 
						|
  // which will usually be different from the scrollport height;
 | 
						|
  // invalidating the difference will cause unnecessary repainting.
 | 
						|
  FinishReflowChild(
 | 
						|
      mScrolledFrame, presContext, *aMetrics, &kidReflowInput, wm,
 | 
						|
      LogicalPoint(wm), dummyContainerSize,
 | 
						|
      ReflowChildFlags::NoMoveFrame | ReflowChildFlags::NoSizeView);
 | 
						|
 | 
						|
  if (mScrolledFrame->HasAnyStateBits(NS_FRAME_CONTAINS_RELATIVE_BSIZE)) {
 | 
						|
    // Propagate NS_FRAME_CONTAINS_RELATIVE_BSIZE from our inner scrolled frame
 | 
						|
    // to ourselves so that our containing block is aware of it.
 | 
						|
    //
 | 
						|
    // Note: If the scrolled frame has any child whose block-size depends on the
 | 
						|
    // containing block's block-size, the NS_FRAME_CONTAINS_RELATIVE_BSIZE bit
 | 
						|
    // is set on the scrolled frame when initializing the child's ReflowInput in
 | 
						|
    // ReflowInput::InitResizeFlags(). Therefore, we propagate the bit here
 | 
						|
    // after we reflowed the scrolled frame.
 | 
						|
    AddStateBits(NS_FRAME_CONTAINS_RELATIVE_BSIZE);
 | 
						|
  }
 | 
						|
 | 
						|
  // XXX Some frames (e.g. nsFrameFrame, nsTextFrame) don't
 | 
						|
  // bother setting their mOverflowArea. This is wrong because every frame
 | 
						|
  // should always set mOverflowArea. In fact nsFrameFrame doesn't
 | 
						|
  // support the 'outline' property because of this. Rather than fix the
 | 
						|
  // world right now, just fix up the overflow area if necessary. Note that we
 | 
						|
  // don't check HasOverflowRect() because it could be set even though the
 | 
						|
  // overflow area doesn't include the frame bounds.
 | 
						|
  aMetrics->UnionOverflowAreasWithDesiredBounds();
 | 
						|
 | 
						|
  auto* disp = StyleDisplay();
 | 
						|
  if (MOZ_UNLIKELY(disp->mOverflowClipBoxInline ==
 | 
						|
                   StyleOverflowClipBox::ContentBox)) {
 | 
						|
    // The scrolled frame is scrollable in the inline axis with
 | 
						|
    // `overflow-clip-box:content-box`. To prevent its content from being
 | 
						|
    // clipped at the scroll container's padding edges, we inflate its
 | 
						|
    // children's scrollable overflow area with its inline padding, and union
 | 
						|
    // its scrollable overflow area with its children's inflated scrollable
 | 
						|
    // overflow area.
 | 
						|
    OverflowAreas childOverflow;
 | 
						|
    mScrolledFrame->UnionChildOverflow(childOverflow);
 | 
						|
    nsRect childScrollableOverflow = childOverflow.ScrollableOverflow();
 | 
						|
 | 
						|
    const LogicalMargin inlinePadding =
 | 
						|
        padding.ApplySkipSides(LogicalSides(wm, eLogicalSideBitsBBoth));
 | 
						|
    childScrollableOverflow.Inflate(inlinePadding.GetPhysicalMargin(wm));
 | 
						|
 | 
						|
    nsRect& so = aMetrics->ScrollableOverflow();
 | 
						|
    so = so.UnionEdges(childScrollableOverflow);
 | 
						|
  }
 | 
						|
 | 
						|
  aState.mContentsOverflowAreas = aMetrics->mOverflowAreas;
 | 
						|
  aState.mScrollbarGutterFromLastReflow = scrollbarGutter;
 | 
						|
  aState.mReflowedContentsWithHScrollbar = aAssumeHScroll;
 | 
						|
  aState.mReflowedContentsWithVScrollbar = aAssumeVScroll;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::GuessHScrollbarNeeded(const ScrollReflowInput& aState) {
 | 
						|
  if (aState.mHScrollbar != ShowScrollbar::Auto) {
 | 
						|
    // no guessing required
 | 
						|
    return aState.mHScrollbar == ShowScrollbar::Always;
 | 
						|
  }
 | 
						|
  // We only care about scrollbars that might take up space when trying to guess
 | 
						|
  // if we need a scrollbar, so we ignore scrollbars only created to scroll the
 | 
						|
  // visual viewport inside the layout viewport because they take up no layout
 | 
						|
  // space.
 | 
						|
  return mHasHorizontalScrollbar && !mOnlyNeedHScrollbarToScrollVVInsideLV;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::GuessVScrollbarNeeded(const ScrollReflowInput& aState) {
 | 
						|
  if (aState.mVScrollbar != ShowScrollbar::Auto) {
 | 
						|
    // no guessing required
 | 
						|
    return aState.mVScrollbar == ShowScrollbar::Always;
 | 
						|
  }
 | 
						|
 | 
						|
  // If we've had at least one non-initial reflow, then just assume
 | 
						|
  // the state of the vertical scrollbar will be what we determined
 | 
						|
  // last time.
 | 
						|
  if (mHadNonInitialReflow) {
 | 
						|
    // We only care about scrollbars that might take up space when trying to
 | 
						|
    // guess if we need a scrollbar, so we ignore scrollbars only created to
 | 
						|
    // scroll the visual viewport inside the layout viewport because they take
 | 
						|
    // up no layout space.
 | 
						|
    return mHasVerticalScrollbar && !mOnlyNeedVScrollbarToScrollVVInsideLV;
 | 
						|
  }
 | 
						|
 | 
						|
  // If this is the initial reflow, guess false because usually
 | 
						|
  // we have very little content by then.
 | 
						|
  if (InInitialReflow()) return false;
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    nsIFrame* f = mScrolledFrame->PrincipalChildList().FirstChild();
 | 
						|
    if (f && f->IsSVGOuterSVGFrame() &&
 | 
						|
        static_cast<SVGOuterSVGFrame*>(f)->VerticalScrollbarNotNeeded()) {
 | 
						|
      // Common SVG case - avoid a bad guess.
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    // Assume that there will be a scrollbar; it seems to me
 | 
						|
    // that 'most pages' do have a scrollbar, and anyway, it's cheaper
 | 
						|
    // to do an extra reflow for the pages that *don't* need a
 | 
						|
    // scrollbar (because on average they will have less content).
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // For non-viewports, just guess that we don't need a scrollbar.
 | 
						|
  // XXX I wonder if statistically this is the right idea; I'm
 | 
						|
  // basically guessing that there are a lot of overflow:auto DIVs
 | 
						|
  // that get their intrinsic size and don't overflow
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::InInitialReflow() const {
 | 
						|
  // We're in an initial reflow if NS_FRAME_FIRST_REFLOW is set, unless we're a
 | 
						|
  // root scrollframe.  In that case we want to skip this clause altogether.
 | 
						|
  // The guess here is that there are lots of overflow:auto divs out there that
 | 
						|
  // end up auto-sizing so they don't overflow, and that the root basically
 | 
						|
  // always needs a scrollbar if it did last time we loaded this page (good
 | 
						|
  // assumption, because our initial reflow is no longer synchronous).
 | 
						|
  return !mIsRoot && HasAnyStateBits(NS_FRAME_FIRST_REFLOW);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ReflowContents(ScrollReflowInput& aState,
 | 
						|
                                       const ReflowOutput& aDesiredSize) {
 | 
						|
  const WritingMode desiredWm = aDesiredSize.GetWritingMode();
 | 
						|
  ReflowOutput kidDesiredSize(desiredWm);
 | 
						|
  ReflowScrolledFrame(aState, GuessHScrollbarNeeded(aState),
 | 
						|
                      GuessVScrollbarNeeded(aState), &kidDesiredSize);
 | 
						|
 | 
						|
  // There's an important special case ... if the child appears to fit
 | 
						|
  // in the inside-border rect (but overflows the scrollport), we
 | 
						|
  // should try laying it out without a vertical scrollbar. It will
 | 
						|
  // usually fit because making the available-width wider will not
 | 
						|
  // normally make the child taller. (The only situation I can think
 | 
						|
  // of is when you have a line containing %-width inline replaced
 | 
						|
  // elements whose percentages sum to more than 100%, so increasing
 | 
						|
  // the available width makes the line break where it was fitting
 | 
						|
  // before.) If we don't treat this case specially, then we will
 | 
						|
  // decide that showing scrollbars is OK because the content
 | 
						|
  // overflows when we're showing scrollbars and we won't try to
 | 
						|
  // remove the vertical scrollbar.
 | 
						|
 | 
						|
  // Detecting when we enter this special case is important for when
 | 
						|
  // people design layouts that exactly fit the container "most of the
 | 
						|
  // time".
 | 
						|
 | 
						|
  // XXX Is this check really sufficient to catch all the incremental cases
 | 
						|
  // where the ideal case doesn't have a scrollbar?
 | 
						|
  if ((aState.mReflowedContentsWithHScrollbar ||
 | 
						|
       aState.mReflowedContentsWithVScrollbar) &&
 | 
						|
      aState.mVScrollbar != ShowScrollbar::Always &&
 | 
						|
      aState.mHScrollbar != ShowScrollbar::Always) {
 | 
						|
    nsSize kidSize = GetContainSizeAxes().ContainSize(
 | 
						|
        kidDesiredSize.PhysicalSize(), *aState.mReflowInput.mFrame);
 | 
						|
    nsSize insideBorderSize = ComputeInsideBorderSize(aState, kidSize);
 | 
						|
    nsRect scrolledRect = GetUnsnappedScrolledRectInternal(
 | 
						|
        kidDesiredSize.ScrollableOverflow(), insideBorderSize);
 | 
						|
    if (nsRect(nsPoint(0, 0), insideBorderSize).Contains(scrolledRect)) {
 | 
						|
      // Let's pretend we had no scrollbars coming in here
 | 
						|
      kidDesiredSize.mOverflowAreas.Clear();
 | 
						|
      ReflowScrolledFrame(aState, false, false, &kidDesiredSize);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (IsRootScrollFrameOfDocument()) {
 | 
						|
    UpdateMinimumScaleSize(aState.mContentsOverflowAreas.ScrollableOverflow(),
 | 
						|
                           kidDesiredSize.PhysicalSize());
 | 
						|
  }
 | 
						|
 | 
						|
  // Try vertical scrollbar settings that leave the vertical scrollbar
 | 
						|
  // unchanged. Do this first because changing the vertical scrollbar setting is
 | 
						|
  // expensive, forcing a reflow always.
 | 
						|
 | 
						|
  // Try leaving the horizontal scrollbar unchanged first. This will be more
 | 
						|
  // efficient.
 | 
						|
  ROOT_SCROLLBAR_LOG("Trying layout1 with %d, %d\n",
 | 
						|
                     aState.mReflowedContentsWithHScrollbar,
 | 
						|
                     aState.mReflowedContentsWithVScrollbar);
 | 
						|
  if (TryLayout(aState, &kidDesiredSize, aState.mReflowedContentsWithHScrollbar,
 | 
						|
                aState.mReflowedContentsWithVScrollbar, false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  ROOT_SCROLLBAR_LOG("Trying layout2 with %d, %d\n",
 | 
						|
                     !aState.mReflowedContentsWithHScrollbar,
 | 
						|
                     aState.mReflowedContentsWithVScrollbar);
 | 
						|
  if (TryLayout(aState, &kidDesiredSize,
 | 
						|
                !aState.mReflowedContentsWithHScrollbar,
 | 
						|
                aState.mReflowedContentsWithVScrollbar, false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // OK, now try toggling the vertical scrollbar. The performance advantage
 | 
						|
  // of trying the status-quo horizontal scrollbar state
 | 
						|
  // does not exist here (we'll have to reflow due to the vertical scrollbar
 | 
						|
  // change), so always try no horizontal scrollbar first.
 | 
						|
  bool newVScrollbarState = !aState.mReflowedContentsWithVScrollbar;
 | 
						|
  ROOT_SCROLLBAR_LOG("Trying layout3 with %d, %d\n", false, newVScrollbarState);
 | 
						|
  if (TryLayout(aState, &kidDesiredSize, false, newVScrollbarState, false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  ROOT_SCROLLBAR_LOG("Trying layout4 with %d, %d\n", true, newVScrollbarState);
 | 
						|
  if (TryLayout(aState, &kidDesiredSize, true, newVScrollbarState, false)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // OK, we're out of ideas. Try again enabling whatever scrollbars we can
 | 
						|
  // enable and force the layout to stick even if it's inconsistent.
 | 
						|
  // This just happens sometimes.
 | 
						|
  ROOT_SCROLLBAR_LOG("Giving up, adding both scrollbars...\n");
 | 
						|
  TryLayout(aState, &kidDesiredSize, aState.mHScrollbar != ShowScrollbar::Never,
 | 
						|
            aState.mVScrollbar != ShowScrollbar::Never, true);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PlaceScrollArea(ScrollReflowInput& aState,
 | 
						|
                                        const nsPoint& aScrollPosition) {
 | 
						|
  // Set the x,y of the scrolled frame to the correct value
 | 
						|
  mScrolledFrame->SetPosition(ScrollPort().TopLeft() - aScrollPosition);
 | 
						|
 | 
						|
  // Recompute our scrollable overflow, taking perspective children into
 | 
						|
  // account. Note that this only recomputes the overflow areas stored on the
 | 
						|
  // helper (which are used to compute scrollable length and scrollbar thumb
 | 
						|
  // sizes) but not the overflow areas stored on the frame. This seems to work
 | 
						|
  // for now, but it's possible that we may need to update both in the future.
 | 
						|
  AdjustForPerspective(aState.mContentsOverflowAreas.ScrollableOverflow());
 | 
						|
 | 
						|
  // Preserve the width or height of empty rects
 | 
						|
  const nsSize portSize = ScrollPort().Size();
 | 
						|
  nsRect scrolledRect = GetUnsnappedScrolledRectInternal(
 | 
						|
      aState.mContentsOverflowAreas.ScrollableOverflow(), portSize);
 | 
						|
  nsRect scrolledArea =
 | 
						|
      scrolledRect.UnionEdges(nsRect(nsPoint(0, 0), portSize));
 | 
						|
 | 
						|
  // Store the new overflow area. Note that this changes where an outline
 | 
						|
  // of the scrolled frame would be painted, but scrolled frames can't have
 | 
						|
  // outlines (the outline would go on this scrollframe instead).
 | 
						|
  // Using FinishAndStoreOverflow is needed so the overflow rect gets set
 | 
						|
  // correctly.  It also messes with the overflow rect in the 'clip' case, but
 | 
						|
  // scrolled frames can't have 'overflow' either.
 | 
						|
  // This needs to happen before SyncFrameViewAfterReflow so
 | 
						|
  // HasOverflowRect() will return the correct value.
 | 
						|
  OverflowAreas overflow(scrolledArea, scrolledArea);
 | 
						|
  mScrolledFrame->FinishAndStoreOverflow(overflow, mScrolledFrame->GetSize());
 | 
						|
 | 
						|
  // Note that making the view *exactly* the size of the scrolled area
 | 
						|
  // is critical, since the view scrolling code uses the size of the
 | 
						|
  // scrolled view to clamp scroll requests.
 | 
						|
  // Normally the mScrolledFrame won't have a view but in some cases it
 | 
						|
  // might create its own.
 | 
						|
  nsContainerFrame::SyncFrameViewAfterReflow(
 | 
						|
      mScrolledFrame->PresContext(), mScrolledFrame, mScrolledFrame->GetView(),
 | 
						|
      scrolledArea, ReflowChildFlags::Default);
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsHTMLScrollFrame::IntrinsicScrollbarGutterSizeAtInlineEdges() const {
 | 
						|
  const auto wm = GetWritingMode();
 | 
						|
  const LogicalMargin gutter(wm, IntrinsicScrollbarGutterSize());
 | 
						|
  return gutter.IStartEnd(wm);
 | 
						|
}
 | 
						|
 | 
						|
nsMargin nsHTMLScrollFrame::IntrinsicScrollbarGutterSize() const {
 | 
						|
  if (PresContext()->UseOverlayScrollbars()) {
 | 
						|
    // Overlay scrollbars do not consume space per spec.
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  const auto* styleForScrollbar = nsLayoutUtils::StyleForScrollbar(this);
 | 
						|
  const auto& styleScrollbarWidth =
 | 
						|
      styleForScrollbar->StyleUIReset()->ScrollbarWidth();
 | 
						|
  if (styleScrollbarWidth == StyleScrollbarWidth::None) {
 | 
						|
    // Scrollbar shouldn't appear at all with "scrollbar-width: none".
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  const auto& styleScrollbarGutter =
 | 
						|
      styleForScrollbar->StyleDisplay()->mScrollbarGutter;
 | 
						|
  nsMargin gutter =
 | 
						|
      ComputeStableScrollbarGutter(styleScrollbarWidth, styleScrollbarGutter);
 | 
						|
  if (gutter.LeftRight() == 0 || gutter.TopBottom() == 0) {
 | 
						|
    // If there is no stable scrollbar-gutter at vertical or horizontal
 | 
						|
    // dimension, check if a scrollbar is always shown at that dimension.
 | 
						|
    ScrollStyles scrollStyles = GetScrollStyles();
 | 
						|
    const nscoord scrollbarSize =
 | 
						|
        GetNonOverlayScrollbarSize(PresContext(), styleScrollbarWidth);
 | 
						|
    if (gutter.LeftRight() == 0 &&
 | 
						|
        scrollStyles.mVertical == StyleOverflow::Scroll) {
 | 
						|
      (IsScrollbarOnRight() ? gutter.right : gutter.left) = scrollbarSize;
 | 
						|
    }
 | 
						|
    if (gutter.TopBottom() == 0 &&
 | 
						|
        scrollStyles.mHorizontal == StyleOverflow::Scroll) {
 | 
						|
      // The horizontal scrollbar is always at the bottom side.
 | 
						|
      gutter.bottom = scrollbarSize;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return gutter;
 | 
						|
}
 | 
						|
 | 
						|
nsMargin nsHTMLScrollFrame::ComputeStableScrollbarGutter(
 | 
						|
    const StyleScrollbarWidth& aStyleScrollbarWidth,
 | 
						|
    const StyleScrollbarGutter& aStyleScrollbarGutter) const {
 | 
						|
  if (PresContext()->UseOverlayScrollbars()) {
 | 
						|
    // Overlay scrollbars do not consume space per spec.
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  if (aStyleScrollbarWidth == StyleScrollbarWidth::None) {
 | 
						|
    // Scrollbar shouldn't appear at all with "scrollbar-width: none".
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  if (aStyleScrollbarGutter == StyleScrollbarGutter::AUTO) {
 | 
						|
    // Scrollbars create space depending on the 'overflow' property and whether
 | 
						|
    // the content overflows. Callers need to check this scenario if they want
 | 
						|
    // to consider the space created by the actual scrollbars.
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  const bool bothEdges =
 | 
						|
      bool(aStyleScrollbarGutter & StyleScrollbarGutter::BOTH_EDGES);
 | 
						|
  const bool isVerticalWM = GetWritingMode().IsVertical();
 | 
						|
  const nscoord scrollbarSize =
 | 
						|
      GetNonOverlayScrollbarSize(PresContext(), aStyleScrollbarWidth);
 | 
						|
 | 
						|
  nsMargin scrollbarGutter;
 | 
						|
  if (bothEdges) {
 | 
						|
    if (isVerticalWM) {
 | 
						|
      scrollbarGutter.top = scrollbarGutter.bottom = scrollbarSize;
 | 
						|
    } else {
 | 
						|
      scrollbarGutter.left = scrollbarGutter.right = scrollbarSize;
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    MOZ_ASSERT(bool(aStyleScrollbarGutter & StyleScrollbarGutter::STABLE),
 | 
						|
               "scrollbar-gutter value should be 'stable'!");
 | 
						|
    if (isVerticalWM) {
 | 
						|
      // The horizontal scrollbar-gutter is always at the bottom side.
 | 
						|
      scrollbarGutter.bottom = scrollbarSize;
 | 
						|
    } else if (IsScrollbarOnRight()) {
 | 
						|
      scrollbarGutter.right = scrollbarSize;
 | 
						|
    } else {
 | 
						|
      scrollbarGutter.left = scrollbarSize;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return scrollbarGutter;
 | 
						|
}
 | 
						|
 | 
						|
// Legacy, this sucks!
 | 
						|
static bool IsMarqueeScrollbox(const nsIFrame& aScrollFrame) {
 | 
						|
  if (!aScrollFrame.GetContent()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if (MOZ_LIKELY(!aScrollFrame.GetContent()->HasBeenInUAWidget())) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  MOZ_ASSERT(aScrollFrame.GetParent() &&
 | 
						|
             aScrollFrame.GetParent()->GetContent());
 | 
						|
  return aScrollFrame.GetParent() &&
 | 
						|
         HTMLMarqueeElement::FromNodeOrNull(
 | 
						|
             aScrollFrame.GetParent()->GetContent());
 | 
						|
}
 | 
						|
 | 
						|
/* virtual */
 | 
						|
nscoord nsHTMLScrollFrame::GetMinISize(gfxContext* aRenderingContext) {
 | 
						|
  nscoord result = [&] {
 | 
						|
    if (const Maybe<nscoord> containISize = ContainIntrinsicISize()) {
 | 
						|
      return *containISize;
 | 
						|
    }
 | 
						|
    if (MOZ_UNLIKELY(IsMarqueeScrollbox(*this))) {
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
    return mScrolledFrame->GetMinISize(aRenderingContext);
 | 
						|
  }();
 | 
						|
 | 
						|
  DISPLAY_MIN_INLINE_SIZE(this, result);
 | 
						|
  return result + IntrinsicScrollbarGutterSizeAtInlineEdges();
 | 
						|
}
 | 
						|
 | 
						|
/* virtual */
 | 
						|
nscoord nsHTMLScrollFrame::GetPrefISize(gfxContext* aRenderingContext) {
 | 
						|
  const Maybe<nscoord> containISize = ContainIntrinsicISize();
 | 
						|
  nscoord result = containISize
 | 
						|
                       ? *containISize
 | 
						|
                       : mScrolledFrame->GetPrefISize(aRenderingContext);
 | 
						|
  DISPLAY_PREF_INLINE_SIZE(this, result);
 | 
						|
  return NSCoordSaturatingAdd(result,
 | 
						|
                              IntrinsicScrollbarGutterSizeAtInlineEdges());
 | 
						|
}
 | 
						|
 | 
						|
// When we have perspective set on the outer scroll frame, and transformed
 | 
						|
// children (possibly with preserve-3d) then the effective transform on the
 | 
						|
// child depends on the offset to the scroll frame, which changes as we scroll.
 | 
						|
// This perspective transform can cause the element to move relative to the
 | 
						|
// scrolled inner frame, which would cause the scrollable length changes during
 | 
						|
// scrolling if we didn't account for it. Since we don't want scrollHeight/Width
 | 
						|
// and the size of scrollbar thumbs to change during scrolling, we compute the
 | 
						|
// scrollable overflow by determining the scroll position at which the child
 | 
						|
// becomes completely visible within the scrollport rather than using the union
 | 
						|
// of the overflow areas at their current position.
 | 
						|
static void GetScrollableOverflowForPerspective(
 | 
						|
    nsIFrame* aScrolledFrame, nsIFrame* aCurrentFrame, const nsRect aScrollPort,
 | 
						|
    nsPoint aOffset, nsRect& aScrolledFrameOverflowArea) {
 | 
						|
  // Iterate over all children except pop-ups.
 | 
						|
  for (const auto& [list, listID] : aCurrentFrame->ChildLists()) {
 | 
						|
    if (listID == FrameChildListID::Popup) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    for (nsIFrame* child : list) {
 | 
						|
      nsPoint offset = aOffset;
 | 
						|
 | 
						|
      // When we reach a direct child of the scroll, then we record the offset
 | 
						|
      // to convert from that frame's coordinate into the scroll frame's
 | 
						|
      // coordinates. Preserve-3d descendant frames use the same offset as their
 | 
						|
      // ancestors, since TransformRect already converts us into the coordinate
 | 
						|
      // space of the preserve-3d root.
 | 
						|
      if (aScrolledFrame == aCurrentFrame) {
 | 
						|
        offset = child->GetPosition();
 | 
						|
      }
 | 
						|
 | 
						|
      if (child->Extend3DContext()) {
 | 
						|
        // If we're a preserve-3d frame, then recurse and include our
 | 
						|
        // descendants since overflow of preserve-3d frames is only included
 | 
						|
        // in the post-transform overflow area of the preserve-3d root frame.
 | 
						|
        GetScrollableOverflowForPerspective(aScrolledFrame, child, aScrollPort,
 | 
						|
                                            offset, aScrolledFrameOverflowArea);
 | 
						|
      }
 | 
						|
 | 
						|
      // If we're transformed, then we want to consider the possibility that
 | 
						|
      // this frame might move relative to the scrolled frame when scrolling.
 | 
						|
      // For preserve-3d, leaf frames have correct overflow rects relative to
 | 
						|
      // themselves. preserve-3d 'nodes' (intermediate frames and the root) have
 | 
						|
      // only their untransformed children included in their overflow relative
 | 
						|
      // to self, which is what we want to include here.
 | 
						|
      if (child->IsTransformed()) {
 | 
						|
        // Compute the overflow rect for this leaf transform frame in the
 | 
						|
        // coordinate space of the scrolled frame.
 | 
						|
        nsPoint scrollPos = aScrolledFrame->GetPosition();
 | 
						|
        nsRect preScroll, postScroll;
 | 
						|
        {
 | 
						|
          // TODO: Can we reuse the reference box?
 | 
						|
          TransformReferenceBox refBox(child);
 | 
						|
          preScroll = nsDisplayTransform::TransformRect(
 | 
						|
              child->ScrollableOverflowRectRelativeToSelf(), child, refBox);
 | 
						|
        }
 | 
						|
 | 
						|
        // Temporarily override the scroll position of the scrolled frame by
 | 
						|
        // 10 CSS pixels, and then recompute what the overflow rect would be.
 | 
						|
        // This scroll position may not be valid, but that shouldn't matter
 | 
						|
        // for our calculations.
 | 
						|
        {
 | 
						|
          aScrolledFrame->SetPosition(scrollPos + nsPoint(600, 600));
 | 
						|
          TransformReferenceBox refBox(child);
 | 
						|
          postScroll = nsDisplayTransform::TransformRect(
 | 
						|
              child->ScrollableOverflowRectRelativeToSelf(), child, refBox);
 | 
						|
          aScrolledFrame->SetPosition(scrollPos);
 | 
						|
        }
 | 
						|
 | 
						|
        // Compute how many app units the overflow rects moves by when we adjust
 | 
						|
        // the scroll position by 1 app unit.
 | 
						|
        double rightDelta =
 | 
						|
            (postScroll.XMost() - preScroll.XMost() + 600.0) / 600.0;
 | 
						|
        double bottomDelta =
 | 
						|
            (postScroll.YMost() - preScroll.YMost() + 600.0) / 600.0;
 | 
						|
 | 
						|
        // We can't ever have negative scrolling.
 | 
						|
        NS_ASSERTION(rightDelta > 0.0f && bottomDelta > 0.0f,
 | 
						|
                     "Scrolling can't be reversed!");
 | 
						|
 | 
						|
        // Move preScroll into the coordinate space of the scrollport.
 | 
						|
        preScroll += offset + scrollPos;
 | 
						|
 | 
						|
        // For each of the four edges of preScroll, figure out how far they
 | 
						|
        // extend beyond the scrollport. Ignore negative values since that means
 | 
						|
        // that side is already scrolled in to view and we don't need to add
 | 
						|
        // overflow to account for it.
 | 
						|
        nsMargin overhang(std::max(0, aScrollPort.Y() - preScroll.Y()),
 | 
						|
                          std::max(0, preScroll.XMost() - aScrollPort.XMost()),
 | 
						|
                          std::max(0, preScroll.YMost() - aScrollPort.YMost()),
 | 
						|
                          std::max(0, aScrollPort.X() - preScroll.X()));
 | 
						|
 | 
						|
        // Scale according to rightDelta/bottomDelta to adjust for the different
 | 
						|
        // scroll rates.
 | 
						|
        overhang.top = NSCoordSaturatingMultiply(
 | 
						|
            overhang.top, static_cast<float>(1 / bottomDelta));
 | 
						|
        overhang.right = NSCoordSaturatingMultiply(
 | 
						|
            overhang.right, static_cast<float>(1 / rightDelta));
 | 
						|
        overhang.bottom = NSCoordSaturatingMultiply(
 | 
						|
            overhang.bottom, static_cast<float>(1 / bottomDelta));
 | 
						|
        overhang.left = NSCoordSaturatingMultiply(
 | 
						|
            overhang.left, static_cast<float>(1 / rightDelta));
 | 
						|
 | 
						|
        // Take the minimum overflow rect that would allow the current scroll
 | 
						|
        // position, using the size of the scroll port and offset by the
 | 
						|
        // inverse of the scroll position.
 | 
						|
        nsRect overflow = aScrollPort - scrollPos;
 | 
						|
 | 
						|
        // Expand it by our margins to get an overflow rect that would allow all
 | 
						|
        // edges of our transformed content to be scrolled into view.
 | 
						|
        overflow.Inflate(overhang);
 | 
						|
 | 
						|
        // Merge it with the combined overflow
 | 
						|
        aScrolledFrameOverflowArea.UnionRect(aScrolledFrameOverflowArea,
 | 
						|
                                             overflow);
 | 
						|
      } else if (aCurrentFrame == aScrolledFrame) {
 | 
						|
        aScrolledFrameOverflowArea.UnionRect(
 | 
						|
            aScrolledFrameOverflowArea,
 | 
						|
            child->ScrollableOverflowRectRelativeToParent());
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
BaselineSharingGroup nsHTMLScrollFrame::GetDefaultBaselineSharingGroup() const {
 | 
						|
  return mScrolledFrame->GetDefaultBaselineSharingGroup();
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsHTMLScrollFrame::SynthesizeFallbackBaseline(
 | 
						|
    mozilla::WritingMode aWM, BaselineSharingGroup aBaselineGroup) const {
 | 
						|
  // Marign-end even for central baselines.
 | 
						|
  if (aWM.IsLineInverted()) {
 | 
						|
    return -GetLogicalUsedMargin(aWM).BStart(aWM);
 | 
						|
  }
 | 
						|
  return aBaselineGroup == BaselineSharingGroup::First
 | 
						|
             ? BSize(aWM) + GetLogicalUsedMargin(aWM).BEnd(aWM)
 | 
						|
             : -GetLogicalUsedMargin(aWM).BEnd(aWM);
 | 
						|
}
 | 
						|
 | 
						|
Maybe<nscoord> nsHTMLScrollFrame::GetNaturalBaselineBOffset(
 | 
						|
    WritingMode aWM, BaselineSharingGroup aBaselineGroup,
 | 
						|
    BaselineExportContext aExportContext) const {
 | 
						|
  // Block containers that are scrollable always have a last baseline
 | 
						|
  // that are synthesized from block-end margin edge.
 | 
						|
  // Note(dshin): This behaviour is really only relevant to `inline-block`
 | 
						|
  // alignment context. In the context of table/flex/grid alignment, first/last
 | 
						|
  // baselines are calculated through `GetFirstLineBaseline`, which does
 | 
						|
  // calculations of its own.
 | 
						|
  // https://drafts.csswg.org/css-align/#baseline-export
 | 
						|
  if (aExportContext == BaselineExportContext::LineLayout &&
 | 
						|
      aBaselineGroup == BaselineSharingGroup::Last &&
 | 
						|
      mScrolledFrame->IsBlockFrameOrSubclass()) {
 | 
						|
    return Some(SynthesizeFallbackBaseline(aWM, aBaselineGroup));
 | 
						|
  }
 | 
						|
 | 
						|
  if (StyleDisplay()->IsContainLayout()) {
 | 
						|
    return Nothing{};
 | 
						|
  }
 | 
						|
 | 
						|
  // OK, here's where we defer to our scrolled frame.
 | 
						|
  return mScrolledFrame
 | 
						|
      ->GetNaturalBaselineBOffset(aWM, aBaselineGroup, aExportContext)
 | 
						|
      .map([this, aWM](nscoord aBaseline) {
 | 
						|
        // We have to add our border BStart thickness to whatever it returns, to
 | 
						|
        // produce an offset in our frame-rect's coordinate system. (We don't
 | 
						|
        // have to add padding, because the scrolled frame handles our padding.)
 | 
						|
        LogicalMargin border = GetLogicalUsedBorder(aWM);
 | 
						|
        const auto bSize = GetLogicalSize(aWM).BSize(aWM);
 | 
						|
        // Clamp the baseline to the border rect. See bug 1791069.
 | 
						|
        return std::clamp(border.BStart(aWM) + aBaseline, 0, bSize);
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AdjustForPerspective(nsRect& aScrollableOverflow) {
 | 
						|
  // If we have perspective that is being applied to our children, then
 | 
						|
  // the effective transform on the child depends on the relative position
 | 
						|
  // of the child to us and changes during scrolling.
 | 
						|
  if (!ChildrenHavePerspective()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  aScrollableOverflow.SetEmpty();
 | 
						|
  GetScrollableOverflowForPerspective(mScrolledFrame, mScrolledFrame,
 | 
						|
                                      ScrollPort(), nsPoint(),
 | 
						|
                                      aScrollableOverflow);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::Reflow(nsPresContext* aPresContext,
 | 
						|
                               ReflowOutput& aDesiredSize,
 | 
						|
                               const ReflowInput& aReflowInput,
 | 
						|
                               nsReflowStatus& aStatus) {
 | 
						|
  MarkInReflow();
 | 
						|
  DO_GLOBAL_REFLOW_COUNT("nsHTMLScrollFrame");
 | 
						|
  DISPLAY_REFLOW(aPresContext, this, aReflowInput, aDesiredSize, aStatus);
 | 
						|
  MOZ_ASSERT(aStatus.IsEmpty(), "Caller should pass a fresh reflow status!");
 | 
						|
 | 
						|
  HandleScrollbarStyleSwitching();
 | 
						|
 | 
						|
  ScrollReflowInput state(this, aReflowInput);
 | 
						|
 | 
						|
  //------------ Handle Incremental Reflow -----------------
 | 
						|
  bool reflowHScrollbar = true;
 | 
						|
  bool reflowVScrollbar = true;
 | 
						|
  bool reflowScrollCorner = true;
 | 
						|
  if (!aReflowInput.ShouldReflowAllKids()) {
 | 
						|
    auto NeedsReflow = [](const nsIFrame* aFrame) {
 | 
						|
      return aFrame && aFrame->IsSubtreeDirty();
 | 
						|
    };
 | 
						|
 | 
						|
    reflowHScrollbar = NeedsReflow(mHScrollbarBox);
 | 
						|
    reflowVScrollbar = NeedsReflow(mVScrollbarBox);
 | 
						|
    reflowScrollCorner =
 | 
						|
        NeedsReflow(mScrollCornerBox) || NeedsReflow(mResizerBox);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    reflowScrollCorner = false;
 | 
						|
  }
 | 
						|
 | 
						|
  const nsRect oldScrollPort = ScrollPort();
 | 
						|
  nsRect oldScrolledAreaBounds =
 | 
						|
      mScrolledFrame->ScrollableOverflowRectRelativeToParent();
 | 
						|
  nsPoint oldScrollPosition = GetScrollPosition();
 | 
						|
 | 
						|
  ReflowContents(state, aDesiredSize);
 | 
						|
 | 
						|
  nsSize layoutSize =
 | 
						|
      mIsUsingMinimumScaleSize ? mMinimumScaleSize : state.mInsideBorderSize;
 | 
						|
  aDesiredSize.Width() = layoutSize.width + state.mComputedBorder.LeftRight();
 | 
						|
  aDesiredSize.Height() = layoutSize.height + state.mComputedBorder.TopBottom();
 | 
						|
 | 
						|
  // Set the size of the frame now since computing the perspective-correct
 | 
						|
  // overflow (within PlaceScrollArea) can rely on it.
 | 
						|
  SetSize(aDesiredSize.GetWritingMode(),
 | 
						|
          aDesiredSize.Size(aDesiredSize.GetWritingMode()));
 | 
						|
 | 
						|
  // Restore the old scroll position, for now, even if that's not valid anymore
 | 
						|
  // because we changed size. We'll fix it up in a post-reflow callback, because
 | 
						|
  // our current size may only be temporary (e.g. we're compute XUL desired
 | 
						|
  // sizes).
 | 
						|
  PlaceScrollArea(state, oldScrollPosition);
 | 
						|
  if (!mPostedReflowCallback) {
 | 
						|
    // Make sure we'll try scrolling to restored position
 | 
						|
    PresShell()->PostReflowCallback(this);
 | 
						|
    mPostedReflowCallback = true;
 | 
						|
  }
 | 
						|
 | 
						|
  bool didOnlyHScrollbar = mOnlyNeedHScrollbarToScrollVVInsideLV;
 | 
						|
  bool didOnlyVScrollbar = mOnlyNeedVScrollbarToScrollVVInsideLV;
 | 
						|
  mOnlyNeedHScrollbarToScrollVVInsideLV =
 | 
						|
      state.mOnlyNeedHScrollbarToScrollVVInsideLV;
 | 
						|
  mOnlyNeedVScrollbarToScrollVVInsideLV =
 | 
						|
      state.mOnlyNeedVScrollbarToScrollVVInsideLV;
 | 
						|
 | 
						|
  bool didHaveHScrollbar = mHasHorizontalScrollbar;
 | 
						|
  bool didHaveVScrollbar = mHasVerticalScrollbar;
 | 
						|
  mHasHorizontalScrollbar = state.mShowHScrollbar;
 | 
						|
  mHasVerticalScrollbar = state.mShowVScrollbar;
 | 
						|
  const nsRect& newScrollPort = ScrollPort();
 | 
						|
  nsRect newScrolledAreaBounds =
 | 
						|
      mScrolledFrame->ScrollableOverflowRectRelativeToParent();
 | 
						|
  if (mSkippedScrollbarLayout || reflowHScrollbar || reflowVScrollbar ||
 | 
						|
      reflowScrollCorner || HasAnyStateBits(NS_FRAME_IS_DIRTY) ||
 | 
						|
      didHaveHScrollbar != state.mShowHScrollbar ||
 | 
						|
      didHaveVScrollbar != state.mShowVScrollbar ||
 | 
						|
      didOnlyHScrollbar != mOnlyNeedHScrollbarToScrollVVInsideLV ||
 | 
						|
      didOnlyVScrollbar != mOnlyNeedVScrollbarToScrollVVInsideLV ||
 | 
						|
      !oldScrollPort.IsEqualEdges(newScrollPort) ||
 | 
						|
      !oldScrolledAreaBounds.IsEqualEdges(newScrolledAreaBounds)) {
 | 
						|
    if (!mSuppressScrollbarUpdate) {
 | 
						|
      mSkippedScrollbarLayout = false;
 | 
						|
      nsHTMLScrollFrame::SetScrollbarVisibility(mHScrollbarBox,
 | 
						|
                                                state.mShowHScrollbar);
 | 
						|
      nsHTMLScrollFrame::SetScrollbarVisibility(mVScrollbarBox,
 | 
						|
                                                state.mShowVScrollbar);
 | 
						|
      // place and reflow scrollbars
 | 
						|
      const nsRect insideBorderArea(
 | 
						|
          nsPoint(state.mComputedBorder.left, state.mComputedBorder.top),
 | 
						|
          layoutSize);
 | 
						|
      LayoutScrollbars(state, insideBorderArea, oldScrollPort);
 | 
						|
    } else {
 | 
						|
      mSkippedScrollbarLayout = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  if (mIsRoot) {
 | 
						|
    if (RefPtr<MobileViewportManager> manager =
 | 
						|
            PresShell()->GetMobileViewportManager()) {
 | 
						|
      // Note that this runs during layout, and when we get here the root
 | 
						|
      // scrollframe has already been laid out. It may have added or removed
 | 
						|
      // scrollbars as a result of that layout, so we need to ensure the
 | 
						|
      // visual viewport is updated to account for that before we read the
 | 
						|
      // visual viewport size.
 | 
						|
      manager->UpdateVisualViewportSizeForPotentialScrollbarChange();
 | 
						|
    } else if (oldScrollPort.Size() != newScrollPort.Size()) {
 | 
						|
      // We want to make sure to send a visual viewport resize event if the
 | 
						|
      // scrollport changed sizes for root scroll frames. The
 | 
						|
      // MobileViewportManager will do that, but if we don't have one (ie we
 | 
						|
      // aren't a root content document for example) we have to send one
 | 
						|
      // ourselves.
 | 
						|
      if (auto* window = nsGlobalWindowInner::Cast(
 | 
						|
              aPresContext->Document()->GetInnerWindow())) {
 | 
						|
        window->VisualViewport()->PostResizeEvent();
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Note that we need to do this after the
 | 
						|
  // UpdateVisualViewportSizeForPotentialScrollbarChange call above because that
 | 
						|
  // is what updates the visual viewport size and we need it to be up to date.
 | 
						|
  if (mIsRoot && !state.OverlayScrollbars() &&
 | 
						|
      (didHaveHScrollbar != state.mShowHScrollbar ||
 | 
						|
       didHaveVScrollbar != state.mShowVScrollbar ||
 | 
						|
       didOnlyHScrollbar != mOnlyNeedHScrollbarToScrollVVInsideLV ||
 | 
						|
       didOnlyVScrollbar != mOnlyNeedVScrollbarToScrollVVInsideLV) &&
 | 
						|
      PresShell()->IsVisualViewportOffsetSet()) {
 | 
						|
    // Removing layout/classic scrollbars can make a previously valid vvoffset
 | 
						|
    // invalid. For example, if we are zoomed in on an overflow hidden document
 | 
						|
    // and then zoom back out, when apz reaches the initial resolution (ie 1.0)
 | 
						|
    // it won't know that we can remove the scrollbars, so the vvoffset can
 | 
						|
    // validly be upto the width/height of the scrollbars. After we reflow and
 | 
						|
    // remove the scrollbars the only valid vvoffset is (0,0). We could wait
 | 
						|
    // until we send the new frame metrics to apz and then have it reply with
 | 
						|
    // the new corrected vvoffset but having an inconsistent vvoffset causes
 | 
						|
    // problems so trigger the vvoffset to be re-set and re-clamped in
 | 
						|
    // ReflowFinished.
 | 
						|
    mReclampVVOffsetInReflowFinished = true;
 | 
						|
  }
 | 
						|
 | 
						|
  aDesiredSize.SetOverflowAreasToDesiredBounds();
 | 
						|
 | 
						|
  UpdateSticky();
 | 
						|
  FinishReflowWithAbsoluteFrames(aPresContext, aDesiredSize, aReflowInput,
 | 
						|
                                 aStatus);
 | 
						|
 | 
						|
  if (!InInitialReflow() && !mHadNonInitialReflow) {
 | 
						|
    mHadNonInitialReflow = true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mIsRoot && !oldScrolledAreaBounds.IsEqualEdges(newScrolledAreaBounds)) {
 | 
						|
    PostScrolledAreaEvent();
 | 
						|
  }
 | 
						|
 | 
						|
  UpdatePrevScrolledRect();
 | 
						|
 | 
						|
  aStatus.Reset();  // This type of frame can't be split.
 | 
						|
  PostOverflowEvent();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::DidReflow(nsPresContext* aPresContext,
 | 
						|
                                  const ReflowInput* aReflowInput) {
 | 
						|
  nsContainerFrame::DidReflow(aPresContext, aReflowInput);
 | 
						|
  if (NeedsResnap()) {
 | 
						|
    PostPendingResnap();
 | 
						|
  } else {
 | 
						|
    PresShell()->PostPendingScrollAnchorAdjustment(Anchor());
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////////////////////////
 | 
						|
 | 
						|
#ifdef DEBUG_FRAME_DUMP
 | 
						|
nsresult nsHTMLScrollFrame::GetFrameName(nsAString& aResult) const {
 | 
						|
  return MakeFrameName(u"HTMLScroll"_ns, aResult);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef ACCESSIBILITY
 | 
						|
a11y::AccType nsHTMLScrollFrame::AccessibleType() {
 | 
						|
  if (IsTableCaption()) {
 | 
						|
    return GetRect().IsEmpty() ? a11y::eNoType : a11y::eHTMLCaptionType;
 | 
						|
  }
 | 
						|
 | 
						|
  // Create an accessible regardless of focusable state because the state can be
 | 
						|
  // changed during frame life cycle without any notifications to accessibility.
 | 
						|
  if (mContent->IsRootOfNativeAnonymousSubtree() ||
 | 
						|
      GetScrollStyles().IsHiddenInBothDirections()) {
 | 
						|
    return a11y::eNoType;
 | 
						|
  }
 | 
						|
 | 
						|
  return a11y::eHyperTextType;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
NS_QUERYFRAME_HEAD(nsHTMLScrollFrame)
 | 
						|
  NS_QUERYFRAME_ENTRY(nsIAnonymousContentCreator)
 | 
						|
  NS_QUERYFRAME_ENTRY(nsIScrollableFrame)
 | 
						|
  NS_QUERYFRAME_ENTRY(nsIStatefulFrame)
 | 
						|
  NS_QUERYFRAME_ENTRY(nsIScrollbarMediator)
 | 
						|
  NS_QUERYFRAME_ENTRY(nsHTMLScrollFrame)
 | 
						|
NS_QUERYFRAME_TAIL_INHERITING(nsContainerFrame)
 | 
						|
 | 
						|
nsMargin nsHTMLScrollFrame::GetDesiredScrollbarSizes() const {
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
  if (pc->UseOverlayScrollbars()) {
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  const auto& style = *nsLayoutUtils::StyleForScrollbar(this);
 | 
						|
  const auto scrollbarWidth = style.StyleUIReset()->ScrollbarWidth();
 | 
						|
  if (scrollbarWidth == StyleScrollbarWidth::None) {
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  ScrollStyles styles = GetScrollStyles();
 | 
						|
  nsMargin result(0, 0, 0, 0);
 | 
						|
 | 
						|
  auto size = GetNonOverlayScrollbarSize(pc, scrollbarWidth);
 | 
						|
  if (styles.mVertical != StyleOverflow::Hidden) {
 | 
						|
    if (IsScrollbarOnRight()) {
 | 
						|
      result.right = size;
 | 
						|
    } else {
 | 
						|
      result.left = size;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (styles.mHorizontal != StyleOverflow::Hidden) {
 | 
						|
    // We don't currently support any scripts that would require a scrollbar
 | 
						|
    // at the top. (Are there any?)
 | 
						|
    result.bottom = size;
 | 
						|
  }
 | 
						|
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsHTMLScrollFrame::GetNonOverlayScrollbarSize(
 | 
						|
    const nsPresContext* aPc, StyleScrollbarWidth aScrollbarWidth) {
 | 
						|
  const auto size = aPc->Theme()->GetScrollbarSize(aPc, aScrollbarWidth,
 | 
						|
                                                   nsITheme::Overlay::No);
 | 
						|
  return aPc->DevPixelsToAppUnits(size);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::HandleScrollbarStyleSwitching() {
 | 
						|
  // Check if we switched between scrollbar styles.
 | 
						|
  if (mScrollbarActivity && !UsesOverlayScrollbars()) {
 | 
						|
    mScrollbarActivity->Destroy();
 | 
						|
    mScrollbarActivity = nullptr;
 | 
						|
  } else if (!mScrollbarActivity && UsesOverlayScrollbars()) {
 | 
						|
    mScrollbarActivity = new ScrollbarActivity(do_QueryFrame(this));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
static bool IsFocused(nsIContent* aContent) {
 | 
						|
  // Some content elements, like the GetContent() of a scroll frame
 | 
						|
  // for a text input field, are inside anonymous subtrees, but the focus
 | 
						|
  // manager always reports a non-anonymous element as the focused one, so
 | 
						|
  // walk up the tree until we reach a non-anonymous element.
 | 
						|
  while (aContent && aContent->IsInNativeAnonymousSubtree()) {
 | 
						|
    aContent = aContent->GetParent();
 | 
						|
  }
 | 
						|
 | 
						|
  return aContent ? nsContentUtils::IsFocusedContent(aContent) : false;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetScrollableByAPZ(bool aScrollable) {
 | 
						|
  mScrollableByAPZ = aScrollable;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetZoomableByAPZ(bool aZoomable) {
 | 
						|
  if (!nsLayoutUtils::UsesAsyncScrolling(this)) {
 | 
						|
    // If APZ is disabled on this window, then we're never actually going to
 | 
						|
    // do any zooming. So we don't need to do any of the setup for it. Note
 | 
						|
    // that this function gets called from ZoomConstraintsClient even if APZ
 | 
						|
    // is disabled to indicate the zoomability of content.
 | 
						|
    aZoomable = false;
 | 
						|
  }
 | 
						|
  if (mZoomableByAPZ != aZoomable) {
 | 
						|
    // We might be changing the result of DecideScrollableLayer() so schedule a
 | 
						|
    // paint to make sure we pick up the result of that change.
 | 
						|
    mZoomableByAPZ = aZoomable;
 | 
						|
    SchedulePaint();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetHasOutOfFlowContentInsideFilter() {
 | 
						|
  mHasOutOfFlowContentInsideFilter = true;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::WantAsyncScroll() const {
 | 
						|
  ScrollStyles styles = GetScrollStyles();
 | 
						|
 | 
						|
  // First, as an optimization because getting the scrollrange is
 | 
						|
  // relatively slow, check overflow hidden and not a zoomed scroll frame.
 | 
						|
  if (styles.mHorizontal == StyleOverflow::Hidden &&
 | 
						|
      styles.mVertical == StyleOverflow::Hidden) {
 | 
						|
    if (!mIsRoot || GetVisualViewportSize() == mScrollPort.Size()) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nscoord oneDevPixel =
 | 
						|
      GetScrolledFrame()->PresContext()->AppUnitsPerDevPixel();
 | 
						|
  nsRect scrollRange = GetLayoutScrollRange();
 | 
						|
 | 
						|
  bool isVScrollable = (scrollRange.height >= oneDevPixel) &&
 | 
						|
                       (styles.mVertical != StyleOverflow::Hidden);
 | 
						|
  bool isHScrollable = (scrollRange.width >= oneDevPixel) &&
 | 
						|
                       (styles.mHorizontal != StyleOverflow::Hidden);
 | 
						|
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
  // Mobile platforms need focus to scroll text inputs.
 | 
						|
  bool canScrollWithoutScrollbars =
 | 
						|
      !IsForTextControlWithNoScrollbars() || IsFocused(GetContent());
 | 
						|
#else
 | 
						|
  bool canScrollWithoutScrollbars = true;
 | 
						|
#endif
 | 
						|
 | 
						|
  // The check for scroll bars was added in bug 825692 to prevent layerization
 | 
						|
  // of text inputs for performance reasons.
 | 
						|
  bool isVAsyncScrollable =
 | 
						|
      isVScrollable && (mVScrollbarBox || canScrollWithoutScrollbars);
 | 
						|
  bool isHAsyncScrollable =
 | 
						|
      isHScrollable && (mHScrollbarBox || canScrollWithoutScrollbars);
 | 
						|
  if (isVAsyncScrollable || isHAsyncScrollable) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // If the page has a visual viewport size that's different from
 | 
						|
  // the layout viewport size at the current zoom level, we need to be
 | 
						|
  // able to scroll the visual viewport inside the layout viewport
 | 
						|
  // even if the page is not zoomable.
 | 
						|
  return mIsRoot && GetVisualViewportSize() != mScrollPort.Size() &&
 | 
						|
         !GetVisualScrollRange().IsEqualInterior(scrollRange);
 | 
						|
}
 | 
						|
 | 
						|
static nsRect GetOnePixelRangeAroundPoint(const nsPoint& aPoint,
 | 
						|
                                          bool aIsHorizontal) {
 | 
						|
  nsRect allowedRange(aPoint, nsSize());
 | 
						|
  nscoord halfPixel = nsPresContext::CSSPixelsToAppUnits(0.5f);
 | 
						|
  if (aIsHorizontal) {
 | 
						|
    allowedRange.x = aPoint.x - halfPixel;
 | 
						|
    allowedRange.width = halfPixel * 2 - 1;
 | 
						|
  } else {
 | 
						|
    allowedRange.y = aPoint.y - halfPixel;
 | 
						|
    allowedRange.height = halfPixel * 2 - 1;
 | 
						|
  }
 | 
						|
  return allowedRange;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollByPage(nsScrollbarFrame* aScrollbar,
 | 
						|
                                     int32_t aDirection,
 | 
						|
                                     ScrollSnapFlags aSnapFlags) {
 | 
						|
  ScrollByUnit(aScrollbar, ScrollMode::Smooth, aDirection, ScrollUnit::PAGES,
 | 
						|
               aSnapFlags);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollByWhole(nsScrollbarFrame* aScrollbar,
 | 
						|
                                      int32_t aDirection,
 | 
						|
                                      ScrollSnapFlags aSnapFlags) {
 | 
						|
  ScrollByUnit(aScrollbar, ScrollMode::Instant, aDirection, ScrollUnit::WHOLE,
 | 
						|
               aSnapFlags);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollByLine(nsScrollbarFrame* aScrollbar,
 | 
						|
                                     int32_t aDirection,
 | 
						|
                                     ScrollSnapFlags aSnapFlags) {
 | 
						|
  bool isHorizontal = aScrollbar->IsHorizontal();
 | 
						|
  nsIntPoint delta;
 | 
						|
  if (isHorizontal) {
 | 
						|
    const double kScrollMultiplier =
 | 
						|
        StaticPrefs::toolkit_scrollbox_horizontalScrollDistance();
 | 
						|
    delta.x = static_cast<int32_t>(aDirection * kScrollMultiplier);
 | 
						|
    if (GetLineScrollAmount().width * delta.x > GetPageScrollAmount().width) {
 | 
						|
      // The scroll frame is so small that the delta would be more
 | 
						|
      // than an entire page.  Scroll by one page instead to maintain
 | 
						|
      // context.
 | 
						|
      ScrollByPage(aScrollbar, aDirection);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    const double kScrollMultiplier =
 | 
						|
        StaticPrefs::toolkit_scrollbox_verticalScrollDistance();
 | 
						|
    delta.y = static_cast<int32_t>(aDirection * kScrollMultiplier);
 | 
						|
    if (GetLineScrollAmount().height * delta.y > GetPageScrollAmount().height) {
 | 
						|
      // The scroll frame is so small that the delta would be more
 | 
						|
      // than an entire page.  Scroll by one page instead to maintain
 | 
						|
      // context.
 | 
						|
      ScrollByPage(aScrollbar, aDirection);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nsIntPoint overflow;
 | 
						|
  ScrollBy(delta, ScrollUnit::LINES, ScrollMode::Smooth, &overflow,
 | 
						|
           ScrollOrigin::Other, nsIScrollableFrame::NOT_MOMENTUM, aSnapFlags);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::RepeatButtonScroll(nsScrollbarFrame* aScrollbar) {
 | 
						|
  aScrollbar->MoveToNewPosition(nsScrollbarFrame::ImplementsScrollByUnit::Yes);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ThumbMoved(nsScrollbarFrame* aScrollbar,
 | 
						|
                                   nscoord aOldPos, nscoord aNewPos) {
 | 
						|
  MOZ_ASSERT(aScrollbar != nullptr);
 | 
						|
  bool isHorizontal = aScrollbar->IsHorizontal();
 | 
						|
  nsPoint current = GetScrollPosition();
 | 
						|
  nsPoint dest = current;
 | 
						|
  if (isHorizontal) {
 | 
						|
    dest.x = IsPhysicalLTR() ? aNewPos : aNewPos - GetLayoutScrollRange().width;
 | 
						|
  } else {
 | 
						|
    dest.y = aNewPos;
 | 
						|
  }
 | 
						|
  nsRect allowedRange = GetOnePixelRangeAroundPoint(dest, isHorizontal);
 | 
						|
 | 
						|
  // Don't try to scroll if we're already at an acceptable place.
 | 
						|
  // Don't call Contains here since Contains returns false when the point is
 | 
						|
  // on the bottom or right edge of the rectangle.
 | 
						|
  if (allowedRange.ClampPoint(current) == current) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      dest, &allowedRange,
 | 
						|
      ScrollOperationParams{ScrollMode::Instant, ScrollOrigin::Other});
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollbarReleased(nsScrollbarFrame* aScrollbar) {
 | 
						|
  // Scrollbar scrolling does not result in fling gestures, clear any
 | 
						|
  // accumulated velocity
 | 
						|
  mVelocityQueue.Reset();
 | 
						|
 | 
						|
  // Perform scroll snapping, if needed.  Scrollbar movement uses the same
 | 
						|
  // smooth scrolling animation as keyboard scrolling.
 | 
						|
  ScrollSnap(mDestination, ScrollMode::Smooth);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollByUnit(nsScrollbarFrame* aScrollbar,
 | 
						|
                                     ScrollMode aMode, int32_t aDirection,
 | 
						|
                                     ScrollUnit aUnit,
 | 
						|
                                     ScrollSnapFlags aSnapFlags) {
 | 
						|
  MOZ_ASSERT(aScrollbar != nullptr);
 | 
						|
  bool isHorizontal = aScrollbar->IsHorizontal();
 | 
						|
  nsIntPoint delta;
 | 
						|
  if (isHorizontal) {
 | 
						|
    delta.x = aDirection;
 | 
						|
  } else {
 | 
						|
    delta.y = aDirection;
 | 
						|
  }
 | 
						|
  nsIntPoint overflow;
 | 
						|
  ScrollBy(delta, aUnit, aMode, &overflow, ScrollOrigin::Other,
 | 
						|
           nsIScrollableFrame::NOT_MOMENTUM, aSnapFlags);
 | 
						|
}
 | 
						|
 | 
						|
//-------------------- Helper ----------------------
 | 
						|
 | 
						|
// AsyncSmoothMSDScroll has ref counting.
 | 
						|
class nsHTMLScrollFrame::AsyncSmoothMSDScroll final
 | 
						|
    : public nsARefreshObserver {
 | 
						|
 public:
 | 
						|
  AsyncSmoothMSDScroll(const nsPoint& aInitialPosition,
 | 
						|
                       const nsPoint& aInitialDestination,
 | 
						|
                       const nsSize& aInitialVelocity, const nsRect& aRange,
 | 
						|
                       const mozilla::TimeStamp& aStartTime,
 | 
						|
                       nsPresContext* aPresContext,
 | 
						|
                       UniquePtr<ScrollSnapTargetIds> aSnapTargetIds,
 | 
						|
                       ScrollTriggeredByScript aTriggeredByScript)
 | 
						|
      : mXAxisModel(aInitialPosition.x, aInitialDestination.x,
 | 
						|
                    aInitialVelocity.width,
 | 
						|
                    StaticPrefs::layout_css_scroll_behavior_spring_constant(),
 | 
						|
                    StaticPrefs::layout_css_scroll_behavior_damping_ratio()),
 | 
						|
        mYAxisModel(aInitialPosition.y, aInitialDestination.y,
 | 
						|
                    aInitialVelocity.height,
 | 
						|
                    StaticPrefs::layout_css_scroll_behavior_spring_constant(),
 | 
						|
                    StaticPrefs::layout_css_scroll_behavior_damping_ratio()),
 | 
						|
        mRange(aRange),
 | 
						|
        mStartPosition(aInitialPosition),
 | 
						|
        mLastRefreshTime(aStartTime),
 | 
						|
        mCallee(nullptr),
 | 
						|
        mOneDevicePixelInAppUnits(aPresContext->DevPixelsToAppUnits(1)),
 | 
						|
        mSnapTargetIds(std::move(aSnapTargetIds)),
 | 
						|
        mTriggeredByScript(aTriggeredByScript) {
 | 
						|
    Telemetry::SetHistogramRecordingEnabled(
 | 
						|
        Telemetry::FX_REFRESH_DRIVER_SYNC_SCROLL_FRAME_DELAY_MS, true);
 | 
						|
  }
 | 
						|
 | 
						|
  NS_INLINE_DECL_REFCOUNTING(AsyncSmoothMSDScroll, override)
 | 
						|
 | 
						|
  nsSize GetVelocity() {
 | 
						|
    // In nscoords per second
 | 
						|
    return nsSize(mXAxisModel.GetVelocity(), mYAxisModel.GetVelocity());
 | 
						|
  }
 | 
						|
 | 
						|
  nsPoint GetPosition() {
 | 
						|
    // In nscoords
 | 
						|
    return nsPoint(NSToCoordRound(mXAxisModel.GetPosition()),
 | 
						|
                   NSToCoordRound(mYAxisModel.GetPosition()));
 | 
						|
  }
 | 
						|
 | 
						|
  void SetDestination(const nsPoint& aDestination,
 | 
						|
                      ScrollTriggeredByScript aTriggeredByScript) {
 | 
						|
    mXAxisModel.SetDestination(static_cast<int32_t>(aDestination.x));
 | 
						|
    mYAxisModel.SetDestination(static_cast<int32_t>(aDestination.y));
 | 
						|
    mTriggeredByScript = aTriggeredByScript;
 | 
						|
  }
 | 
						|
 | 
						|
  void SetRange(const nsRect& aRange) { mRange = aRange; }
 | 
						|
 | 
						|
  nsRect GetRange() { return mRange; }
 | 
						|
 | 
						|
  nsPoint GetStartPosition() { return mStartPosition; }
 | 
						|
 | 
						|
  void Simulate(const TimeDuration& aDeltaTime) {
 | 
						|
    mXAxisModel.Simulate(aDeltaTime);
 | 
						|
    mYAxisModel.Simulate(aDeltaTime);
 | 
						|
 | 
						|
    nsPoint desired = GetPosition();
 | 
						|
    nsPoint clamped = mRange.ClampPoint(desired);
 | 
						|
    if (desired.x != clamped.x) {
 | 
						|
      // The scroll has hit the "wall" at the left or right edge of the allowed
 | 
						|
      // scroll range.
 | 
						|
      // Absorb the impact to avoid bounceback effect.
 | 
						|
      mXAxisModel.SetVelocity(0.0);
 | 
						|
      mXAxisModel.SetPosition(clamped.x);
 | 
						|
    }
 | 
						|
 | 
						|
    if (desired.y != clamped.y) {
 | 
						|
      // The scroll has hit the "wall" at the left or right edge of the allowed
 | 
						|
      // scroll range.
 | 
						|
      // Absorb the impact to avoid bounceback effect.
 | 
						|
      mYAxisModel.SetVelocity(0.0);
 | 
						|
      mYAxisModel.SetPosition(clamped.y);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  bool IsFinished() {
 | 
						|
    return mXAxisModel.IsFinished(mOneDevicePixelInAppUnits) &&
 | 
						|
           mYAxisModel.IsFinished(mOneDevicePixelInAppUnits);
 | 
						|
  }
 | 
						|
 | 
						|
  virtual void WillRefresh(mozilla::TimeStamp aTime) override {
 | 
						|
    mozilla::TimeDuration deltaTime = aTime - mLastRefreshTime;
 | 
						|
    mLastRefreshTime = aTime;
 | 
						|
 | 
						|
    // The callback may release "this".
 | 
						|
    // We don't access members after returning, so no need for KungFuDeathGrip.
 | 
						|
    nsHTMLScrollFrame::AsyncSmoothMSDScrollCallback(mCallee, deltaTime);
 | 
						|
  }
 | 
						|
 | 
						|
  /*
 | 
						|
   * Set a refresh observer for smooth scroll iterations (and start observing).
 | 
						|
   * Should be used at most once during the lifetime of this object.
 | 
						|
   */
 | 
						|
  void SetRefreshObserver(nsHTMLScrollFrame* aCallee) {
 | 
						|
    MOZ_ASSERT(aCallee,
 | 
						|
               "AsyncSmoothMSDScroll::SetRefreshObserver needs "
 | 
						|
               "a non-null aCallee in order to get a refresh driver");
 | 
						|
    MOZ_RELEASE_ASSERT(!mCallee,
 | 
						|
                       "AsyncSmoothMSDScroll::SetRefreshObserver "
 | 
						|
                       "shouldn't be called if we're already registered with "
 | 
						|
                       "a refresh driver, via a preexisting mCallee");
 | 
						|
 | 
						|
    RefreshDriver(aCallee)->AddRefreshObserver(this, FlushType::Style,
 | 
						|
                                               "Smooth scroll (MSD) animation");
 | 
						|
    mCallee = aCallee;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * The mCallee holds a strong ref to us since the refresh driver doesn't.
 | 
						|
   * Our dtor and mCallee's Destroy() method both call RemoveObserver() -
 | 
						|
   * whichever comes first removes us from the refresh driver.
 | 
						|
   */
 | 
						|
  void RemoveObserver() {
 | 
						|
    if (mCallee) {
 | 
						|
      RefreshDriver(mCallee)->RemoveRefreshObserver(this, FlushType::Style);
 | 
						|
      mCallee = nullptr;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  UniquePtr<ScrollSnapTargetIds> TakeSnapTargetIds() {
 | 
						|
    return std::move(mSnapTargetIds);
 | 
						|
  }
 | 
						|
 | 
						|
  bool WasTriggeredByScript() const {
 | 
						|
    return mTriggeredByScript == ScrollTriggeredByScript::Yes;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  // Private destructor, to discourage deletion outside of Release():
 | 
						|
  ~AsyncSmoothMSDScroll() {
 | 
						|
    RemoveObserver();
 | 
						|
    Telemetry::SetHistogramRecordingEnabled(
 | 
						|
        Telemetry::FX_REFRESH_DRIVER_SYNC_SCROLL_FRAME_DELAY_MS, false);
 | 
						|
  }
 | 
						|
 | 
						|
  nsRefreshDriver* RefreshDriver(nsHTMLScrollFrame* aCallee) {
 | 
						|
    return aCallee->PresContext()->RefreshDriver();
 | 
						|
  }
 | 
						|
 | 
						|
  mozilla::layers::AxisPhysicsMSDModel mXAxisModel, mYAxisModel;
 | 
						|
  nsRect mRange;
 | 
						|
  nsPoint mStartPosition;
 | 
						|
  mozilla::TimeStamp mLastRefreshTime;
 | 
						|
  nsHTMLScrollFrame* mCallee;
 | 
						|
  nscoord mOneDevicePixelInAppUnits;
 | 
						|
  UniquePtr<ScrollSnapTargetIds> mSnapTargetIds;
 | 
						|
  ScrollTriggeredByScript mTriggeredByScript;
 | 
						|
};
 | 
						|
 | 
						|
// AsyncScroll has ref counting.
 | 
						|
class nsHTMLScrollFrame::AsyncScroll final : public nsARefreshObserver {
 | 
						|
 public:
 | 
						|
  typedef mozilla::TimeStamp TimeStamp;
 | 
						|
  typedef mozilla::TimeDuration TimeDuration;
 | 
						|
 | 
						|
  explicit AsyncScroll(UniquePtr<ScrollSnapTargetIds> aSnapTargetIds,
 | 
						|
                       ScrollTriggeredByScript aTriggeredByScript)
 | 
						|
      : mOrigin(ScrollOrigin::NotSpecified),
 | 
						|
        mCallee(nullptr),
 | 
						|
        mSnapTargetIds(std::move(aSnapTargetIds)),
 | 
						|
        mTriggeredByScript(aTriggeredByScript) {
 | 
						|
    Telemetry::SetHistogramRecordingEnabled(
 | 
						|
        Telemetry::FX_REFRESH_DRIVER_SYNC_SCROLL_FRAME_DELAY_MS, true);
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  // Private destructor, to discourage deletion outside of Release():
 | 
						|
  ~AsyncScroll() {
 | 
						|
    RemoveObserver();
 | 
						|
    Telemetry::SetHistogramRecordingEnabled(
 | 
						|
        Telemetry::FX_REFRESH_DRIVER_SYNC_SCROLL_FRAME_DELAY_MS, false);
 | 
						|
  }
 | 
						|
 | 
						|
 public:
 | 
						|
  void InitSmoothScroll(TimeStamp aTime, nsPoint aInitialPosition,
 | 
						|
                        nsPoint aDestination, ScrollOrigin aOrigin,
 | 
						|
                        const nsRect& aRange, const nsSize& aCurrentVelocity);
 | 
						|
  void Init(nsPoint aInitialPosition, const nsRect& aRange) {
 | 
						|
    mAnimationPhysics = nullptr;
 | 
						|
    mRange = aRange;
 | 
						|
    mStartPosition = aInitialPosition;
 | 
						|
  }
 | 
						|
 | 
						|
  bool IsSmoothScroll() { return mAnimationPhysics != nullptr; }
 | 
						|
 | 
						|
  bool IsFinished(const TimeStamp& aTime) const {
 | 
						|
    MOZ_RELEASE_ASSERT(mAnimationPhysics);
 | 
						|
    return mAnimationPhysics->IsFinished(aTime);
 | 
						|
  }
 | 
						|
 | 
						|
  nsPoint PositionAt(const TimeStamp& aTime) const {
 | 
						|
    MOZ_RELEASE_ASSERT(mAnimationPhysics);
 | 
						|
    return mAnimationPhysics->PositionAt(aTime);
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize VelocityAt(const TimeStamp& aTime) const {
 | 
						|
    MOZ_RELEASE_ASSERT(mAnimationPhysics);
 | 
						|
    return mAnimationPhysics->VelocityAt(aTime);
 | 
						|
  }
 | 
						|
 | 
						|
  nsPoint GetStartPosition() const { return mStartPosition; }
 | 
						|
 | 
						|
  // Most recent scroll origin.
 | 
						|
  ScrollOrigin mOrigin;
 | 
						|
 | 
						|
  // Allowed destination positions around mDestination
 | 
						|
  nsRect mRange;
 | 
						|
 | 
						|
  // Initial position where the async scroll was triggered.
 | 
						|
  nsPoint mStartPosition;
 | 
						|
 | 
						|
 private:
 | 
						|
  void InitPreferences(TimeStamp aTime, nsAtom* aOrigin);
 | 
						|
 | 
						|
  UniquePtr<ScrollAnimationPhysics> mAnimationPhysics;
 | 
						|
 | 
						|
  // The next section is observer/callback management
 | 
						|
  // Bodies of WillRefresh and RefreshDriver contain nsHTMLScrollFrame specific
 | 
						|
  // code.
 | 
						|
 public:
 | 
						|
  NS_INLINE_DECL_REFCOUNTING(AsyncScroll, override)
 | 
						|
 | 
						|
  /*
 | 
						|
   * Set a refresh observer for smooth scroll iterations (and start observing).
 | 
						|
   * Should be used at most once during the lifetime of this object.
 | 
						|
   */
 | 
						|
  void SetRefreshObserver(nsHTMLScrollFrame* aCallee) {
 | 
						|
    MOZ_ASSERT(aCallee,
 | 
						|
               "AsyncScroll::SetRefreshObserver needs "
 | 
						|
               "a non-null aCallee in order to get a refresh driver");
 | 
						|
    MOZ_RELEASE_ASSERT(!mCallee,
 | 
						|
                       "AsyncScroll::SetRefreshObserver "
 | 
						|
                       "shouldn't be called if we're already registered with "
 | 
						|
                       "a refresh driver, via a preexisting mCallee");
 | 
						|
 | 
						|
    RefreshDriver(aCallee)->AddRefreshObserver(this, FlushType::Style,
 | 
						|
                                               "Smooth scroll animation");
 | 
						|
    mCallee = aCallee;
 | 
						|
    auto* presShell = mCallee->PresShell();
 | 
						|
    MOZ_ASSERT(presShell);
 | 
						|
    presShell->SuppressDisplayport(true);
 | 
						|
  }
 | 
						|
 | 
						|
  virtual void WillRefresh(mozilla::TimeStamp aTime) override {
 | 
						|
    // The callback may release "this".
 | 
						|
    // We don't access members after returning, so no need for KungFuDeathGrip.
 | 
						|
    nsHTMLScrollFrame::AsyncScrollCallback(mCallee, aTime);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * The mCallee holds a strong ref to us since the refresh driver doesn't.
 | 
						|
   * Our dtor and mCallee's Destroy() method both call RemoveObserver() -
 | 
						|
   * whichever comes first removes us from the refresh driver.
 | 
						|
   */
 | 
						|
  void RemoveObserver() {
 | 
						|
    if (mCallee) {
 | 
						|
      RefreshDriver(mCallee)->RemoveRefreshObserver(this, FlushType::Style);
 | 
						|
      auto* presShell = mCallee->PresShell();
 | 
						|
      MOZ_ASSERT(presShell);
 | 
						|
      presShell->SuppressDisplayport(false);
 | 
						|
      mCallee = nullptr;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  UniquePtr<ScrollSnapTargetIds> TakeSnapTargetIds() {
 | 
						|
    return std::move(mSnapTargetIds);
 | 
						|
  }
 | 
						|
 | 
						|
  bool WasTriggeredByScript() const {
 | 
						|
    return mTriggeredByScript == ScrollTriggeredByScript::Yes;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mCallee;
 | 
						|
  UniquePtr<ScrollSnapTargetIds> mSnapTargetIds;
 | 
						|
  ScrollTriggeredByScript mTriggeredByScript;
 | 
						|
 | 
						|
  nsRefreshDriver* RefreshDriver(nsHTMLScrollFrame* aCallee) {
 | 
						|
    return aCallee->PresContext()->RefreshDriver();
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AsyncScroll::InitSmoothScroll(
 | 
						|
    TimeStamp aTime, nsPoint aInitialPosition, nsPoint aDestination,
 | 
						|
    ScrollOrigin aOrigin, const nsRect& aRange,
 | 
						|
    const nsSize& aCurrentVelocity) {
 | 
						|
  switch (aOrigin) {
 | 
						|
    case ScrollOrigin::NotSpecified:
 | 
						|
    case ScrollOrigin::Restore:
 | 
						|
    case ScrollOrigin::Relative:
 | 
						|
      // We don't have special prefs for "restore", just treat it as "other".
 | 
						|
      // "restore" scrolls are (for now) always instant anyway so unless
 | 
						|
      // something changes we should never have aOrigin ==
 | 
						|
      // ScrollOrigin::Restore here.
 | 
						|
      aOrigin = ScrollOrigin::Other;
 | 
						|
      break;
 | 
						|
    case ScrollOrigin::Apz:
 | 
						|
      // Likewise we should never get APZ-triggered scrolls here, and if that
 | 
						|
      // changes something is likely broken somewhere.
 | 
						|
      MOZ_ASSERT(false);
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      break;
 | 
						|
  };
 | 
						|
 | 
						|
  // Read preferences only on first iteration or for a different event origin.
 | 
						|
  if (!mAnimationPhysics || aOrigin != mOrigin) {
 | 
						|
    mOrigin = aOrigin;
 | 
						|
    if (StaticPrefs::general_smoothScroll_msdPhysics_enabled()) {
 | 
						|
      mAnimationPhysics =
 | 
						|
          MakeUnique<ScrollAnimationMSDPhysics>(aInitialPosition);
 | 
						|
    } else {
 | 
						|
      ScrollAnimationBezierPhysicsSettings settings =
 | 
						|
          layers::apz::ComputeBezierAnimationSettingsForOrigin(mOrigin);
 | 
						|
      mAnimationPhysics =
 | 
						|
          MakeUnique<ScrollAnimationBezierPhysics>(aInitialPosition, settings);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mStartPosition = aInitialPosition;
 | 
						|
  mRange = aRange;
 | 
						|
 | 
						|
  mAnimationPhysics->Update(aTime, aDestination, aCurrentVelocity);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Callback function from AsyncSmoothMSDScroll, used in
 | 
						|
 * nsHTMLScrollFrame::ScrollTo
 | 
						|
 */
 | 
						|
void nsHTMLScrollFrame::AsyncSmoothMSDScrollCallback(
 | 
						|
    nsHTMLScrollFrame* aInstance, mozilla::TimeDuration aDeltaTime) {
 | 
						|
  NS_ASSERTION(aInstance != nullptr, "aInstance must not be null");
 | 
						|
  NS_ASSERTION(aInstance->mAsyncSmoothMSDScroll,
 | 
						|
               "Did not expect AsyncSmoothMSDScrollCallback without an active "
 | 
						|
               "MSD scroll.");
 | 
						|
 | 
						|
  nsRect range = aInstance->mAsyncSmoothMSDScroll->GetRange();
 | 
						|
  aInstance->mAsyncSmoothMSDScroll->Simulate(aDeltaTime);
 | 
						|
 | 
						|
  if (!aInstance->mAsyncSmoothMSDScroll->IsFinished()) {
 | 
						|
    nsPoint destination = aInstance->mAsyncSmoothMSDScroll->GetPosition();
 | 
						|
    // Allow this scroll operation to land on any pixel boundary within the
 | 
						|
    // allowed scroll range for this frame.
 | 
						|
    // If the MSD is under-dampened or the destination is changed rapidly,
 | 
						|
    // it is expected (and desired) that the scrolling may overshoot.
 | 
						|
    nsRect intermediateRange = nsRect(destination, nsSize()).UnionEdges(range);
 | 
						|
    aInstance->ScrollToImpl(destination, intermediateRange);
 | 
						|
    // 'aInstance' might be destroyed here
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  aInstance->CompleteAsyncScroll(
 | 
						|
      aInstance->mAsyncSmoothMSDScroll->GetStartPosition(), range,
 | 
						|
      aInstance->mAsyncSmoothMSDScroll->TakeSnapTargetIds());
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Callback function from AsyncScroll, used in nsHTMLScrollFrame::ScrollTo
 | 
						|
 */
 | 
						|
void nsHTMLScrollFrame::AsyncScrollCallback(nsHTMLScrollFrame* aInstance,
 | 
						|
                                            mozilla::TimeStamp aTime) {
 | 
						|
  MOZ_ASSERT(aInstance != nullptr, "aInstance must not be null");
 | 
						|
  MOZ_ASSERT(
 | 
						|
      aInstance->mAsyncScroll,
 | 
						|
      "Did not expect AsyncScrollCallback without an active async scroll.");
 | 
						|
 | 
						|
  if (!aInstance || !aInstance->mAsyncScroll) {
 | 
						|
    return;  // XXX wallpaper bug 1107353 for now.
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect range = aInstance->mAsyncScroll->mRange;
 | 
						|
  if (aInstance->mAsyncScroll->IsSmoothScroll()) {
 | 
						|
    if (!aInstance->mAsyncScroll->IsFinished(aTime)) {
 | 
						|
      nsPoint destination = aInstance->mAsyncScroll->PositionAt(aTime);
 | 
						|
      // Allow this scroll operation to land on any pixel boundary between the
 | 
						|
      // current position and the final allowed range.  (We don't want
 | 
						|
      // intermediate steps to be more constrained than the final step!)
 | 
						|
      nsRect intermediateRange =
 | 
						|
          nsRect(aInstance->GetScrollPosition(), nsSize()).UnionEdges(range);
 | 
						|
      aInstance->ScrollToImpl(destination, intermediateRange);
 | 
						|
      // 'aInstance' might be destroyed here
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  aInstance->CompleteAsyncScroll(aInstance->mAsyncScroll->GetStartPosition(),
 | 
						|
                                 range,
 | 
						|
                                 aInstance->mAsyncScroll->TakeSnapTargetIds());
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetTransformingByAPZ(bool aTransforming) {
 | 
						|
  if (mTransformingByAPZ && !aTransforming) {
 | 
						|
    PostScrollEndEvent();
 | 
						|
  }
 | 
						|
  mTransformingByAPZ = aTransforming;
 | 
						|
  if (!mozilla::css::TextOverflow::HasClippedTextOverflow(this) ||
 | 
						|
      mozilla::css::TextOverflow::HasBlockEllipsis(mScrolledFrame)) {
 | 
						|
    // If the block has some overflow marker stuff we should kick off a paint
 | 
						|
    // because we have special behaviour for it when APZ scrolling is active.
 | 
						|
    SchedulePaint();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::CompleteAsyncScroll(
 | 
						|
    const nsPoint& aStartPosition, const nsRect& aRange,
 | 
						|
    UniquePtr<ScrollSnapTargetIds> aSnapTargetIds, ScrollOrigin aOrigin) {
 | 
						|
  SetLastSnapTargetIds(std::move(aSnapTargetIds));
 | 
						|
 | 
						|
  bool scrollPositionChanged = mDestination != aStartPosition;
 | 
						|
  bool isNotHandledByApz =
 | 
						|
      nsLayoutUtils::CanScrollOriginClobberApz(aOrigin) ||
 | 
						|
      ScrollAnimationState().contains(AnimationState::MainThread);
 | 
						|
 | 
						|
  // Apply desired destination range since this is the last step of scrolling.
 | 
						|
  RemoveObservers();
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  ScrollToImpl(mDestination, aRange, aOrigin);
 | 
						|
  if (!weakFrame.IsAlive()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  // We are done scrolling, set our destination to wherever we actually ended
 | 
						|
  // up scrolling to.
 | 
						|
  mDestination = GetScrollPosition();
 | 
						|
  // Post a `scrollend` event for scrolling not handled by APZ, including:
 | 
						|
  //
 | 
						|
  //  - programmatic instant scrolls
 | 
						|
  //  - the end of a smooth scroll animation running on the main thread
 | 
						|
  //
 | 
						|
  // For scrolling handled by APZ, the `scrollend` event is posted in
 | 
						|
  // SetTransformingByAPZ() when the APZC is transitioning from a transforming
 | 
						|
  // to a non-transforming state (e.g. a transition from PANNING to NOTHING).
 | 
						|
  // The scrollend event should not be fired for a scroll that does not
 | 
						|
  // result in a scroll position change.
 | 
						|
  if (isNotHandledByApz && scrollPositionChanged) {
 | 
						|
    PostScrollEndEvent();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::HasBgAttachmentLocal() const {
 | 
						|
  const nsStyleBackground* bg = StyleBackground();
 | 
						|
  return bg->HasLocalBackground();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollToInternal(
 | 
						|
    nsPoint aScrollPosition, ScrollMode aMode, ScrollOrigin aOrigin,
 | 
						|
    const nsRect* aRange, ScrollSnapFlags aSnapFlags,
 | 
						|
    ScrollTriggeredByScript aTriggeredByScript) {
 | 
						|
  if (aOrigin == ScrollOrigin::NotSpecified) {
 | 
						|
    aOrigin = ScrollOrigin::Other;
 | 
						|
  }
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      aScrollPosition, aRange,
 | 
						|
      ScrollOperationParams{aMode, aOrigin, aSnapFlags, aTriggeredByScript});
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollToCSSPixels(const CSSIntPoint& aScrollPosition,
 | 
						|
                                          ScrollMode aMode) {
 | 
						|
  CSSIntPoint currentCSSPixels = GetRoundedScrollPositionCSSPixels();
 | 
						|
  // Transmogrify this scroll to a relative one if there's any on-going
 | 
						|
  // animation in APZ triggered by __user__.
 | 
						|
  // Bug 1740164: We will apply it for cases there's no animation in APZ.
 | 
						|
 | 
						|
  auto scrollAnimationState = ScrollAnimationState();
 | 
						|
  bool isScrollAnimating =
 | 
						|
      scrollAnimationState.contains(AnimationState::MainThread) ||
 | 
						|
      scrollAnimationState.contains(AnimationState::APZPending) ||
 | 
						|
      scrollAnimationState.contains(AnimationState::APZRequested);
 | 
						|
  if (mCurrentAPZScrollAnimationType ==
 | 
						|
          APZScrollAnimationType::TriggeredByUserInput &&
 | 
						|
      !isScrollAnimating) {
 | 
						|
    CSSIntPoint delta = aScrollPosition - currentCSSPixels;
 | 
						|
    // This transmogrification need to be an intended end position scroll
 | 
						|
    // operation.
 | 
						|
    ScrollByCSSPixelsInternal(delta, aMode,
 | 
						|
                              ScrollSnapFlags::IntendedEndPosition);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nscoord halfPixel = nsPresContext::CSSPixelsToAppUnits(0.5f);
 | 
						|
  nsPoint pt = CSSPoint::ToAppUnits(aScrollPosition);
 | 
						|
  nsRect range(pt.x - halfPixel, pt.y - halfPixel, 2 * halfPixel - 1,
 | 
						|
               2 * halfPixel - 1);
 | 
						|
  // XXX I don't think the following blocks are needed anymore, now that
 | 
						|
  // ScrollToImpl simply tries to scroll an integer number of layer
 | 
						|
  // pixels from the current position
 | 
						|
  nsPoint current = GetScrollPosition();
 | 
						|
  if (currentCSSPixels.x == aScrollPosition.x) {
 | 
						|
    pt.x = current.x;
 | 
						|
    range.x = pt.x;
 | 
						|
    range.width = 0;
 | 
						|
  }
 | 
						|
  if (currentCSSPixels.y == aScrollPosition.y) {
 | 
						|
    pt.y = current.y;
 | 
						|
    range.y = pt.y;
 | 
						|
    range.height = 0;
 | 
						|
  }
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      pt, &range,
 | 
						|
      ScrollOperationParams{
 | 
						|
          aMode, ScrollOrigin::Other,
 | 
						|
          // This ScrollToCSSPixels is used for Element.scrollTo,
 | 
						|
          // Element.scrollTop, Element.scrollLeft and for Window.scrollTo.
 | 
						|
          ScrollSnapFlags::IntendedEndPosition, ScrollTriggeredByScript::Yes});
 | 
						|
  // 'this' might be destroyed here
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollToCSSPixelsForApz(
 | 
						|
    const CSSPoint& aScrollPosition, ScrollSnapTargetIds&& aLastSnapTargetIds) {
 | 
						|
  nsPoint pt = CSSPoint::ToAppUnits(aScrollPosition);
 | 
						|
  nscoord halfRange = nsPresContext::CSSPixelsToAppUnits(1000);
 | 
						|
  nsRect range(pt.x - halfRange, pt.y - halfRange, 2 * halfRange - 1,
 | 
						|
               2 * halfRange - 1);
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      pt, &range,
 | 
						|
      ScrollOperationParams{ScrollMode::Instant, ScrollOrigin::Apz,
 | 
						|
                            std::move(aLastSnapTargetIds)});
 | 
						|
  // 'this' might be destroyed here
 | 
						|
}
 | 
						|
 | 
						|
CSSIntPoint nsHTMLScrollFrame::GetRoundedScrollPositionCSSPixels() {
 | 
						|
  return CSSIntPoint::FromAppUnitsRounded(GetScrollPosition());
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * this method wraps calls to ScrollToImpl(), either in one shot or
 | 
						|
 * incrementally, based on the setting of the smoothness scroll pref
 | 
						|
 */
 | 
						|
void nsHTMLScrollFrame::ScrollToWithOrigin(nsPoint aScrollPosition,
 | 
						|
                                           const nsRect* aRange,
 | 
						|
                                           ScrollOperationParams&& aParams) {
 | 
						|
  // None is never a valid scroll origin to be passed in.
 | 
						|
  MOZ_ASSERT(aParams.mOrigin != ScrollOrigin::None);
 | 
						|
 | 
						|
  if (aParams.mOrigin != ScrollOrigin::Restore) {
 | 
						|
    // If we're doing a non-restore scroll, we don't want to later
 | 
						|
    // override it by restoring our saved scroll position.
 | 
						|
    SCROLLRESTORE_LOG("%p: Clearing mRestorePos (cur=%s, dst=%s)\n", this,
 | 
						|
                      ToString(GetScrollPosition()).c_str(),
 | 
						|
                      ToString(aScrollPosition).c_str());
 | 
						|
    mRestorePos.x = mRestorePos.y = -1;
 | 
						|
  }
 | 
						|
 | 
						|
  Maybe<SnapDestination> snapDestination;
 | 
						|
  if (!aParams.IsScrollSnapDisabled()) {
 | 
						|
    snapDestination = GetSnapPointForDestination(ScrollUnit::DEVICE_PIXELS,
 | 
						|
                                                 aParams.mSnapFlags,
 | 
						|
                                                 mDestination, aScrollPosition);
 | 
						|
    if (snapDestination) {
 | 
						|
      aScrollPosition = snapDestination->mPosition;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect scrollRange = GetLayoutScrollRange();
 | 
						|
  mDestination = scrollRange.ClampPoint(aScrollPosition);
 | 
						|
  if (mDestination != aScrollPosition &&
 | 
						|
      aParams.mOrigin == ScrollOrigin::Restore &&
 | 
						|
      GetPageLoadingState() != LoadingState::Loading) {
 | 
						|
    // If we're doing a restore but the scroll position is clamped, promote
 | 
						|
    // the origin from one that APZ can clobber to one that it can't clobber.
 | 
						|
    aParams.mOrigin = ScrollOrigin::Other;
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect range = aRange && snapDestination.isNothing()
 | 
						|
                     ? *aRange
 | 
						|
                     : nsRect(aScrollPosition, nsSize(0, 0));
 | 
						|
 | 
						|
  UniquePtr<ScrollSnapTargetIds> snapTargetIds;
 | 
						|
  if (snapDestination) {
 | 
						|
    snapTargetIds =
 | 
						|
        MakeUnique<ScrollSnapTargetIds>(std::move(snapDestination->mTargetIds));
 | 
						|
  } else {
 | 
						|
    snapTargetIds =
 | 
						|
        MakeUnique<ScrollSnapTargetIds>(std::move(aParams.mTargetIds));
 | 
						|
  }
 | 
						|
  if (aParams.IsInstant()) {
 | 
						|
    // Asynchronous scrolling is not allowed, so we'll kill any existing
 | 
						|
    // async-scrolling process and do an instant scroll.
 | 
						|
    CompleteAsyncScroll(GetScrollPosition(), range, std::move(snapTargetIds),
 | 
						|
                        aParams.mOrigin);
 | 
						|
    mApzSmoothScrollDestination = Nothing();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!aParams.IsSmoothMsd()) {
 | 
						|
    // If we get a non-smooth-scroll, reset the cached APZ scroll destination,
 | 
						|
    // so that we know to process the next smooth-scroll destined for APZ.
 | 
						|
    mApzSmoothScrollDestination = Nothing();
 | 
						|
  }
 | 
						|
 | 
						|
  nsPresContext* presContext = PresContext();
 | 
						|
  TimeStamp now =
 | 
						|
      presContext->RefreshDriver()->IsTestControllingRefreshesEnabled()
 | 
						|
          ? presContext->RefreshDriver()->MostRecentRefresh()
 | 
						|
          : TimeStamp::Now();
 | 
						|
 | 
						|
  nsSize currentVelocity(0, 0);
 | 
						|
 | 
						|
  const bool canHandoffToApz =
 | 
						|
      nsLayoutUtils::AsyncPanZoomEnabled(this) && WantAsyncScroll() &&
 | 
						|
      CanApzScrollInTheseDirections(
 | 
						|
          DirectionsInDelta(mDestination - GetScrollPosition()));
 | 
						|
 | 
						|
  if (aParams.IsSmoothMsd()) {
 | 
						|
    mIgnoreMomentumScroll = true;
 | 
						|
    if (!mAsyncSmoothMSDScroll) {
 | 
						|
      nsPoint sv = mVelocityQueue.GetVelocity();
 | 
						|
      currentVelocity.width = sv.x;
 | 
						|
      currentVelocity.height = sv.y;
 | 
						|
      if (mAsyncScroll) {
 | 
						|
        if (mAsyncScroll->IsSmoothScroll()) {
 | 
						|
          currentVelocity = mAsyncScroll->VelocityAt(now);
 | 
						|
        }
 | 
						|
        mAsyncScroll = nullptr;
 | 
						|
      }
 | 
						|
 | 
						|
      if (canHandoffToApz) {
 | 
						|
        ApzSmoothScrollTo(mDestination, ScrollMode::SmoothMsd, aParams.mOrigin,
 | 
						|
                          aParams.mTriggeredByScript, std::move(snapTargetIds));
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      mAsyncSmoothMSDScroll = new AsyncSmoothMSDScroll(
 | 
						|
          GetScrollPosition(), mDestination, currentVelocity,
 | 
						|
          GetLayoutScrollRange(), now, presContext, std::move(snapTargetIds),
 | 
						|
          aParams.mTriggeredByScript);
 | 
						|
 | 
						|
      mAsyncSmoothMSDScroll->SetRefreshObserver(this);
 | 
						|
    } else {
 | 
						|
      // A previous smooth MSD scroll is still in progress, so we just need to
 | 
						|
      // update its range and destination.
 | 
						|
      mAsyncSmoothMSDScroll->SetRange(GetLayoutScrollRange());
 | 
						|
      mAsyncSmoothMSDScroll->SetDestination(mDestination,
 | 
						|
                                            aParams.mTriggeredByScript);
 | 
						|
    }
 | 
						|
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mAsyncSmoothMSDScroll) {
 | 
						|
    currentVelocity = mAsyncSmoothMSDScroll->GetVelocity();
 | 
						|
    mAsyncSmoothMSDScroll = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  const bool isSmoothScroll =
 | 
						|
      aParams.IsSmooth() && nsLayoutUtils::IsSmoothScrollingEnabled();
 | 
						|
  if (!mAsyncScroll) {
 | 
						|
    if (isSmoothScroll && canHandoffToApz) {
 | 
						|
      ApzSmoothScrollTo(mDestination, ScrollMode::Smooth, aParams.mOrigin,
 | 
						|
                        aParams.mTriggeredByScript, std::move(snapTargetIds));
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    mAsyncScroll =
 | 
						|
        new AsyncScroll(std::move(snapTargetIds), aParams.mTriggeredByScript);
 | 
						|
    mAsyncScroll->SetRefreshObserver(this);
 | 
						|
  }
 | 
						|
 | 
						|
  if (isSmoothScroll) {
 | 
						|
    mAsyncScroll->InitSmoothScroll(now, GetScrollPosition(), mDestination,
 | 
						|
                                   aParams.mOrigin, range, currentVelocity);
 | 
						|
  } else {
 | 
						|
    mAsyncScroll->Init(GetScrollPosition(), range);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// We can't use nsContainerFrame::PositionChildViews here because
 | 
						|
// we don't want to invalidate views that have moved.
 | 
						|
static void AdjustViews(nsIFrame* aFrame) {
 | 
						|
  nsView* view = aFrame->GetView();
 | 
						|
  if (view) {
 | 
						|
    nsPoint pt;
 | 
						|
    aFrame->GetParent()->GetClosestView(&pt);
 | 
						|
    pt += aFrame->GetPosition();
 | 
						|
    view->SetPosition(pt.x, pt.y);
 | 
						|
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!aFrame->HasAnyStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Call AdjustViews recursively for all child frames except the popup list as
 | 
						|
  // the views for popups are not scrolled.
 | 
						|
  for (const auto& [list, listID] : aFrame->ChildLists()) {
 | 
						|
    if (listID == FrameChildListID::Popup) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
    for (nsIFrame* child : list) {
 | 
						|
      AdjustViews(child);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::MarkScrollbarsDirtyForReflow() const {
 | 
						|
  auto* presShell = PresShell();
 | 
						|
  if (mVScrollbarBox) {
 | 
						|
    presShell->FrameNeedsReflow(mVScrollbarBox,
 | 
						|
                                IntrinsicDirty::FrameAncestorsAndDescendants,
 | 
						|
                                NS_FRAME_IS_DIRTY);
 | 
						|
  }
 | 
						|
  if (mHScrollbarBox) {
 | 
						|
    presShell->FrameNeedsReflow(mHScrollbarBox,
 | 
						|
                                IntrinsicDirty::FrameAncestorsAndDescendants,
 | 
						|
                                NS_FRAME_IS_DIRTY);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::InvalidateScrollbars() const {
 | 
						|
  if (mHScrollbarBox) {
 | 
						|
    mHScrollbarBox->InvalidateFrameSubtree();
 | 
						|
  }
 | 
						|
  if (mVScrollbarBox) {
 | 
						|
    mVScrollbarBox->InvalidateFrameSubtree();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsAlwaysActive() const {
 | 
						|
  if (nsDisplayItem::ForceActiveLayers()) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // Unless this is the root scrollframe for a non-chrome document
 | 
						|
  // which is the direct child of a chrome document, we default to not
 | 
						|
  // being "active".
 | 
						|
  if (!(mIsRoot && PresContext()->IsRootContentDocumentCrossProcess())) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // If we have scrolled before, then we should stay active.
 | 
						|
  if (mHasBeenScrolled) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // If we're overflow:hidden, then start as inactive until
 | 
						|
  // we get scrolled manually.
 | 
						|
  ScrollStyles styles = GetScrollStyles();
 | 
						|
  return (styles.mHorizontal != StyleOverflow::Hidden &&
 | 
						|
          styles.mVertical != StyleOverflow::Hidden);
 | 
						|
}
 | 
						|
 | 
						|
static void RemoveDisplayPortCallback(nsITimer* aTimer, void* aClosure) {
 | 
						|
  nsHTMLScrollFrame* sf = static_cast<nsHTMLScrollFrame*>(aClosure);
 | 
						|
 | 
						|
  // This function only ever gets called from the expiry timer, so it must
 | 
						|
  // be non-null here. Set it to null here so that we don't keep resetting
 | 
						|
  // it unnecessarily in MarkRecentlyScrolled().
 | 
						|
  MOZ_ASSERT(sf->mDisplayPortExpiryTimer);
 | 
						|
  sf->mDisplayPortExpiryTimer = nullptr;
 | 
						|
 | 
						|
  if (!sf->AllowDisplayPortExpiration() || sf->mIsParentToActiveScrollFrames) {
 | 
						|
    // If this is a scroll parent for some other scrollable frame, don't
 | 
						|
    // expire the displayport because it would break scroll handoff. Once the
 | 
						|
    // descendant scrollframes have their displayports expired, they will
 | 
						|
    // trigger the displayport expiration on this scrollframe as well, and
 | 
						|
    // mIsParentToActiveScrollFrames will presumably be false when that kicks
 | 
						|
    // in.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Remove the displayport from this scrollframe if it's been a while
 | 
						|
  // since it's scrolled, except if it needs to be always active. Note that
 | 
						|
  // there is one scrollframe that doesn't fall under this general rule, and
 | 
						|
  // that is the one that nsLayoutUtils::MaybeCreateDisplayPort decides to put
 | 
						|
  // a displayport on (i.e. the first scrollframe that WantAsyncScroll()s).
 | 
						|
  // If that scrollframe is this one, we remove the displayport anyway, and
 | 
						|
  // as part of the next paint MaybeCreateDisplayPort will put another
 | 
						|
  // displayport back on it. Although the displayport will "flicker" off and
 | 
						|
  // back on, the layer itself should never disappear, because this all
 | 
						|
  // happens between actual painting. If the displayport is reset to a
 | 
						|
  // different position that's ok; this scrollframe hasn't been scrolled
 | 
						|
  // recently and so the reset should be correct.
 | 
						|
 | 
						|
  nsIContent* content = sf->GetContent();
 | 
						|
 | 
						|
  if (nsHTMLScrollFrame::ShouldActivateAllScrollFrames()) {
 | 
						|
    // If we are activating all scroll frames then we only want to remove the
 | 
						|
    // regular display port and downgrade to a minimal display port.
 | 
						|
    MOZ_ASSERT(!content->GetProperty(nsGkAtoms::MinimalDisplayPort));
 | 
						|
    content->SetProperty(nsGkAtoms::MinimalDisplayPort,
 | 
						|
                         reinterpret_cast<void*>(true));
 | 
						|
  } else {
 | 
						|
    content->RemoveProperty(nsGkAtoms::MinimalDisplayPort);
 | 
						|
    DisplayPortUtils::RemoveDisplayPort(content);
 | 
						|
    // Be conservative and unflag this this scrollframe as being scrollable by
 | 
						|
    // APZ. If it is still scrollable this will get flipped back soon enough.
 | 
						|
    sf->mScrollableByAPZ = false;
 | 
						|
  }
 | 
						|
 | 
						|
  DisplayPortUtils::ExpireDisplayPortOnAsyncScrollableAncestor(sf);
 | 
						|
  sf->SchedulePaint();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::MarkEverScrolled() {
 | 
						|
  // Mark this frame as having been scrolled. If this is the root
 | 
						|
  // scroll frame of a content document, then IsAlwaysActive()
 | 
						|
  // will return true from now on and MarkNotRecentlyScrolled() won't
 | 
						|
  // have any effect.
 | 
						|
  mHasBeenScrolled = true;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::MarkNotRecentlyScrolled() {
 | 
						|
  if (!mHasBeenScrolledRecently) return;
 | 
						|
 | 
						|
  mHasBeenScrolledRecently = false;
 | 
						|
  SchedulePaint();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::MarkRecentlyScrolled() {
 | 
						|
  mHasBeenScrolledRecently = true;
 | 
						|
  if (IsAlwaysActive()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mActivityExpirationState.IsTracked()) {
 | 
						|
    gScrollFrameActivityTracker->MarkUsed(this);
 | 
						|
  } else {
 | 
						|
    if (!gScrollFrameActivityTracker) {
 | 
						|
      gScrollFrameActivityTracker =
 | 
						|
          new ScrollFrameActivityTracker(GetMainThreadSerialEventTarget());
 | 
						|
    }
 | 
						|
    gScrollFrameActivityTracker->AddObject(this);
 | 
						|
  }
 | 
						|
 | 
						|
  // If we just scrolled and there's a displayport expiry timer in place,
 | 
						|
  // reset the timer.
 | 
						|
  ResetDisplayPortExpiryTimer();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ResetDisplayPortExpiryTimer() {
 | 
						|
  if (mDisplayPortExpiryTimer) {
 | 
						|
    mDisplayPortExpiryTimer->InitWithNamedFuncCallback(
 | 
						|
        RemoveDisplayPortCallback, this,
 | 
						|
        StaticPrefs::apz_displayport_expiry_ms(), nsITimer::TYPE_ONE_SHOT,
 | 
						|
        "nsHTMLScrollFrame::ResetDisplayPortExpiryTimer");
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::AllowDisplayPortExpiration() {
 | 
						|
  if (IsAlwaysActive()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mIsRoot && PresContext()->IsRoot()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // If this was the first scrollable frame found, this displayport should
 | 
						|
  // not expire.
 | 
						|
  if (IsFirstScrollableFrameSequenceNumber().isSome()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (ShouldActivateAllScrollFrames() &&
 | 
						|
      GetContent()->GetProperty(nsGkAtoms::MinimalDisplayPort)) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::TriggerDisplayPortExpiration() {
 | 
						|
  if (!AllowDisplayPortExpiration()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!StaticPrefs::apz_displayport_expiry_ms()) {
 | 
						|
    // a zero time disables the expiry
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mDisplayPortExpiryTimer) {
 | 
						|
    mDisplayPortExpiryTimer = NS_NewTimer();
 | 
						|
  }
 | 
						|
  ResetDisplayPortExpiryTimer();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollVisual() {
 | 
						|
  MarkEverScrolled();
 | 
						|
 | 
						|
  AdjustViews(mScrolledFrame);
 | 
						|
  // We need to call this after fixing up the view positions
 | 
						|
  // to be consistent with the frame hierarchy.
 | 
						|
  MarkRecentlyScrolled();
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Clamp desired scroll position aDesired and range [aDestLower, aDestUpper]
 | 
						|
 * to [aBoundLower, aBoundUpper] and then select the appunit value from among
 | 
						|
 * aBoundLower, aBoundUpper and those such that (aDesired - aCurrent) *
 | 
						|
 * aRes/aAppUnitsPerPixel is an integer (or as close as we can get
 | 
						|
 * modulo rounding to appunits) that is in [aDestLower, aDestUpper] and
 | 
						|
 * closest to aDesired.  If no such value exists, return the nearest in
 | 
						|
 * [aDestLower, aDestUpper].
 | 
						|
 */
 | 
						|
static nscoord ClampAndAlignWithPixels(nscoord aDesired, nscoord aBoundLower,
 | 
						|
                                       nscoord aBoundUpper, nscoord aDestLower,
 | 
						|
                                       nscoord aDestUpper,
 | 
						|
                                       nscoord aAppUnitsPerPixel, double aRes,
 | 
						|
                                       nscoord aCurrent) {
 | 
						|
  // Intersect scroll range with allowed range, by clamping the ends
 | 
						|
  // of aRange to be within bounds
 | 
						|
  nscoord destLower = clamped(aDestLower, aBoundLower, aBoundUpper);
 | 
						|
  nscoord destUpper = clamped(aDestUpper, aBoundLower, aBoundUpper);
 | 
						|
 | 
						|
  nscoord desired = clamped(aDesired, destLower, destUpper);
 | 
						|
  if (StaticPrefs::layout_scroll_disable_pixel_alignment()) {
 | 
						|
    return desired;
 | 
						|
  }
 | 
						|
 | 
						|
  double currentLayerVal = (aRes * aCurrent) / aAppUnitsPerPixel;
 | 
						|
  double desiredLayerVal = (aRes * desired) / aAppUnitsPerPixel;
 | 
						|
  double delta = desiredLayerVal - currentLayerVal;
 | 
						|
  double nearestLayerVal = NS_round(delta) + currentLayerVal;
 | 
						|
 | 
						|
  // Convert back from PaintedLayer space to appunits relative to the top-left
 | 
						|
  // of the scrolled frame.
 | 
						|
  nscoord aligned =
 | 
						|
      aRes == 0.0
 | 
						|
          ? 0.0
 | 
						|
          : NSToCoordRoundWithClamp(nearestLayerVal * aAppUnitsPerPixel / aRes);
 | 
						|
 | 
						|
  // Use a bound if it is within the allowed range and closer to desired than
 | 
						|
  // the nearest pixel-aligned value.
 | 
						|
  if (aBoundUpper == destUpper &&
 | 
						|
      static_cast<decltype(Abs(desired))>(aBoundUpper - desired) <
 | 
						|
          Abs(desired - aligned)) {
 | 
						|
    return aBoundUpper;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aBoundLower == destLower &&
 | 
						|
      static_cast<decltype(Abs(desired))>(desired - aBoundLower) <
 | 
						|
          Abs(aligned - desired)) {
 | 
						|
    return aBoundLower;
 | 
						|
  }
 | 
						|
 | 
						|
  // Accept the nearest pixel-aligned value if it is within the allowed range.
 | 
						|
  if (aligned >= destLower && aligned <= destUpper) {
 | 
						|
    return aligned;
 | 
						|
  }
 | 
						|
 | 
						|
  // Check if opposite pixel boundary fits into allowed range.
 | 
						|
  double oppositeLayerVal =
 | 
						|
      nearestLayerVal + ((nearestLayerVal < desiredLayerVal) ? 1.0 : -1.0);
 | 
						|
  nscoord opposite = aRes == 0.0
 | 
						|
                         ? 0.0
 | 
						|
                         : NSToCoordRoundWithClamp(oppositeLayerVal *
 | 
						|
                                                   aAppUnitsPerPixel / aRes);
 | 
						|
  if (opposite >= destLower && opposite <= destUpper) {
 | 
						|
    return opposite;
 | 
						|
  }
 | 
						|
 | 
						|
  // No alignment available.
 | 
						|
  return desired;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Clamp desired scroll position aPt to aBounds and then snap
 | 
						|
 * it to the same layer pixel edges as aCurrent, keeping it within aRange
 | 
						|
 * during snapping. aCurrent is the current scroll position.
 | 
						|
 */
 | 
						|
static nsPoint ClampAndAlignWithLayerPixels(const nsPoint& aPt,
 | 
						|
                                            const nsRect& aBounds,
 | 
						|
                                            const nsRect& aRange,
 | 
						|
                                            const nsPoint& aCurrent,
 | 
						|
                                            nscoord aAppUnitsPerPixel,
 | 
						|
                                            const MatrixScales& aScale) {
 | 
						|
  return nsPoint(
 | 
						|
      ClampAndAlignWithPixels(aPt.x, aBounds.x, aBounds.XMost(), aRange.x,
 | 
						|
                              aRange.XMost(), aAppUnitsPerPixel, aScale.xScale,
 | 
						|
                              aCurrent.x),
 | 
						|
      ClampAndAlignWithPixels(aPt.y, aBounds.y, aBounds.YMost(), aRange.y,
 | 
						|
                              aRange.YMost(), aAppUnitsPerPixel, aScale.yScale,
 | 
						|
                              aCurrent.y));
 | 
						|
}
 | 
						|
 | 
						|
/* static */
 | 
						|
void nsHTMLScrollFrame::ScrollActivityCallback(nsITimer* aTimer,
 | 
						|
                                               void* anInstance) {
 | 
						|
  nsHTMLScrollFrame* self = static_cast<nsHTMLScrollFrame*>(anInstance);
 | 
						|
 | 
						|
  // Fire the synth mouse move.
 | 
						|
  self->mScrollActivityTimer->Cancel();
 | 
						|
  self->mScrollActivityTimer = nullptr;
 | 
						|
  self->PresShell()->SynthesizeMouseMove(true);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScheduleSyntheticMouseMove() {
 | 
						|
  if (!mScrollActivityTimer) {
 | 
						|
    mScrollActivityTimer = NS_NewTimer(GetMainThreadSerialEventTarget());
 | 
						|
    if (!mScrollActivityTimer) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mScrollActivityTimer->InitWithNamedFuncCallback(
 | 
						|
      ScrollActivityCallback, this, 100, nsITimer::TYPE_ONE_SHOT,
 | 
						|
      "nsHTMLScrollFrame::ScheduleSyntheticMouseMove");
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::NotifyApproximateFrameVisibilityUpdate(
 | 
						|
    bool aIgnoreDisplayPort) {
 | 
						|
  mLastUpdateFramesPos = GetScrollPosition();
 | 
						|
  if (aIgnoreDisplayPort) {
 | 
						|
    mHadDisplayPortAtLastFrameUpdate = false;
 | 
						|
    mDisplayPortAtLastFrameUpdate = nsRect();
 | 
						|
  } else {
 | 
						|
    mHadDisplayPortAtLastFrameUpdate = DisplayPortUtils::GetDisplayPort(
 | 
						|
        GetContent(), &mDisplayPortAtLastFrameUpdate);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::GetDisplayPortAtLastApproximateFrameVisibilityUpdate(
 | 
						|
    nsRect* aDisplayPort) {
 | 
						|
  if (mHadDisplayPortAtLastFrameUpdate) {
 | 
						|
    *aDisplayPort = mDisplayPortAtLastFrameUpdate;
 | 
						|
  }
 | 
						|
  return mHadDisplayPortAtLastFrameUpdate;
 | 
						|
}
 | 
						|
 | 
						|
/* aIncludeCSSTransform controls if we include CSS transforms that are in this
 | 
						|
 * process (the BrowserChild EffectsInfo mTransformToAncestorScale will include
 | 
						|
 * CSS transforms in ancestor processes in all cases). */
 | 
						|
MatrixScales GetPaintedLayerScaleForFrame(nsIFrame* aFrame,
 | 
						|
                                          bool aIncludeCSSTransform) {
 | 
						|
  MOZ_ASSERT(aFrame, "need a frame");
 | 
						|
 | 
						|
  nsPresContext* presCtx = aFrame->PresContext()->GetRootPresContext();
 | 
						|
 | 
						|
  if (!presCtx) {
 | 
						|
    presCtx = aFrame->PresContext();
 | 
						|
    MOZ_ASSERT(presCtx);
 | 
						|
  }
 | 
						|
 | 
						|
  ParentLayerToScreenScale2D transformToAncestorScale;
 | 
						|
  if (aIncludeCSSTransform) {
 | 
						|
    transformToAncestorScale =
 | 
						|
        nsLayoutUtils::GetTransformToAncestorScaleCrossProcessForFrameMetrics(
 | 
						|
            aFrame);
 | 
						|
  } else {
 | 
						|
    if (BrowserChild* browserChild =
 | 
						|
            BrowserChild::GetFrom(aFrame->PresShell())) {
 | 
						|
      transformToAncestorScale =
 | 
						|
          browserChild->GetEffectsInfo().mTransformToAncestorScale;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  transformToAncestorScale =
 | 
						|
      ParentLayerToParentLayerScale(
 | 
						|
          presCtx->PresShell()->GetCumulativeResolution()) *
 | 
						|
      transformToAncestorScale;
 | 
						|
 | 
						|
  return transformToAncestorScale.ToUnknownScale();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollToImpl(
 | 
						|
    nsPoint aPt, const nsRect& aRange, ScrollOrigin aOrigin,
 | 
						|
    ScrollTriggeredByScript aTriggeredByScript) {
 | 
						|
  // None is never a valid scroll origin to be passed in.
 | 
						|
  MOZ_ASSERT(aOrigin != ScrollOrigin::None);
 | 
						|
 | 
						|
  // Figure out the effective origin for this scroll request.
 | 
						|
  if (aOrigin == ScrollOrigin::NotSpecified) {
 | 
						|
    // If no origin was specified, we still want to set it to something that's
 | 
						|
    // non-unknown, so that we can use eUnknown to distinguish if the frame was
 | 
						|
    // scrolled at all. Default it to some generic placeholder.
 | 
						|
    aOrigin = ScrollOrigin::Other;
 | 
						|
  }
 | 
						|
 | 
						|
  // If this scroll is |relative|, but we've already had a user scroll that
 | 
						|
  // was not relative, promote this origin to |other|. This ensures that we
 | 
						|
  // may only transmit a relative update to APZ if all scrolls since the last
 | 
						|
  // transaction or repaint request have been relative.
 | 
						|
  if (aOrigin == ScrollOrigin::Relative &&
 | 
						|
      (mLastScrollOrigin != ScrollOrigin::None &&
 | 
						|
       mLastScrollOrigin != ScrollOrigin::NotSpecified &&
 | 
						|
       mLastScrollOrigin != ScrollOrigin::Relative &&
 | 
						|
       mLastScrollOrigin != ScrollOrigin::Apz)) {
 | 
						|
    aOrigin = ScrollOrigin::Other;
 | 
						|
  }
 | 
						|
 | 
						|
  // If the origin is a downgrade, and downgrades are allowed, process the
 | 
						|
  // downgrade even if we're going to early-exit because we're already at
 | 
						|
  // the correct scroll position. This ensures that if there wasn't a main-
 | 
						|
  // thread scroll update pending before a frame reconstruction (as indicated
 | 
						|
  // by mAllowScrollOriginDowngrade=true), then after the frame reconstruction
 | 
						|
  // the origin is downgraded to "restore" even if the layout scroll offset to
 | 
						|
  // be restored is (0,0) (which will take the early-exit below). This is
 | 
						|
  // important so that restoration of a *visual* scroll offset (which might be
 | 
						|
  // to something other than (0,0)) isn't clobbered.
 | 
						|
  bool isScrollOriginDowngrade =
 | 
						|
      nsLayoutUtils::CanScrollOriginClobberApz(mLastScrollOrigin) &&
 | 
						|
      !nsLayoutUtils::CanScrollOriginClobberApz(aOrigin);
 | 
						|
  bool allowScrollOriginChange =
 | 
						|
      mAllowScrollOriginDowngrade && isScrollOriginDowngrade;
 | 
						|
 | 
						|
  if (allowScrollOriginChange) {
 | 
						|
    mLastScrollOrigin = aOrigin;
 | 
						|
    mAllowScrollOriginDowngrade = false;
 | 
						|
  }
 | 
						|
 | 
						|
  nsPresContext* presContext = PresContext();
 | 
						|
  nscoord appUnitsPerDevPixel = presContext->AppUnitsPerDevPixel();
 | 
						|
  // 'scale' is our estimate of the scale factor that will be applied
 | 
						|
  // when rendering the scrolled content to its own PaintedLayer.
 | 
						|
  MatrixScales scale = GetPaintedLayerScaleForFrame(
 | 
						|
      mScrolledFrame, /* aIncludeCSSTransform = */ true);
 | 
						|
  nsPoint curPos = GetScrollPosition();
 | 
						|
 | 
						|
  // Try to align aPt with curPos so they have an integer number of layer
 | 
						|
  // pixels between them. This gives us the best chance of scrolling without
 | 
						|
  // having to invalidate due to changes in subpixel rendering.
 | 
						|
  // Note that when we actually draw into a PaintedLayer, the coordinates
 | 
						|
  // that get mapped onto the layer buffer pixels are from the display list,
 | 
						|
  // which are relative to the display root frame's top-left increasing down,
 | 
						|
  // whereas here our coordinates are scroll positions which increase upward
 | 
						|
  // and are relative to the scrollport top-left. This difference doesn't
 | 
						|
  // actually matter since all we are about is that there be an integer number
 | 
						|
  // of layer pixels between pt and curPos.
 | 
						|
  nsPoint pt = ClampAndAlignWithLayerPixels(aPt, GetLayoutScrollRange(), aRange,
 | 
						|
                                            curPos, appUnitsPerDevPixel, scale);
 | 
						|
  if (pt == curPos) {
 | 
						|
    // Even if we are bailing out due to no-op main-thread scroll position
 | 
						|
    // change, we might need to cancel an APZ smooth scroll that we already
 | 
						|
    // kicked off. It might be reasonable to eventually remove the
 | 
						|
    // mApzSmoothScrollDestination clause from this if statement, as that
 | 
						|
    // may simplify this a bit and should be fine from the APZ side.
 | 
						|
    if (mApzSmoothScrollDestination && aOrigin != ScrollOrigin::Clamp) {
 | 
						|
      if (aOrigin == ScrollOrigin::Relative) {
 | 
						|
        AppendScrollUpdate(
 | 
						|
            ScrollPositionUpdate::NewRelativeScroll(mApzScrollPos, pt));
 | 
						|
        mApzScrollPos = pt;
 | 
						|
      } else if (aOrigin != ScrollOrigin::Apz) {
 | 
						|
        ScrollOrigin origin =
 | 
						|
            (mAllowScrollOriginDowngrade || !isScrollOriginDowngrade)
 | 
						|
                ? aOrigin
 | 
						|
                : mLastScrollOrigin;
 | 
						|
        AppendScrollUpdate(ScrollPositionUpdate::NewScroll(origin, pt));
 | 
						|
      }
 | 
						|
    }
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // If we are scrolling the RCD-RSF, and a visual scroll update is pending,
 | 
						|
  // cancel it; otherwise, it will clobber this scroll.
 | 
						|
  if (IsRootScrollFrameOfDocument() &&
 | 
						|
      presContext->IsRootContentDocumentCrossProcess()) {
 | 
						|
    auto* ps = presContext->GetPresShell();
 | 
						|
    if (const auto& visualScrollUpdate = ps->GetPendingVisualScrollUpdate()) {
 | 
						|
      if (visualScrollUpdate->mVisualScrollOffset != aPt) {
 | 
						|
        // Only clobber if the scroll was originated by the main thread.
 | 
						|
        // Respect the priority of origins (an "eRestore" layout scroll should
 | 
						|
        // not clobber an "eMainThread" visual scroll.)
 | 
						|
        bool shouldClobber =
 | 
						|
            aOrigin == ScrollOrigin::Other ||
 | 
						|
            (aOrigin == ScrollOrigin::Restore &&
 | 
						|
             visualScrollUpdate->mUpdateType == FrameMetrics::eRestore);
 | 
						|
        if (shouldClobber) {
 | 
						|
          ps->AcknowledgePendingVisualScrollUpdate();
 | 
						|
          ps->ClearPendingVisualScrollUpdate();
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  bool needFrameVisibilityUpdate = mLastUpdateFramesPos == nsPoint(-1, -1);
 | 
						|
 | 
						|
  nsPoint dist(std::abs(pt.x - mLastUpdateFramesPos.x),
 | 
						|
               std::abs(pt.y - mLastUpdateFramesPos.y));
 | 
						|
  nsSize visualViewportSize = GetVisualViewportSize();
 | 
						|
  nscoord horzAllowance = std::max(
 | 
						|
      visualViewportSize.width /
 | 
						|
          std::max(
 | 
						|
              StaticPrefs::
 | 
						|
                  layout_framevisibility_amountscrollbeforeupdatehorizontal(),
 | 
						|
              1),
 | 
						|
      AppUnitsPerCSSPixel());
 | 
						|
  nscoord vertAllowance = std::max(
 | 
						|
      visualViewportSize.height /
 | 
						|
          std::max(
 | 
						|
              StaticPrefs::
 | 
						|
                  layout_framevisibility_amountscrollbeforeupdatevertical(),
 | 
						|
              1),
 | 
						|
      AppUnitsPerCSSPixel());
 | 
						|
  if (dist.x >= horzAllowance || dist.y >= vertAllowance) {
 | 
						|
    needFrameVisibilityUpdate = true;
 | 
						|
  }
 | 
						|
 | 
						|
  // notify the listeners.
 | 
						|
  for (uint32_t i = 0; i < mListeners.Length(); i++) {
 | 
						|
    mListeners[i]->ScrollPositionWillChange(pt.x, pt.y);
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect oldDisplayPort;
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
  DisplayPortUtils::GetDisplayPort(content, &oldDisplayPort);
 | 
						|
  oldDisplayPort.MoveBy(-mScrolledFrame->GetPosition());
 | 
						|
 | 
						|
  // Update frame position for scrolling
 | 
						|
  mScrolledFrame->SetPosition(mScrollPort.TopLeft() - pt);
 | 
						|
 | 
						|
  // If |mLastScrollOrigin| is already set to something that can clobber APZ's
 | 
						|
  // scroll offset, then we don't want to change it to something that can't.
 | 
						|
  // If we allowed this, then we could end up in a state where APZ ignores
 | 
						|
  // legitimate scroll offset updates because the origin has been masked by
 | 
						|
  // a later change within the same refresh driver tick.
 | 
						|
  allowScrollOriginChange =
 | 
						|
      (mAllowScrollOriginDowngrade || !isScrollOriginDowngrade);
 | 
						|
 | 
						|
  if (allowScrollOriginChange) {
 | 
						|
    mLastScrollOrigin = aOrigin;
 | 
						|
    mAllowScrollOriginDowngrade = false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aOrigin == ScrollOrigin::Relative) {
 | 
						|
    MOZ_ASSERT(!isScrollOriginDowngrade);
 | 
						|
    MOZ_ASSERT(mLastScrollOrigin == ScrollOrigin::Relative);
 | 
						|
    AppendScrollUpdate(
 | 
						|
        ScrollPositionUpdate::NewRelativeScroll(mApzScrollPos, pt));
 | 
						|
    mApzScrollPos = pt;
 | 
						|
  } else if (aOrigin != ScrollOrigin::Apz) {
 | 
						|
    AppendScrollUpdate(ScrollPositionUpdate::NewScroll(mLastScrollOrigin, pt));
 | 
						|
  }
 | 
						|
 | 
						|
  if (mLastScrollOrigin == ScrollOrigin::Apz) {
 | 
						|
    mApzScrollPos = GetScrollPosition();
 | 
						|
  }
 | 
						|
 | 
						|
  ScrollVisual();
 | 
						|
  mAnchor.UserScrolled();
 | 
						|
 | 
						|
  // Only report user-triggered scrolling interactions
 | 
						|
  bool jsOnStack = nsContentUtils::GetCurrentJSContext() != nullptr;
 | 
						|
  bool scrollingToAnchor = ScrollingInteractionContext::IsScrollingToAnchor();
 | 
						|
  if (!jsOnStack && !scrollingToAnchor) {
 | 
						|
    nsPoint distanceScrolled(std::abs(pt.x - curPos.x),
 | 
						|
                             std::abs(pt.y - curPos.y));
 | 
						|
    ScrollingMetrics::OnScrollingInteraction(
 | 
						|
        CSSPoint::FromAppUnits(distanceScrolled).Length());
 | 
						|
  }
 | 
						|
 | 
						|
  bool schedulePaint = true;
 | 
						|
  if (nsLayoutUtils::AsyncPanZoomEnabled(this) &&
 | 
						|
      !nsLayoutUtils::ShouldDisableApzForElement(content) &&
 | 
						|
      !content->GetProperty(nsGkAtoms::MinimalDisplayPort) &&
 | 
						|
      StaticPrefs::apz_paint_skipping_enabled()) {
 | 
						|
    // If APZ is enabled with paint-skipping, there are certain conditions in
 | 
						|
    // which we can skip paints:
 | 
						|
    // 1) If APZ triggered this scroll, and the tile-aligned displayport is
 | 
						|
    //    unchanged.
 | 
						|
    // 2) If non-APZ triggered this scroll, but we can handle it by just asking
 | 
						|
    //    APZ to update the scroll position. Again we make this conditional on
 | 
						|
    //    the tile-aligned displayport being unchanged.
 | 
						|
    // We do the displayport check first since it's common to all scenarios,
 | 
						|
    // and then if the displayport is unchanged, we check if APZ triggered,
 | 
						|
    // or can handle, this scroll. If so, we set schedulePaint to false and
 | 
						|
    // skip the paint.
 | 
						|
    // Because of bug 1264297, we also don't do paint-skipping for elements with
 | 
						|
    // perspective, because the displayport may not have captured everything
 | 
						|
    // that needs to be painted. So even if the final tile-aligned displayport
 | 
						|
    // is the same, we force a repaint for these elements. Bug 1254260 tracks
 | 
						|
    // fixing this properly.
 | 
						|
    nsRect displayPort;
 | 
						|
    bool usingDisplayPort =
 | 
						|
        DisplayPortUtils::GetDisplayPort(content, &displayPort);
 | 
						|
    displayPort.MoveBy(-mScrolledFrame->GetPosition());
 | 
						|
 | 
						|
    PAINT_SKIP_LOG(
 | 
						|
        "New scrollpos %s usingDP %d dpEqual %d scrollableByApz "
 | 
						|
        "%d perspective %d bglocal %d filter %d\n",
 | 
						|
        ToString(CSSPoint::FromAppUnits(GetScrollPosition())).c_str(),
 | 
						|
        usingDisplayPort, displayPort.IsEqualEdges(oldDisplayPort),
 | 
						|
        mScrollableByAPZ, HasPerspective(), HasBgAttachmentLocal(),
 | 
						|
        mHasOutOfFlowContentInsideFilter);
 | 
						|
    if (usingDisplayPort && displayPort.IsEqualEdges(oldDisplayPort) &&
 | 
						|
        !HasPerspective() && !HasBgAttachmentLocal() &&
 | 
						|
        !mHasOutOfFlowContentInsideFilter) {
 | 
						|
      bool haveScrollLinkedEffects =
 | 
						|
          content->GetComposedDoc()->HasScrollLinkedEffect();
 | 
						|
      bool apzDisabled = haveScrollLinkedEffects &&
 | 
						|
                         StaticPrefs::apz_disable_for_scroll_linked_effects();
 | 
						|
      if (!apzDisabled) {
 | 
						|
        if (LastScrollOrigin() == ScrollOrigin::Apz) {
 | 
						|
          schedulePaint = false;
 | 
						|
          PAINT_SKIP_LOG("Skipping due to APZ scroll\n");
 | 
						|
        } else if (mScrollableByAPZ) {
 | 
						|
          nsIWidget* widget = presContext->GetNearestWidget();
 | 
						|
          WindowRenderer* renderer =
 | 
						|
              widget ? widget->GetWindowRenderer() : nullptr;
 | 
						|
          if (renderer) {
 | 
						|
            mozilla::layers::ScrollableLayerGuid::ViewID id;
 | 
						|
            bool success = nsLayoutUtils::FindIDFor(content, &id);
 | 
						|
            MOZ_ASSERT(success);  // we have a displayport, we better have an ID
 | 
						|
 | 
						|
            // Schedule an empty transaction to carry over the scroll offset
 | 
						|
            // update, instead of a full transaction. This empty transaction
 | 
						|
            // might still get squashed into a full transaction if something
 | 
						|
            // happens to trigger one.
 | 
						|
            MOZ_ASSERT(!mScrollUpdates.IsEmpty());
 | 
						|
            success = renderer->AddPendingScrollUpdateForNextTransaction(
 | 
						|
                id, mScrollUpdates.LastElement());
 | 
						|
            if (success) {
 | 
						|
              schedulePaint = false;
 | 
						|
              SchedulePaint(nsIFrame::PAINT_COMPOSITE_ONLY);
 | 
						|
              PAINT_SKIP_LOG(
 | 
						|
                  "Skipping due to APZ-forwarded main-thread scroll\n");
 | 
						|
            } else {
 | 
						|
              PAINT_SKIP_LOG(
 | 
						|
                  "Failed to set pending scroll update on layer manager\n");
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // If the new scroll offset is going to clobber APZ's scroll offset, for
 | 
						|
  // the RCD-RSF this will have the effect of updating the visual viewport
 | 
						|
  // offset in a way that keeps the relative offset between the layout and
 | 
						|
  // visual viewports constant. This will cause APZ to send us a new visual
 | 
						|
  // viewport offset, but instead of waiting for  that, just set the value
 | 
						|
  // we expect APZ will set ourselves, to minimize the chances of
 | 
						|
  // inconsistencies from querying a stale value.
 | 
						|
  if (mIsRoot && nsLayoutUtils::CanScrollOriginClobberApz(aOrigin)) {
 | 
						|
    AutoWeakFrame weakFrame(this);
 | 
						|
    AutoScrollbarRepaintSuppression repaintSuppression(this, weakFrame,
 | 
						|
                                                       !schedulePaint);
 | 
						|
 | 
						|
    nsPoint visualViewportOffset = curPos;
 | 
						|
    if (presContext->PresShell()->IsVisualViewportOffsetSet()) {
 | 
						|
      visualViewportOffset =
 | 
						|
          presContext->PresShell()->GetVisualViewportOffset();
 | 
						|
    }
 | 
						|
    nsPoint relativeOffset = visualViewportOffset - curPos;
 | 
						|
 | 
						|
    presContext->PresShell()->SetVisualViewportOffset(pt + relativeOffset,
 | 
						|
                                                      curPos);
 | 
						|
    if (!weakFrame.IsAlive()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (schedulePaint) {
 | 
						|
    SchedulePaint();
 | 
						|
 | 
						|
    if (needFrameVisibilityUpdate) {
 | 
						|
      presContext->PresShell()->ScheduleApproximateFrameVisibilityUpdateNow();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (ChildrenHavePerspective()) {
 | 
						|
    // The overflow areas of descendants may depend on the scroll position,
 | 
						|
    // so ensure they get updated.
 | 
						|
 | 
						|
    // First we recompute the overflow areas of the transformed children
 | 
						|
    // that use the perspective. FinishAndStoreOverflow only calls this
 | 
						|
    // if the size changes, so we need to do it manually.
 | 
						|
    RecomputePerspectiveChildrenOverflow(this);
 | 
						|
 | 
						|
    // Update the overflow for the scrolled frame to take any changes from the
 | 
						|
    // children into account.
 | 
						|
    mScrolledFrame->UpdateOverflow();
 | 
						|
 | 
						|
    // Update the overflow for the outer so that we recompute scrollbars.
 | 
						|
    UpdateOverflow();
 | 
						|
  }
 | 
						|
 | 
						|
  ScheduleSyntheticMouseMove();
 | 
						|
 | 
						|
  nsAutoScriptBlocker scriptBlocker;
 | 
						|
  PresShell::AutoAssertNoFlush noFlush(*PresShell());
 | 
						|
 | 
						|
  {  // scope the AutoScrollbarRepaintSuppression
 | 
						|
    AutoWeakFrame weakFrame(this);
 | 
						|
    AutoScrollbarRepaintSuppression repaintSuppression(this, weakFrame,
 | 
						|
                                                       !schedulePaint);
 | 
						|
    UpdateScrollbarPosition();
 | 
						|
    if (!weakFrame.IsAlive()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  presContext->RecordInteractionTime(
 | 
						|
      nsPresContext::InteractionType::ScrollInteraction, TimeStamp::Now());
 | 
						|
 | 
						|
  PostScrollEvent();
 | 
						|
  // If this is a viewport scroll, this could affect the relative offset
 | 
						|
  // between layout and visual viewport, so we might have to fire a visual
 | 
						|
  // viewport scroll event as well.
 | 
						|
  if (mIsRoot) {
 | 
						|
    if (auto* window = nsGlobalWindowInner::Cast(
 | 
						|
            PresContext()->Document()->GetInnerWindow())) {
 | 
						|
      window->VisualViewport()->PostScrollEvent(
 | 
						|
          presContext->PresShell()->GetVisualViewportOffset(), curPos);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Schedule the scroll-timelines linked to its scrollable frame.
 | 
						|
  // if `pt == curPos`, we early return, so the position must be changed at
 | 
						|
  // this moment. Therefore, we can schedule scroll animations directly.
 | 
						|
  ScheduleScrollAnimations();
 | 
						|
 | 
						|
  // notify the listeners.
 | 
						|
  for (uint32_t i = 0; i < mListeners.Length(); i++) {
 | 
						|
    mListeners[i]->ScrollPositionDidChange(pt.x, pt.y);
 | 
						|
  }
 | 
						|
 | 
						|
  if (nsCOMPtr<nsIDocShell> docShell = presContext->GetDocShell()) {
 | 
						|
    docShell->NotifyScrollObservers();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Finds the max z-index of the items in aList that meet the following
 | 
						|
// conditions
 | 
						|
//   1) have z-index auto or z-index >= 0, and
 | 
						|
//   2) aFrame is a proper ancestor of the item's frame.
 | 
						|
// Returns Nothing() if there is no such item.
 | 
						|
static Maybe<int32_t> MaxZIndexInListOfItemsContainedInFrame(
 | 
						|
    nsDisplayList* aList, nsIFrame* aFrame) {
 | 
						|
  Maybe<int32_t> maxZIndex = Nothing();
 | 
						|
  for (nsDisplayItem* item : *aList) {
 | 
						|
    int32_t zIndex = item->ZIndex();
 | 
						|
    if (zIndex < 0 ||
 | 
						|
        !nsLayoutUtils::IsProperAncestorFrame(aFrame, item->Frame())) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
    if (!maxZIndex) {
 | 
						|
      maxZIndex = Some(zIndex);
 | 
						|
    } else {
 | 
						|
      maxZIndex = Some(std::max(maxZIndex.value(), zIndex));
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return maxZIndex;
 | 
						|
}
 | 
						|
 | 
						|
template <class T>
 | 
						|
static void AppendInternalItemToTop(const nsDisplayListSet& aLists, T* aItem,
 | 
						|
                                    const Maybe<int32_t>& aZIndex) {
 | 
						|
  if (aZIndex) {
 | 
						|
    aItem->SetOverrideZIndex(aZIndex.value());
 | 
						|
    aLists.PositionedDescendants()->AppendToTop(aItem);
 | 
						|
  } else {
 | 
						|
    aLists.Content()->AppendToTop(aItem);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static const uint32_t APPEND_OWN_LAYER = 0x1;
 | 
						|
static const uint32_t APPEND_POSITIONED = 0x2;
 | 
						|
static const uint32_t APPEND_SCROLLBAR_CONTAINER = 0x4;
 | 
						|
static const uint32_t APPEND_OVERLAY = 0x8;
 | 
						|
static const uint32_t APPEND_TOP = 0x10;
 | 
						|
 | 
						|
static void AppendToTop(nsDisplayListBuilder* aBuilder,
 | 
						|
                        const nsDisplayListSet& aLists, nsDisplayList* aSource,
 | 
						|
                        nsIFrame* aSourceFrame, nsIFrame* aScrollFrame,
 | 
						|
                        uint32_t aFlags) {
 | 
						|
  if (aSource->IsEmpty()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsDisplayWrapList* newItem;
 | 
						|
  const ActiveScrolledRoot* asr = aBuilder->CurrentActiveScrolledRoot();
 | 
						|
  if (aFlags & APPEND_OWN_LAYER) {
 | 
						|
    ScrollbarData scrollbarData;
 | 
						|
    if (aFlags & APPEND_SCROLLBAR_CONTAINER) {
 | 
						|
      scrollbarData = ScrollbarData::CreateForScrollbarContainer(
 | 
						|
          aBuilder->GetCurrentScrollbarDirection(),
 | 
						|
          aBuilder->GetCurrentScrollbarTarget());
 | 
						|
      // Direction should be set
 | 
						|
      MOZ_ASSERT(scrollbarData.mDirection.isSome());
 | 
						|
    }
 | 
						|
 | 
						|
    newItem = MakeDisplayItemWithIndex<nsDisplayOwnLayer>(
 | 
						|
        aBuilder, aSourceFrame,
 | 
						|
        /* aIndex = */ nsDisplayOwnLayer::OwnLayerForScrollbar, aSource, asr,
 | 
						|
        nsDisplayOwnLayerFlags::None, scrollbarData, true, false);
 | 
						|
  } else {
 | 
						|
    // Build the wrap list with an index of 1, since the scrollbar frame itself
 | 
						|
    // might have already built an nsDisplayWrapList.
 | 
						|
    newItem = MakeDisplayItemWithIndex<nsDisplayWrapper>(
 | 
						|
        aBuilder, aSourceFrame, 1, aSource, asr, false);
 | 
						|
  }
 | 
						|
  if (!newItem) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aFlags & APPEND_POSITIONED) {
 | 
						|
    // We want overlay scrollbars to always be on top of the scrolled content,
 | 
						|
    // but we don't want them to unnecessarily cover overlapping elements from
 | 
						|
    // outside our scroll frame.
 | 
						|
    Maybe<int32_t> zIndex = Nothing();
 | 
						|
    if (aFlags & APPEND_TOP) {
 | 
						|
      zIndex = Some(INT32_MAX);
 | 
						|
    } else if (aFlags & APPEND_OVERLAY) {
 | 
						|
      zIndex = MaxZIndexInListOfItemsContainedInFrame(
 | 
						|
          aLists.PositionedDescendants(), aScrollFrame);
 | 
						|
    } else if (aSourceFrame->StylePosition()->mZIndex.IsInteger()) {
 | 
						|
      zIndex = Some(aSourceFrame->StylePosition()->mZIndex.integer._0);
 | 
						|
    }
 | 
						|
    AppendInternalItemToTop(aLists, newItem, zIndex);
 | 
						|
  } else {
 | 
						|
    aLists.BorderBackground()->AppendToTop(newItem);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
struct HoveredStateComparator {
 | 
						|
  static bool Hovered(const nsIFrame* aFrame) {
 | 
						|
    return aFrame->GetContent()->IsElement() &&
 | 
						|
           aFrame->GetContent()->AsElement()->HasAttr(nsGkAtoms::hover);
 | 
						|
  }
 | 
						|
 | 
						|
  bool Equals(nsIFrame* A, nsIFrame* B) const {
 | 
						|
    return Hovered(A) == Hovered(B);
 | 
						|
  }
 | 
						|
 | 
						|
  bool LessThan(nsIFrame* A, nsIFrame* B) const {
 | 
						|
    return !Hovered(A) && Hovered(B);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AppendScrollPartsTo(nsDisplayListBuilder* aBuilder,
 | 
						|
                                            const nsDisplayListSet& aLists,
 | 
						|
                                            bool aCreateLayer,
 | 
						|
                                            bool aPositioned) {
 | 
						|
  const bool overlayScrollbars = UsesOverlayScrollbars();
 | 
						|
 | 
						|
  AutoTArray<nsIFrame*, 3> scrollParts;
 | 
						|
  for (nsIFrame* kid : PrincipalChildList()) {
 | 
						|
    if (kid == mScrolledFrame ||
 | 
						|
        (kid->IsAbsPosContainingBlock() || overlayScrollbars) != aPositioned) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    scrollParts.AppendElement(kid);
 | 
						|
  }
 | 
						|
  if (scrollParts.IsEmpty()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // We can't check will-change budget during display list building phase.
 | 
						|
  // This means that we will build scroll bar layers for out of budget
 | 
						|
  // will-change: scroll position.
 | 
						|
  const mozilla::layers::ScrollableLayerGuid::ViewID scrollTargetId =
 | 
						|
      IsScrollingActive()
 | 
						|
          ? nsLayoutUtils::FindOrCreateIDFor(mScrolledFrame->GetContent())
 | 
						|
          : mozilla::layers::ScrollableLayerGuid::NULL_SCROLL_ID;
 | 
						|
 | 
						|
  scrollParts.Sort(HoveredStateComparator());
 | 
						|
 | 
						|
  DisplayListClipState::AutoSaveRestore clipState(aBuilder);
 | 
						|
  // Don't let scrollparts extent outside our frame's border-box, if these are
 | 
						|
  // viewport scrollbars. They would create layerization problems. This wouldn't
 | 
						|
  // normally be an issue but themes can add overflow areas to scrollbar parts.
 | 
						|
  if (mIsRoot) {
 | 
						|
    nsRect scrollPartsClip(aBuilder->ToReferenceFrame(this),
 | 
						|
                           TrueOuterSize(aBuilder));
 | 
						|
    clipState.ClipContentDescendants(scrollPartsClip);
 | 
						|
  }
 | 
						|
 | 
						|
  for (uint32_t i = 0; i < scrollParts.Length(); ++i) {
 | 
						|
    MOZ_ASSERT(scrollParts[i]);
 | 
						|
    Maybe<ScrollDirection> scrollDirection;
 | 
						|
    uint32_t appendToTopFlags = 0;
 | 
						|
    if (scrollParts[i] == mVScrollbarBox) {
 | 
						|
      scrollDirection.emplace(ScrollDirection::eVertical);
 | 
						|
      appendToTopFlags |= APPEND_SCROLLBAR_CONTAINER;
 | 
						|
    }
 | 
						|
    if (scrollParts[i] == mHScrollbarBox) {
 | 
						|
      MOZ_ASSERT(!scrollDirection.isSome());
 | 
						|
      scrollDirection.emplace(ScrollDirection::eHorizontal);
 | 
						|
      appendToTopFlags |= APPEND_SCROLLBAR_CONTAINER;
 | 
						|
    }
 | 
						|
 | 
						|
    // The display port doesn't necessarily include the scrollbars, so just
 | 
						|
    // include all of the scrollbars if we are in a RCD-RSF. We only do
 | 
						|
    // this for the root scrollframe of the root content document, which is
 | 
						|
    // zoomable, and where the scrollbar sizes are bounded by the widget.
 | 
						|
    const nsRect visible =
 | 
						|
        mIsRoot && PresContext()->IsRootContentDocumentCrossProcess()
 | 
						|
            ? scrollParts[i]->InkOverflowRectRelativeToParent()
 | 
						|
            : aBuilder->GetVisibleRect();
 | 
						|
    if (visible.IsEmpty()) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
    const nsRect dirty =
 | 
						|
        mIsRoot && PresContext()->IsRootContentDocumentCrossProcess()
 | 
						|
            ? scrollParts[i]->InkOverflowRectRelativeToParent()
 | 
						|
            : aBuilder->GetDirtyRect();
 | 
						|
 | 
						|
    // Always create layers for overlay scrollbars so that we don't create a
 | 
						|
    // giant layer covering the whole scrollport if both scrollbars are visible.
 | 
						|
    const bool isOverlayScrollbar =
 | 
						|
        scrollDirection.isSome() && overlayScrollbars;
 | 
						|
    const bool createLayer =
 | 
						|
        aCreateLayer || isOverlayScrollbar ||
 | 
						|
        StaticPrefs::layout_scrollbars_always_layerize_track();
 | 
						|
 | 
						|
    nsDisplayListCollection partList(aBuilder);
 | 
						|
    {
 | 
						|
      nsDisplayListBuilder::AutoBuildingDisplayList buildingForChild(
 | 
						|
          aBuilder, this, visible, dirty);
 | 
						|
 | 
						|
      nsDisplayListBuilder::AutoCurrentScrollbarInfoSetter infoSetter(
 | 
						|
          aBuilder, scrollTargetId, scrollDirection, createLayer);
 | 
						|
      BuildDisplayListForChild(
 | 
						|
          aBuilder, scrollParts[i], partList,
 | 
						|
          nsIFrame::DisplayChildFlag::ForceStackingContext);
 | 
						|
    }
 | 
						|
 | 
						|
    // DisplayChildFlag::ForceStackingContext put everything into
 | 
						|
    // partList.PositionedDescendants().
 | 
						|
    if (partList.PositionedDescendants()->IsEmpty()) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    if (createLayer) {
 | 
						|
      appendToTopFlags |= APPEND_OWN_LAYER;
 | 
						|
    }
 | 
						|
    if (aPositioned) {
 | 
						|
      appendToTopFlags |= APPEND_POSITIONED;
 | 
						|
    }
 | 
						|
 | 
						|
    if (isOverlayScrollbar || scrollParts[i] == mResizerBox) {
 | 
						|
      if (isOverlayScrollbar && mIsRoot) {
 | 
						|
        appendToTopFlags |= APPEND_TOP;
 | 
						|
      } else {
 | 
						|
        appendToTopFlags |= APPEND_OVERLAY;
 | 
						|
        aBuilder->SetDisablePartialUpdates(true);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    {
 | 
						|
      nsDisplayListBuilder::AutoBuildingDisplayList buildingForChild(
 | 
						|
          aBuilder, scrollParts[i], visible + GetOffsetTo(scrollParts[i]),
 | 
						|
          dirty + GetOffsetTo(scrollParts[i]));
 | 
						|
      if (scrollParts[i]->IsTransformed()) {
 | 
						|
        nsPoint toOuterReferenceFrame;
 | 
						|
        const nsIFrame* outerReferenceFrame = aBuilder->FindReferenceFrameFor(
 | 
						|
            scrollParts[i]->GetParent(), &toOuterReferenceFrame);
 | 
						|
        toOuterReferenceFrame += scrollParts[i]->GetPosition();
 | 
						|
 | 
						|
        buildingForChild.SetReferenceFrameAndCurrentOffset(
 | 
						|
            outerReferenceFrame, toOuterReferenceFrame);
 | 
						|
      }
 | 
						|
      nsDisplayListBuilder::AutoCurrentScrollbarInfoSetter infoSetter(
 | 
						|
          aBuilder, scrollTargetId, scrollDirection, createLayer);
 | 
						|
 | 
						|
      ::AppendToTop(aBuilder, aLists, partList.PositionedDescendants(),
 | 
						|
                    scrollParts[i], this, appendToTopFlags);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::ExpandRectToNearlyVisible(const nsRect& aRect) const {
 | 
						|
  // We don't want to expand a rect in a direction that we can't scroll, so we
 | 
						|
  // check the scroll range.
 | 
						|
  nsRect scrollRange = GetLayoutScrollRange();
 | 
						|
  nsPoint scrollPos = GetScrollPosition();
 | 
						|
  nsMargin expand(0, 0, 0, 0);
 | 
						|
 | 
						|
  nscoord vertShift =
 | 
						|
      StaticPrefs::layout_framevisibility_numscrollportheights() * aRect.height;
 | 
						|
  if (scrollRange.y < scrollPos.y) {
 | 
						|
    expand.top = vertShift;
 | 
						|
  }
 | 
						|
  if (scrollPos.y < scrollRange.YMost()) {
 | 
						|
    expand.bottom = vertShift;
 | 
						|
  }
 | 
						|
 | 
						|
  nscoord horzShift =
 | 
						|
      StaticPrefs::layout_framevisibility_numscrollportwidths() * aRect.width;
 | 
						|
  if (scrollRange.x < scrollPos.x) {
 | 
						|
    expand.left = horzShift;
 | 
						|
  }
 | 
						|
  if (scrollPos.x < scrollRange.XMost()) {
 | 
						|
    expand.right = horzShift;
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect rect = aRect;
 | 
						|
  rect.Inflate(expand);
 | 
						|
  return rect;
 | 
						|
}
 | 
						|
 | 
						|
static bool ShouldBeClippedByFrame(nsIFrame* aClipFrame,
 | 
						|
                                   nsIFrame* aClippedFrame) {
 | 
						|
  return nsLayoutUtils::IsProperAncestorFrame(aClipFrame, aClippedFrame);
 | 
						|
}
 | 
						|
 | 
						|
static void ClipItemsExceptCaret(
 | 
						|
    nsDisplayList* aList, nsDisplayListBuilder* aBuilder, nsIFrame* aClipFrame,
 | 
						|
    const DisplayItemClipChain* aExtraClip,
 | 
						|
    nsTHashMap<nsPtrHashKey<const DisplayItemClipChain>,
 | 
						|
               const DisplayItemClipChain*>& aCache) {
 | 
						|
  for (nsDisplayItem* i : *aList) {
 | 
						|
    if (!ShouldBeClippedByFrame(aClipFrame, i->Frame())) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    const DisplayItemType type = i->GetType();
 | 
						|
    if (type != DisplayItemType::TYPE_CARET &&
 | 
						|
        type != DisplayItemType::TYPE_CONTAINER) {
 | 
						|
      const DisplayItemClipChain* clip = i->GetClipChain();
 | 
						|
      const DisplayItemClipChain* intersection = nullptr;
 | 
						|
      if (aCache.Get(clip, &intersection)) {
 | 
						|
        i->SetClipChain(intersection, true);
 | 
						|
      } else {
 | 
						|
        i->IntersectClip(aBuilder, aExtraClip, true);
 | 
						|
        aCache.InsertOrUpdate(clip, i->GetClipChain());
 | 
						|
      }
 | 
						|
    }
 | 
						|
    nsDisplayList* children = i->GetSameCoordinateSystemChildren();
 | 
						|
    if (children) {
 | 
						|
      ClipItemsExceptCaret(children, aBuilder, aClipFrame, aExtraClip, aCache);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static void ClipListsExceptCaret(nsDisplayListCollection* aLists,
 | 
						|
                                 nsDisplayListBuilder* aBuilder,
 | 
						|
                                 nsIFrame* aClipFrame,
 | 
						|
                                 const DisplayItemClipChain* aExtraClip) {
 | 
						|
  nsTHashMap<nsPtrHashKey<const DisplayItemClipChain>,
 | 
						|
             const DisplayItemClipChain*>
 | 
						|
      cache;
 | 
						|
  ClipItemsExceptCaret(aLists->BorderBackground(), aBuilder, aClipFrame,
 | 
						|
                       aExtraClip, cache);
 | 
						|
  ClipItemsExceptCaret(aLists->BlockBorderBackgrounds(), aBuilder, aClipFrame,
 | 
						|
                       aExtraClip, cache);
 | 
						|
  ClipItemsExceptCaret(aLists->Floats(), aBuilder, aClipFrame, aExtraClip,
 | 
						|
                       cache);
 | 
						|
  ClipItemsExceptCaret(aLists->PositionedDescendants(), aBuilder, aClipFrame,
 | 
						|
                       aExtraClip, cache);
 | 
						|
  ClipItemsExceptCaret(aLists->Outlines(), aBuilder, aClipFrame, aExtraClip,
 | 
						|
                       cache);
 | 
						|
  ClipItemsExceptCaret(aLists->Content(), aBuilder, aClipFrame, aExtraClip,
 | 
						|
                       cache);
 | 
						|
}
 | 
						|
 | 
						|
// This is similar to a "save-restore" RAII class for
 | 
						|
// DisplayListBuilder::ContainsBlendMode(), with a slight enhancement.
 | 
						|
// If this class is put on the stack and then unwound, the DL builder's
 | 
						|
// ContainsBlendMode flag will be effectively the same as if this class wasn't
 | 
						|
// put on the stack. However, if the CaptureContainsBlendMode method is called,
 | 
						|
// there will be a difference - the blend mode in the descendant display lists
 | 
						|
// will be "captured" and extracted.
 | 
						|
// The main goal here is to allow conditionally capturing the flag that
 | 
						|
// indicates whether or not a blend mode was encountered in the descendant part
 | 
						|
// of the display list.
 | 
						|
class MOZ_RAII AutoContainsBlendModeCapturer {
 | 
						|
  nsDisplayListBuilder& mBuilder;
 | 
						|
  bool mSavedContainsBlendMode;
 | 
						|
 | 
						|
 public:
 | 
						|
  explicit AutoContainsBlendModeCapturer(nsDisplayListBuilder& aBuilder)
 | 
						|
      : mBuilder(aBuilder),
 | 
						|
        mSavedContainsBlendMode(aBuilder.ContainsBlendMode()) {
 | 
						|
    mBuilder.SetContainsBlendMode(false);
 | 
						|
  }
 | 
						|
 | 
						|
  bool CaptureContainsBlendMode() {
 | 
						|
    // "Capture" the flag by extracting and clearing the ContainsBlendMode flag
 | 
						|
    // on the builder.
 | 
						|
    bool capturedBlendMode = mBuilder.ContainsBlendMode();
 | 
						|
    mBuilder.SetContainsBlendMode(false);
 | 
						|
    return capturedBlendMode;
 | 
						|
  }
 | 
						|
 | 
						|
  ~AutoContainsBlendModeCapturer() {
 | 
						|
    // If CaptureContainsBlendMode() was called, the descendant blend mode was
 | 
						|
    // "captured" and so uncapturedContainsBlendMode will be false. If
 | 
						|
    // CaptureContainsBlendMode() wasn't called, then no capture occurred, and
 | 
						|
    // uncapturedContainsBlendMode may be true if there was a descendant blend
 | 
						|
    // mode. In that case, we set the flag on the DL builder so that we restore
 | 
						|
    // state to what it would have been without this RAII class on the stack.
 | 
						|
    bool uncapturedContainsBlendMode = mBuilder.ContainsBlendMode();
 | 
						|
    mBuilder.SetContainsBlendMode(mSavedContainsBlendMode ||
 | 
						|
                                  uncapturedContainsBlendMode);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
void nsHTMLScrollFrame::MaybeCreateTopLayerAndWrapRootItems(
 | 
						|
    nsDisplayListBuilder* aBuilder, nsDisplayListCollection& aSet,
 | 
						|
    bool aCreateAsyncZoom,
 | 
						|
    AutoContainsBlendModeCapturer* aAsyncZoomBlendCapture,
 | 
						|
    const nsRect& aAsyncZoomClipRect, nscoord* aRadii) {
 | 
						|
  if (!mIsRoot) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Create any required items for the 'top layer' and check if they'll be
 | 
						|
  // opaque over the entire area of the viewport. If they are, then we can
 | 
						|
  // skip building display items for the rest of the page.
 | 
						|
  if (ViewportFrame* viewport = do_QueryFrame(GetParent())) {
 | 
						|
    bool topLayerIsOpaque = false;
 | 
						|
    if (nsDisplayWrapList* topLayerWrapList =
 | 
						|
            viewport->BuildDisplayListForTopLayer(aBuilder,
 | 
						|
                                                  &topLayerIsOpaque)) {
 | 
						|
      // If the top layer content is opaque, and we're the root content document
 | 
						|
      // in the process, we can drop the display items behind it. We only
 | 
						|
      // support doing this for the root content document in the process, since
 | 
						|
      // the top layer content might have fixed position items that have a
 | 
						|
      // scrolltarget referencing the APZ data for the document. APZ builds this
 | 
						|
      // data implicitly for the root content document in the process, but
 | 
						|
      // subdocuments etc need their display items to generate it, so we can't
 | 
						|
      // cull those.
 | 
						|
      if (topLayerIsOpaque && PresContext()->IsRootContentDocumentInProcess()) {
 | 
						|
        aSet.DeleteAll(aBuilder);
 | 
						|
      }
 | 
						|
      aSet.PositionedDescendants()->AppendToTop(topLayerWrapList);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nsDisplayList rootResultList(aBuilder);
 | 
						|
 | 
						|
  bool serializedList = false;
 | 
						|
  auto SerializeList = [&] {
 | 
						|
    if (!serializedList) {
 | 
						|
      serializedList = true;
 | 
						|
      aSet.SerializeWithCorrectZOrder(&rootResultList, GetContent());
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  if (nsIFrame* rootStyleFrame = GetFrameForStyle()) {
 | 
						|
    bool usingBackdropFilter =
 | 
						|
        rootStyleFrame->StyleEffects()->HasBackdropFilters() &&
 | 
						|
        rootStyleFrame->IsVisibleForPainting();
 | 
						|
 | 
						|
    if (rootStyleFrame->StyleEffects()->HasFilters() &&
 | 
						|
        !aBuilder->IsForGenerateGlyphMask()) {
 | 
						|
      SerializeList();
 | 
						|
      rootResultList.AppendNewToTop<nsDisplayFilters>(
 | 
						|
          aBuilder, this, &rootResultList, rootStyleFrame, usingBackdropFilter);
 | 
						|
    }
 | 
						|
 | 
						|
    if (usingBackdropFilter) {
 | 
						|
      SerializeList();
 | 
						|
      DisplayListClipState::AutoSaveRestore clipState(aBuilder);
 | 
						|
      nsRect backdropRect =
 | 
						|
          GetRectRelativeToSelf() + aBuilder->ToReferenceFrame(this);
 | 
						|
      rootResultList.AppendNewToTop<nsDisplayBackdropFilters>(
 | 
						|
          aBuilder, this, &rootResultList, backdropRect, rootStyleFrame);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (aCreateAsyncZoom) {
 | 
						|
    MOZ_ASSERT(mIsRoot);
 | 
						|
 | 
						|
    // Wrap all our scrolled contents in an nsDisplayAsyncZoom. This will be
 | 
						|
    // the layer that gets scaled for APZ zooming. It does not have the
 | 
						|
    // scrolled ASR, but it does have the composition bounds clip applied to
 | 
						|
    // it. The children have the layout viewport clip applied to them (above).
 | 
						|
    // Effectively we are double clipping to the viewport, at potentially
 | 
						|
    // different async scales.
 | 
						|
    SerializeList();
 | 
						|
 | 
						|
    if (aAsyncZoomBlendCapture->CaptureContainsBlendMode()) {
 | 
						|
      // The async zoom contents contain a mix-blend mode, so let's wrap all
 | 
						|
      // those contents into a blend container, and then wrap the blend
 | 
						|
      // container in the async zoom container. Otherwise the blend container
 | 
						|
      // ends up outside the zoom container which results in blend failure for
 | 
						|
      // WebRender.
 | 
						|
      nsDisplayItem* blendContainer =
 | 
						|
          nsDisplayBlendContainer::CreateForMixBlendMode(
 | 
						|
              aBuilder, this, &rootResultList,
 | 
						|
              aBuilder->CurrentActiveScrolledRoot());
 | 
						|
      rootResultList.AppendToTop(blendContainer);
 | 
						|
 | 
						|
      // Blend containers can be created or omitted during partial updates
 | 
						|
      // depending on the dirty rect. So we basically can't do partial updates
 | 
						|
      // if there's a blend container involved. There is equivalent code to this
 | 
						|
      // in the BuildDisplayListForStackingContext function as well, with a more
 | 
						|
      // detailed comment explaining things better.
 | 
						|
      if (aBuilder->IsRetainingDisplayList()) {
 | 
						|
        if (aBuilder->IsPartialUpdate()) {
 | 
						|
          aBuilder->SetPartialBuildFailed(true);
 | 
						|
        } else {
 | 
						|
          aBuilder->SetDisablePartialUpdates(true);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    mozilla::layers::FrameMetrics::ViewID viewID =
 | 
						|
        nsLayoutUtils::FindOrCreateIDFor(mScrolledFrame->GetContent());
 | 
						|
 | 
						|
    DisplayListClipState::AutoSaveRestore clipState(aBuilder);
 | 
						|
    clipState.ClipContentDescendants(aAsyncZoomClipRect, aRadii);
 | 
						|
 | 
						|
    rootResultList.AppendNewToTop<nsDisplayAsyncZoom>(
 | 
						|
        aBuilder, this, &rootResultList, aBuilder->CurrentActiveScrolledRoot(),
 | 
						|
        viewID);
 | 
						|
  }
 | 
						|
 | 
						|
  if (serializedList) {
 | 
						|
    aSet.Content()->AppendToTop(&rootResultList);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::BuildDisplayList(nsDisplayListBuilder* aBuilder,
 | 
						|
                                         const nsDisplayListSet& aLists) {
 | 
						|
  SetAndNullOnExit<const nsIFrame> tmpBuilder(
 | 
						|
      mReferenceFrameDuringPainting, aBuilder->GetCurrentReferenceFrame());
 | 
						|
  if (aBuilder->IsForFrameVisibility()) {
 | 
						|
    NotifyApproximateFrameVisibilityUpdate(false);
 | 
						|
  }
 | 
						|
 | 
						|
  DisplayBorderBackgroundOutline(aBuilder, aLists);
 | 
						|
 | 
						|
  const bool isRootContent =
 | 
						|
      mIsRoot && PresContext()->IsRootContentDocumentCrossProcess();
 | 
						|
 | 
						|
  nsRect effectiveScrollPort = mScrollPort;
 | 
						|
  if (isRootContent && PresContext()->HasDynamicToolbar()) {
 | 
						|
    // Expand the scroll port to the size including the area covered by dynamic
 | 
						|
    // toolbar in the case where the dynamic toolbar is being used since
 | 
						|
    // position:fixed elements attached to this root scroller might be taller
 | 
						|
    // than its scroll port (e.g 100vh). Even if the dynamic toolbar covers the
 | 
						|
    // taller area, it doesn't mean the area is clipped by the toolbar because
 | 
						|
    // the dynamic toolbar is laid out outside of our topmost window and it
 | 
						|
    // transitions without changing our topmost window size.
 | 
						|
    effectiveScrollPort.SizeTo(nsLayoutUtils::ExpandHeightForDynamicToolbar(
 | 
						|
        PresContext(), effectiveScrollPort.Size()));
 | 
						|
  }
 | 
						|
 | 
						|
  // It's safe to get this value before the DecideScrollableLayer call below
 | 
						|
  // because that call cannot create a displayport for root scroll frames,
 | 
						|
  // and hence it cannot create an ignore scroll frame.
 | 
						|
  const bool ignoringThisScrollFrame = aBuilder->GetIgnoreScrollFrame() == this;
 | 
						|
 | 
						|
  // Overflow clipping can never clip frames outside our subtree, so there
 | 
						|
  // is no need to worry about whether we are a moving frame that might clip
 | 
						|
  // non-moving frames.
 | 
						|
  // Not all our descendants will be clipped by overflow clipping, but all
 | 
						|
  // the ones that aren't clipped will be out of flow frames that have already
 | 
						|
  // had dirty rects saved for them by their parent frames calling
 | 
						|
  // MarkOutOfFlowChildrenForDisplayList, so it's safe to restrict our
 | 
						|
  // dirty rect here.
 | 
						|
  nsRect visibleRect = aBuilder->GetVisibleRect();
 | 
						|
  nsRect dirtyRect = aBuilder->GetDirtyRect();
 | 
						|
  if (!ignoringThisScrollFrame) {
 | 
						|
    visibleRect = visibleRect.Intersect(effectiveScrollPort);
 | 
						|
    dirtyRect = dirtyRect.Intersect(effectiveScrollPort);
 | 
						|
  }
 | 
						|
 | 
						|
  bool dirtyRectHasBeenOverriden = false;
 | 
						|
  Unused << DecideScrollableLayer(aBuilder, &visibleRect, &dirtyRect,
 | 
						|
                                  /* aSetBase = */ !mIsRoot,
 | 
						|
                                  &dirtyRectHasBeenOverriden);
 | 
						|
 | 
						|
  if (aBuilder->IsForFrameVisibility()) {
 | 
						|
    // We expand the dirty rect to catch frames just outside of the scroll port.
 | 
						|
    // We use the dirty rect instead of the whole scroll port to prevent
 | 
						|
    // too much expansion in the presence of very large (bigger than the
 | 
						|
    // viewport) scroll ports.
 | 
						|
    dirtyRect = ExpandRectToNearlyVisible(dirtyRect);
 | 
						|
    visibleRect = dirtyRect;
 | 
						|
  }
 | 
						|
 | 
						|
  // We put non-overlay scrollbars in their own layers when this is the root
 | 
						|
  // scroll frame and we are a toplevel content document. In this situation,
 | 
						|
  // the scrollbar(s) would normally be assigned their own layer anyway, since
 | 
						|
  // they're not scrolled with the rest of the document. But when both
 | 
						|
  // scrollbars are visible, the layer's visible rectangle would be the size
 | 
						|
  // of the viewport, so most layer implementations would create a layer buffer
 | 
						|
  // that's much larger than necessary. Creating independent layers for each
 | 
						|
  // scrollbar works around the problem.
 | 
						|
  const bool createLayersForScrollbars = isRootContent;
 | 
						|
 | 
						|
  nsDisplayListCollection set(aBuilder);
 | 
						|
 | 
						|
  if (ignoringThisScrollFrame) {
 | 
						|
    // If we are a root scroll frame that has a display port we want to add
 | 
						|
    // scrollbars, they will be children of the scrollable layer, but they get
 | 
						|
    // adjusted by the APZC automatically.
 | 
						|
    bool addScrollBars =
 | 
						|
        mIsRoot && mWillBuildScrollableLayer && aBuilder->IsPaintingToWindow();
 | 
						|
 | 
						|
    if (addScrollBars) {
 | 
						|
      // Add classic scrollbars.
 | 
						|
      AppendScrollPartsTo(aBuilder, set, createLayersForScrollbars, false);
 | 
						|
    }
 | 
						|
 | 
						|
    {
 | 
						|
      nsDisplayListBuilder::AutoBuildingDisplayList building(
 | 
						|
          aBuilder, this, visibleRect, dirtyRect);
 | 
						|
 | 
						|
      // Don't clip the scrolled child, and don't paint scrollbars/scrollcorner.
 | 
						|
      // The scrolled frame shouldn't have its own background/border, so we
 | 
						|
      // can just pass aLists directly.
 | 
						|
      BuildDisplayListForChild(aBuilder, mScrolledFrame, set);
 | 
						|
    }
 | 
						|
 | 
						|
    MaybeCreateTopLayerAndWrapRootItems(aBuilder, set,
 | 
						|
                                        /* aCreateAsyncZoom = */ false, nullptr,
 | 
						|
                                        nsRect(), nullptr);
 | 
						|
 | 
						|
    if (addScrollBars) {
 | 
						|
      // Add overlay scrollbars.
 | 
						|
      AppendScrollPartsTo(aBuilder, set, createLayersForScrollbars, true);
 | 
						|
    }
 | 
						|
 | 
						|
    set.MoveTo(aLists);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Whether we might want to build a scrollable layer for this scroll frame
 | 
						|
  // at some point in the future. This controls whether we add the information
 | 
						|
  // to the layer tree (a scroll info layer if necessary, and add the right
 | 
						|
  // area to the dispatch to content layer event regions) necessary to activate
 | 
						|
  // a scroll frame so it creates a scrollable layer.
 | 
						|
  const bool couldBuildLayer = [&] {
 | 
						|
    if (!aBuilder->IsPaintingToWindow()) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    if (mWillBuildScrollableLayer) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
    return StyleVisibility()->IsVisible() &&
 | 
						|
           nsLayoutUtils::AsyncPanZoomEnabled(this) && WantAsyncScroll();
 | 
						|
  }();
 | 
						|
 | 
						|
  // Now display the scrollbars and scrollcorner. These parts are drawn
 | 
						|
  // in the border-background layer, on top of our own background and
 | 
						|
  // borders and underneath borders and backgrounds of later elements
 | 
						|
  // in the tree.
 | 
						|
  // Note that this does not apply for overlay scrollbars; those are drawn
 | 
						|
  // in the positioned-elements layer on top of everything else by the call
 | 
						|
  // to AppendScrollPartsTo(..., true) further down.
 | 
						|
  AppendScrollPartsTo(aBuilder, aLists, createLayersForScrollbars, false);
 | 
						|
 | 
						|
  const nsStyleDisplay* disp = StyleDisplay();
 | 
						|
  if (aBuilder->IsForPainting() &&
 | 
						|
      disp->mWillChange.bits & StyleWillChangeBits::SCROLL) {
 | 
						|
    aBuilder->AddToWillChangeBudget(this, GetVisualViewportSize());
 | 
						|
  }
 | 
						|
 | 
						|
  mScrollParentID = aBuilder->GetCurrentScrollParentId();
 | 
						|
 | 
						|
  Maybe<nsRect> contentBoxClip;
 | 
						|
  Maybe<const DisplayItemClipChain*> extraContentBoxClipForNonCaretContent;
 | 
						|
  if (MOZ_UNLIKELY(
 | 
						|
          disp->mOverflowClipBoxBlock == StyleOverflowClipBox::ContentBox ||
 | 
						|
          disp->mOverflowClipBoxInline == StyleOverflowClipBox::ContentBox)) {
 | 
						|
    WritingMode wm = mScrolledFrame->GetWritingMode();
 | 
						|
    bool cbH = (wm.IsVertical() ? disp->mOverflowClipBoxBlock
 | 
						|
                                : disp->mOverflowClipBoxInline) ==
 | 
						|
               StyleOverflowClipBox::ContentBox;
 | 
						|
    bool cbV = (wm.IsVertical() ? disp->mOverflowClipBoxInline
 | 
						|
                                : disp->mOverflowClipBoxBlock) ==
 | 
						|
               StyleOverflowClipBox::ContentBox;
 | 
						|
    // We only clip if there is *scrollable* overflow, to avoid clipping
 | 
						|
    // *ink* overflow unnecessarily.
 | 
						|
    nsRect clipRect = effectiveScrollPort + aBuilder->ToReferenceFrame(this);
 | 
						|
    nsMargin padding = GetUsedPadding();
 | 
						|
    if (!cbH) {
 | 
						|
      padding.left = padding.right = nscoord(0);
 | 
						|
    }
 | 
						|
    if (!cbV) {
 | 
						|
      padding.top = padding.bottom = nscoord(0);
 | 
						|
    }
 | 
						|
    clipRect.Deflate(padding);
 | 
						|
 | 
						|
    nsRect so = mScrolledFrame->ScrollableOverflowRect();
 | 
						|
    if ((cbH && (clipRect.width != so.width || so.x < 0)) ||
 | 
						|
        (cbV && (clipRect.height != so.height || so.y < 0))) {
 | 
						|
      // The non-inflated clip needs to be set on all non-caret items.
 | 
						|
      // We prepare an extra DisplayItemClipChain here that will be intersected
 | 
						|
      // with those items after they've been created.
 | 
						|
      const ActiveScrolledRoot* asr = aBuilder->CurrentActiveScrolledRoot();
 | 
						|
 | 
						|
      DisplayItemClip newClip;
 | 
						|
      newClip.SetTo(clipRect);
 | 
						|
 | 
						|
      const DisplayItemClipChain* extraClip =
 | 
						|
          aBuilder->AllocateDisplayItemClipChain(newClip, asr, nullptr);
 | 
						|
 | 
						|
      extraContentBoxClipForNonCaretContent = Some(extraClip);
 | 
						|
 | 
						|
      nsIFrame* caretFrame = aBuilder->GetCaretFrame();
 | 
						|
      // Avoid clipping it in a zero-height line box (heuristic only).
 | 
						|
      if (caretFrame && caretFrame->GetRect().height != 0) {
 | 
						|
        nsRect caretRect = aBuilder->GetCaretRect();
 | 
						|
        // Allow the caret to stick out of the content box clip by half the
 | 
						|
        // caret height on the top, and its full width on the right.
 | 
						|
        nsRect inflatedClip = clipRect;
 | 
						|
        inflatedClip.Inflate(
 | 
						|
            nsMargin(caretRect.height / 2, caretRect.width, 0, 0));
 | 
						|
        contentBoxClip = Some(inflatedClip);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  AutoContainsBlendModeCapturer blendCapture(*aBuilder);
 | 
						|
 | 
						|
  const bool willBuildAsyncZoomContainer =
 | 
						|
      mWillBuildScrollableLayer && aBuilder->ShouldBuildAsyncZoomContainer() &&
 | 
						|
      isRootContent;
 | 
						|
 | 
						|
  nsRect scrollPortClip =
 | 
						|
      effectiveScrollPort + aBuilder->ToReferenceFrame(this);
 | 
						|
  nsRect clipRect = scrollPortClip;
 | 
						|
  // Our override of GetBorderRadii ensures we never have a radius at
 | 
						|
  // the corners where we have a scrollbar.
 | 
						|
  nscoord radii[8];
 | 
						|
  const bool haveRadii = GetPaddingBoxBorderRadii(radii);
 | 
						|
  if (mIsRoot) {
 | 
						|
    clipRect.SizeTo(nsLayoutUtils::CalculateCompositionSizeForFrame(
 | 
						|
        this, true /* aSubtractScrollbars */,
 | 
						|
        nullptr /* aOverrideScrollPortSize */,
 | 
						|
        // With the dynamic toolbar, this CalculateCompositionSizeForFrame call
 | 
						|
        // basically expands the region being covered up by the dynamic toolbar,
 | 
						|
        // but if the root scroll container is not scrollable, e.g. the root
 | 
						|
        // element has `overflow: hidden` or `position: fixed`, the function
 | 
						|
        // doesn't expand the region since expanding the region in such cases
 | 
						|
        // will prevent the content from restoring zooming to 1.0 zoom level
 | 
						|
        // such as bug 1652190. That said, this `clipRect` which will be used
 | 
						|
        // for the async zoom container needs to be expanded because zoomed-in
 | 
						|
        // contents can be scrollable __visually__ so that the region under the
 | 
						|
        // dynamic toolbar area will be revealed.
 | 
						|
        nsLayoutUtils::IncludeDynamicToolbar::Force));
 | 
						|
 | 
						|
    // The composition size is essentially in visual coordinates.
 | 
						|
    // If we are hit-testing in layout coordinates, transform the clip rect
 | 
						|
    // to layout coordinates to match.
 | 
						|
    if (aBuilder->IsRelativeToLayoutViewport() && isRootContent) {
 | 
						|
      clipRect = ViewportUtils::VisualToLayout(clipRect, PresShell());
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  {
 | 
						|
    DisplayListClipState::AutoSaveRestore clipState(aBuilder);
 | 
						|
 | 
						|
    // If we're building an async zoom container, clip the contents inside
 | 
						|
    // to the layout viewport (scrollPortClip). The composition bounds clip
 | 
						|
    // (clipRect) will be applied to the zoom container itself in
 | 
						|
    // MaybeCreateTopLayerAndWrapRootItems.
 | 
						|
    nsRect clipRectForContents =
 | 
						|
        willBuildAsyncZoomContainer ? scrollPortClip : clipRect;
 | 
						|
    if (mIsRoot) {
 | 
						|
      clipState.ClipContentDescendants(clipRectForContents,
 | 
						|
                                       haveRadii ? radii : nullptr);
 | 
						|
    } else {
 | 
						|
      clipState.ClipContainingBlockDescendants(clipRectForContents,
 | 
						|
                                               haveRadii ? radii : nullptr);
 | 
						|
    }
 | 
						|
 | 
						|
    Maybe<DisplayListClipState::AutoSaveRestore> contentBoxClipState;
 | 
						|
    if (contentBoxClip) {
 | 
						|
      contentBoxClipState.emplace(aBuilder);
 | 
						|
      if (mIsRoot) {
 | 
						|
        contentBoxClipState->ClipContentDescendants(*contentBoxClip);
 | 
						|
      } else {
 | 
						|
        contentBoxClipState->ClipContainingBlockDescendants(*contentBoxClip);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    nsDisplayListBuilder::AutoCurrentActiveScrolledRootSetter asrSetter(
 | 
						|
        aBuilder);
 | 
						|
 | 
						|
    if (mWillBuildScrollableLayer && aBuilder->IsPaintingToWindow()) {
 | 
						|
      // If this scroll frame has a first scrollable frame sequence number,
 | 
						|
      // ensure that it matches the current paint sequence number. If it does
 | 
						|
      // not, reset it so that we can expire the displayport. The stored
 | 
						|
      // sequence number will not match that of the current paint if the dom
 | 
						|
      // was mutated in some way that alters the order of scroll frames.
 | 
						|
      if (IsFirstScrollableFrameSequenceNumber().isSome() &&
 | 
						|
          *IsFirstScrollableFrameSequenceNumber() !=
 | 
						|
              nsDisplayListBuilder::GetPaintSequenceNumber()) {
 | 
						|
        SetIsFirstScrollableFrameSequenceNumber(Nothing());
 | 
						|
      }
 | 
						|
      asrSetter.EnterScrollFrame(this);
 | 
						|
    }
 | 
						|
 | 
						|
    if (couldBuildLayer && mScrolledFrame->GetContent()) {
 | 
						|
      asrSetter.SetCurrentScrollParentId(
 | 
						|
          nsLayoutUtils::FindOrCreateIDFor(mScrolledFrame->GetContent()));
 | 
						|
    }
 | 
						|
 | 
						|
    if (mWillBuildScrollableLayer && aBuilder->BuildCompositorHitTestInfo()) {
 | 
						|
      // Create a hit test info item for the scrolled content that's not
 | 
						|
      // clipped to the displayport. This ensures that within the bounds
 | 
						|
      // of the scroll frame, the scrolled content is always hit, even
 | 
						|
      // if we are checkerboarding.
 | 
						|
      CompositorHitTestInfo info =
 | 
						|
          mScrolledFrame->GetCompositorHitTestInfo(aBuilder);
 | 
						|
 | 
						|
      if (info != CompositorHitTestInvisibleToHit) {
 | 
						|
        auto* hitInfo =
 | 
						|
            MakeDisplayItemWithIndex<nsDisplayCompositorHitTestInfo>(
 | 
						|
                aBuilder, mScrolledFrame, 1);
 | 
						|
        if (hitInfo) {
 | 
						|
          aBuilder->SetCompositorHitTestInfo(info);
 | 
						|
          set.BorderBackground()->AppendToTop(hitInfo);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    {
 | 
						|
      // Clip our contents to the unsnapped scrolled rect. This makes sure
 | 
						|
      // that we don't have display items over the subpixel seam at the edge
 | 
						|
      // of the scrolled area.
 | 
						|
      DisplayListClipState::AutoSaveRestore scrolledRectClipState(aBuilder);
 | 
						|
      nsRect scrolledRectClip =
 | 
						|
          GetUnsnappedScrolledRectInternal(
 | 
						|
              mScrolledFrame->ScrollableOverflowRect(), mScrollPort.Size()) +
 | 
						|
          mScrolledFrame->GetPosition();
 | 
						|
      bool clippedToDisplayPort = false;
 | 
						|
      if (mWillBuildScrollableLayer && aBuilder->IsPaintingToWindow()) {
 | 
						|
        // Clip the contents to the display port.
 | 
						|
        // The dirty rect already acts kind of like a clip, in that
 | 
						|
        // FrameLayerBuilder intersects item bounds and opaque regions with
 | 
						|
        // it, but it doesn't have the consistent snapping behavior of a
 | 
						|
        // true clip.
 | 
						|
        // For a case where this makes a difference, imagine the following
 | 
						|
        // scenario: The display port has an edge that falls on a fractional
 | 
						|
        // layer pixel, and there's an opaque display item that covers the
 | 
						|
        // whole display port up until that fractional edge, and there is a
 | 
						|
        // transparent display item that overlaps the edge. We want to prevent
 | 
						|
        // this transparent item from enlarging the scrolled layer's visible
 | 
						|
        // region beyond its opaque region. The dirty rect doesn't do that -
 | 
						|
        // it gets rounded out, whereas a true clip gets rounded to nearest
 | 
						|
        // pixels.
 | 
						|
        // If there is no display port, we don't need this because the clip
 | 
						|
        // from the scroll port is still applied.
 | 
						|
        scrolledRectClip = scrolledRectClip.Intersect(visibleRect);
 | 
						|
        clippedToDisplayPort = scrolledRectClip.IsEqualEdges(visibleRect);
 | 
						|
      }
 | 
						|
      scrolledRectClipState.ClipContainingBlockDescendants(
 | 
						|
          scrolledRectClip + aBuilder->ToReferenceFrame(this));
 | 
						|
      if (clippedToDisplayPort) {
 | 
						|
        // We have to do this after the ClipContainingBlockDescendants call
 | 
						|
        // above, otherwise that call will clobber the flag set by this call
 | 
						|
        // to SetClippedToDisplayPort.
 | 
						|
        scrolledRectClipState.SetClippedToDisplayPort();
 | 
						|
      }
 | 
						|
 | 
						|
      nsRect visibleRectForChildren = visibleRect;
 | 
						|
      nsRect dirtyRectForChildren = dirtyRect;
 | 
						|
 | 
						|
      // If we are entering the RCD-RSF, we are crossing the async zoom
 | 
						|
      // container boundary, and need to convert from visual to layout
 | 
						|
      // coordinates.
 | 
						|
      if (willBuildAsyncZoomContainer && aBuilder->IsForEventDelivery()) {
 | 
						|
        MOZ_ASSERT(ViewportUtils::IsZoomedContentRoot(mScrolledFrame));
 | 
						|
        visibleRectForChildren =
 | 
						|
            ViewportUtils::VisualToLayout(visibleRectForChildren, PresShell());
 | 
						|
        dirtyRectForChildren =
 | 
						|
            ViewportUtils::VisualToLayout(dirtyRectForChildren, PresShell());
 | 
						|
      }
 | 
						|
 | 
						|
      nsDisplayListBuilder::AutoBuildingDisplayList building(
 | 
						|
          aBuilder, this, visibleRectForChildren, dirtyRectForChildren);
 | 
						|
 | 
						|
      BuildDisplayListForChild(aBuilder, mScrolledFrame, set);
 | 
						|
 | 
						|
      if (dirtyRectHasBeenOverriden &&
 | 
						|
          StaticPrefs::layout_display_list_show_rebuild_area()) {
 | 
						|
        nsDisplaySolidColor* color = MakeDisplayItem<nsDisplaySolidColor>(
 | 
						|
            aBuilder, this,
 | 
						|
            dirtyRect + aBuilder->GetCurrentFrameOffsetToReferenceFrame(),
 | 
						|
            NS_RGBA(0, 0, 255, 64), false);
 | 
						|
        if (color) {
 | 
						|
          color->SetOverrideZIndex(INT32_MAX);
 | 
						|
          set.PositionedDescendants()->AppendToTop(color);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (extraContentBoxClipForNonCaretContent) {
 | 
						|
      // The items were built while the inflated content box clip was in
 | 
						|
      // effect, so that the caret wasn't clipped unnecessarily. We apply
 | 
						|
      // the non-inflated clip to the non-caret items now, by intersecting
 | 
						|
      // it with their existing clip.
 | 
						|
      ClipListsExceptCaret(&set, aBuilder, mScrolledFrame,
 | 
						|
                           *extraContentBoxClipForNonCaretContent);
 | 
						|
    }
 | 
						|
 | 
						|
    if (aBuilder->IsPaintingToWindow()) {
 | 
						|
      mIsParentToActiveScrollFrames =
 | 
						|
          ShouldActivateAllScrollFrames()
 | 
						|
              ? asrSetter.GetContainsNonMinimalDisplayPort()
 | 
						|
              : asrSetter.ShouldForceLayerForScrollParent();
 | 
						|
    }
 | 
						|
 | 
						|
    if (asrSetter.ShouldForceLayerForScrollParent()) {
 | 
						|
      // Note that forcing layerization of scroll parents follows the scroll
 | 
						|
      // handoff chain which is subject to the out-of-flow-frames caveat noted
 | 
						|
      // above (where the asrSetter variable is created).
 | 
						|
      MOZ_ASSERT(couldBuildLayer && mScrolledFrame->GetContent() &&
 | 
						|
                 aBuilder->IsPaintingToWindow());
 | 
						|
      if (!mWillBuildScrollableLayer) {
 | 
						|
        // Set a displayport so next paint we don't have to force layerization
 | 
						|
        // after the fact. It's ok to pass DoNotRepaint here, since we've
 | 
						|
        // already painted the change and we're just optimizing it to be
 | 
						|
        // detected earlier. We also won't confuse RetainedDisplayLists
 | 
						|
        // with the silent change, since we explicitly request partial updates
 | 
						|
        // to be disabled on the next paint.
 | 
						|
        DisplayPortUtils::SetDisplayPortMargins(
 | 
						|
            GetContent(), PresShell(), DisplayPortMargins::Empty(GetContent()),
 | 
						|
            DisplayPortUtils::ClearMinimalDisplayPortProperty::Yes, 0,
 | 
						|
            DisplayPortUtils::RepaintMode::DoNotRepaint);
 | 
						|
        // Call DecideScrollableLayer to recompute mWillBuildScrollableLayer
 | 
						|
        // and recompute the current animated geometry root if needed. It's
 | 
						|
        // too late to change the dirty rect so pass a copy.
 | 
						|
        nsRect copyOfDirtyRect = dirtyRect;
 | 
						|
        nsRect copyOfVisibleRect = visibleRect;
 | 
						|
        Unused << DecideScrollableLayer(aBuilder, ©OfVisibleRect,
 | 
						|
                                        ©OfDirtyRect,
 | 
						|
                                        /* aSetBase = */ false, nullptr);
 | 
						|
        if (mWillBuildScrollableLayer) {
 | 
						|
#ifndef MOZ_WIDGET_ANDROID
 | 
						|
          gfxCriticalNoteOnce << "inserted scroll frame";
 | 
						|
#endif
 | 
						|
          asrSetter.InsertScrollFrame(this);
 | 
						|
          aBuilder->SetDisablePartialUpdates(true);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (mWillBuildScrollableLayer && aBuilder->IsPaintingToWindow()) {
 | 
						|
    aBuilder->ForceLayerForScrollParent();
 | 
						|
  }
 | 
						|
 | 
						|
  MaybeCreateTopLayerAndWrapRootItems(
 | 
						|
      aBuilder, set, willBuildAsyncZoomContainer, &blendCapture, clipRect,
 | 
						|
      haveRadii ? radii : nullptr);
 | 
						|
 | 
						|
  // We want to call SetContainsNonMinimalDisplayPort if
 | 
						|
  // mWillBuildScrollableLayer is true for any reason other than having a
 | 
						|
  // minimal display port.
 | 
						|
  if (aBuilder->IsPaintingToWindow()) {
 | 
						|
    if (DisplayPortUtils::HasNonMinimalDisplayPort(GetContent()) ||
 | 
						|
        mZoomableByAPZ || nsContentUtils::HasScrollgrab(GetContent())) {
 | 
						|
      aBuilder->SetContainsNonMinimalDisplayPort();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (couldBuildLayer) {
 | 
						|
    CompositorHitTestInfo info(CompositorHitTestFlags::eVisibleToHitTest,
 | 
						|
                               CompositorHitTestFlags::eInactiveScrollframe);
 | 
						|
    // If the scroll frame has non-default overscroll-behavior, instruct
 | 
						|
    // APZ to require a target confirmation before processing events that
 | 
						|
    // hit this scroll frame (that is, to drop the events if a
 | 
						|
    // confirmation does not arrive within the timeout period). Otherwise,
 | 
						|
    // APZ's fallback behaviour of scrolling the enclosing scroll frame
 | 
						|
    // would violate the specified overscroll-behavior.
 | 
						|
    auto overscroll = GetOverscrollBehaviorInfo();
 | 
						|
    if (overscroll.mBehaviorX != OverscrollBehavior::Auto ||
 | 
						|
        overscroll.mBehaviorY != OverscrollBehavior::Auto) {
 | 
						|
      info += CompositorHitTestFlags::eRequiresTargetConfirmation;
 | 
						|
    }
 | 
						|
 | 
						|
    nsRect area = effectiveScrollPort + aBuilder->ToReferenceFrame(this);
 | 
						|
 | 
						|
    // Make sure that APZ will dispatch events back to content so we can
 | 
						|
    // create a displayport for this frame. We'll add the item later on.
 | 
						|
    if (!mWillBuildScrollableLayer && aBuilder->BuildCompositorHitTestInfo()) {
 | 
						|
      // Make sure the z-index of the inactive item is at least zero.
 | 
						|
      // Otherwise, it will end up behind non-positioned items in the scrolled
 | 
						|
      // content.
 | 
						|
      int32_t zIndex = MaxZIndexInListOfItemsContainedInFrame(
 | 
						|
                           set.PositionedDescendants(), this)
 | 
						|
                           .valueOr(0);
 | 
						|
      if (aBuilder->IsPartialUpdate()) {
 | 
						|
        for (nsDisplayItem* item : mScrolledFrame->DisplayItems()) {
 | 
						|
          if (item->GetType() ==
 | 
						|
              DisplayItemType::TYPE_COMPOSITOR_HITTEST_INFO) {
 | 
						|
            auto* hitTestItem =
 | 
						|
                static_cast<nsDisplayCompositorHitTestInfo*>(item);
 | 
						|
            if (hitTestItem->GetHitTestInfo().Info().contains(
 | 
						|
                    CompositorHitTestFlags::eInactiveScrollframe)) {
 | 
						|
              zIndex = std::max(zIndex, hitTestItem->ZIndex());
 | 
						|
              item->SetCantBeReused();
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
      nsDisplayCompositorHitTestInfo* hitInfo =
 | 
						|
          MakeDisplayItemWithIndex<nsDisplayCompositorHitTestInfo>(
 | 
						|
              aBuilder, mScrolledFrame, 1, area, info);
 | 
						|
      if (hitInfo) {
 | 
						|
        AppendInternalItemToTop(set, hitInfo, Some(zIndex));
 | 
						|
        aBuilder->SetCompositorHitTestInfo(info);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (aBuilder->ShouldBuildScrollInfoItemsForHoisting()) {
 | 
						|
      aBuilder->AppendNewScrollInfoItemForHoisting(
 | 
						|
          MakeDisplayItem<nsDisplayScrollInfoLayer>(aBuilder, mScrolledFrame,
 | 
						|
                                                    this, info, area));
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Now display overlay scrollbars and the resizer, if we have one.
 | 
						|
  AppendScrollPartsTo(aBuilder, set, createLayersForScrollbars, true);
 | 
						|
 | 
						|
  set.MoveTo(aLists);
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::RestrictToRootDisplayPort(
 | 
						|
    const nsRect& aDisplayportBase) {
 | 
						|
  // This function clips aDisplayportBase so that it is no larger than the
 | 
						|
  // root frame's displayport (or the root composition bounds, if we can't
 | 
						|
  // obtain the root frame's displayport). This is useful for ensuring that
 | 
						|
  // the displayport of a tall scrollframe doesn't gobble up all the memory.
 | 
						|
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
  const nsPresContext* rootPresContext =
 | 
						|
      pc->GetInProcessRootContentDocumentPresContext();
 | 
						|
  if (!rootPresContext) {
 | 
						|
    rootPresContext = pc->GetRootPresContext();
 | 
						|
  }
 | 
						|
  if (!rootPresContext) {
 | 
						|
    return aDisplayportBase;
 | 
						|
  }
 | 
						|
  const mozilla::PresShell* const rootPresShell = rootPresContext->PresShell();
 | 
						|
  nsIFrame* rootFrame = rootPresShell->GetRootScrollFrame();
 | 
						|
  if (!rootFrame) {
 | 
						|
    rootFrame = rootPresShell->GetRootFrame();
 | 
						|
  }
 | 
						|
  if (!rootFrame) {
 | 
						|
    return aDisplayportBase;
 | 
						|
  }
 | 
						|
 | 
						|
  // Make sure we aren't trying to restrict to our own displayport, which is a
 | 
						|
  // circular dependency.
 | 
						|
  MOZ_ASSERT(!mIsRoot || rootPresContext != pc);
 | 
						|
 | 
						|
  nsRect rootDisplayPort;
 | 
						|
  bool hasDisplayPort =
 | 
						|
      rootFrame->GetContent() && DisplayPortUtils::GetDisplayPort(
 | 
						|
                                     rootFrame->GetContent(), &rootDisplayPort);
 | 
						|
  if (hasDisplayPort) {
 | 
						|
    // The display port of the root frame already factors in it's callback
 | 
						|
    // transform, so subtract it out here, the GetCumulativeApzCallbackTransform
 | 
						|
    // call below will add it back.
 | 
						|
    MOZ_LOG(sDisplayportLog, LogLevel::Verbose,
 | 
						|
            ("RestrictToRootDisplayPort: Existing root displayport is %s\n",
 | 
						|
             ToString(rootDisplayPort).c_str()));
 | 
						|
    if (nsIContent* content = rootFrame->GetContent()) {
 | 
						|
      if (void* property =
 | 
						|
              content->GetProperty(nsGkAtoms::apzCallbackTransform)) {
 | 
						|
        rootDisplayPort -=
 | 
						|
            CSSPoint::ToAppUnits(*static_cast<CSSPoint*>(property));
 | 
						|
      }
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    // If we don't have a display port on the root frame let's fall back to
 | 
						|
    // the root composition bounds instead.
 | 
						|
    nsRect rootCompBounds =
 | 
						|
        nsRect(nsPoint(0, 0),
 | 
						|
               nsLayoutUtils::CalculateCompositionSizeForFrame(rootFrame));
 | 
						|
 | 
						|
    // If rootFrame is the RCD-RSF then
 | 
						|
    // CalculateCompositionSizeForFrame did not take the document's
 | 
						|
    // resolution into account, so we must.
 | 
						|
    if (rootPresContext->IsRootContentDocumentCrossProcess() &&
 | 
						|
        rootFrame == rootPresShell->GetRootScrollFrame()) {
 | 
						|
      MOZ_LOG(
 | 
						|
          sDisplayportLog, LogLevel::Verbose,
 | 
						|
          ("RestrictToRootDisplayPort: Removing resolution %f from root "
 | 
						|
           "composition bounds %s\n",
 | 
						|
           rootPresShell->GetResolution(), ToString(rootCompBounds).c_str()));
 | 
						|
      rootCompBounds =
 | 
						|
          rootCompBounds.RemoveResolution(rootPresShell->GetResolution());
 | 
						|
    }
 | 
						|
 | 
						|
    rootDisplayPort = rootCompBounds;
 | 
						|
  }
 | 
						|
  MOZ_LOG(sDisplayportLog, LogLevel::Verbose,
 | 
						|
          ("RestrictToRootDisplayPort: Intermediate root displayport %s\n",
 | 
						|
           ToString(rootDisplayPort).c_str()));
 | 
						|
 | 
						|
  // We want to convert the root display port from the
 | 
						|
  // coordinate space of |rootFrame| to the coordinate space of
 | 
						|
  // |this|. We do that with the TransformRect call below.
 | 
						|
  // However, since we care about the root display port
 | 
						|
  // relative to what the user is actually seeing, we also need to
 | 
						|
  // incorporate the APZ callback transforms into this. Most of the
 | 
						|
  // time those transforms are negligible, but in some cases (e.g.
 | 
						|
  // when a zoom is applied on an overflow:hidden document) it is
 | 
						|
  // not (see bug 1280013).
 | 
						|
  // XXX: Eventually we may want to create a modified version of
 | 
						|
  // TransformRect that includes the APZ callback transforms
 | 
						|
  // directly.
 | 
						|
  nsLayoutUtils::TransformRect(rootFrame, this, rootDisplayPort);
 | 
						|
  MOZ_LOG(sDisplayportLog, LogLevel::Verbose,
 | 
						|
          ("RestrictToRootDisplayPort: Transformed root displayport %s\n",
 | 
						|
           ToString(rootDisplayPort).c_str()));
 | 
						|
  rootDisplayPort += CSSPoint::ToAppUnits(
 | 
						|
      nsLayoutUtils::GetCumulativeApzCallbackTransform(this));
 | 
						|
  MOZ_LOG(sDisplayportLog, LogLevel::Verbose,
 | 
						|
          ("RestrictToRootDisplayPort: Final root displayport %s\n",
 | 
						|
           ToString(rootDisplayPort).c_str()));
 | 
						|
 | 
						|
  // We want to limit aDisplayportBase to be no larger than
 | 
						|
  // rootDisplayPort on either axis, but we don't want to just
 | 
						|
  // blindly intersect the two, because rootDisplayPort might be
 | 
						|
  // offset from where aDisplayportBase is (see bug 1327095 comment
 | 
						|
  // 8). Instead, we translate rootDisplayPort so as to maximize the
 | 
						|
  // overlap with aDisplayportBase, and *then* do the intersection.
 | 
						|
  if (rootDisplayPort.x > aDisplayportBase.x &&
 | 
						|
      rootDisplayPort.XMost() > aDisplayportBase.XMost()) {
 | 
						|
    // rootDisplayPort is at a greater x-position for both left and
 | 
						|
    // right, so translate it such that the XMost() values are the
 | 
						|
    // same. This will line up the right edge of the two rects, and
 | 
						|
    // might mean that rootDisplayPort.x is smaller than
 | 
						|
    // aDisplayportBase.x. We can avoid that by taking the min of the
 | 
						|
    // x delta and XMost() delta, but it doesn't really matter
 | 
						|
    // because the intersection between the two rects below will end
 | 
						|
    // up the same.
 | 
						|
    rootDisplayPort.x -= (rootDisplayPort.XMost() - aDisplayportBase.XMost());
 | 
						|
  } else if (rootDisplayPort.x < aDisplayportBase.x &&
 | 
						|
             rootDisplayPort.XMost() < aDisplayportBase.XMost()) {
 | 
						|
    // Analaogous code for when the rootDisplayPort is at a smaller
 | 
						|
    // x-position.
 | 
						|
    rootDisplayPort.x = aDisplayportBase.x;
 | 
						|
  }
 | 
						|
  // Do the same for y-axis
 | 
						|
  if (rootDisplayPort.y > aDisplayportBase.y &&
 | 
						|
      rootDisplayPort.YMost() > aDisplayportBase.YMost()) {
 | 
						|
    rootDisplayPort.y -= (rootDisplayPort.YMost() - aDisplayportBase.YMost());
 | 
						|
  } else if (rootDisplayPort.y < aDisplayportBase.y &&
 | 
						|
             rootDisplayPort.YMost() < aDisplayportBase.YMost()) {
 | 
						|
    rootDisplayPort.y = aDisplayportBase.y;
 | 
						|
  }
 | 
						|
  MOZ_LOG(
 | 
						|
      sDisplayportLog, LogLevel::Verbose,
 | 
						|
      ("RestrictToRootDisplayPort: Root displayport translated to %s to "
 | 
						|
       "better enclose %s\n",
 | 
						|
       ToString(rootDisplayPort).c_str(), ToString(aDisplayportBase).c_str()));
 | 
						|
 | 
						|
  // Now we can do the intersection
 | 
						|
  return aDisplayportBase.Intersect(rootDisplayPort);
 | 
						|
}
 | 
						|
 | 
						|
/* static */ bool nsHTMLScrollFrame::ShouldActivateAllScrollFrames() {
 | 
						|
  return (StaticPrefs::apz_wr_activate_all_scroll_frames() ||
 | 
						|
          (StaticPrefs::apz_wr_activate_all_scroll_frames_when_fission() &&
 | 
						|
           FissionAutostart()));
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::DecideScrollableLayer(
 | 
						|
    nsDisplayListBuilder* aBuilder, nsRect* aVisibleRect, nsRect* aDirtyRect,
 | 
						|
    bool aSetBase, bool* aDirtyRectHasBeenOverriden) {
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
  bool hasDisplayPort = DisplayPortUtils::HasDisplayPort(content);
 | 
						|
  // For hit testing purposes with fission we want to create a
 | 
						|
  // minimal display port for every scroll frame that could be active. (We only
 | 
						|
  // do this when aSetBase is true because we only want to do this the first
 | 
						|
  // time this function is called for the same scroll frame.)
 | 
						|
  if (aSetBase && !hasDisplayPort && aBuilder->IsPaintingToWindow() &&
 | 
						|
      ShouldActivateAllScrollFrames() &&
 | 
						|
      nsLayoutUtils::AsyncPanZoomEnabled(this) && WantAsyncScroll()) {
 | 
						|
    // SetDisplayPortMargins calls TriggerDisplayPortExpiration which starts a
 | 
						|
    // display port expiry timer for display ports that do expire. However
 | 
						|
    // minimal display ports do not expire, so the display port has to be
 | 
						|
    // marked before the SetDisplayPortMargins call so the expiry timer
 | 
						|
    // doesn't get started.
 | 
						|
    content->SetProperty(nsGkAtoms::MinimalDisplayPort,
 | 
						|
                         reinterpret_cast<void*>(true));
 | 
						|
 | 
						|
    DisplayPortUtils::SetDisplayPortMargins(
 | 
						|
        content, PresShell(), DisplayPortMargins::Empty(content),
 | 
						|
        DisplayPortUtils::ClearMinimalDisplayPortProperty::No, 0,
 | 
						|
        DisplayPortUtils::RepaintMode::DoNotRepaint);
 | 
						|
    hasDisplayPort = true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aBuilder->IsPaintingToWindow()) {
 | 
						|
    if (aSetBase) {
 | 
						|
      nsRect displayportBase = *aVisibleRect;
 | 
						|
      nsPresContext* pc = PresContext();
 | 
						|
 | 
						|
      bool isChromeRootDoc =
 | 
						|
          !pc->Document()->IsContentDocument() && !pc->GetParentPresContext();
 | 
						|
 | 
						|
      if (mIsRoot &&
 | 
						|
          (pc->IsRootContentDocumentCrossProcess() || isChromeRootDoc)) {
 | 
						|
        displayportBase =
 | 
						|
            nsRect(nsPoint(0, 0),
 | 
						|
                   nsLayoutUtils::CalculateCompositionSizeForFrame(this));
 | 
						|
      } else {
 | 
						|
        // Make the displayport base equal to the visible rect restricted to
 | 
						|
        // the scrollport and the root composition bounds, relative to the
 | 
						|
        // scrollport.
 | 
						|
        displayportBase = aVisibleRect->Intersect(mScrollPort);
 | 
						|
 | 
						|
        mozilla::layers::ScrollableLayerGuid::ViewID viewID =
 | 
						|
            mozilla::layers::ScrollableLayerGuid::NULL_SCROLL_ID;
 | 
						|
        if (MOZ_LOG_TEST(sDisplayportLog, LogLevel::Verbose)) {
 | 
						|
          nsLayoutUtils::FindIDFor(GetContent(), &viewID);
 | 
						|
          MOZ_LOG(
 | 
						|
              sDisplayportLog, LogLevel::Verbose,
 | 
						|
              ("Scroll id %" PRIu64 " has visible rect %s, scroll port %s\n",
 | 
						|
               viewID, ToString(*aVisibleRect).c_str(),
 | 
						|
               ToString(mScrollPort).c_str()));
 | 
						|
        }
 | 
						|
 | 
						|
        // Only restrict to the root displayport bounds if necessary,
 | 
						|
        // as the required coordinate transformation is expensive.
 | 
						|
        // And don't call RestrictToRootDisplayPort if we would be trying to
 | 
						|
        // restrict to our own display port, which doesn't make sense (ie if we
 | 
						|
        // are a root scroll frame in a process root prescontext).
 | 
						|
        if (hasDisplayPort && (!mIsRoot || pc->GetParentPresContext()) &&
 | 
						|
            !DisplayPortUtils::WillUseEmptyDisplayPortMargins(content)) {
 | 
						|
          displayportBase = RestrictToRootDisplayPort(displayportBase);
 | 
						|
          MOZ_LOG(sDisplayportLog, LogLevel::Verbose,
 | 
						|
                  ("Scroll id %" PRIu64 " has restricted base %s\n", viewID,
 | 
						|
                   ToString(displayportBase).c_str()));
 | 
						|
        }
 | 
						|
        displayportBase -= mScrollPort.TopLeft();
 | 
						|
      }
 | 
						|
 | 
						|
      DisplayPortUtils::SetDisplayPortBase(GetContent(), displayportBase);
 | 
						|
    }
 | 
						|
 | 
						|
    // If we don't have aSetBase == true then should have already
 | 
						|
    // been called with aSetBase == true which should have set a
 | 
						|
    // displayport base.
 | 
						|
    MOZ_ASSERT(content->GetProperty(nsGkAtoms::DisplayPortBase));
 | 
						|
    nsRect displayPort;
 | 
						|
    hasDisplayPort = DisplayPortUtils::GetDisplayPort(
 | 
						|
        content, &displayPort,
 | 
						|
        DisplayPortOptions().With(DisplayportRelativeTo::ScrollFrame));
 | 
						|
 | 
						|
    auto OverrideDirtyRect = [&](const nsRect& aRect) {
 | 
						|
      *aDirtyRect = aRect;
 | 
						|
      if (aDirtyRectHasBeenOverriden) {
 | 
						|
        *aDirtyRectHasBeenOverriden = true;
 | 
						|
      }
 | 
						|
    };
 | 
						|
 | 
						|
    if (hasDisplayPort) {
 | 
						|
      // Override the dirty rectangle if the displayport has been set.
 | 
						|
      *aVisibleRect = displayPort;
 | 
						|
      if (aBuilder->IsReusingStackingContextItems() ||
 | 
						|
          !aBuilder->IsPartialUpdate() || aBuilder->InInvalidSubtree() ||
 | 
						|
          IsFrameModified()) {
 | 
						|
        OverrideDirtyRect(displayPort);
 | 
						|
      } else if (HasOverrideDirtyRegion()) {
 | 
						|
        nsRect* rect = GetProperty(
 | 
						|
            nsDisplayListBuilder::DisplayListBuildingDisplayPortRect());
 | 
						|
        if (rect) {
 | 
						|
          OverrideDirtyRect(*rect);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    } else if (mIsRoot) {
 | 
						|
      // The displayPort getter takes care of adjusting for resolution. So if
 | 
						|
      // we have resolution but no displayPort then we need to adjust for
 | 
						|
      // resolution here.
 | 
						|
      auto* presShell = PresShell();
 | 
						|
      *aVisibleRect =
 | 
						|
          aVisibleRect->RemoveResolution(presShell->GetResolution());
 | 
						|
      *aDirtyRect = aDirtyRect->RemoveResolution(presShell->GetResolution());
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Since making new layers is expensive, only create a scrollable layer
 | 
						|
  // for some scroll frames.
 | 
						|
  // When a displayport is being used, force building of a layer so that
 | 
						|
  // the compositor can find the scrollable layer for async scrolling.
 | 
						|
  // If the element is marked 'scrollgrab', also force building of a layer
 | 
						|
  // so that APZ can implement scroll grabbing.
 | 
						|
  mWillBuildScrollableLayer = hasDisplayPort ||
 | 
						|
                              nsContentUtils::HasScrollgrab(content) ||
 | 
						|
                              mZoomableByAPZ;
 | 
						|
  return mWillBuildScrollableLayer;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::NotifyApzTransaction() {
 | 
						|
  mAllowScrollOriginDowngrade = true;
 | 
						|
  mApzScrollPos = GetScrollPosition();
 | 
						|
  mApzAnimationRequested = IsLastScrollUpdateAnimating();
 | 
						|
  mApzAnimationTriggeredByScriptRequested =
 | 
						|
      IsLastScrollUpdateTriggeredByScriptAnimating();
 | 
						|
  mScrollUpdates.Clear();
 | 
						|
  if (mIsRoot) {
 | 
						|
    PresShell()->SetResolutionUpdated(false);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Maybe<ScrollMetadata> nsHTMLScrollFrame::ComputeScrollMetadata(
 | 
						|
    WebRenderLayerManager* aLayerManager, const nsIFrame* aItemFrame,
 | 
						|
    const nsPoint& aOffsetToReferenceFrame) const {
 | 
						|
  if (!mWillBuildScrollableLayer) {
 | 
						|
    return Nothing();
 | 
						|
  }
 | 
						|
 | 
						|
  bool isRootContent =
 | 
						|
      mIsRoot && PresContext()->IsRootContentDocumentCrossProcess();
 | 
						|
 | 
						|
  MOZ_ASSERT(mScrolledFrame->GetContent());
 | 
						|
 | 
						|
  return Some(nsLayoutUtils::ComputeScrollMetadata(
 | 
						|
      mScrolledFrame, this, GetContent(), aItemFrame, aOffsetToReferenceFrame,
 | 
						|
      aLayerManager, mScrollParentID, mScrollPort.Size(), isRootContent));
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsRectNearlyVisible(const nsRect& aRect) const {
 | 
						|
  // Use the right rect depending on if a display port is set.
 | 
						|
  nsRect displayPort;
 | 
						|
  bool usingDisplayport = DisplayPortUtils::GetDisplayPort(
 | 
						|
      GetContent(), &displayPort,
 | 
						|
      DisplayPortOptions().With(DisplayportRelativeTo::ScrollFrame));
 | 
						|
 | 
						|
  if (mIsRoot && !usingDisplayport &&
 | 
						|
      PresContext()->IsRootContentDocumentInProcess() &&
 | 
						|
      !PresContext()->IsRootContentDocumentCrossProcess()) {
 | 
						|
    // In the case of the root scroller of OOP iframes, there are cases where
 | 
						|
    // any display port value isn't set, e.g. the iframe element is out of view
 | 
						|
    // in the parent document. In such cases we'd consider the iframe is not
 | 
						|
    // visible.
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  return aRect.Intersects(
 | 
						|
      ExpandRectToNearlyVisible(usingDisplayport ? displayPort : mScrollPort));
 | 
						|
}
 | 
						|
 | 
						|
OverscrollBehaviorInfo nsHTMLScrollFrame::GetOverscrollBehaviorInfo() const {
 | 
						|
  nsIFrame* frame = GetFrameForStyle();
 | 
						|
  if (!frame) {
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  auto& disp = *frame->StyleDisplay();
 | 
						|
  return OverscrollBehaviorInfo::FromStyleConstants(disp.mOverscrollBehaviorX,
 | 
						|
                                                    disp.mOverscrollBehaviorY);
 | 
						|
}
 | 
						|
 | 
						|
ScrollStyles nsHTMLScrollFrame::GetScrollStyles() const {
 | 
						|
  nsPresContext* presContext = PresContext();
 | 
						|
  if (!presContext->IsDynamic() &&
 | 
						|
      !(mIsRoot && presContext->HasPaginatedScrolling())) {
 | 
						|
    return ScrollStyles(StyleOverflow::Hidden, StyleOverflow::Hidden);
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mIsRoot) {
 | 
						|
    return ScrollStyles(*StyleDisplay(),
 | 
						|
                        ScrollStyles::MapOverflowToValidScrollStyle);
 | 
						|
  }
 | 
						|
 | 
						|
  ScrollStyles result = presContext->GetViewportScrollStylesOverride();
 | 
						|
  if (nsDocShell* ds = presContext->GetDocShell()) {
 | 
						|
    switch (ds->ScrollbarPreference()) {
 | 
						|
      case ScrollbarPreference::Auto:
 | 
						|
        break;
 | 
						|
      case ScrollbarPreference::Never:
 | 
						|
        result.mHorizontal = result.mVertical = StyleOverflow::Hidden;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetLayoutScrollRange() const {
 | 
						|
  return GetScrollRange(mScrollPort.width, mScrollPort.height);
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetScrollRange(nscoord aWidth,
 | 
						|
                                         nscoord aHeight) const {
 | 
						|
  nsRect range = GetScrolledRect();
 | 
						|
  range.width = std::max(range.width - aWidth, 0);
 | 
						|
  range.height = std::max(range.height - aHeight, 0);
 | 
						|
  return range;
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetVisualScrollRange() const {
 | 
						|
  nsSize visualViewportSize = GetVisualViewportSize();
 | 
						|
  return GetScrollRange(visualViewportSize.width, visualViewportSize.height);
 | 
						|
}
 | 
						|
 | 
						|
nsSize nsHTMLScrollFrame::GetVisualViewportSize() const {
 | 
						|
  auto* presShell = PresShell();
 | 
						|
  if (mIsRoot && presShell->IsVisualViewportSizeSet()) {
 | 
						|
    return presShell->GetVisualViewportSize();
 | 
						|
  }
 | 
						|
  return mScrollPort.Size();
 | 
						|
}
 | 
						|
 | 
						|
nsPoint nsHTMLScrollFrame::GetVisualViewportOffset() const {
 | 
						|
  if (mIsRoot) {
 | 
						|
    auto* presShell = PresShell();
 | 
						|
    if (auto pendingUpdate = presShell->GetPendingVisualScrollUpdate()) {
 | 
						|
      // The pending visual scroll update on the PresShell contains a raw,
 | 
						|
      // unclamped offset (basically, whatever was passed to ScrollToVisual()).
 | 
						|
      // It will be clamped on the APZ side, but if we use it as the
 | 
						|
      // main-thread's visual viewport offset we need to clamp it ourselves.
 | 
						|
      // Use GetScrollRangeForUserInputEvents() to do the clamping because this
 | 
						|
      // the scroll range that APZ will use.
 | 
						|
      return GetScrollRangeForUserInputEvents().ClampPoint(
 | 
						|
          pendingUpdate->mVisualScrollOffset);
 | 
						|
    }
 | 
						|
    return presShell->GetVisualViewportOffset();
 | 
						|
  }
 | 
						|
  return GetScrollPosition();
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::SetVisualViewportOffset(const nsPoint& aOffset,
 | 
						|
                                                bool aRepaint) {
 | 
						|
  MOZ_ASSERT(mIsRoot);
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  AutoScrollbarRepaintSuppression repaintSuppression(this, weakFrame,
 | 
						|
                                                     !aRepaint);
 | 
						|
 | 
						|
  bool retVal =
 | 
						|
      PresShell()->SetVisualViewportOffset(aOffset, GetScrollPosition());
 | 
						|
  if (!weakFrame.IsAlive()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  return retVal;
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetVisualOptimalViewingRect() const {
 | 
						|
  auto* presShell = PresShell();
 | 
						|
  nsRect rect = mScrollPort;
 | 
						|
  if (mIsRoot && presShell->IsVisualViewportSizeSet() &&
 | 
						|
      presShell->IsVisualViewportOffsetSet()) {
 | 
						|
    rect = nsRect(mScrollPort.TopLeft() - GetScrollPosition() +
 | 
						|
                      presShell->GetVisualViewportOffset(),
 | 
						|
                  presShell->GetVisualViewportSize());
 | 
						|
  }
 | 
						|
  // NOTE: We intentionally resolve scroll-padding percentages against the
 | 
						|
  // scrollport even when the visual viewport is set, see
 | 
						|
  // https://github.com/w3c/csswg-drafts/issues/4393.
 | 
						|
  rect.Deflate(GetScrollPadding());
 | 
						|
  return rect;
 | 
						|
}
 | 
						|
 | 
						|
static void AdjustDestinationForWholeDelta(const nsIntPoint& aDelta,
 | 
						|
                                           const nsRect& aScrollRange,
 | 
						|
                                           nsPoint& aPoint) {
 | 
						|
  if (aDelta.x < 0) {
 | 
						|
    aPoint.x = aScrollRange.X();
 | 
						|
  } else if (aDelta.x > 0) {
 | 
						|
    aPoint.x = aScrollRange.XMost();
 | 
						|
  }
 | 
						|
  if (aDelta.y < 0) {
 | 
						|
    aPoint.y = aScrollRange.Y();
 | 
						|
  } else if (aDelta.y > 0) {
 | 
						|
    aPoint.y = aScrollRange.YMost();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Calculate lower/upper scrollBy range in given direction.
 | 
						|
 * @param aDelta specifies scrollBy direction, if 0 then range will be 0 size
 | 
						|
 * @param aPos desired destination in AppUnits
 | 
						|
 * @param aNeg/PosTolerance defines relative range distance
 | 
						|
 *   below and above of aPos point
 | 
						|
 * @param aMultiplier used for conversion of tolerance into appUnis
 | 
						|
 */
 | 
						|
static void CalcRangeForScrollBy(int32_t aDelta, nscoord aPos,
 | 
						|
                                 float aNegTolerance, float aPosTolerance,
 | 
						|
                                 nscoord aMultiplier, nscoord* aLower,
 | 
						|
                                 nscoord* aUpper) {
 | 
						|
  if (!aDelta) {
 | 
						|
    *aLower = *aUpper = aPos;
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  *aLower = aPos - NSToCoordRound(aMultiplier *
 | 
						|
                                  (aDelta > 0 ? aNegTolerance : aPosTolerance));
 | 
						|
  *aUpper = aPos + NSToCoordRound(aMultiplier *
 | 
						|
                                  (aDelta > 0 ? aPosTolerance : aNegTolerance));
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollBy(nsIntPoint aDelta, ScrollUnit aUnit,
 | 
						|
                                 ScrollMode aMode, nsIntPoint* aOverflow,
 | 
						|
                                 ScrollOrigin aOrigin,
 | 
						|
                                 nsIScrollableFrame::ScrollMomentum aMomentum,
 | 
						|
                                 ScrollSnapFlags aSnapFlags) {
 | 
						|
  // When a smooth scroll is being processed on a frame, mouse wheel and
 | 
						|
  // trackpad momentum scroll event updates must notcancel the SMOOTH or
 | 
						|
  // SMOOTH_MSD scroll animations, enabling Javascript that depends on them to
 | 
						|
  // be responsive without forcing the user to wait for the fling animations to
 | 
						|
  // completely stop.
 | 
						|
  switch (aMomentum) {
 | 
						|
    case nsIScrollableFrame::NOT_MOMENTUM:
 | 
						|
      mIgnoreMomentumScroll = false;
 | 
						|
      break;
 | 
						|
    case nsIScrollableFrame::SYNTHESIZED_MOMENTUM_EVENT:
 | 
						|
      if (mIgnoreMomentumScroll) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      break;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mAsyncSmoothMSDScroll != nullptr) {
 | 
						|
    // When CSSOM-View scroll-behavior smooth scrolling is interrupted,
 | 
						|
    // the scroll is not completed to avoid non-smooth snapping to the
 | 
						|
    // prior smooth scroll's destination.
 | 
						|
    mDestination = GetScrollPosition();
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize deltaMultiplier;
 | 
						|
  float negativeTolerance;
 | 
						|
  float positiveTolerance;
 | 
						|
  if (aOrigin == ScrollOrigin::NotSpecified) {
 | 
						|
    aOrigin = ScrollOrigin::Other;
 | 
						|
  }
 | 
						|
  bool isGenericOrigin = (aOrigin == ScrollOrigin::Other);
 | 
						|
 | 
						|
  bool askApzToDoTheScroll = false;
 | 
						|
  if ((aSnapFlags == ScrollSnapFlags::Disabled || !NeedsScrollSnap()) &&
 | 
						|
      gfxPlatform::UseDesktopZoomingScrollbars() &&
 | 
						|
      nsLayoutUtils::AsyncPanZoomEnabled(this) &&
 | 
						|
      !nsLayoutUtils::ShouldDisableApzForElement(GetContent()) &&
 | 
						|
      (WantAsyncScroll() || mZoomableByAPZ) &&
 | 
						|
      CanApzScrollInTheseDirections(DirectionsInDelta(aDelta))) {
 | 
						|
    askApzToDoTheScroll = true;
 | 
						|
  }
 | 
						|
 | 
						|
  switch (aUnit) {
 | 
						|
    case ScrollUnit::DEVICE_PIXELS: {
 | 
						|
      nscoord appUnitsPerDevPixel = PresContext()->AppUnitsPerDevPixel();
 | 
						|
      deltaMultiplier = nsSize(appUnitsPerDevPixel, appUnitsPerDevPixel);
 | 
						|
      if (isGenericOrigin) {
 | 
						|
        aOrigin = ScrollOrigin::Pixels;
 | 
						|
      }
 | 
						|
      negativeTolerance = positiveTolerance = 0.5f;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    case ScrollUnit::LINES: {
 | 
						|
      deltaMultiplier = GetLineScrollAmount();
 | 
						|
      if (isGenericOrigin) {
 | 
						|
        aOrigin = ScrollOrigin::Lines;
 | 
						|
      }
 | 
						|
      negativeTolerance = positiveTolerance = 0.1f;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    case ScrollUnit::PAGES: {
 | 
						|
      deltaMultiplier = GetPageScrollAmount();
 | 
						|
      if (isGenericOrigin) {
 | 
						|
        aOrigin = ScrollOrigin::Pages;
 | 
						|
      }
 | 
						|
      negativeTolerance = 0.05f;
 | 
						|
      positiveTolerance = 0;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    case ScrollUnit::WHOLE: {
 | 
						|
      if (askApzToDoTheScroll) {
 | 
						|
        MOZ_ASSERT(aDelta.x >= -1 && aDelta.x <= 1 && aDelta.y >= -1 &&
 | 
						|
                   aDelta.y <= 1);
 | 
						|
        deltaMultiplier = GetScrollRangeForUserInputEvents().Size();
 | 
						|
        break;
 | 
						|
      } else {
 | 
						|
        nsPoint pos = GetScrollPosition();
 | 
						|
        AdjustDestinationForWholeDelta(aDelta, GetLayoutScrollRange(), pos);
 | 
						|
        ScrollToWithOrigin(
 | 
						|
            pos, nullptr /* range */,
 | 
						|
            ScrollOperationParams{aMode, ScrollOrigin::Other, aSnapFlags,
 | 
						|
                                  ScrollTriggeredByScript::No});
 | 
						|
        // 'this' might be destroyed here
 | 
						|
        if (aOverflow) {
 | 
						|
          *aOverflow = nsIntPoint(0, 0);
 | 
						|
        }
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    default:
 | 
						|
      NS_ERROR("Invalid scroll mode");
 | 
						|
      return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (askApzToDoTheScroll) {
 | 
						|
    nsPoint delta(
 | 
						|
        NSCoordSaturatingNonnegativeMultiply(aDelta.x, deltaMultiplier.width),
 | 
						|
        NSCoordSaturatingNonnegativeMultiply(aDelta.y, deltaMultiplier.height));
 | 
						|
 | 
						|
    AppendScrollUpdate(
 | 
						|
        ScrollPositionUpdate::NewPureRelativeScroll(aOrigin, aMode, delta));
 | 
						|
 | 
						|
    nsIContent* content = GetContent();
 | 
						|
    if (!DisplayPortUtils::HasNonMinimalNonZeroDisplayPort(content)) {
 | 
						|
      if (MOZ_LOG_TEST(sDisplayportLog, LogLevel::Debug)) {
 | 
						|
        mozilla::layers::ScrollableLayerGuid::ViewID viewID =
 | 
						|
            mozilla::layers::ScrollableLayerGuid::NULL_SCROLL_ID;
 | 
						|
        nsLayoutUtils::FindIDFor(content, &viewID);
 | 
						|
        MOZ_LOG(
 | 
						|
            sDisplayportLog, LogLevel::Debug,
 | 
						|
            ("ScrollBy setting displayport on scrollId=%" PRIu64 "\n", viewID));
 | 
						|
      }
 | 
						|
 | 
						|
      DisplayPortUtils::CalculateAndSetDisplayPortMargins(
 | 
						|
          GetScrollTargetFrame(), DisplayPortUtils::RepaintMode::Repaint);
 | 
						|
      nsIFrame* frame = do_QueryFrame(GetScrollTargetFrame());
 | 
						|
      DisplayPortUtils::SetZeroMarginDisplayPortOnAsyncScrollableAncestors(
 | 
						|
          frame);
 | 
						|
    }
 | 
						|
 | 
						|
    SchedulePaint();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsPoint newPos(NSCoordSaturatingAdd(mDestination.x,
 | 
						|
                                      NSCoordSaturatingNonnegativeMultiply(
 | 
						|
                                          aDelta.x, deltaMultiplier.width)),
 | 
						|
                 NSCoordSaturatingAdd(mDestination.y,
 | 
						|
                                      NSCoordSaturatingNonnegativeMultiply(
 | 
						|
                                          aDelta.y, deltaMultiplier.height)));
 | 
						|
 | 
						|
  Maybe<SnapDestination> snapDestination;
 | 
						|
  if (aSnapFlags != ScrollSnapFlags::Disabled) {
 | 
						|
    if (NeedsScrollSnap()) {
 | 
						|
      nscoord appUnitsPerDevPixel = PresContext()->AppUnitsPerDevPixel();
 | 
						|
      deltaMultiplier = nsSize(appUnitsPerDevPixel, appUnitsPerDevPixel);
 | 
						|
      negativeTolerance = 0.1f;
 | 
						|
      positiveTolerance = 0;
 | 
						|
      ScrollUnit snapUnit = aUnit;
 | 
						|
      if (aOrigin == ScrollOrigin::MouseWheel) {
 | 
						|
        // When using a clicky scroll wheel, snap point selection works the same
 | 
						|
        // as keyboard up/down/left/right navigation, but with varying amounts
 | 
						|
        // of scroll delta.
 | 
						|
        snapUnit = ScrollUnit::LINES;
 | 
						|
      }
 | 
						|
      snapDestination = GetSnapPointForDestination(snapUnit, aSnapFlags,
 | 
						|
                                                   mDestination, newPos);
 | 
						|
      if (snapDestination) {
 | 
						|
        newPos = snapDestination->mPosition;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Calculate desired range values.
 | 
						|
  nscoord rangeLowerX, rangeUpperX, rangeLowerY, rangeUpperY;
 | 
						|
  CalcRangeForScrollBy(aDelta.x, newPos.x, negativeTolerance, positiveTolerance,
 | 
						|
                       deltaMultiplier.width, &rangeLowerX, &rangeUpperX);
 | 
						|
  CalcRangeForScrollBy(aDelta.y, newPos.y, negativeTolerance, positiveTolerance,
 | 
						|
                       deltaMultiplier.height, &rangeLowerY, &rangeUpperY);
 | 
						|
  nsRect range(rangeLowerX, rangeLowerY, rangeUpperX - rangeLowerX,
 | 
						|
               rangeUpperY - rangeLowerY);
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      newPos, &range,
 | 
						|
      snapDestination
 | 
						|
          ? ScrollOperationParams{aMode, aOrigin,
 | 
						|
                                  std::move(snapDestination->mTargetIds)}
 | 
						|
          : ScrollOperationParams{aMode, aOrigin});
 | 
						|
  if (!weakFrame.IsAlive()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aOverflow) {
 | 
						|
    nsPoint clampAmount = newPos - mDestination;
 | 
						|
    float appUnitsPerDevPixel = PresContext()->AppUnitsPerDevPixel();
 | 
						|
    *aOverflow =
 | 
						|
        nsIntPoint(NSAppUnitsToIntPixels(clampAmount.x, appUnitsPerDevPixel),
 | 
						|
                   NSAppUnitsToIntPixels(clampAmount.y, appUnitsPerDevPixel));
 | 
						|
  }
 | 
						|
 | 
						|
  if (aUnit == ScrollUnit::DEVICE_PIXELS &&
 | 
						|
      !nsLayoutUtils::AsyncPanZoomEnabled(this)) {
 | 
						|
    // When APZ is disabled, we must track the velocity
 | 
						|
    // on the main thread; otherwise, the APZC will manage this.
 | 
						|
    mVelocityQueue.Sample(GetScrollPosition());
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollByCSSPixelsInternal(const CSSIntPoint& aDelta,
 | 
						|
                                                  ScrollMode aMode,
 | 
						|
                                                  ScrollSnapFlags aSnapFlags) {
 | 
						|
  nsPoint current = GetScrollPosition();
 | 
						|
  // `current` value above might be a value which was aligned to in
 | 
						|
  // layer-pixels, so starting from such points will make the difference between
 | 
						|
  // the given position in script (e.g. scrollTo) and the aligned position
 | 
						|
  // larger, in the worst case the difference can be observed in CSS pixels.
 | 
						|
  // To avoid it, we use the current position in CSS pixels as the start
 | 
						|
  // position.  Hopefully it exactly matches the position where it was given by
 | 
						|
  // the previous scrolling operation, but there may be some edge cases where
 | 
						|
  // the current position in CSS pixels differs from the given position, the
 | 
						|
  // cases should be fixed in bug 1556685.
 | 
						|
  CSSPoint currentCSSPixels;
 | 
						|
  if (StaticPrefs::layout_scroll_disable_pixel_alignment()) {
 | 
						|
    currentCSSPixels = GetScrollPositionCSSPixels();
 | 
						|
  } else {
 | 
						|
    currentCSSPixels = GetRoundedScrollPositionCSSPixels();
 | 
						|
  }
 | 
						|
  nsPoint pt = CSSPoint::ToAppUnits(currentCSSPixels + aDelta);
 | 
						|
 | 
						|
  nscoord halfPixel = nsPresContext::CSSPixelsToAppUnits(0.5f);
 | 
						|
  nsRect range(pt.x - halfPixel, pt.y - halfPixel, 2 * halfPixel - 1,
 | 
						|
               2 * halfPixel - 1);
 | 
						|
  // XXX I don't think the following blocks are needed anymore, now that
 | 
						|
  // ScrollToImpl simply tries to scroll an integer number of layer
 | 
						|
  // pixels from the current position
 | 
						|
  if (aDelta.x == 0.0f) {
 | 
						|
    pt.x = current.x;
 | 
						|
    range.x = pt.x;
 | 
						|
    range.width = 0;
 | 
						|
  }
 | 
						|
  if (aDelta.y == 0.0f) {
 | 
						|
    pt.y = current.y;
 | 
						|
    range.y = pt.y;
 | 
						|
    range.height = 0;
 | 
						|
  }
 | 
						|
  ScrollToWithOrigin(
 | 
						|
      pt, &range,
 | 
						|
      ScrollOperationParams{aMode, ScrollOrigin::Relative, aSnapFlags,
 | 
						|
                            ScrollTriggeredByScript::Yes});
 | 
						|
  // 'this' might be destroyed here
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollSnap(ScrollMode aMode) {
 | 
						|
  float flingSensitivity =
 | 
						|
      StaticPrefs::layout_css_scroll_snap_prediction_sensitivity();
 | 
						|
  int maxVelocity =
 | 
						|
      StaticPrefs::layout_css_scroll_snap_prediction_max_velocity();
 | 
						|
  maxVelocity = nsPresContext::CSSPixelsToAppUnits(maxVelocity);
 | 
						|
  int maxOffset = maxVelocity * flingSensitivity;
 | 
						|
  nsPoint velocity = mVelocityQueue.GetVelocity();
 | 
						|
  // Multiply each component individually to avoid integer multiply
 | 
						|
  nsPoint predictedOffset =
 | 
						|
      nsPoint(velocity.x * flingSensitivity, velocity.y * flingSensitivity);
 | 
						|
  predictedOffset.Clamp(maxOffset);
 | 
						|
  nsPoint pos = GetScrollPosition();
 | 
						|
  nsPoint destinationPos = pos + predictedOffset;
 | 
						|
  ScrollSnap(destinationPos, aMode);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScrollSnap(const nsPoint& aDestination,
 | 
						|
                                   ScrollMode aMode) {
 | 
						|
  nsRect scrollRange = GetLayoutScrollRange();
 | 
						|
  nsPoint pos = GetScrollPosition();
 | 
						|
  nsPoint destination = scrollRange.ClampPoint(aDestination);
 | 
						|
  ScrollSnapFlags snapFlags = ScrollSnapFlags::IntendedEndPosition;
 | 
						|
  if (mVelocityQueue.GetVelocity() != nsPoint()) {
 | 
						|
    snapFlags |= ScrollSnapFlags::IntendedDirection;
 | 
						|
  }
 | 
						|
 | 
						|
  // Bug 1776624 : Consider using mDestination as |aStartPos| argument for this
 | 
						|
  // GetSnapPointForDestination call, this function call is the only one call
 | 
						|
  // site using `GetScrollPosition()` as |aStartPos|.
 | 
						|
  if (auto snapDestination = GetSnapPointForDestination(
 | 
						|
          ScrollUnit::DEVICE_PIXELS, snapFlags, pos, destination)) {
 | 
						|
    destination = snapDestination->mPosition;
 | 
						|
    ScrollToWithOrigin(
 | 
						|
        destination, nullptr /* range */,
 | 
						|
        ScrollOperationParams{aMode, ScrollOrigin::Other,
 | 
						|
                              std::move(snapDestination->mTargetIds)});
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsSize nsHTMLScrollFrame::GetLineScrollAmount() const {
 | 
						|
  RefPtr<nsFontMetrics> fm =
 | 
						|
      nsLayoutUtils::GetInflatedFontMetricsForFrame(this);
 | 
						|
  NS_ASSERTION(fm, "FontMetrics is null, assuming fontHeight == 1 appunit");
 | 
						|
  int32_t appUnitsPerDevPixel = PresContext()->AppUnitsPerDevPixel();
 | 
						|
  nscoord minScrollAmountInAppUnits =
 | 
						|
      std::max(1, StaticPrefs::mousewheel_min_line_scroll_amount()) *
 | 
						|
      appUnitsPerDevPixel;
 | 
						|
  nscoord horizontalAmount = fm ? fm->AveCharWidth() : 0;
 | 
						|
  nscoord verticalAmount = fm ? fm->MaxHeight() : 0;
 | 
						|
  return nsSize(std::max(horizontalAmount, minScrollAmountInAppUnits),
 | 
						|
                std::max(verticalAmount, minScrollAmountInAppUnits));
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Compute the scrollport size excluding any fixed-pos and sticky-pos (that are
 | 
						|
 * stuck) headers and footers. A header or footer is an box that spans that
 | 
						|
 * entire width of the viewport and touches the top (or bottom, respectively) of
 | 
						|
 * the viewport. We also want to consider fixed/sticky elements that stack or
 | 
						|
 * overlap to effectively create a larger header or footer. Headers and footers
 | 
						|
 * that cover more than a third of the the viewport are ignored since they
 | 
						|
 * probably aren't true headers and footers and we don't want to restrict
 | 
						|
 * scrolling too much in such cases. This is a bit conservative --- some
 | 
						|
 * pages use elements as headers or footers that don't span the entire width
 | 
						|
 * of the viewport --- but it should be a good start.
 | 
						|
 *
 | 
						|
 * If aViewportFrame is non-null then the scroll frame is the root scroll
 | 
						|
 * frame and we should consider fixed-pos items.
 | 
						|
 */
 | 
						|
struct TopAndBottom {
 | 
						|
  TopAndBottom(nscoord aTop, nscoord aBottom) : top(aTop), bottom(aBottom) {}
 | 
						|
 | 
						|
  nscoord top, bottom;
 | 
						|
};
 | 
						|
struct TopComparator {
 | 
						|
  bool Equals(const TopAndBottom& A, const TopAndBottom& B) const {
 | 
						|
    return A.top == B.top;
 | 
						|
  }
 | 
						|
  bool LessThan(const TopAndBottom& A, const TopAndBottom& B) const {
 | 
						|
    return A.top < B.top;
 | 
						|
  }
 | 
						|
};
 | 
						|
struct ReverseBottomComparator {
 | 
						|
  bool Equals(const TopAndBottom& A, const TopAndBottom& B) const {
 | 
						|
    return A.bottom == B.bottom;
 | 
						|
  }
 | 
						|
  bool LessThan(const TopAndBottom& A, const TopAndBottom& B) const {
 | 
						|
    return A.bottom > B.bottom;
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
static void AddToListIfHeaderFooter(nsIFrame* aFrame,
 | 
						|
                                    nsIFrame* aScrollPortFrame,
 | 
						|
                                    const nsRect& aScrollPort,
 | 
						|
                                    nsTArray<TopAndBottom>& aList) {
 | 
						|
  nsRect r = aFrame->GetRectRelativeToSelf();
 | 
						|
  r = nsLayoutUtils::TransformFrameRectToAncestor(aFrame, r, aScrollPortFrame);
 | 
						|
  r = r.Intersect(aScrollPort);
 | 
						|
  if ((r.width >= aScrollPort.width / 2 ||
 | 
						|
       r.width >= NSIntPixelsToAppUnits(800, AppUnitsPerCSSPixel())) &&
 | 
						|
      r.height <= aScrollPort.height / 3) {
 | 
						|
    aList.AppendElement(TopAndBottom(r.y, r.YMost()));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static nsSize GetScrollPortSizeExcludingHeadersAndFooters(
 | 
						|
    nsIFrame* aScrollFrame, nsIFrame* aViewportFrame,
 | 
						|
    const nsRect& aScrollPort) {
 | 
						|
  AutoTArray<TopAndBottom, 10> list;
 | 
						|
  if (aViewportFrame) {
 | 
						|
    for (nsIFrame* f : aViewportFrame->GetChildList(FrameChildListID::Fixed)) {
 | 
						|
      AddToListIfHeaderFooter(f, aViewportFrame, aScrollPort, list);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Add sticky frames that are currently in "fixed" positions
 | 
						|
  StickyScrollContainer* ssc =
 | 
						|
      StickyScrollContainer::GetStickyScrollContainerForScrollFrame(
 | 
						|
          aScrollFrame);
 | 
						|
  if (ssc) {
 | 
						|
    const nsTArray<nsIFrame*>& stickyFrames = ssc->GetFrames();
 | 
						|
    for (nsIFrame* f : stickyFrames) {
 | 
						|
      // If it's acting like fixed position.
 | 
						|
      if (ssc->IsStuckInYDirection(f)) {
 | 
						|
        AddToListIfHeaderFooter(f, aScrollFrame, aScrollPort, list);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  list.Sort(TopComparator());
 | 
						|
  nscoord headerBottom = 0;
 | 
						|
  for (uint32_t i = 0; i < list.Length(); ++i) {
 | 
						|
    if (list[i].top <= headerBottom) {
 | 
						|
      headerBottom = std::max(headerBottom, list[i].bottom);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  list.Sort(ReverseBottomComparator());
 | 
						|
  nscoord footerTop = aScrollPort.height;
 | 
						|
  for (uint32_t i = 0; i < list.Length(); ++i) {
 | 
						|
    if (list[i].bottom >= footerTop) {
 | 
						|
      footerTop = std::min(footerTop, list[i].top);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  headerBottom = std::min(aScrollPort.height / 3, headerBottom);
 | 
						|
  footerTop = std::max(aScrollPort.height - aScrollPort.height / 3, footerTop);
 | 
						|
 | 
						|
  return nsSize(aScrollPort.width, footerTop - headerBottom);
 | 
						|
}
 | 
						|
 | 
						|
nsSize nsHTMLScrollFrame::GetPageScrollAmount() const {
 | 
						|
  nsSize effectiveScrollPortSize;
 | 
						|
 | 
						|
  if (GetVisualViewportSize() != mScrollPort.Size()) {
 | 
						|
    // We want to use the visual viewport size if one is set.
 | 
						|
    // The headers/footers adjustment is too complicated to do if there is a
 | 
						|
    // visual viewport that differs from the layout viewport, this is probably
 | 
						|
    // okay.
 | 
						|
    effectiveScrollPortSize = GetVisualViewportSize();
 | 
						|
  } else {
 | 
						|
    // Reduce effective scrollport height by the height of any
 | 
						|
    // fixed-pos/sticky-pos headers or footers
 | 
						|
    effectiveScrollPortSize = GetScrollPortSizeExcludingHeadersAndFooters(
 | 
						|
        const_cast<nsHTMLScrollFrame*>(this),
 | 
						|
        mIsRoot ? PresShell()->GetRootFrame() : nullptr, mScrollPort);
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize lineScrollAmount = GetLineScrollAmount();
 | 
						|
 | 
						|
  // The page increment is the size of the page, minus the smaller of
 | 
						|
  // 10% of the size or 2 lines.
 | 
						|
  return nsSize(effectiveScrollPortSize.width -
 | 
						|
                    std::min(effectiveScrollPortSize.width / 10,
 | 
						|
                             2 * lineScrollAmount.width),
 | 
						|
                effectiveScrollPortSize.height -
 | 
						|
                    std::min(effectiveScrollPortSize.height / 10,
 | 
						|
                             2 * lineScrollAmount.height));
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * this code is resposible for restoring the scroll position back to some
 | 
						|
 * saved position. if the user has not moved the scroll position manually
 | 
						|
 * we keep scrolling down until we get to our original position. keep in
 | 
						|
 * mind that content could incrementally be coming in. we only want to stop
 | 
						|
 * when we reach our new position.
 | 
						|
 */
 | 
						|
void nsHTMLScrollFrame::ScrollToRestoredPosition() {
 | 
						|
  if (mRestorePos.y == -1 || mLastPos.x == -1 || mLastPos.y == -1) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  // make sure our scroll position did not change for where we last put
 | 
						|
  // it. if it does then the user must have moved it, and we no longer
 | 
						|
  // need to restore.
 | 
						|
  //
 | 
						|
  // In the RTL case, we check whether the scroll position changed using the
 | 
						|
  // logical scroll position, but we scroll to the physical scroll position in
 | 
						|
  // all cases
 | 
						|
 | 
						|
  // The layout offset we want to restore is the same as the visual offset
 | 
						|
  // (for now, may change in bug 1499210), but clamped to the layout scroll
 | 
						|
  // range (which can be a subset of the visual scroll range).
 | 
						|
  // Note that we can't do the clamping when initializing mRestorePos in
 | 
						|
  // RestoreState(), since the scrollable rect (which the clamping depends
 | 
						|
  // on) can change over the course of the restoration process.
 | 
						|
  nsPoint layoutRestorePos = GetLayoutScrollRange().ClampPoint(mRestorePos);
 | 
						|
  nsPoint visualRestorePos = GetVisualScrollRange().ClampPoint(mRestorePos);
 | 
						|
 | 
						|
  // Continue restoring until both the layout and visual scroll positions
 | 
						|
  // reach the destination. (Note that the two can only be different for
 | 
						|
  // the root content document's root scroll frame, and when zoomed in).
 | 
						|
  // This is necessary to avoid situations where the two offsets get stuck
 | 
						|
  // at different values and nothing reconciles them (see bug 1519621 comment
 | 
						|
  // 8).
 | 
						|
  nsPoint logicalLayoutScrollPos = GetLogicalScrollPosition();
 | 
						|
 | 
						|
  SCROLLRESTORE_LOG(
 | 
						|
      "%p: ScrollToRestoredPosition (mRestorePos=%s, mLastPos=%s, "
 | 
						|
      "layoutRestorePos=%s, visualRestorePos=%s, "
 | 
						|
      "logicalLayoutScrollPos=%s, "
 | 
						|
      "GetLogicalVisualViewportOffset()=%s)\n",
 | 
						|
      this, ToString(mRestorePos).c_str(), ToString(mLastPos).c_str(),
 | 
						|
      ToString(layoutRestorePos).c_str(), ToString(visualRestorePos).c_str(),
 | 
						|
      ToString(logicalLayoutScrollPos).c_str(),
 | 
						|
      ToString(GetLogicalVisualViewportOffset()).c_str());
 | 
						|
 | 
						|
  // if we didn't move, we still need to restore
 | 
						|
  if (GetLogicalVisualViewportOffset() == mLastPos ||
 | 
						|
      logicalLayoutScrollPos == mLastPos) {
 | 
						|
    // if our desired position is different to the scroll position, scroll.
 | 
						|
    // remember that we could be incrementally loading so we may enter
 | 
						|
    // and scroll many times.
 | 
						|
    if (mRestorePos != mLastPos /* GetLogicalVisualViewportOffset() */ ||
 | 
						|
        layoutRestorePos != logicalLayoutScrollPos) {
 | 
						|
      LoadingState state = GetPageLoadingState();
 | 
						|
      if (state == LoadingState::Stopped && !IsSubtreeDirty()) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      nsPoint visualScrollToPos = visualRestorePos;
 | 
						|
      nsPoint layoutScrollToPos = layoutRestorePos;
 | 
						|
      if (!IsPhysicalLTR()) {
 | 
						|
        // convert from logical to physical scroll position
 | 
						|
        visualScrollToPos.x -=
 | 
						|
            (GetVisualViewportSize().width - mScrolledFrame->GetRect().width);
 | 
						|
        layoutScrollToPos.x -=
 | 
						|
            (GetVisualViewportSize().width - mScrolledFrame->GetRect().width);
 | 
						|
      }
 | 
						|
      AutoWeakFrame weakFrame(this);
 | 
						|
      // It's very important to pass ScrollOrigin::Restore here, so
 | 
						|
      // ScrollToWithOrigin won't clear out mRestorePos.
 | 
						|
      ScrollToWithOrigin(
 | 
						|
          layoutScrollToPos, nullptr,
 | 
						|
          ScrollOperationParams{ScrollMode::Instant, ScrollOrigin::Restore});
 | 
						|
      if (!weakFrame.IsAlive()) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      if (mIsRoot) {
 | 
						|
        PresShell()->ScrollToVisual(visualScrollToPos, FrameMetrics::eRestore,
 | 
						|
                                    ScrollMode::Instant);
 | 
						|
      }
 | 
						|
      if (state == LoadingState::Loading || IsSubtreeDirty()) {
 | 
						|
        // If we're trying to do a history scroll restore, then we want to
 | 
						|
        // keep trying this until we succeed, because the page can be loading
 | 
						|
        // incrementally. So re-get the scroll position for the next iteration,
 | 
						|
        // it might not be exactly equal to mRestorePos due to rounding and
 | 
						|
        // clamping.
 | 
						|
        mLastPos = GetLogicalVisualViewportOffset();
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    // If we get here, either we reached the desired position (mLastPos ==
 | 
						|
    // mRestorePos) or we're not trying to do a history scroll restore, so
 | 
						|
    // we can stop after the scroll attempt above.
 | 
						|
    mRestorePos.y = -1;
 | 
						|
    mLastPos.x = -1;
 | 
						|
    mLastPos.y = -1;
 | 
						|
  } else {
 | 
						|
    // user moved the position, so we won't need to restore
 | 
						|
    mLastPos.x = -1;
 | 
						|
    mLastPos.y = -1;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
auto nsHTMLScrollFrame::GetPageLoadingState() -> LoadingState {
 | 
						|
  bool loadCompleted = false, stopped = false;
 | 
						|
  nsCOMPtr<nsIDocShell> ds = GetContent()->GetComposedDoc()->GetDocShell();
 | 
						|
  if (ds) {
 | 
						|
    nsCOMPtr<nsIDocumentViewer> viewer;
 | 
						|
    ds->GetDocViewer(getter_AddRefs(viewer));
 | 
						|
    if (viewer) {
 | 
						|
      loadCompleted = viewer->GetLoadCompleted();
 | 
						|
      stopped = viewer->GetIsStopped();
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return loadCompleted
 | 
						|
             ? (stopped ? LoadingState::Stopped : LoadingState::Loaded)
 | 
						|
             : LoadingState::Loading;
 | 
						|
}
 | 
						|
 | 
						|
nsHTMLScrollFrame::OverflowState nsHTMLScrollFrame::GetOverflowState() const {
 | 
						|
  nsSize scrollportSize = mScrollPort.Size();
 | 
						|
  nsSize childSize = GetScrolledRect().Size();
 | 
						|
 | 
						|
  OverflowState result = OverflowState::None;
 | 
						|
 | 
						|
  if (childSize.height > scrollportSize.height) {
 | 
						|
    result |= OverflowState::Vertical;
 | 
						|
  }
 | 
						|
 | 
						|
  if (childSize.width > scrollportSize.width) {
 | 
						|
    result |= OverflowState::Horizontal;
 | 
						|
  }
 | 
						|
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
nsresult nsHTMLScrollFrame::FireScrollPortEvent() {
 | 
						|
  mAsyncScrollPortEvent.Forget();
 | 
						|
 | 
						|
  // TODO(emilio): why do we need the whole WillPaintObserver infrastructure and
 | 
						|
  // can't use AddScriptRunner & co? I guess it made sense when we used
 | 
						|
  // WillPaintObserver for scroll events too, or when this used to flush.
 | 
						|
  //
 | 
						|
  // Should we remove this?
 | 
						|
 | 
						|
  OverflowState overflowState = GetOverflowState();
 | 
						|
 | 
						|
  bool newVerticalOverflow = !!(overflowState & OverflowState::Vertical);
 | 
						|
  bool vertChanged = mVerticalOverflow != newVerticalOverflow;
 | 
						|
 | 
						|
  bool newHorizontalOverflow = !!(overflowState & OverflowState::Horizontal);
 | 
						|
  bool horizChanged = mHorizontalOverflow != newHorizontalOverflow;
 | 
						|
 | 
						|
  if (!vertChanged && !horizChanged) {
 | 
						|
    return NS_OK;
 | 
						|
  }
 | 
						|
 | 
						|
  // If both either overflowed or underflowed then we dispatch only one
 | 
						|
  // DOM event.
 | 
						|
  bool both = vertChanged && horizChanged &&
 | 
						|
              newVerticalOverflow == newHorizontalOverflow;
 | 
						|
  InternalScrollPortEvent::OrientType orient;
 | 
						|
  if (both) {
 | 
						|
    orient = InternalScrollPortEvent::eBoth;
 | 
						|
    mHorizontalOverflow = newHorizontalOverflow;
 | 
						|
    mVerticalOverflow = newVerticalOverflow;
 | 
						|
  } else if (vertChanged) {
 | 
						|
    orient = InternalScrollPortEvent::eVertical;
 | 
						|
    mVerticalOverflow = newVerticalOverflow;
 | 
						|
    if (horizChanged) {
 | 
						|
      // We need to dispatch a separate horizontal DOM event. Do that the next
 | 
						|
      // time around since dispatching the vertical DOM event might destroy
 | 
						|
      // the frame.
 | 
						|
      PostOverflowEvent();
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    orient = InternalScrollPortEvent::eHorizontal;
 | 
						|
    mHorizontalOverflow = newHorizontalOverflow;
 | 
						|
  }
 | 
						|
 | 
						|
  InternalScrollPortEvent event(
 | 
						|
      true,
 | 
						|
      (orient == InternalScrollPortEvent::eHorizontal ? mHorizontalOverflow
 | 
						|
                                                      : mVerticalOverflow)
 | 
						|
          ? eScrollPortOverflow
 | 
						|
          : eScrollPortUnderflow,
 | 
						|
      nullptr);
 | 
						|
  event.mOrient = orient;
 | 
						|
 | 
						|
  RefPtr<nsIContent> content = GetContent();
 | 
						|
  RefPtr<nsPresContext> presContext = PresContext();
 | 
						|
  return EventDispatcher::Dispatch(content, presContext, &event);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostScrollEndEvent() {
 | 
						|
  if (mScrollEndEvent) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // The ScrollEndEvent constructor registers itself with the refresh driver.
 | 
						|
  mScrollEndEvent = new ScrollEndEvent(this);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::FireScrollEndEvent() {
 | 
						|
  MOZ_ASSERT(GetContent());
 | 
						|
  MOZ_ASSERT(mScrollEndEvent);
 | 
						|
 | 
						|
  RefPtr<nsPresContext> presContext = PresContext();
 | 
						|
  mScrollEndEvent->Revoke();
 | 
						|
  mScrollEndEvent = nullptr;
 | 
						|
 | 
						|
  nsEventStatus status = nsEventStatus_eIgnore;
 | 
						|
  WidgetGUIEvent event(true, eScrollend, nullptr);
 | 
						|
  event.mFlags.mBubbles = mIsRoot;
 | 
						|
  event.mFlags.mCancelable = false;
 | 
						|
  RefPtr<nsINode> target =
 | 
						|
      mIsRoot ? static_cast<nsINode*>(presContext->Document()) : GetContent();
 | 
						|
  EventDispatcher::Dispatch(target, presContext, &event, nullptr, &status);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ReloadChildFrames() {
 | 
						|
  mScrolledFrame = nullptr;
 | 
						|
  mHScrollbarBox = nullptr;
 | 
						|
  mVScrollbarBox = nullptr;
 | 
						|
  mScrollCornerBox = nullptr;
 | 
						|
  mResizerBox = nullptr;
 | 
						|
 | 
						|
  for (nsIFrame* frame : PrincipalChildList()) {
 | 
						|
    nsIContent* content = frame->GetContent();
 | 
						|
    if (content == GetContent()) {
 | 
						|
      NS_ASSERTION(!mScrolledFrame, "Already found the scrolled frame");
 | 
						|
      mScrolledFrame = frame;
 | 
						|
    } else {
 | 
						|
      nsAutoString value;
 | 
						|
      if (content->IsElement()) {
 | 
						|
        content->AsElement()->GetAttr(nsGkAtoms::orient, value);
 | 
						|
      }
 | 
						|
      if (!value.IsEmpty()) {
 | 
						|
        // probably a scrollbar then
 | 
						|
        if (value.LowerCaseEqualsLiteral("horizontal")) {
 | 
						|
          NS_ASSERTION(!mHScrollbarBox,
 | 
						|
                       "Found multiple horizontal scrollbars?");
 | 
						|
          mHScrollbarBox = do_QueryFrame(frame);
 | 
						|
          MOZ_ASSERT(mHScrollbarBox, "Not a scrollbar?");
 | 
						|
        } else {
 | 
						|
          NS_ASSERTION(!mVScrollbarBox, "Found multiple vertical scrollbars?");
 | 
						|
          mVScrollbarBox = do_QueryFrame(frame);
 | 
						|
          MOZ_ASSERT(mVScrollbarBox, "Not a scrollbar?");
 | 
						|
        }
 | 
						|
      } else if (content->IsXULElement(nsGkAtoms::resizer)) {
 | 
						|
        NS_ASSERTION(!mResizerBox, "Found multiple resizers");
 | 
						|
        mResizerBox = frame;
 | 
						|
      } else if (content->IsXULElement(nsGkAtoms::scrollcorner)) {
 | 
						|
        // probably a scrollcorner
 | 
						|
        NS_ASSERTION(!mScrollCornerBox, "Found multiple scrollcorners");
 | 
						|
        mScrollCornerBox = frame;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
already_AddRefed<Element> nsHTMLScrollFrame::MakeScrollbar(
 | 
						|
    NodeInfo* aNodeInfo, bool aVertical, AnonymousContentKey& aKey) {
 | 
						|
  MOZ_ASSERT(aNodeInfo);
 | 
						|
  MOZ_ASSERT(
 | 
						|
      aNodeInfo->Equals(nsGkAtoms::scrollbar, nullptr, kNameSpaceID_XUL));
 | 
						|
 | 
						|
  static constexpr nsLiteralString kOrientValues[2] = {
 | 
						|
      u"horizontal"_ns,
 | 
						|
      u"vertical"_ns,
 | 
						|
  };
 | 
						|
 | 
						|
  aKey = AnonymousContentKey::Type_Scrollbar;
 | 
						|
  if (aVertical) {
 | 
						|
    aKey |= AnonymousContentKey::Flag_Vertical;
 | 
						|
  }
 | 
						|
 | 
						|
  RefPtr<Element> e;
 | 
						|
  NS_TrustedNewXULElement(getter_AddRefs(e), do_AddRef(aNodeInfo));
 | 
						|
 | 
						|
#ifdef DEBUG
 | 
						|
  // Scrollbars can get restyled by theme changes.  Whether such a restyle
 | 
						|
  // will actually reconstruct them correctly if it involves a frame
 | 
						|
  // reconstruct... I don't know.  :(
 | 
						|
  e->SetProperty(nsGkAtoms::restylableAnonymousNode,
 | 
						|
                 reinterpret_cast<void*>(true));
 | 
						|
#endif  // DEBUG
 | 
						|
 | 
						|
  e->SetAttr(kNameSpaceID_None, nsGkAtoms::orient, kOrientValues[aVertical],
 | 
						|
             false);
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    e->SetProperty(nsGkAtoms::docLevelNativeAnonymousContent,
 | 
						|
                   reinterpret_cast<void*>(true));
 | 
						|
    e->SetAttr(kNameSpaceID_None, nsGkAtoms::root_, u"true"_ns, false);
 | 
						|
 | 
						|
    // Don't bother making style caching take [root="true"] styles into account.
 | 
						|
    aKey = AnonymousContentKey::None;
 | 
						|
  }
 | 
						|
 | 
						|
  return e.forget();
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsForTextControlWithNoScrollbars() const {
 | 
						|
  // FIXME(emilio): we should probably make the scroller inside <input> an
 | 
						|
  // internal pseudo-element, and then this would be simpler.
 | 
						|
  //
 | 
						|
  // Also, this could just use scrollbar-width these days.
 | 
						|
  auto* content = GetContent();
 | 
						|
  if (!content) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  auto* input = content->GetClosestNativeAnonymousSubtreeRootParentOrHost();
 | 
						|
  return input && input->IsHTMLElement(nsGkAtoms::input);
 | 
						|
}
 | 
						|
 | 
						|
auto nsHTMLScrollFrame::GetCurrentAnonymousContent() const
 | 
						|
    -> EnumSet<AnonymousContentType> {
 | 
						|
  EnumSet<AnonymousContentType> result;
 | 
						|
  if (mHScrollbarContent) {
 | 
						|
    result += AnonymousContentType::HorizontalScrollbar;
 | 
						|
  }
 | 
						|
  if (mVScrollbarContent) {
 | 
						|
    result += AnonymousContentType::VerticalScrollbar;
 | 
						|
  }
 | 
						|
  if (mResizerContent) {
 | 
						|
    result += AnonymousContentType::Resizer;
 | 
						|
  }
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
auto nsHTMLScrollFrame::GetNeededAnonymousContent() const
 | 
						|
    -> EnumSet<AnonymousContentType> {
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
 | 
						|
  // Don't create scrollbars if we're an SVG document being used as an image,
 | 
						|
  // or if we're printing/print previewing.
 | 
						|
  // (In the printing case, we allow scrollbars if this is the child of the
 | 
						|
  // viewport & paginated scrolling is enabled, because then we must be the
 | 
						|
  // scroll frame for the print preview window, & that does need scrollbars.)
 | 
						|
  if (pc->Document()->IsBeingUsedAsImage() ||
 | 
						|
      (!pc->IsDynamic() && !(mIsRoot && pc->HasPaginatedScrolling()))) {
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  if (IsForTextControlWithNoScrollbars()) {
 | 
						|
    return {};
 | 
						|
  }
 | 
						|
 | 
						|
  EnumSet<AnonymousContentType> result;
 | 
						|
  // If we're the scrollframe for the root, then we want to construct our
 | 
						|
  // scrollbar frames no matter what.  That way later dynamic changes to
 | 
						|
  // propagated overflow styles will show or hide scrollbars on the viewport
 | 
						|
  // without requiring frame reconstruction of the viewport (good!).
 | 
						|
  //
 | 
						|
  // TODO(emilio): Figure out if we can remove this special-case now that we
 | 
						|
  // have more targeted optimizations.
 | 
						|
  if (mIsRoot) {
 | 
						|
    result += AnonymousContentType::HorizontalScrollbar;
 | 
						|
    result += AnonymousContentType::VerticalScrollbar;
 | 
						|
    // If scrollbar-width is none, don't generate scrollbars.
 | 
						|
  } else if (StyleUIReset()->ScrollbarWidth() != StyleScrollbarWidth::None) {
 | 
						|
    ScrollStyles styles = GetScrollStyles();
 | 
						|
    if (styles.mHorizontal != StyleOverflow::Hidden) {
 | 
						|
      result += AnonymousContentType::HorizontalScrollbar;
 | 
						|
    }
 | 
						|
    if (styles.mVertical != StyleOverflow::Hidden) {
 | 
						|
      result += AnonymousContentType::VerticalScrollbar;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Check if the frame is resizable. Note:
 | 
						|
  // "The effect of the resize property on generated content is undefined.
 | 
						|
  //  Implementations should not apply the resize property to generated
 | 
						|
  //  content." [1]
 | 
						|
  // For info on what is generated content, see [2].
 | 
						|
  // [1]: https://drafts.csswg.org/css-ui/#resize
 | 
						|
  // [2]: https://www.w3.org/TR/CSS2/generate.html#content
 | 
						|
  auto resizeStyle = StyleDisplay()->mResize;
 | 
						|
  if (resizeStyle != StyleResize::None &&
 | 
						|
      !HasAnyStateBits(NS_FRAME_GENERATED_CONTENT)) {
 | 
						|
    result += AnonymousContentType::Resizer;
 | 
						|
  }
 | 
						|
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
nsresult nsHTMLScrollFrame::CreateAnonymousContent(
 | 
						|
    nsTArray<nsIAnonymousContentCreator::ContentInfo>& aElements) {
 | 
						|
  typedef nsIAnonymousContentCreator::ContentInfo ContentInfo;
 | 
						|
 | 
						|
  nsPresContext* presContext = PresContext();
 | 
						|
  nsNodeInfoManager* nodeInfoManager =
 | 
						|
      presContext->Document()->NodeInfoManager();
 | 
						|
 | 
						|
  auto neededAnonContent = GetNeededAnonymousContent();
 | 
						|
  if (neededAnonContent.isEmpty()) {
 | 
						|
    return NS_OK;
 | 
						|
  }
 | 
						|
 | 
						|
  {
 | 
						|
    RefPtr<NodeInfo> nodeInfo = nodeInfoManager->GetNodeInfo(
 | 
						|
        nsGkAtoms::scrollbar, nullptr, kNameSpaceID_XUL, nsINode::ELEMENT_NODE);
 | 
						|
    NS_ENSURE_TRUE(nodeInfo, NS_ERROR_OUT_OF_MEMORY);
 | 
						|
 | 
						|
    if (neededAnonContent.contains(AnonymousContentType::HorizontalScrollbar)) {
 | 
						|
      AnonymousContentKey key;
 | 
						|
      mHScrollbarContent = MakeScrollbar(nodeInfo, /* aVertical */ false, key);
 | 
						|
      aElements.AppendElement(ContentInfo(mHScrollbarContent, key));
 | 
						|
    }
 | 
						|
 | 
						|
    if (neededAnonContent.contains(AnonymousContentType::VerticalScrollbar)) {
 | 
						|
      AnonymousContentKey key;
 | 
						|
      mVScrollbarContent = MakeScrollbar(nodeInfo, /* aVertical */ true, key);
 | 
						|
      aElements.AppendElement(ContentInfo(mVScrollbarContent, key));
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (neededAnonContent.contains(AnonymousContentType::Resizer)) {
 | 
						|
    MOZ_ASSERT(!mIsRoot, "Root scroll frame shouldn't be resizable");
 | 
						|
 | 
						|
    RefPtr<NodeInfo> nodeInfo;
 | 
						|
    nodeInfo = nodeInfoManager->GetNodeInfo(
 | 
						|
        nsGkAtoms::resizer, nullptr, kNameSpaceID_XUL, nsINode::ELEMENT_NODE);
 | 
						|
    NS_ENSURE_TRUE(nodeInfo, NS_ERROR_OUT_OF_MEMORY);
 | 
						|
 | 
						|
    NS_TrustedNewXULElement(getter_AddRefs(mResizerContent), nodeInfo.forget());
 | 
						|
 | 
						|
    nsAutoString dir;
 | 
						|
    switch (StyleDisplay()->mResize) {
 | 
						|
      case StyleResize::Horizontal:
 | 
						|
        if (IsScrollbarOnRight()) {
 | 
						|
          dir.AssignLiteral("right");
 | 
						|
        } else {
 | 
						|
          dir.AssignLiteral("left");
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      case StyleResize::Vertical:
 | 
						|
        dir.AssignLiteral("bottom");
 | 
						|
        if (!IsScrollbarOnRight()) {
 | 
						|
          mResizerContent->SetAttr(kNameSpaceID_None, nsGkAtoms::flip, u""_ns,
 | 
						|
                                   false);
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      case StyleResize::Both:
 | 
						|
        if (IsScrollbarOnRight()) {
 | 
						|
          dir.AssignLiteral("bottomright");
 | 
						|
        } else {
 | 
						|
          dir.AssignLiteral("bottomleft");
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      default:
 | 
						|
        NS_WARNING("only resizable types should have resizers");
 | 
						|
    }
 | 
						|
    mResizerContent->SetAttr(kNameSpaceID_None, nsGkAtoms::dir, dir, false);
 | 
						|
    aElements.AppendElement(mResizerContent);
 | 
						|
  }
 | 
						|
 | 
						|
  if (neededAnonContent.contains(AnonymousContentType::HorizontalScrollbar) &&
 | 
						|
      neededAnonContent.contains(AnonymousContentType::VerticalScrollbar)) {
 | 
						|
    AnonymousContentKey key = AnonymousContentKey::Type_ScrollCorner;
 | 
						|
 | 
						|
    RefPtr<NodeInfo> nodeInfo =
 | 
						|
        nodeInfoManager->GetNodeInfo(nsGkAtoms::scrollcorner, nullptr,
 | 
						|
                                     kNameSpaceID_XUL, nsINode::ELEMENT_NODE);
 | 
						|
    NS_TrustedNewXULElement(getter_AddRefs(mScrollCornerContent),
 | 
						|
                            nodeInfo.forget());
 | 
						|
    if (mIsRoot) {
 | 
						|
      mScrollCornerContent->SetProperty(
 | 
						|
          nsGkAtoms::docLevelNativeAnonymousContent,
 | 
						|
          reinterpret_cast<void*>(true));
 | 
						|
      mScrollCornerContent->SetAttr(kNameSpaceID_None, nsGkAtoms::root_,
 | 
						|
                                    u"true"_ns, false);
 | 
						|
 | 
						|
      // Don't bother making style caching take [root="true"] styles into
 | 
						|
      // account.
 | 
						|
      key = AnonymousContentKey::None;
 | 
						|
    }
 | 
						|
    aElements.AppendElement(ContentInfo(mScrollCornerContent, key));
 | 
						|
  }
 | 
						|
 | 
						|
  // Don't cache styles if we are a child of a <select> element, since we have
 | 
						|
  // some UA style sheet rules that depend on the <select>'s attributes.
 | 
						|
  if (GetContent()->IsHTMLElement(nsGkAtoms::select)) {
 | 
						|
    for (auto& info : aElements) {
 | 
						|
      info.mKey = AnonymousContentKey::None;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AppendAnonymousContentTo(
 | 
						|
    nsTArray<nsIContent*>& aElements, uint32_t aFilter) {
 | 
						|
  if (mHScrollbarContent) {
 | 
						|
    aElements.AppendElement(mHScrollbarContent);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mVScrollbarContent) {
 | 
						|
    aElements.AppendElement(mVScrollbarContent);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mScrollCornerContent) {
 | 
						|
    aElements.AppendElement(mScrollCornerContent);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mResizerContent) {
 | 
						|
    aElements.AppendElement(mResizerContent);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::DidSetComputedStyle(ComputedStyle* aOldComputedStyle) {
 | 
						|
  nsContainerFrame::DidSetComputedStyle(aOldComputedStyle);
 | 
						|
  if (aOldComputedStyle && !mIsRoot &&
 | 
						|
      StyleDisplay()->mScrollSnapType !=
 | 
						|
          aOldComputedStyle->StyleDisplay()->mScrollSnapType) {
 | 
						|
    PostPendingResnap();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::RemoveObservers() {
 | 
						|
  if (mAsyncScroll) {
 | 
						|
    mAsyncScroll->RemoveObserver();
 | 
						|
    mAsyncScroll = nullptr;
 | 
						|
  }
 | 
						|
  if (mAsyncSmoothMSDScroll) {
 | 
						|
    mAsyncSmoothMSDScroll->RemoveObserver();
 | 
						|
    mAsyncSmoothMSDScroll = nullptr;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Called when we want to update the scrollbar position, either because
 | 
						|
 * scrolling happened or the user moved the scrollbar position and we need to
 | 
						|
 * undo that (e.g., when the user clicks to scroll and we're using smooth
 | 
						|
 * scrolling, so we need to put the thumb back to its initial position for the
 | 
						|
 * start of the smooth sequence).
 | 
						|
 */
 | 
						|
void nsHTMLScrollFrame::UpdateScrollbarPosition() {
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  mFrameIsUpdatingScrollbar = true;
 | 
						|
 | 
						|
  nsPoint pt = GetScrollPosition();
 | 
						|
  nsRect scrollRange = GetVisualScrollRange();
 | 
						|
 | 
						|
  if (gfxPlatform::UseDesktopZoomingScrollbars()) {
 | 
						|
    pt = GetVisualViewportOffset();
 | 
						|
    scrollRange = GetScrollRangeForUserInputEvents();
 | 
						|
  }
 | 
						|
 | 
						|
  if (mVScrollbarBox) {
 | 
						|
    SetCoordAttribute(mVScrollbarBox->GetContent()->AsElement(),
 | 
						|
                      nsGkAtoms::curpos, pt.y - scrollRange.y);
 | 
						|
    if (!weakFrame.IsAlive()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  if (mHScrollbarBox) {
 | 
						|
    SetCoordAttribute(mHScrollbarBox->GetContent()->AsElement(),
 | 
						|
                      nsGkAtoms::curpos, pt.x - scrollRange.x);
 | 
						|
    if (!weakFrame.IsAlive()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mFrameIsUpdatingScrollbar = false;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::CurPosAttributeChangedInternal(nsIContent* aContent,
 | 
						|
                                                       bool aDoScroll) {
 | 
						|
  NS_ASSERTION(aContent, "aContent must not be null");
 | 
						|
  NS_ASSERTION((mHScrollbarBox && mHScrollbarBox->GetContent() == aContent) ||
 | 
						|
                   (mVScrollbarBox && mVScrollbarBox->GetContent() == aContent),
 | 
						|
               "unexpected child");
 | 
						|
  MOZ_ASSERT(aContent->IsElement());
 | 
						|
 | 
						|
  // Attribute changes on the scrollbars happen in one of three ways:
 | 
						|
  // 1) The scrollbar changed the attribute in response to some user event
 | 
						|
  // 2) We changed the attribute in response to a ScrollPositionDidChange
 | 
						|
  // callback from the scrolling view
 | 
						|
  // 3) We changed the attribute to adjust the scrollbars for the start
 | 
						|
  // of a smooth scroll operation
 | 
						|
  //
 | 
						|
  // In cases 2 and 3 we do not need to scroll because we're just
 | 
						|
  // updating our scrollbar.
 | 
						|
  if (mFrameIsUpdatingScrollbar) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect scrollRange = GetVisualScrollRange();
 | 
						|
 | 
						|
  nsPoint current = GetScrollPosition() - scrollRange.TopLeft();
 | 
						|
 | 
						|
  if (gfxPlatform::UseDesktopZoomingScrollbars()) {
 | 
						|
    scrollRange = GetScrollRangeForUserInputEvents();
 | 
						|
    current = GetVisualViewportOffset() - scrollRange.TopLeft();
 | 
						|
  }
 | 
						|
 | 
						|
  nsPoint dest;
 | 
						|
  nsRect allowedRange;
 | 
						|
  dest.x = GetCoordAttribute(mHScrollbarBox, nsGkAtoms::curpos, current.x,
 | 
						|
                             &allowedRange.x, &allowedRange.width);
 | 
						|
  dest.y = GetCoordAttribute(mVScrollbarBox, nsGkAtoms::curpos, current.y,
 | 
						|
                             &allowedRange.y, &allowedRange.height);
 | 
						|
  current += scrollRange.TopLeft();
 | 
						|
  dest += scrollRange.TopLeft();
 | 
						|
  allowedRange += scrollRange.TopLeft();
 | 
						|
 | 
						|
  // Don't try to scroll if we're already at an acceptable place.
 | 
						|
  // Don't call Contains here since Contains returns false when the point is
 | 
						|
  // on the bottom or right edge of the rectangle.
 | 
						|
  if (allowedRange.ClampPoint(current) == current) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mScrollbarActivity &&
 | 
						|
      (mHasHorizontalScrollbar || mHasVerticalScrollbar)) {
 | 
						|
    RefPtr<ScrollbarActivity> scrollbarActivity(mScrollbarActivity);
 | 
						|
    scrollbarActivity->ActivityOccurred();
 | 
						|
  }
 | 
						|
 | 
						|
  const bool isSmooth = aContent->AsElement()->HasAttr(nsGkAtoms::smooth);
 | 
						|
  if (isSmooth) {
 | 
						|
    // Make sure an attribute-setting callback occurs even if the view
 | 
						|
    // didn't actually move yet.  We need to make sure other listeners
 | 
						|
    // see that the scroll position is not (yet) what they thought it
 | 
						|
    // was.
 | 
						|
    AutoWeakFrame weakFrame(this);
 | 
						|
    UpdateScrollbarPosition();
 | 
						|
    if (!weakFrame.IsAlive()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (aDoScroll) {
 | 
						|
    ScrollToWithOrigin(dest, &allowedRange,
 | 
						|
                       ScrollOperationParams{
 | 
						|
                           isSmooth ? ScrollMode::Smooth : ScrollMode::Instant,
 | 
						|
                           ScrollOrigin::Scrollbars});
 | 
						|
  }
 | 
						|
  // 'this' might be destroyed here
 | 
						|
}
 | 
						|
 | 
						|
/* ============= Scroll events ========== */
 | 
						|
 | 
						|
nsHTMLScrollFrame::ScrollEvent::ScrollEvent(nsHTMLScrollFrame* aHelper,
 | 
						|
                                            bool aDelayed)
 | 
						|
    : Runnable("nsHTMLScrollFrame::ScrollEvent"), mHelper(aHelper) {
 | 
						|
  mHelper->PresContext()->RefreshDriver()->PostScrollEvent(this, aDelayed);
 | 
						|
}
 | 
						|
 | 
						|
// TODO: Convert this to MOZ_CAN_RUN_SCRIPT (bug 1415230, bug 1535398)
 | 
						|
MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP
 | 
						|
nsHTMLScrollFrame::ScrollEvent::Run() {
 | 
						|
  if (mHelper) {
 | 
						|
    mHelper->FireScrollEvent();
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
nsHTMLScrollFrame::ScrollEndEvent::ScrollEndEvent(nsHTMLScrollFrame* aHelper)
 | 
						|
    : Runnable("nsHTMLScrollFrame::ScrollEndEvent"), mHelper(aHelper) {
 | 
						|
  mHelper->PresContext()->RefreshDriver()->PostScrollEvent(this);
 | 
						|
}
 | 
						|
 | 
						|
MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP
 | 
						|
nsHTMLScrollFrame::ScrollEndEvent::Run() {
 | 
						|
  if (mHelper) {
 | 
						|
    mHelper->FireScrollEndEvent();
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::FireScrollEvent() {
 | 
						|
  RefPtr<nsIContent> content = GetContent();
 | 
						|
  RefPtr<nsPresContext> presContext = PresContext();
 | 
						|
  AUTO_PROFILER_TRACING_MARKER_DOCSHELL("Paint", "FireScrollEvent", GRAPHICS,
 | 
						|
                                        presContext->GetDocShell());
 | 
						|
 | 
						|
  MOZ_ASSERT(mScrollEvent);
 | 
						|
  mScrollEvent->Revoke();
 | 
						|
  mScrollEvent = nullptr;
 | 
						|
 | 
						|
  // If event handling is suppressed, keep posting the scroll event to the
 | 
						|
  // refresh driver until it is unsuppressed. The event is marked as delayed so
 | 
						|
  // that the refresh driver does not continue ticking.
 | 
						|
  if (content->GetComposedDoc() &&
 | 
						|
      content->GetComposedDoc()->EventHandlingSuppressed()) {
 | 
						|
    content->GetComposedDoc()->SetHasDelayedRefreshEvent();
 | 
						|
    PostScrollEvent(/* aDelayed = */ true);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  bool oldProcessing = mProcessingScrollEvent;
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  auto RestoreProcessingScrollEvent = mozilla::MakeScopeExit([&] {
 | 
						|
    if (weakFrame.IsAlive()) {  // Otherwise `this` will be dead too.
 | 
						|
      mProcessingScrollEvent = oldProcessing;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  mProcessingScrollEvent = true;
 | 
						|
 | 
						|
  WidgetGUIEvent event(true, eScroll, nullptr);
 | 
						|
  nsEventStatus status = nsEventStatus_eIgnore;
 | 
						|
  // Fire viewport scroll events at the document (where they
 | 
						|
  // will bubble to the window)
 | 
						|
  mozilla::layers::ScrollLinkedEffectDetector detector(
 | 
						|
      content->GetComposedDoc(),
 | 
						|
      presContext->RefreshDriver()->MostRecentRefresh());
 | 
						|
  if (mIsRoot) {
 | 
						|
    if (RefPtr<Document> doc = content->GetUncomposedDoc()) {
 | 
						|
      EventDispatcher::Dispatch(doc, presContext, &event, nullptr, &status);
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    // scroll events fired at elements don't bubble (although scroll events
 | 
						|
    // fired at documents do, to the window)
 | 
						|
    event.mFlags.mBubbles = false;
 | 
						|
    EventDispatcher::Dispatch(content, presContext, &event, nullptr, &status);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostScrollEvent(bool aDelayed) {
 | 
						|
  if (mScrollEvent) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // The ScrollEvent constructor registers itself with the refresh driver.
 | 
						|
  mScrollEvent = new ScrollEvent(this, aDelayed);
 | 
						|
}
 | 
						|
 | 
						|
// TODO: Convert this to MOZ_CAN_RUN_SCRIPT (bug 1415230, bug 1535398)
 | 
						|
MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP
 | 
						|
nsHTMLScrollFrame::AsyncScrollPortEvent::Run() {
 | 
						|
  return mHelper ? mHelper->FireScrollPortEvent() : NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostOverflowEvent() {
 | 
						|
  if (mAsyncScrollPortEvent.IsPending()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  OverflowState overflowState = GetOverflowState();
 | 
						|
 | 
						|
  bool newVerticalOverflow = !!(overflowState & OverflowState::Vertical);
 | 
						|
  bool vertChanged = mVerticalOverflow != newVerticalOverflow;
 | 
						|
 | 
						|
  bool newHorizontalOverflow = !!(overflowState & OverflowState::Horizontal);
 | 
						|
  bool horizChanged = mHorizontalOverflow != newHorizontalOverflow;
 | 
						|
 | 
						|
  if (!vertChanged && !horizChanged) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsRootPresContext* rpc = PresContext()->GetRootPresContext();
 | 
						|
  if (!rpc) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mAsyncScrollPortEvent = new AsyncScrollPortEvent(this);
 | 
						|
  rpc->AddWillPaintObserver(mAsyncScrollPortEvent.get());
 | 
						|
}
 | 
						|
 | 
						|
nsIFrame* nsHTMLScrollFrame::GetFrameForStyle() const {
 | 
						|
  nsIFrame* styleFrame = nullptr;
 | 
						|
  if (mIsRoot) {
 | 
						|
    if (const Element* rootElement =
 | 
						|
            PresContext()->Document()->GetRootElement()) {
 | 
						|
      styleFrame = rootElement->GetPrimaryFrame();
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    styleFrame = const_cast<nsHTMLScrollFrame*>(this);
 | 
						|
  }
 | 
						|
 | 
						|
  return styleFrame;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::NeedsScrollSnap() const {
 | 
						|
  nsIFrame* scrollSnapFrame = GetFrameForStyle();
 | 
						|
  if (!scrollSnapFrame) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  return scrollSnapFrame->StyleDisplay()->mScrollSnapType.strictness !=
 | 
						|
         StyleScrollSnapStrictness::None;
 | 
						|
}
 | 
						|
 | 
						|
nsSize nsHTMLScrollFrame::GetSnapportSize() const {
 | 
						|
  nsRect snapport = GetScrollPortRect();
 | 
						|
  nsMargin scrollPadding = GetScrollPadding();
 | 
						|
  snapport.Deflate(scrollPadding);
 | 
						|
  return snapport.Size();
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsScrollbarOnRight() const {
 | 
						|
  // The position of the scrollbar in top-level windows depends on the pref
 | 
						|
  // layout.scrollbar.side. For non-top-level elements, it depends only on the
 | 
						|
  // directionaliy of the element (equivalent to a value of "1" for the pref).
 | 
						|
  if (!mIsRoot) {
 | 
						|
    return IsPhysicalLTR();
 | 
						|
  }
 | 
						|
  switch (StaticPrefs::layout_scrollbar_side()) {
 | 
						|
    default:
 | 
						|
    case 0:  // UI directionality
 | 
						|
      return StaticPrefs::bidi_direction() == IBMBIDI_TEXTDIRECTION_LTR;
 | 
						|
    case 1:  // Document / content directionality
 | 
						|
      return IsPhysicalLTR();
 | 
						|
    case 2:  // Always right
 | 
						|
      return true;
 | 
						|
    case 3:  // Always left
 | 
						|
      return false;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsScrollingActive() const {
 | 
						|
  const nsStyleDisplay* disp = StyleDisplay();
 | 
						|
  if (disp->mWillChange.bits & StyleWillChangeBits::SCROLL) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
  return mHasBeenScrolledRecently || IsAlwaysActive() ||
 | 
						|
         DisplayPortUtils::HasDisplayPort(content) ||
 | 
						|
         nsContentUtils::HasScrollgrab(content);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::FinishReflowForScrollbar(Element* aElement,
 | 
						|
                                                 nscoord aMinXY, nscoord aMaxXY,
 | 
						|
                                                 nscoord aCurPosXY,
 | 
						|
                                                 nscoord aPageIncrement,
 | 
						|
                                                 nscoord aIncrement) {
 | 
						|
  // Scrollbars assume zero is the minimum position, so translate for them.
 | 
						|
  SetCoordAttribute(aElement, nsGkAtoms::curpos, aCurPosXY - aMinXY);
 | 
						|
  SetScrollbarEnabled(aElement, aMaxXY - aMinXY);
 | 
						|
  SetCoordAttribute(aElement, nsGkAtoms::maxpos, aMaxXY - aMinXY);
 | 
						|
  SetCoordAttribute(aElement, nsGkAtoms::pageincrement, aPageIncrement);
 | 
						|
  SetCoordAttribute(aElement, nsGkAtoms::increment, aIncrement);
 | 
						|
}
 | 
						|
 | 
						|
class MOZ_RAII AutoMinimumScaleSizeChangeDetector final {
 | 
						|
 public:
 | 
						|
  explicit AutoMinimumScaleSizeChangeDetector(
 | 
						|
      nsHTMLScrollFrame* ansHTMLScrollFrame)
 | 
						|
      : mHelper(ansHTMLScrollFrame) {
 | 
						|
    MOZ_ASSERT(mHelper);
 | 
						|
    MOZ_ASSERT(mHelper->mIsRoot);
 | 
						|
 | 
						|
    mPreviousMinimumScaleSize = ansHTMLScrollFrame->mMinimumScaleSize;
 | 
						|
    mPreviousIsUsingMinimumScaleSize =
 | 
						|
        ansHTMLScrollFrame->mIsUsingMinimumScaleSize;
 | 
						|
  }
 | 
						|
  ~AutoMinimumScaleSizeChangeDetector() {
 | 
						|
    if (mPreviousMinimumScaleSize != mHelper->mMinimumScaleSize ||
 | 
						|
        mPreviousIsUsingMinimumScaleSize != mHelper->mIsUsingMinimumScaleSize) {
 | 
						|
      mHelper->mMinimumScaleSizeChanged = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  nsHTMLScrollFrame* mHelper;
 | 
						|
 | 
						|
  nsSize mPreviousMinimumScaleSize;
 | 
						|
  bool mPreviousIsUsingMinimumScaleSize;
 | 
						|
};
 | 
						|
 | 
						|
nsSize nsHTMLScrollFrame::TrueOuterSize(nsDisplayListBuilder* aBuilder) const {
 | 
						|
  if (!PresShell()->UsesMobileViewportSizing()) {
 | 
						|
    return GetSize();
 | 
						|
  }
 | 
						|
 | 
						|
  RefPtr<MobileViewportManager> manager =
 | 
						|
      PresShell()->GetMobileViewportManager();
 | 
						|
  MOZ_ASSERT(manager);
 | 
						|
 | 
						|
  LayoutDeviceIntSize displaySize = manager->DisplaySize();
 | 
						|
 | 
						|
  MOZ_ASSERT(aBuilder);
 | 
						|
  // In case of WebRender, we expand the outer size to include the dynamic
 | 
						|
  // toolbar area here.
 | 
						|
  // In case of non WebRender, we expand the size dynamically in
 | 
						|
  // MoveScrollbarForLayerMargin in AsyncCompositionManager.cpp.
 | 
						|
  WebRenderLayerManager* layerManager = aBuilder->GetWidgetLayerManager();
 | 
						|
  if (layerManager) {
 | 
						|
    displaySize.height += ViewAs<LayoutDevicePixel>(
 | 
						|
        PresContext()->GetDynamicToolbarMaxHeight(),
 | 
						|
        PixelCastJustification::LayoutDeviceIsScreenForBounds);
 | 
						|
  }
 | 
						|
 | 
						|
  return LayoutDeviceSize::ToAppUnits(displaySize,
 | 
						|
                                      PresContext()->AppUnitsPerDevPixel());
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::UpdateMinimumScaleSize(
 | 
						|
    const nsRect& aScrollableOverflow, const nsSize& aICBSize) {
 | 
						|
  MOZ_ASSERT(mIsRoot);
 | 
						|
 | 
						|
  AutoMinimumScaleSizeChangeDetector minimumScaleSizeChangeDetector(this);
 | 
						|
 | 
						|
  mIsUsingMinimumScaleSize = false;
 | 
						|
 | 
						|
  if (!PresShell()->UsesMobileViewportSizing()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
  MOZ_ASSERT(pc->IsRootContentDocumentCrossProcess(),
 | 
						|
             "The pres context should be for the root content document");
 | 
						|
 | 
						|
  RefPtr<MobileViewportManager> manager =
 | 
						|
      PresShell()->GetMobileViewportManager();
 | 
						|
  MOZ_ASSERT(manager);
 | 
						|
 | 
						|
  ScreenIntSize displaySize = ViewAs<ScreenPixel>(
 | 
						|
      manager->DisplaySize(),
 | 
						|
      PixelCastJustification::LayoutDeviceIsScreenForBounds);
 | 
						|
  if (displaySize.width == 0 || displaySize.height == 0) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  if (aScrollableOverflow.IsEmpty()) {
 | 
						|
    // Bail if the scrollable overflow rect is empty, as we're going to be
 | 
						|
    // dividing by it.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  Document* doc = pc->Document();
 | 
						|
  MOZ_ASSERT(doc, "The document should be valid");
 | 
						|
  if (doc->GetFullscreenElement()) {
 | 
						|
    // Don't use the minimum scale size in the case of fullscreen state.
 | 
						|
    // FIXME: 1508177: We will no longer need this.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsViewportInfo viewportInfo = doc->GetViewportInfo(displaySize);
 | 
						|
  if (!viewportInfo.IsZoomAllowed()) {
 | 
						|
    // Don't apply the minimum scale size if user-scalable=no is specified.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // The intrinsic minimum scale is the scale that fits the entire content
 | 
						|
  // width into the visual viewport.
 | 
						|
  CSSToScreenScale intrinsicMinScale(
 | 
						|
      displaySize.width / CSSRect::FromAppUnits(aScrollableOverflow).XMost());
 | 
						|
 | 
						|
  // The scale used to compute the minimum-scale size is the larger of the
 | 
						|
  // intrinsic minimum and the min-scale from the meta viewport tag.
 | 
						|
  CSSToScreenScale minScale =
 | 
						|
      std::max(intrinsicMinScale, viewportInfo.GetMinZoom());
 | 
						|
 | 
						|
  // The minimum-scale size is the size of the visual viewport when zoomed
 | 
						|
  // to be the minimum scale.
 | 
						|
  mMinimumScaleSize = CSSSize::ToAppUnits(ScreenSize(displaySize) / minScale);
 | 
						|
 | 
						|
  // Ensure the minimum-scale size is never smaller than the ICB size.
 | 
						|
  // That could happen if a page has a meta viewport tag with large explicitly
 | 
						|
  // specified viewport dimensions (making the ICB large) and also a large
 | 
						|
  // minimum scale (making the min-scale size small).
 | 
						|
  mMinimumScaleSize = Max(aICBSize, mMinimumScaleSize);
 | 
						|
 | 
						|
  mIsUsingMinimumScaleSize = true;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::ReflowFinished() {
 | 
						|
  mPostedReflowCallback = false;
 | 
						|
 | 
						|
  TryScheduleScrollAnimations();
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    if (mMinimumScaleSizeChanged && PresShell()->UsesMobileViewportSizing() &&
 | 
						|
        !PresShell()->IsResolutionUpdatedByApz()) {
 | 
						|
      RefPtr<MobileViewportManager> manager =
 | 
						|
          PresShell()->GetMobileViewportManager();
 | 
						|
      MOZ_ASSERT(manager);
 | 
						|
 | 
						|
      manager->ShrinkToDisplaySizeIfNeeded();
 | 
						|
      mMinimumScaleSizeChanged = false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!UsesOverlayScrollbars()) {
 | 
						|
      // Layout scrollbars may have added or removed during reflow, so let's
 | 
						|
      // update the visual viewport accordingly. Note that this may be a no-op
 | 
						|
      // because we might have recomputed the visual viewport size during the
 | 
						|
      // reflow itself, just before laying out the fixed-pos items. But there
 | 
						|
      // might be cases where that code doesn't run, so this is a sort of
 | 
						|
      // backstop to ensure we do that recomputation.
 | 
						|
      if (RefPtr<MobileViewportManager> manager =
 | 
						|
              PresShell()->GetMobileViewportManager()) {
 | 
						|
        manager->UpdateVisualViewportSizeForPotentialScrollbarChange();
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
    const bool hasVerticalOverflow =
 | 
						|
        GetOverflowState() & OverflowState::Vertical &&
 | 
						|
        GetScrollStyles().mVertical != StyleOverflow::Hidden;
 | 
						|
    if (!mFirstReflow && mHasVerticalOverflowForDynamicToolbar &&
 | 
						|
        !hasVerticalOverflow) {
 | 
						|
      PresShell()->MaybeNotifyShowDynamicToolbar();
 | 
						|
    }
 | 
						|
    mHasVerticalOverflowForDynamicToolbar = hasVerticalOverflow;
 | 
						|
#endif  // defined(MOZ_WIDGET_ANDROID)
 | 
						|
  }
 | 
						|
 | 
						|
  bool doScroll = true;
 | 
						|
  if (IsSubtreeDirty()) {
 | 
						|
    // We will get another call after the next reflow and scrolling
 | 
						|
    // later is less janky.
 | 
						|
    doScroll = false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mFirstReflow) {
 | 
						|
    nsPoint currentScrollPos = GetScrollPosition();
 | 
						|
    if (!mScrollUpdates.IsEmpty() &&
 | 
						|
        mScrollUpdates.LastElement().GetOrigin() == ScrollOrigin::None &&
 | 
						|
        currentScrollPos != nsPoint()) {
 | 
						|
      // With frame reconstructions, the reconstructed frame may have a nonzero
 | 
						|
      // scroll position by the end of the reflow, but without going through
 | 
						|
      // RestoreState. In particular this can happen with RTL XUL scrollframes,
 | 
						|
      // see https://bugzilla.mozilla.org/show_bug.cgi?id=1664638#c14.
 | 
						|
      // Upon construction, the nsHTMLScrollFrame constructor will have inserted
 | 
						|
      // a ScrollPositionUpdate into mScrollUpdates with origin None and a zero
 | 
						|
      // scroll position, but here we update that to hold the correct scroll
 | 
						|
      // position. Otherwise APZ may end up resetting the scroll position to
 | 
						|
      // zero incorrectly. If we ever hit this codepath, it must be on a reflow
 | 
						|
      // immediately following the scrollframe construction, so there should be
 | 
						|
      // exactly one ScrollPositionUpdate in mScrollUpdates.
 | 
						|
      MOZ_ASSERT(mScrollUpdates.Length() == 1);
 | 
						|
      MOZ_ASSERT(mScrollUpdates.LastElement().GetGeneration() ==
 | 
						|
                 mScrollGeneration);
 | 
						|
      MOZ_ASSERT(mScrollUpdates.LastElement().GetDestination() == CSSPoint());
 | 
						|
      SCROLLRESTORE_LOG("%p: updating initial SPU to pos %s\n", this,
 | 
						|
                        ToString(currentScrollPos).c_str());
 | 
						|
      mScrollUpdates.Clear();
 | 
						|
      AppendScrollUpdate(
 | 
						|
          ScrollPositionUpdate::NewScrollframe(currentScrollPos));
 | 
						|
    }
 | 
						|
 | 
						|
    mFirstReflow = false;
 | 
						|
  }
 | 
						|
 | 
						|
  nsAutoScriptBlocker scriptBlocker;
 | 
						|
 | 
						|
  if (mReclampVVOffsetInReflowFinished) {
 | 
						|
    MOZ_ASSERT(mIsRoot && PresShell()->IsVisualViewportOffsetSet());
 | 
						|
    mReclampVVOffsetInReflowFinished = false;
 | 
						|
    AutoWeakFrame weakFrame(this);
 | 
						|
    PresShell()->SetVisualViewportOffset(PresShell()->GetVisualViewportOffset(),
 | 
						|
                                         GetScrollPosition());
 | 
						|
    NS_ENSURE_TRUE(weakFrame.IsAlive(), false);
 | 
						|
  }
 | 
						|
 | 
						|
  if (doScroll) {
 | 
						|
    ScrollToRestoredPosition();
 | 
						|
 | 
						|
    // Clamp current scroll position to new bounds. Normally this won't
 | 
						|
    // do anything.
 | 
						|
    nsPoint currentScrollPos = GetScrollPosition();
 | 
						|
    ScrollToImpl(currentScrollPos, nsRect(currentScrollPos, nsSize(0, 0)),
 | 
						|
                 ScrollOrigin::Clamp);
 | 
						|
    if (ScrollAnimationState().isEmpty()) {
 | 
						|
      // We need to have mDestination track the current scroll position,
 | 
						|
      // in case it falls outside the new reflow area. mDestination is used
 | 
						|
      // by ScrollBy as its starting position.
 | 
						|
      mDestination = GetScrollPosition();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mUpdateScrollbarAttributes) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  mUpdateScrollbarAttributes = false;
 | 
						|
 | 
						|
  // Update scrollbar attributes.
 | 
						|
  if (mMayHaveDirtyFixedChildren) {
 | 
						|
    mMayHaveDirtyFixedChildren = false;
 | 
						|
    nsIFrame* parentFrame = GetParent();
 | 
						|
    for (nsIFrame* fixedChild =
 | 
						|
             parentFrame->GetChildList(FrameChildListID::Fixed).FirstChild();
 | 
						|
         fixedChild; fixedChild = fixedChild->GetNextSibling()) {
 | 
						|
      // force a reflow of the fixed child
 | 
						|
      PresShell()->FrameNeedsReflow(fixedChild, IntrinsicDirty::None,
 | 
						|
                                    NS_FRAME_HAS_DIRTY_CHILDREN);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Suppress handling of the curpos attribute changes we make here.
 | 
						|
  NS_ASSERTION(!mFrameIsUpdatingScrollbar, "We shouldn't be reentering here");
 | 
						|
  mFrameIsUpdatingScrollbar = true;
 | 
						|
 | 
						|
  // FIXME(emilio): Why this instead of mHScrollbarContent / mVScrollbarContent?
 | 
						|
  RefPtr<Element> vScroll =
 | 
						|
      mVScrollbarBox ? mVScrollbarBox->GetContent()->AsElement() : nullptr;
 | 
						|
  RefPtr<Element> hScroll =
 | 
						|
      mHScrollbarBox ? mHScrollbarBox->GetContent()->AsElement() : nullptr;
 | 
						|
 | 
						|
  // Note, in some cases this may get deleted while finishing reflow
 | 
						|
  // for scrollbars. XXXmats is this still true now that we have a script
 | 
						|
  // blocker in this scope? (if not, remove the weak frame checks below).
 | 
						|
  if (vScroll || hScroll) {
 | 
						|
    nsSize visualViewportSize = GetVisualViewportSize();
 | 
						|
    nsRect scrollRange = GetVisualScrollRange();
 | 
						|
    nsPoint scrollPos = GetScrollPosition();
 | 
						|
    nsSize lineScrollAmount = GetLineScrollAmount();
 | 
						|
 | 
						|
    if (gfxPlatform::UseDesktopZoomingScrollbars()) {
 | 
						|
      scrollRange = GetScrollRangeForUserInputEvents();
 | 
						|
      scrollPos = GetVisualViewportOffset();
 | 
						|
    }
 | 
						|
 | 
						|
    // If modifying the logic here, be sure to modify the corresponding
 | 
						|
    // compositor-side calculation in ScrollThumbUtils::ApplyTransformForAxis().
 | 
						|
    AutoWeakFrame weakFrame(this);
 | 
						|
    if (vScroll) {
 | 
						|
      const double kScrollMultiplier =
 | 
						|
          StaticPrefs::toolkit_scrollbox_verticalScrollDistance();
 | 
						|
      nscoord increment = lineScrollAmount.height * kScrollMultiplier;
 | 
						|
      // We normally use (visualViewportSize.height - increment) for height of
 | 
						|
      // page scrolling.  However, it is too small when increment is very large.
 | 
						|
      // (If increment is larger than visualViewportSize.height, direction of
 | 
						|
      // scrolling will be opposite). To avoid it, we use
 | 
						|
      // (float(visualViewportSize.height) * 0.8) as lower bound value of height
 | 
						|
      // of page scrolling. (bug 383267)
 | 
						|
      // XXX shouldn't we use GetPageScrollAmount here?
 | 
						|
      nscoord pageincrement = nscoord(visualViewportSize.height - increment);
 | 
						|
      nscoord pageincrementMin =
 | 
						|
          nscoord(float(visualViewportSize.height) * 0.8);
 | 
						|
      FinishReflowForScrollbar(
 | 
						|
          vScroll, scrollRange.y, scrollRange.YMost(), scrollPos.y,
 | 
						|
          std::max(pageincrement, pageincrementMin), increment);
 | 
						|
    }
 | 
						|
    if (hScroll) {
 | 
						|
      const double kScrollMultiplier =
 | 
						|
          StaticPrefs::toolkit_scrollbox_horizontalScrollDistance();
 | 
						|
      nscoord increment = lineScrollAmount.width * kScrollMultiplier;
 | 
						|
      FinishReflowForScrollbar(
 | 
						|
          hScroll, scrollRange.x, scrollRange.XMost(), scrollPos.x,
 | 
						|
          nscoord(float(visualViewportSize.width) * 0.8), increment);
 | 
						|
    }
 | 
						|
    NS_ENSURE_TRUE(weakFrame.IsAlive(), false);
 | 
						|
  }
 | 
						|
 | 
						|
  mFrameIsUpdatingScrollbar = false;
 | 
						|
  // We used to rely on the curpos attribute changes above to scroll the
 | 
						|
  // view.  However, for scrolling to the left of the viewport, we
 | 
						|
  // rescale the curpos attribute, which means that operations like
 | 
						|
  // resizing the window while it is scrolled all the way to the left
 | 
						|
  // hold the curpos attribute constant at 0 while still requiring
 | 
						|
  // scrolling.  So we suppress the effect of the changes above with
 | 
						|
  // mFrameIsUpdatingScrollbar and call CurPosAttributeChanged here.
 | 
						|
  // (It actually even works some of the time without this, thanks to
 | 
						|
  // nsSliderFrame::AttributeChanged's handling of maxpos, but not when
 | 
						|
  // we hide the scrollbar on a large size change, such as
 | 
						|
  // maximization.)
 | 
						|
  if (!mHScrollbarBox && !mVScrollbarBox) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  CurPosAttributeChangedInternal(
 | 
						|
      mVScrollbarBox ? mVScrollbarBox->GetContent()->AsElement()
 | 
						|
                     : mHScrollbarBox->GetContent()->AsElement(),
 | 
						|
      doScroll);
 | 
						|
  return doScroll;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ReflowCallbackCanceled() {
 | 
						|
  mPostedReflowCallback = false;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::ComputeCustomOverflow(OverflowAreas& aOverflowAreas) {
 | 
						|
  ScrollStyles ss = GetScrollStyles();
 | 
						|
 | 
						|
  // Reflow when the change in overflow leads to one of our scrollbars
 | 
						|
  // changing or might require repositioning the scrolled content due to
 | 
						|
  // reduced extents.
 | 
						|
  nsRect scrolledRect = GetScrolledRect();
 | 
						|
  ScrollDirections overflowChange =
 | 
						|
      GetOverflowChange(scrolledRect, mPrevScrolledRect);
 | 
						|
  mPrevScrolledRect = scrolledRect;
 | 
						|
 | 
						|
  bool needReflow = false;
 | 
						|
  nsPoint scrollPosition = GetScrollPosition();
 | 
						|
  if (overflowChange.contains(ScrollDirection::eHorizontal)) {
 | 
						|
    if (ss.mHorizontal != StyleOverflow::Hidden || scrollPosition.x) {
 | 
						|
      needReflow = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  if (overflowChange.contains(ScrollDirection::eVertical)) {
 | 
						|
    if (ss.mVertical != StyleOverflow::Hidden || scrollPosition.y) {
 | 
						|
      needReflow = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (needReflow) {
 | 
						|
    // If there are scrollbars, or we're not at the beginning of the pane,
 | 
						|
    // the scroll position may change. In this case, mark the frame as
 | 
						|
    // needing reflow. Don't use NS_FRAME_IS_DIRTY as dirty as that means
 | 
						|
    // we have to reflow the frame and all its descendants, and we don't
 | 
						|
    // have to do that here. Only this frame needs to be reflowed.
 | 
						|
    PresShell()->FrameNeedsReflow(this, IntrinsicDirty::None,
 | 
						|
                                  NS_FRAME_HAS_DIRTY_CHILDREN);
 | 
						|
    // Ensure that next time nsHTMLScrollFrame::Reflow runs, we don't skip
 | 
						|
    // updating the scrollbars. (Because the overflow area of the scrolled
 | 
						|
    // frame has probably just been updated, Reflow won't see it change.)
 | 
						|
    mSkippedScrollbarLayout = true;
 | 
						|
    return false;  // reflowing will update overflow
 | 
						|
  }
 | 
						|
  PostOverflowEvent();
 | 
						|
  return nsContainerFrame::ComputeCustomOverflow(aOverflowAreas);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::UpdateSticky() {
 | 
						|
  StickyScrollContainer* ssc =
 | 
						|
      StickyScrollContainer::GetStickyScrollContainerForScrollFrame(this);
 | 
						|
  if (ssc) {
 | 
						|
    ssc->UpdatePositions(GetScrollPosition(), this);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::UpdatePrevScrolledRect() {
 | 
						|
  // The layout scroll range is determinated by the scrolled rect and the scroll
 | 
						|
  // port, so if the scrolled rect is updated, we may have to schedule the
 | 
						|
  // associated scroll-driven animations' restyles.
 | 
						|
  nsRect currScrolledRect = GetScrolledRect();
 | 
						|
  if (!currScrolledRect.IsEqualEdges(mPrevScrolledRect)) {
 | 
						|
    mMayScheduleScrollAnimations = true;
 | 
						|
  }
 | 
						|
  mPrevScrolledRect = currScrolledRect;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AdjustScrollbarRectForResizer(
 | 
						|
    nsIFrame* aFrame, nsPresContext* aPresContext, nsRect& aRect,
 | 
						|
    bool aHasResizer, ScrollDirection aDirection) {
 | 
						|
  if ((aDirection == ScrollDirection::eVertical ? aRect.width : aRect.height) ==
 | 
						|
      0) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // if a content resizer is present, use its size. Otherwise, check if the
 | 
						|
  // widget has a resizer.
 | 
						|
  nsRect resizerRect;
 | 
						|
  if (aHasResizer) {
 | 
						|
    resizerRect = mResizerBox->GetRect();
 | 
						|
  } else {
 | 
						|
    nsPoint offset;
 | 
						|
    nsIWidget* widget = aFrame->GetNearestWidget(offset);
 | 
						|
    LayoutDeviceIntRect widgetRect;
 | 
						|
    if (!widget || !widget->ShowsResizeIndicator(&widgetRect)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    resizerRect =
 | 
						|
        nsRect(aPresContext->DevPixelsToAppUnits(widgetRect.x) - offset.x,
 | 
						|
               aPresContext->DevPixelsToAppUnits(widgetRect.y) - offset.y,
 | 
						|
               aPresContext->DevPixelsToAppUnits(widgetRect.width),
 | 
						|
               aPresContext->DevPixelsToAppUnits(widgetRect.height));
 | 
						|
  }
 | 
						|
 | 
						|
  if (resizerRect.Contains(aRect.BottomRight() - nsPoint(1, 1))) {
 | 
						|
    switch (aDirection) {
 | 
						|
      case ScrollDirection::eVertical:
 | 
						|
        aRect.height = std::max(0, resizerRect.y - aRect.y);
 | 
						|
        break;
 | 
						|
      case ScrollDirection::eHorizontal:
 | 
						|
        aRect.width = std::max(0, resizerRect.x - aRect.x);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  } else if (resizerRect.Contains(aRect.BottomLeft() + nsPoint(1, -1))) {
 | 
						|
    switch (aDirection) {
 | 
						|
      case ScrollDirection::eVertical:
 | 
						|
        aRect.height = std::max(0, resizerRect.y - aRect.y);
 | 
						|
        break;
 | 
						|
      case ScrollDirection::eHorizontal: {
 | 
						|
        nscoord xmost = aRect.XMost();
 | 
						|
        aRect.x = std::max(aRect.x, resizerRect.XMost());
 | 
						|
        aRect.width = xmost - aRect.x;
 | 
						|
        break;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static void AdjustOverlappingScrollbars(nsRect& aVRect, nsRect& aHRect) {
 | 
						|
  if (aVRect.IsEmpty() || aHRect.IsEmpty()) return;
 | 
						|
 | 
						|
  const nsRect oldVRect = aVRect;
 | 
						|
  const nsRect oldHRect = aHRect;
 | 
						|
  if (oldVRect.Contains(oldHRect.BottomRight() - nsPoint(1, 1))) {
 | 
						|
    aHRect.width = std::max(0, oldVRect.x - oldHRect.x);
 | 
						|
  } else if (oldVRect.Contains(oldHRect.BottomLeft() - nsPoint(0, 1))) {
 | 
						|
    nscoord overlap = std::min(oldHRect.width, oldVRect.XMost() - oldHRect.x);
 | 
						|
    aHRect.x += overlap;
 | 
						|
    aHRect.width -= overlap;
 | 
						|
  }
 | 
						|
  if (oldHRect.Contains(oldVRect.BottomRight() - nsPoint(1, 1))) {
 | 
						|
    aVRect.height = std::max(0, oldHRect.y - oldVRect.y);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::LayoutScrollbarPartAtRect(
 | 
						|
    const ScrollReflowInput& aState, ReflowInput& aKidReflowInput,
 | 
						|
    const nsRect& aRect) {
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
  nsIFrame* kid = aKidReflowInput.mFrame;
 | 
						|
  const auto wm = kid->GetWritingMode();
 | 
						|
  ReflowOutput desiredSize(wm);
 | 
						|
  MOZ_ASSERT(!wm.IsVertical(),
 | 
						|
             "Scrollbar parts should have writing-mode: initial");
 | 
						|
  MOZ_ASSERT(!wm.IsInlineReversed(),
 | 
						|
             "Scrollbar parts should have writing-mode: initial");
 | 
						|
  // XXX Maybe get a meaningful container size or something. Shouldn't matter
 | 
						|
  // given our asserts above.
 | 
						|
  const nsSize containerSize;
 | 
						|
  aKidReflowInput.SetComputedISize(aRect.Width());
 | 
						|
  aKidReflowInput.SetComputedBSize(aRect.Height());
 | 
						|
 | 
						|
  const LogicalPoint pos(wm, aRect.TopLeft(), containerSize);
 | 
						|
  const auto flags = ReflowChildFlags::Default;
 | 
						|
  nsReflowStatus status;
 | 
						|
  ReflowOutput kidDesiredSize(wm);
 | 
						|
  ReflowChild(kid, pc, kidDesiredSize, aKidReflowInput, wm, pos, containerSize,
 | 
						|
              flags, status);
 | 
						|
  FinishReflowChild(kid, pc, kidDesiredSize, &aKidReflowInput, wm, pos,
 | 
						|
                    containerSize, flags);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::LayoutScrollbars(ScrollReflowInput& aState,
 | 
						|
                                         const nsRect& aInsideBorderArea,
 | 
						|
                                         const nsRect& aOldScrollPort) {
 | 
						|
  NS_ASSERTION(!mSuppressScrollbarUpdate, "This should have been suppressed");
 | 
						|
 | 
						|
  const bool scrollbarOnLeft = !IsScrollbarOnRight();
 | 
						|
  const bool overlayScrollbars = UsesOverlayScrollbars();
 | 
						|
  const bool overlayScrollBarsOnRoot = overlayScrollbars && mIsRoot;
 | 
						|
  const bool showVScrollbar = mVScrollbarBox && mHasVerticalScrollbar;
 | 
						|
  const bool showHScrollbar = mHScrollbarBox && mHasHorizontalScrollbar;
 | 
						|
 | 
						|
  nsSize compositionSize = mScrollPort.Size();
 | 
						|
  if (overlayScrollBarsOnRoot) {
 | 
						|
    compositionSize = nsLayoutUtils::CalculateCompositionSizeForFrame(
 | 
						|
        this, false, &compositionSize);
 | 
						|
  }
 | 
						|
 | 
						|
  nsPresContext* presContext = mScrolledFrame->PresContext();
 | 
						|
  nsRect vRect;
 | 
						|
  if (showVScrollbar) {
 | 
						|
    vRect.height =
 | 
						|
        overlayScrollBarsOnRoot ? compositionSize.height : mScrollPort.height;
 | 
						|
    vRect.y = mScrollPort.y;
 | 
						|
    if (scrollbarOnLeft) {
 | 
						|
      vRect.width = mScrollPort.x - aInsideBorderArea.x;
 | 
						|
      vRect.x = aInsideBorderArea.x;
 | 
						|
    } else {
 | 
						|
      vRect.width = aInsideBorderArea.XMost() - mScrollPort.XMost();
 | 
						|
      vRect.x = mScrollPort.x + compositionSize.width;
 | 
						|
    }
 | 
						|
    if (overlayScrollbars || mOnlyNeedVScrollbarToScrollVVInsideLV) {
 | 
						|
      const nscoord width = aState.VScrollbarPrefWidth();
 | 
						|
      // There is no space reserved for the layout scrollbar, it is currently
 | 
						|
      // not visible because it is positioned just outside the scrollport. But
 | 
						|
      // we know that it needs to be made visible so we shift it back in.
 | 
						|
      vRect.width += width;
 | 
						|
      if (!scrollbarOnLeft) {
 | 
						|
        vRect.x -= width;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  nsRect hRect;
 | 
						|
  if (showHScrollbar) {
 | 
						|
    hRect.width =
 | 
						|
        overlayScrollBarsOnRoot ? compositionSize.width : mScrollPort.width;
 | 
						|
    hRect.x = mScrollPort.x;
 | 
						|
    hRect.height = aInsideBorderArea.YMost() - mScrollPort.YMost();
 | 
						|
    hRect.y = mScrollPort.y + compositionSize.height;
 | 
						|
 | 
						|
    if (overlayScrollbars || mOnlyNeedHScrollbarToScrollVVInsideLV) {
 | 
						|
      const nscoord height = aState.HScrollbarPrefHeight();
 | 
						|
      hRect.height += height;
 | 
						|
      // There is no space reserved for the layout scrollbar, it is currently
 | 
						|
      // not visible because it is positioned just outside the scrollport. But
 | 
						|
      // we know that it needs to be made visible so we shift it back in.
 | 
						|
      hRect.y -= height;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  const bool hasVisualOnlyScrollbarsOnBothDirections =
 | 
						|
      !overlayScrollbars && showHScrollbar &&
 | 
						|
      mOnlyNeedHScrollbarToScrollVVInsideLV && showVScrollbar &&
 | 
						|
      mOnlyNeedVScrollbarToScrollVVInsideLV;
 | 
						|
  nsPresContext* pc = PresContext();
 | 
						|
 | 
						|
  // place the scrollcorner
 | 
						|
  if (mScrollCornerBox) {
 | 
						|
    nsRect r(0, 0, 0, 0);
 | 
						|
    if (scrollbarOnLeft) {
 | 
						|
      // scrollbar (if any) on left
 | 
						|
      r.width = showVScrollbar ? mScrollPort.x - aInsideBorderArea.x : 0;
 | 
						|
      r.x = aInsideBorderArea.x;
 | 
						|
    } else {
 | 
						|
      // scrollbar (if any) on right
 | 
						|
      r.width =
 | 
						|
          showVScrollbar ? aInsideBorderArea.XMost() - mScrollPort.XMost() : 0;
 | 
						|
      r.x = aInsideBorderArea.XMost() - r.width;
 | 
						|
    }
 | 
						|
    NS_ASSERTION(r.width >= 0, "Scroll area should be inside client rect");
 | 
						|
 | 
						|
    if (showHScrollbar) {
 | 
						|
      // scrollbar (if any) on bottom
 | 
						|
      // Note we don't support the horizontal scrollbar at the top side.
 | 
						|
      r.height = aInsideBorderArea.YMost() - mScrollPort.YMost();
 | 
						|
      NS_ASSERTION(r.height >= 0, "Scroll area should be inside client rect");
 | 
						|
    }
 | 
						|
    r.y = aInsideBorderArea.YMost() - r.height;
 | 
						|
 | 
						|
    // If we have layout scrollbars and both scrollbars are present and both are
 | 
						|
    // only needed to scroll the VV inside the LV then we need a scrollcorner
 | 
						|
    // but the above calculation will result in an empty rect, so adjust it.
 | 
						|
    if (r.IsEmpty() && hasVisualOnlyScrollbarsOnBothDirections) {
 | 
						|
      r.width = vRect.width;
 | 
						|
      r.height = hRect.height;
 | 
						|
      r.x = scrollbarOnLeft ? mScrollPort.x : mScrollPort.XMost() - r.width;
 | 
						|
      r.y = mScrollPort.YMost() - r.height;
 | 
						|
    }
 | 
						|
 | 
						|
    ReflowInput scrollCornerRI(
 | 
						|
        pc, aState.mReflowInput, mScrollCornerBox,
 | 
						|
        LogicalSize(mScrollCornerBox->GetWritingMode(), r.Size()));
 | 
						|
    LayoutScrollbarPartAtRect(aState, scrollCornerRI, r);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mResizerBox) {
 | 
						|
    // If a resizer is present, get its size.
 | 
						|
    //
 | 
						|
    // TODO(emilio): Should this really account for scrollbar-width?
 | 
						|
    auto scrollbarWidth = nsLayoutUtils::StyleForScrollbar(this)
 | 
						|
                              ->StyleUIReset()
 | 
						|
                              ->ScrollbarWidth();
 | 
						|
    const nscoord scrollbarSize =
 | 
						|
        GetNonOverlayScrollbarSize(pc, scrollbarWidth);
 | 
						|
    ReflowInput resizerRI(pc, aState.mReflowInput, mResizerBox,
 | 
						|
                          LogicalSize(mResizerBox->GetWritingMode()));
 | 
						|
    nsSize resizerMinSize = {resizerRI.ComputedMinWidth(),
 | 
						|
                             resizerRI.ComputedMinHeight()};
 | 
						|
 | 
						|
    nsRect r;
 | 
						|
    r.width = std::max(std::max(r.width, scrollbarSize), resizerMinSize.width);
 | 
						|
    r.x = scrollbarOnLeft ? aInsideBorderArea.x
 | 
						|
                          : aInsideBorderArea.XMost() - r.width;
 | 
						|
    r.height =
 | 
						|
        std::max(std::max(r.height, scrollbarSize), resizerMinSize.height);
 | 
						|
    r.y = aInsideBorderArea.YMost() - r.height;
 | 
						|
 | 
						|
    LayoutScrollbarPartAtRect(aState, resizerRI, r);
 | 
						|
  }
 | 
						|
 | 
						|
  // Note that AdjustScrollbarRectForResizer has to be called after the
 | 
						|
  // resizer has been laid out immediately above this because it gets the rect
 | 
						|
  // of the resizer frame.
 | 
						|
  if (mVScrollbarBox) {
 | 
						|
    AdjustScrollbarRectForResizer(this, presContext, vRect, mResizerBox,
 | 
						|
                                  ScrollDirection::eVertical);
 | 
						|
  }
 | 
						|
  if (mHScrollbarBox) {
 | 
						|
    AdjustScrollbarRectForResizer(this, presContext, hRect, mResizerBox,
 | 
						|
                                  ScrollDirection::eHorizontal);
 | 
						|
  }
 | 
						|
 | 
						|
  // Layout scrollbars can overlap at this point if they are both present and
 | 
						|
  // both only needed to scroll the VV inside the LV.
 | 
						|
  if (!LookAndFeel::GetInt(LookAndFeel::IntID::AllowOverlayScrollbarsOverlap) ||
 | 
						|
      hasVisualOnlyScrollbarsOnBothDirections) {
 | 
						|
    AdjustOverlappingScrollbars(vRect, hRect);
 | 
						|
  }
 | 
						|
  if (mVScrollbarBox) {
 | 
						|
    ReflowInput vScrollbarRI(
 | 
						|
        pc, aState.mReflowInput, mVScrollbarBox,
 | 
						|
        LogicalSize(mVScrollbarBox->GetWritingMode(), vRect.Size()));
 | 
						|
    LayoutScrollbarPartAtRect(aState, vScrollbarRI, vRect);
 | 
						|
  }
 | 
						|
  if (mHScrollbarBox) {
 | 
						|
    ReflowInput hScrollbarRI(
 | 
						|
        pc, aState.mReflowInput, mHScrollbarBox,
 | 
						|
        LogicalSize(mHScrollbarBox->GetWritingMode(), hRect.Size()));
 | 
						|
    LayoutScrollbarPartAtRect(aState, hScrollbarRI, hRect);
 | 
						|
  }
 | 
						|
 | 
						|
  // may need to update fixed position children of the viewport,
 | 
						|
  // if the client area changed size because of an incremental
 | 
						|
  // reflow of a descendant.  (If the outer frame is dirty, the fixed
 | 
						|
  // children will be re-laid out anyway)
 | 
						|
  if (aOldScrollPort.Size() != mScrollPort.Size() &&
 | 
						|
      !HasAnyStateBits(NS_FRAME_IS_DIRTY) && mIsRoot) {
 | 
						|
    mMayHaveDirtyFixedChildren = true;
 | 
						|
  }
 | 
						|
 | 
						|
  // post reflow callback to modify scrollbar attributes
 | 
						|
  mUpdateScrollbarAttributes = true;
 | 
						|
  if (!mPostedReflowCallback) {
 | 
						|
    PresShell()->PostReflowCallback(this);
 | 
						|
    mPostedReflowCallback = true;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
#if DEBUG
 | 
						|
static bool ShellIsAlive(nsWeakPtr& aWeakPtr) {
 | 
						|
  RefPtr<PresShell> presShell = do_QueryReferent(aWeakPtr);
 | 
						|
  return !!presShell;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetScrollbarEnabled(Element* aElement,
 | 
						|
                                            nscoord aMaxPos) {
 | 
						|
  DebugOnly<nsWeakPtr> weakShell(do_GetWeakReference(PresShell()));
 | 
						|
  if (aMaxPos) {
 | 
						|
    aElement->UnsetAttr(kNameSpaceID_None, nsGkAtoms::disabled, true);
 | 
						|
  } else {
 | 
						|
    aElement->SetAttr(kNameSpaceID_None, nsGkAtoms::disabled, u"true"_ns, true);
 | 
						|
  }
 | 
						|
  MOZ_ASSERT(ShellIsAlive(weakShell), "pres shell was destroyed by scrolling");
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetCoordAttribute(Element* aElement, nsAtom* aAtom,
 | 
						|
                                          nscoord aSize) {
 | 
						|
  DebugOnly<nsWeakPtr> weakShell(do_GetWeakReference(PresShell()));
 | 
						|
  // convert to pixels
 | 
						|
  int32_t pixelSize = nsPresContext::AppUnitsToIntCSSPixels(aSize);
 | 
						|
 | 
						|
  // only set the attribute if it changed.
 | 
						|
 | 
						|
  nsAutoString newValue;
 | 
						|
  newValue.AppendInt(pixelSize);
 | 
						|
 | 
						|
  if (aElement->AttrValueIs(kNameSpaceID_None, aAtom, newValue, eCaseMatters)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  AutoWeakFrame weakFrame(this);
 | 
						|
  RefPtr<Element> kungFuDeathGrip = aElement;
 | 
						|
  aElement->SetAttr(kNameSpaceID_None, aAtom, newValue, true);
 | 
						|
  MOZ_ASSERT(ShellIsAlive(weakShell), "pres shell was destroyed by scrolling");
 | 
						|
  if (!weakFrame.IsAlive()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mScrollbarActivity &&
 | 
						|
      (mHasHorizontalScrollbar || mHasVerticalScrollbar)) {
 | 
						|
    RefPtr<ScrollbarActivity> scrollbarActivity(mScrollbarActivity);
 | 
						|
    scrollbarActivity->ActivityOccurred();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static void ReduceRadii(nscoord aXBorder, nscoord aYBorder, nscoord& aXRadius,
 | 
						|
                        nscoord& aYRadius) {
 | 
						|
  // In order to ensure that the inside edge of the border has no
 | 
						|
  // curvature, we need at least one of its radii to be zero.
 | 
						|
  if (aXRadius <= aXBorder || aYRadius <= aYBorder) return;
 | 
						|
 | 
						|
  // For any corner where we reduce the radii, preserve the corner's shape.
 | 
						|
  double ratio =
 | 
						|
      std::max(double(aXBorder) / aXRadius, double(aYBorder) / aYRadius);
 | 
						|
  aXRadius *= ratio;
 | 
						|
  aYRadius *= ratio;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Implement an override for nsIFrame::GetBorderRadii to ensure that
 | 
						|
 * the clipping region for the border radius does not clip the scrollbars.
 | 
						|
 *
 | 
						|
 * In other words, we require that the border radius be reduced until the
 | 
						|
 * inner border radius at the inner edge of the border is 0 wherever we
 | 
						|
 * have scrollbars.
 | 
						|
 */
 | 
						|
bool nsHTMLScrollFrame::GetBorderRadii(const nsSize& aFrameSize,
 | 
						|
                                       const nsSize& aBorderArea,
 | 
						|
                                       Sides aSkipSides,
 | 
						|
                                       nscoord aRadii[8]) const {
 | 
						|
  if (!nsContainerFrame::GetBorderRadii(aFrameSize, aBorderArea, aSkipSides,
 | 
						|
                                        aRadii)) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // Since we can use GetActualScrollbarSizes (rather than
 | 
						|
  // GetDesiredScrollbarSizes) since this doesn't affect reflow, we
 | 
						|
  // probably should.
 | 
						|
  nsMargin sb = GetActualScrollbarSizes();
 | 
						|
  nsMargin border = GetUsedBorder();
 | 
						|
 | 
						|
  if (sb.left > 0 || sb.top > 0) {
 | 
						|
    ReduceRadii(border.left, border.top, aRadii[eCornerTopLeftX],
 | 
						|
                aRadii[eCornerTopLeftY]);
 | 
						|
  }
 | 
						|
 | 
						|
  if (sb.top > 0 || sb.right > 0) {
 | 
						|
    ReduceRadii(border.right, border.top, aRadii[eCornerTopRightX],
 | 
						|
                aRadii[eCornerTopRightY]);
 | 
						|
  }
 | 
						|
 | 
						|
  if (sb.right > 0 || sb.bottom > 0) {
 | 
						|
    ReduceRadii(border.right, border.bottom, aRadii[eCornerBottomRightX],
 | 
						|
                aRadii[eCornerBottomRightY]);
 | 
						|
  }
 | 
						|
 | 
						|
  if (sb.bottom > 0 || sb.left > 0) {
 | 
						|
    ReduceRadii(border.left, border.bottom, aRadii[eCornerBottomLeftX],
 | 
						|
                aRadii[eCornerBottomLeftY]);
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
static nscoord SnapCoord(nscoord aCoord, double aRes,
 | 
						|
                         nscoord aAppUnitsPerPixel) {
 | 
						|
  if (StaticPrefs::layout_scroll_disable_pixel_alignment()) {
 | 
						|
    return aCoord;
 | 
						|
  }
 | 
						|
  double snappedToLayerPixels = NS_round((aRes * aCoord) / aAppUnitsPerPixel);
 | 
						|
  return NSToCoordRoundWithClamp(snappedToLayerPixels * aAppUnitsPerPixel /
 | 
						|
                                 aRes);
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetScrolledRect() const {
 | 
						|
  nsRect result = GetUnsnappedScrolledRectInternal(
 | 
						|
      mScrolledFrame->ScrollableOverflowRect(), mScrollPort.Size());
 | 
						|
 | 
						|
#if 0
 | 
						|
  // This happens often enough.
 | 
						|
  if (result.width < mScrollPort.width || result.height < mScrollPort.height) {
 | 
						|
    NS_WARNING("Scrolled rect smaller than scrollport?");
 | 
						|
  }
 | 
						|
#endif
 | 
						|
 | 
						|
  // Expand / contract the result by up to half a layer pixel so that scrolling
 | 
						|
  // to the right / bottom edge does not change the layer pixel alignment of
 | 
						|
  // the scrolled contents.
 | 
						|
 | 
						|
  if (result.x == 0 && result.y == 0 && result.width == mScrollPort.width &&
 | 
						|
      result.height == mScrollPort.height) {
 | 
						|
    // The edges that we would snap are already aligned with the scroll port,
 | 
						|
    // so we can skip all the work below.
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  // For that, we first convert the scroll port and the scrolled rect to rects
 | 
						|
  // relative to the reference frame, since that's the space where painting does
 | 
						|
  // snapping.
 | 
						|
  nsSize visualViewportSize = GetVisualViewportSize();
 | 
						|
  const nsIFrame* referenceFrame =
 | 
						|
      mReferenceFrameDuringPainting ? mReferenceFrameDuringPainting
 | 
						|
                                    : nsLayoutUtils::GetReferenceFrame(
 | 
						|
                                          const_cast<nsHTMLScrollFrame*>(this));
 | 
						|
  nsPoint toReferenceFrame = GetOffsetToCrossDoc(referenceFrame);
 | 
						|
  nsRect scrollPort(mScrollPort.TopLeft() + toReferenceFrame,
 | 
						|
                    visualViewportSize);
 | 
						|
  nsRect scrolledRect = result + scrollPort.TopLeft();
 | 
						|
 | 
						|
  if (scrollPort.Overflows() || scrolledRect.Overflows()) {
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  // Now, snap the bottom right corner of both of these rects.
 | 
						|
  // We snap to layer pixels, so we need to respect the layer's scale.
 | 
						|
  nscoord appUnitsPerDevPixel =
 | 
						|
      mScrolledFrame->PresContext()->AppUnitsPerDevPixel();
 | 
						|
  MatrixScales scale = GetPaintedLayerScaleForFrame(
 | 
						|
      mScrolledFrame, /* aIncludeCSSTransform = */ false);
 | 
						|
  if (scale.xScale == 0 || scale.yScale == 0) {
 | 
						|
    scale = MatrixScales();
 | 
						|
  }
 | 
						|
 | 
						|
  // Compute bounds for the scroll position, and computed the snapped scrolled
 | 
						|
  // rect from the scroll position bounds.
 | 
						|
  nscoord snappedScrolledAreaBottom =
 | 
						|
      SnapCoord(scrolledRect.YMost(), scale.yScale, appUnitsPerDevPixel);
 | 
						|
  nscoord snappedScrollPortBottom =
 | 
						|
      SnapCoord(scrollPort.YMost(), scale.yScale, appUnitsPerDevPixel);
 | 
						|
  nscoord maximumScrollOffsetY =
 | 
						|
      snappedScrolledAreaBottom - snappedScrollPortBottom;
 | 
						|
  result.SetBottomEdge(scrollPort.height + maximumScrollOffsetY);
 | 
						|
 | 
						|
  if (GetScrolledFrameDir() == StyleDirection::Ltr) {
 | 
						|
    nscoord snappedScrolledAreaRight =
 | 
						|
        SnapCoord(scrolledRect.XMost(), scale.xScale, appUnitsPerDevPixel);
 | 
						|
    nscoord snappedScrollPortRight =
 | 
						|
        SnapCoord(scrollPort.XMost(), scale.xScale, appUnitsPerDevPixel);
 | 
						|
    nscoord maximumScrollOffsetX =
 | 
						|
        snappedScrolledAreaRight - snappedScrollPortRight;
 | 
						|
    result.SetRightEdge(scrollPort.width + maximumScrollOffsetX);
 | 
						|
  } else {
 | 
						|
    // In RTL, the scrolled area's right edge is at scrollPort.XMost(),
 | 
						|
    // and the scrolled area's x position is zero or negative. We want
 | 
						|
    // the right edge to stay flush with the scroll port, so we snap the
 | 
						|
    // left edge.
 | 
						|
    nscoord snappedScrolledAreaLeft =
 | 
						|
        SnapCoord(scrolledRect.x, scale.xScale, appUnitsPerDevPixel);
 | 
						|
    nscoord snappedScrollPortLeft =
 | 
						|
        SnapCoord(scrollPort.x, scale.xScale, appUnitsPerDevPixel);
 | 
						|
    nscoord minimumScrollOffsetX =
 | 
						|
        snappedScrolledAreaLeft - snappedScrollPortLeft;
 | 
						|
    result.SetLeftEdge(minimumScrollOffsetX);
 | 
						|
  }
 | 
						|
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
StyleDirection nsHTMLScrollFrame::GetScrolledFrameDir() const {
 | 
						|
  // If the scrolled frame has unicode-bidi: plaintext, the paragraph
 | 
						|
  // direction set by the text content overrides the direction of the frame
 | 
						|
  if (mScrolledFrame->StyleTextReset()->mUnicodeBidi ==
 | 
						|
      StyleUnicodeBidi::Plaintext) {
 | 
						|
    if (nsIFrame* child = mScrolledFrame->PrincipalChildList().FirstChild()) {
 | 
						|
      return nsBidiPresUtils::ParagraphDirection(child) ==
 | 
						|
                     mozilla::intl::BidiDirection::LTR
 | 
						|
                 ? StyleDirection::Ltr
 | 
						|
                 : StyleDirection::Rtl;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return IsBidiLTR() ? StyleDirection::Ltr : StyleDirection::Rtl;
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetUnsnappedScrolledRectInternal(
 | 
						|
    const nsRect& aScrolledOverflowArea, const nsSize& aScrollPortSize) const {
 | 
						|
  return nsLayoutUtils::GetScrolledRect(mScrolledFrame, aScrolledOverflowArea,
 | 
						|
                                        aScrollPortSize, GetScrolledFrameDir());
 | 
						|
}
 | 
						|
 | 
						|
nsMargin nsHTMLScrollFrame::GetActualScrollbarSizes(
 | 
						|
    nsIScrollableFrame::ScrollbarSizesOptions
 | 
						|
        aOptions /* = nsIScrollableFrame::ScrollbarSizesOptions::NONE */)
 | 
						|
    const {
 | 
						|
  nsRect r = GetPaddingRectRelativeToSelf();
 | 
						|
 | 
						|
  nsMargin m(mScrollPort.y - r.y, r.XMost() - mScrollPort.XMost(),
 | 
						|
             r.YMost() - mScrollPort.YMost(), mScrollPort.x - r.x);
 | 
						|
 | 
						|
  if (aOptions == nsIScrollableFrame::ScrollbarSizesOptions::
 | 
						|
                      INCLUDE_VISUAL_VIEWPORT_SCROLLBARS &&
 | 
						|
      !UsesOverlayScrollbars()) {
 | 
						|
    // If we are using layout scrollbars and they only exist to scroll the
 | 
						|
    // visual viewport then they do not take up any layout space (so the
 | 
						|
    // scrollport is the same as the padding rect) but they do cover everything
 | 
						|
    // below them so some callers may want to include this special type of
 | 
						|
    // scrollbars in the returned value.
 | 
						|
    if (mHScrollbarBox && mHasHorizontalScrollbar &&
 | 
						|
        mOnlyNeedHScrollbarToScrollVVInsideLV) {
 | 
						|
      m.bottom += mHScrollbarBox->GetRect().height;
 | 
						|
    }
 | 
						|
    if (mVScrollbarBox && mHasVerticalScrollbar &&
 | 
						|
        mOnlyNeedVScrollbarToScrollVVInsideLV) {
 | 
						|
      if (IsScrollbarOnRight()) {
 | 
						|
        m.right += mVScrollbarBox->GetRect().width;
 | 
						|
      } else {
 | 
						|
        m.left += mVScrollbarBox->GetRect().width;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return m;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetScrollbarVisibility(nsIFrame* aScrollbar,
 | 
						|
                                               bool aVisible) {
 | 
						|
  nsScrollbarFrame* scrollbar = do_QueryFrame(aScrollbar);
 | 
						|
  if (scrollbar) {
 | 
						|
    // See if we have a mediator.
 | 
						|
    nsIScrollbarMediator* mediator = scrollbar->GetScrollbarMediator();
 | 
						|
    if (mediator) {
 | 
						|
      // Inform the mediator of the visibility change.
 | 
						|
      mediator->VisibilityChanged(aVisible);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsHTMLScrollFrame::GetCoordAttribute(nsIFrame* aBox, nsAtom* aAtom,
 | 
						|
                                             nscoord aDefaultValue,
 | 
						|
                                             nscoord* aRangeStart,
 | 
						|
                                             nscoord* aRangeLength) {
 | 
						|
  if (aBox) {
 | 
						|
    nsIContent* content = aBox->GetContent();
 | 
						|
 | 
						|
    nsAutoString value;
 | 
						|
    if (content->IsElement()) {
 | 
						|
      content->AsElement()->GetAttr(aAtom, value);
 | 
						|
    }
 | 
						|
    if (!value.IsEmpty()) {
 | 
						|
      nsresult error;
 | 
						|
      // convert it to appunits
 | 
						|
      nscoord result =
 | 
						|
          nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
 | 
						|
      nscoord halfPixel = nsPresContext::CSSPixelsToAppUnits(0.5f);
 | 
						|
      // Any nscoord value that would round to the attribute value when
 | 
						|
      // converted to CSS pixels is allowed.
 | 
						|
      *aRangeStart = result - halfPixel;
 | 
						|
      *aRangeLength = halfPixel * 2 - 1;
 | 
						|
      return result;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Only this exact default value is allowed.
 | 
						|
  *aRangeStart = aDefaultValue;
 | 
						|
  *aRangeLength = 0;
 | 
						|
  return aDefaultValue;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsLastScrollUpdateAnimating() const {
 | 
						|
  if (!mScrollUpdates.IsEmpty()) {
 | 
						|
    switch (mScrollUpdates.LastElement().GetMode()) {
 | 
						|
      case ScrollMode::Smooth:
 | 
						|
      case ScrollMode::SmoothMsd:
 | 
						|
        return true;
 | 
						|
      case ScrollMode::Instant:
 | 
						|
      case ScrollMode::Normal:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsLastScrollUpdateTriggeredByScriptAnimating() const {
 | 
						|
  if (!mScrollUpdates.IsEmpty()) {
 | 
						|
    const ScrollPositionUpdate& lastUpdate = mScrollUpdates.LastElement();
 | 
						|
    if (lastUpdate.WasTriggeredByScript() &&
 | 
						|
        (mScrollUpdates.LastElement().GetMode() == ScrollMode::Smooth ||
 | 
						|
         mScrollUpdates.LastElement().GetMode() == ScrollMode::SmoothMsd)) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
using AnimationState = nsIScrollableFrame::AnimationState;
 | 
						|
EnumSet<AnimationState> nsHTMLScrollFrame::ScrollAnimationState() const {
 | 
						|
  EnumSet<AnimationState> retval;
 | 
						|
  if (IsApzAnimationInProgress()) {
 | 
						|
    retval += AnimationState::APZInProgress;
 | 
						|
    if (mCurrentAPZScrollAnimationType ==
 | 
						|
        APZScrollAnimationType::TriggeredByScript) {
 | 
						|
      retval += AnimationState::TriggeredByScript;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (mApzAnimationRequested) {
 | 
						|
    retval += AnimationState::APZRequested;
 | 
						|
    if (mApzAnimationTriggeredByScriptRequested) {
 | 
						|
      retval += AnimationState::TriggeredByScript;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (IsLastScrollUpdateAnimating()) {
 | 
						|
    retval += AnimationState::APZPending;
 | 
						|
    if (IsLastScrollUpdateTriggeredByScriptAnimating()) {
 | 
						|
      retval += AnimationState::TriggeredByScript;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  if (mAsyncScroll) {
 | 
						|
    retval += AnimationState::MainThread;
 | 
						|
    if (mAsyncScroll->WasTriggeredByScript()) {
 | 
						|
      retval += AnimationState::TriggeredByScript;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (mAsyncSmoothMSDScroll) {
 | 
						|
    retval += AnimationState::MainThread;
 | 
						|
    if (mAsyncSmoothMSDScroll->WasTriggeredByScript()) {
 | 
						|
      retval += AnimationState::TriggeredByScript;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return retval;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ResetScrollInfoIfNeeded(
 | 
						|
    const MainThreadScrollGeneration& aGeneration,
 | 
						|
    const APZScrollGeneration& aGenerationOnApz,
 | 
						|
    APZScrollAnimationType aAPZScrollAnimationType,
 | 
						|
    InScrollingGesture aInScrollingGesture) {
 | 
						|
  if (aGeneration == mScrollGeneration) {
 | 
						|
    mLastScrollOrigin = ScrollOrigin::None;
 | 
						|
    mApzAnimationRequested = false;
 | 
						|
    mApzAnimationTriggeredByScriptRequested = false;
 | 
						|
  }
 | 
						|
 | 
						|
  mScrollGenerationOnApz = aGenerationOnApz;
 | 
						|
  // We can reset this regardless of scroll generation, as this is only set
 | 
						|
  // here, as a response to APZ requesting a repaint.
 | 
						|
  mCurrentAPZScrollAnimationType = aAPZScrollAnimationType;
 | 
						|
 | 
						|
  mInScrollingGesture = aInScrollingGesture;
 | 
						|
}
 | 
						|
 | 
						|
UniquePtr<PresState> nsHTMLScrollFrame::SaveState() {
 | 
						|
  nsIScrollbarMediator* mediator = do_QueryFrame(GetScrolledFrame());
 | 
						|
  if (mediator) {
 | 
						|
    // child handles its own scroll state, so don't bother saving state here
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  // Don't store a scroll state if we never have been scrolled or restored
 | 
						|
  // a previous scroll state, and we're not in the middle of a smooth scroll.
 | 
						|
  auto scrollAnimationState = ScrollAnimationState();
 | 
						|
  bool isScrollAnimating =
 | 
						|
      scrollAnimationState.contains(AnimationState::MainThread) ||
 | 
						|
      scrollAnimationState.contains(AnimationState::APZPending) ||
 | 
						|
      scrollAnimationState.contains(AnimationState::APZRequested);
 | 
						|
  if (!mHasBeenScrolled && !mDidHistoryRestore && !isScrollAnimating) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  UniquePtr<PresState> state = NewPresState();
 | 
						|
  bool allowScrollOriginDowngrade =
 | 
						|
      !nsLayoutUtils::CanScrollOriginClobberApz(mLastScrollOrigin) ||
 | 
						|
      mAllowScrollOriginDowngrade;
 | 
						|
  // Save mRestorePos instead of our actual current scroll position, if it's
 | 
						|
  // valid and we haven't moved since the last update of mLastPos (same check
 | 
						|
  // that ScrollToRestoredPosition uses). This ensures if a reframe occurs
 | 
						|
  // while we're in the process of loading content to scroll to a restored
 | 
						|
  // position, we'll keep trying after the reframe. Similarly, if we're in the
 | 
						|
  // middle of a smooth scroll, store the destination so that when we restore
 | 
						|
  // we'll jump straight to the end of the scroll animation, rather than
 | 
						|
  // effectively dropping it. Note that the mRestorePos will override the
 | 
						|
  // smooth scroll destination if both are present.
 | 
						|
  nsPoint pt = GetLogicalVisualViewportOffset();
 | 
						|
  if (isScrollAnimating) {
 | 
						|
    pt = mDestination;
 | 
						|
    allowScrollOriginDowngrade = false;
 | 
						|
  }
 | 
						|
  SCROLLRESTORE_LOG("%p: SaveState, pt=%s, mLastPos=%s, mRestorePos=%s\n", this,
 | 
						|
                    ToString(pt).c_str(), ToString(mLastPos).c_str(),
 | 
						|
                    ToString(mRestorePos).c_str());
 | 
						|
  if (mRestorePos.y != -1 && pt == mLastPos) {
 | 
						|
    pt = mRestorePos;
 | 
						|
  }
 | 
						|
  state->scrollState() = pt;
 | 
						|
  state->allowScrollOriginDowngrade() = allowScrollOriginDowngrade;
 | 
						|
  if (mIsRoot) {
 | 
						|
    // Only save resolution properties for root scroll frames
 | 
						|
    state->resolution() = PresShell()->GetResolution();
 | 
						|
  }
 | 
						|
  return state;
 | 
						|
}
 | 
						|
 | 
						|
NS_IMETHODIMP nsHTMLScrollFrame::RestoreState(PresState* aState) {
 | 
						|
  mRestorePos = aState->scrollState();
 | 
						|
  MOZ_ASSERT(mLastScrollOrigin == ScrollOrigin::None);
 | 
						|
  mAllowScrollOriginDowngrade = aState->allowScrollOriginDowngrade();
 | 
						|
  // When restoring state, we promote mLastScrollOrigin to a stronger value
 | 
						|
  // from the default of eNone, to restore the behaviour that existed when
 | 
						|
  // the state was saved. If mLastScrollOrigin was a weaker value previously,
 | 
						|
  // then mAllowScrollOriginDowngrade will be true, and so the combination of
 | 
						|
  // mAllowScrollOriginDowngrade and the stronger mLastScrollOrigin will allow
 | 
						|
  // the same types of scrolls as before. It might be possible to also just
 | 
						|
  // save and restore the mAllowScrollOriginDowngrade and mLastScrollOrigin
 | 
						|
  // values directly without this sort of fiddling. Something to try in the
 | 
						|
  // future or if we tinker with this code more.
 | 
						|
  mLastScrollOrigin = ScrollOrigin::Other;
 | 
						|
  mDidHistoryRestore = true;
 | 
						|
  mLastPos = mScrolledFrame ? GetLogicalVisualViewportOffset() : nsPoint(0, 0);
 | 
						|
  SCROLLRESTORE_LOG("%p: RestoreState, set mRestorePos=%s mLastPos=%s\n", this,
 | 
						|
                    ToString(mRestorePos).c_str(), ToString(mLastPos).c_str());
 | 
						|
 | 
						|
  // Resolution properties should only exist on root scroll frames.
 | 
						|
  MOZ_ASSERT(mIsRoot || aState->resolution() == 1.0);
 | 
						|
 | 
						|
  if (mIsRoot) {
 | 
						|
    PresShell()->SetResolutionAndScaleTo(
 | 
						|
        aState->resolution(), ResolutionChangeOrigin::MainThreadRestore);
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostScrolledAreaEvent() {
 | 
						|
  if (mScrolledAreaEvent.IsPending()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  mScrolledAreaEvent = new ScrolledAreaEvent(this);
 | 
						|
  nsContentUtils::AddScriptRunner(mScrolledAreaEvent.get());
 | 
						|
}
 | 
						|
 | 
						|
////////////////////////////////////////////////////////////////////////////////
 | 
						|
// ScrolledArea change event dispatch
 | 
						|
 | 
						|
// TODO: Convert this to MOZ_CAN_RUN_SCRIPT (bug 1415230, bug 1535398)
 | 
						|
MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP
 | 
						|
nsHTMLScrollFrame::ScrolledAreaEvent::Run() {
 | 
						|
  if (mHelper) {
 | 
						|
    mHelper->FireScrolledAreaEvent();
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::FireScrolledAreaEvent() {
 | 
						|
  mScrolledAreaEvent.Forget();
 | 
						|
 | 
						|
  InternalScrollAreaEvent event(true, eScrolledAreaChanged, nullptr);
 | 
						|
  RefPtr<nsPresContext> presContext = PresContext();
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
 | 
						|
  event.mArea = mScrolledFrame->ScrollableOverflowRectRelativeToParent();
 | 
						|
  if (RefPtr<Document> doc = content->GetUncomposedDoc()) {
 | 
						|
    EventDispatcher::Dispatch(doc, presContext, &event, nullptr);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
ScrollDirections nsIScrollableFrame::GetAvailableScrollingDirections() const {
 | 
						|
  nscoord oneDevPixel =
 | 
						|
      GetScrolledFrame()->PresContext()->AppUnitsPerDevPixel();
 | 
						|
  ScrollDirections directions;
 | 
						|
  nsRect scrollRange = GetScrollRange();
 | 
						|
  if (scrollRange.width >= oneDevPixel) {
 | 
						|
    directions += ScrollDirection::eHorizontal;
 | 
						|
  }
 | 
						|
  if (scrollRange.height >= oneDevPixel) {
 | 
						|
    directions += ScrollDirection::eVertical;
 | 
						|
  }
 | 
						|
  return directions;
 | 
						|
}
 | 
						|
 | 
						|
nsRect nsHTMLScrollFrame::GetScrollRangeForUserInputEvents() const {
 | 
						|
  // This function computes a scroll range based on a scrolled rect and scroll
 | 
						|
  // port defined as follows:
 | 
						|
  //   scrollable rect = overflow:hidden ? layout viewport : scrollable rect
 | 
						|
  //   scroll port = have visual viewport ? visual viewport : layout viewport
 | 
						|
  // The results in the same notion of scroll range that APZ uses (the combined
 | 
						|
  // effect of FrameMetrics::CalculateScrollRange() and
 | 
						|
  // nsLayoutUtils::CalculateScrollableRectForFrame).
 | 
						|
 | 
						|
  ScrollStyles ss = GetScrollStyles();
 | 
						|
 | 
						|
  nsPoint scrollPos = GetScrollPosition();
 | 
						|
 | 
						|
  nsRect scrolledRect = GetScrolledRect();
 | 
						|
  if (StyleOverflow::Hidden == ss.mHorizontal) {
 | 
						|
    scrolledRect.width = mScrollPort.width;
 | 
						|
    scrolledRect.x = scrollPos.x;
 | 
						|
  }
 | 
						|
  if (StyleOverflow::Hidden == ss.mVertical) {
 | 
						|
    scrolledRect.height = mScrollPort.height;
 | 
						|
    scrolledRect.y = scrollPos.y;
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize scrollPort = GetVisualViewportSize();
 | 
						|
 | 
						|
  nsRect scrollRange = scrolledRect;
 | 
						|
  scrollRange.width = std::max(scrolledRect.width - scrollPort.width, 0);
 | 
						|
  scrollRange.height = std::max(scrolledRect.height - scrollPort.height, 0);
 | 
						|
 | 
						|
  return scrollRange;
 | 
						|
}
 | 
						|
 | 
						|
ScrollDirections
 | 
						|
nsHTMLScrollFrame::GetAvailableScrollingDirectionsForUserInputEvents() const {
 | 
						|
  nsRect scrollRange = GetScrollRangeForUserInputEvents();
 | 
						|
 | 
						|
  // We check if there is at least one half of a screen pixel of scroll range to
 | 
						|
  // roughly match what apz does when it checks if the change in scroll position
 | 
						|
  // in screen pixels round to zero or not.
 | 
						|
  // (https://searchfox.org/mozilla-central/rev/2f09184ec781a2667feec87499d4b81b32b6c48e/gfx/layers/apz/src/AsyncPanZoomController.cpp#3210)
 | 
						|
  // This isn't quite half a screen pixel, it doesn't take into account CSS
 | 
						|
  // transforms, but should be good enough.
 | 
						|
  float halfScreenPixel =
 | 
						|
      GetScrolledFrame()->PresContext()->AppUnitsPerDevPixel() /
 | 
						|
      (PresShell()->GetCumulativeResolution() * 2.f);
 | 
						|
  ScrollDirections directions;
 | 
						|
  if (scrollRange.width >= halfScreenPixel) {
 | 
						|
    directions += ScrollDirection::eHorizontal;
 | 
						|
  }
 | 
						|
  if (scrollRange.height >= halfScreenPixel) {
 | 
						|
    directions += ScrollDirection::eVertical;
 | 
						|
  }
 | 
						|
  return directions;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Append scroll positions for valid snap positions into |aSnapInfo| if
 | 
						|
 * applicable.
 | 
						|
 */
 | 
						|
static void AppendScrollPositionsForSnap(
 | 
						|
    const nsIFrame* aFrame, const nsIFrame* aScrolledFrame,
 | 
						|
    const nsRect& aScrolledRect, const nsMargin& aScrollPadding,
 | 
						|
    const nsRect& aScrollRange, WritingMode aWritingModeOnScroller,
 | 
						|
    ScrollSnapInfo& aSnapInfo, nsHTMLScrollFrame::SnapTargetSet* aSnapTargets) {
 | 
						|
  ScrollSnapTargetId targetId = ScrollSnapUtils::GetTargetIdFor(aFrame);
 | 
						|
 | 
						|
  nsRect snapArea =
 | 
						|
      ScrollSnapUtils::GetSnapAreaFor(aFrame, aScrolledFrame, aScrolledRect);
 | 
						|
  // Use the writing-mode on the target element if the snap area is larger than
 | 
						|
  // the snapport.
 | 
						|
  // https://drafts.csswg.org/css-scroll-snap/#snap-scope
 | 
						|
  WritingMode writingMode = ScrollSnapUtils::NeedsToRespectTargetWritingMode(
 | 
						|
                                snapArea.Size(), aSnapInfo.mSnapportSize)
 | 
						|
                                ? aFrame->GetWritingMode()
 | 
						|
                                : aWritingModeOnScroller;
 | 
						|
 | 
						|
  // These snap range shouldn't be involved with scroll-margin since we just
 | 
						|
  // need the visible range of the target element.
 | 
						|
  if (snapArea.width > aSnapInfo.mSnapportSize.width) {
 | 
						|
    aSnapInfo.mXRangeWiderThanSnapport.AppendElement(
 | 
						|
        ScrollSnapInfo::ScrollSnapRange(snapArea, ScrollDirection::eHorizontal,
 | 
						|
                                        targetId));
 | 
						|
  }
 | 
						|
  if (snapArea.height > aSnapInfo.mSnapportSize.height) {
 | 
						|
    aSnapInfo.mYRangeWiderThanSnapport.AppendElement(
 | 
						|
        ScrollSnapInfo::ScrollSnapRange(snapArea, ScrollDirection::eVertical,
 | 
						|
                                        targetId));
 | 
						|
  }
 | 
						|
 | 
						|
  // Shift target rect position by the scroll padding to get the padded
 | 
						|
  // position thus we don't need to take account scroll-padding values in
 | 
						|
  // ScrollSnapUtils::GetSnapPointForDestination() when it gets called from
 | 
						|
  // the compositor thread.
 | 
						|
  snapArea.y -= aScrollPadding.top;
 | 
						|
  snapArea.x -= aScrollPadding.left;
 | 
						|
 | 
						|
  LogicalRect logicalTargetRect(writingMode, snapArea, aSnapInfo.mSnapportSize);
 | 
						|
  LogicalSize logicalSnapportRect(writingMode, aSnapInfo.mSnapportSize);
 | 
						|
  LogicalRect logicalScrollRange(aWritingModeOnScroller, aScrollRange,
 | 
						|
                                 // The origin of this logical coordinate system
 | 
						|
                                 // what we need here is (0, 0), so we use an
 | 
						|
                                 // empty size.
 | 
						|
                                 nsSize());
 | 
						|
 | 
						|
  Maybe<nscoord> blockDirectionPosition;
 | 
						|
  const nsStyleDisplay* styleDisplay = aFrame->StyleDisplay();
 | 
						|
  nscoord containerBSize = logicalSnapportRect.BSize(writingMode);
 | 
						|
  switch (styleDisplay->mScrollSnapAlign.block) {
 | 
						|
    case StyleScrollSnapAlignKeyword::None:
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAlignKeyword::Start:
 | 
						|
      blockDirectionPosition.emplace(
 | 
						|
          writingMode.IsVerticalRL() ? -logicalTargetRect.BStart(writingMode)
 | 
						|
                                     : logicalTargetRect.BStart(writingMode));
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAlignKeyword::End: {
 | 
						|
      nscoord candidate = std::clamp(
 | 
						|
          // What we need here is the scroll position instead of the snap
 | 
						|
          // position itself, so we need, for example, the top edge of the
 | 
						|
          // scroll port on horizontal-tb when the frame is positioned at
 | 
						|
          // the bottom edge of the scroll port. For this reason we subtract
 | 
						|
          // containerBSize from BEnd of the target and clamp it inside the
 | 
						|
          // scrollable range.
 | 
						|
          logicalTargetRect.BEnd(writingMode) - containerBSize,
 | 
						|
          logicalScrollRange.BStart(writingMode),
 | 
						|
          logicalScrollRange.BEnd(writingMode));
 | 
						|
      blockDirectionPosition.emplace(writingMode.IsVerticalRL() ? -candidate
 | 
						|
                                                                : candidate);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    case StyleScrollSnapAlignKeyword::Center: {
 | 
						|
      nscoord targetCenter = (logicalTargetRect.BStart(writingMode) +
 | 
						|
                              logicalTargetRect.BEnd(writingMode)) /
 | 
						|
                             2;
 | 
						|
      nscoord halfSnapportSize = containerBSize / 2;
 | 
						|
      // Get the center of the target to align with the center of the snapport
 | 
						|
      // depending on direction.
 | 
						|
      nscoord candidate = std::clamp(targetCenter - halfSnapportSize,
 | 
						|
                                     logicalScrollRange.BStart(writingMode),
 | 
						|
                                     logicalScrollRange.BEnd(writingMode));
 | 
						|
      blockDirectionPosition.emplace(writingMode.IsVerticalRL() ? -candidate
 | 
						|
                                                                : candidate);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  Maybe<nscoord> inlineDirectionPosition;
 | 
						|
  nscoord containerISize = logicalSnapportRect.ISize(writingMode);
 | 
						|
  switch (styleDisplay->mScrollSnapAlign.inline_) {
 | 
						|
    case StyleScrollSnapAlignKeyword::None:
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAlignKeyword::Start:
 | 
						|
      inlineDirectionPosition.emplace(
 | 
						|
          writingMode.IsInlineReversed()
 | 
						|
              ? -logicalTargetRect.IStart(writingMode)
 | 
						|
              : logicalTargetRect.IStart(writingMode));
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAlignKeyword::End: {
 | 
						|
      nscoord candidate = std::clamp(
 | 
						|
          // Same as above BEnd case, we subtract containerISize.
 | 
						|
          //
 | 
						|
          // Note that the logical scroll range is mapped to [0, x] range even
 | 
						|
          // if it's in RTL contents. So for example, if the physical range is
 | 
						|
          // [-200, 0], it's mapped to [0, 200], i.e. IStart() is 0, IEnd() is
 | 
						|
          // 200. So we can just use std::clamp with the same arguments in both
 | 
						|
          // RTL/LTR cases.
 | 
						|
          logicalTargetRect.IEnd(writingMode) - containerISize,
 | 
						|
          logicalScrollRange.IStart(writingMode),
 | 
						|
          logicalScrollRange.IEnd(writingMode));
 | 
						|
      inlineDirectionPosition.emplace(
 | 
						|
          writingMode.IsInlineReversed() ? -candidate : candidate);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    case StyleScrollSnapAlignKeyword::Center: {
 | 
						|
      nscoord targetCenter = (logicalTargetRect.IStart(writingMode) +
 | 
						|
                              logicalTargetRect.IEnd(writingMode)) /
 | 
						|
                             2;
 | 
						|
      nscoord halfSnapportSize = containerISize / 2;
 | 
						|
      // Get the center of the target to align with the center of the snapport
 | 
						|
      // depending on direction.
 | 
						|
      nscoord candidate = std::clamp(targetCenter - halfSnapportSize,
 | 
						|
                                     logicalScrollRange.IStart(writingMode),
 | 
						|
                                     logicalScrollRange.IEnd(writingMode));
 | 
						|
      inlineDirectionPosition.emplace(
 | 
						|
          writingMode.IsInlineReversed() ? -candidate : candidate);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (blockDirectionPosition || inlineDirectionPosition) {
 | 
						|
    aSnapInfo.mSnapTargets.AppendElement(
 | 
						|
        writingMode.IsVertical()
 | 
						|
            ? ScrollSnapInfo::SnapTarget(
 | 
						|
                  std::move(blockDirectionPosition),
 | 
						|
                  std::move(inlineDirectionPosition), std::move(snapArea),
 | 
						|
                  styleDisplay->mScrollSnapStop, targetId)
 | 
						|
            : ScrollSnapInfo::SnapTarget(
 | 
						|
                  std::move(inlineDirectionPosition),
 | 
						|
                  std::move(blockDirectionPosition), std::move(snapArea),
 | 
						|
                  styleDisplay->mScrollSnapStop, targetId));
 | 
						|
    if (aSnapTargets) {
 | 
						|
      aSnapTargets->EnsureInserted(aFrame->GetContent());
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Collect the scroll positions corresponding to snap positions of frames in the
 | 
						|
 * subtree rooted at |aFrame|, relative to |aScrolledFrame|, into |aSnapInfo|.
 | 
						|
 */
 | 
						|
static void CollectScrollPositionsForSnap(
 | 
						|
    nsIFrame* aFrame, nsIFrame* aScrolledFrame, const nsRect& aScrolledRect,
 | 
						|
    const nsMargin& aScrollPadding, const nsRect& aScrollRange,
 | 
						|
    WritingMode aWritingModeOnScroller, ScrollSnapInfo& aSnapInfo,
 | 
						|
    nsHTMLScrollFrame::SnapTargetSet* aSnapTargets) {
 | 
						|
  // Snap positions only affect the nearest ancestor scroll container on the
 | 
						|
  // element's containing block chain.
 | 
						|
  nsIScrollableFrame* sf = do_QueryFrame(aFrame);
 | 
						|
  if (sf) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  for (const auto& childList : aFrame->ChildLists()) {
 | 
						|
    for (nsIFrame* f : childList.mList) {
 | 
						|
      const nsStyleDisplay* styleDisplay = f->StyleDisplay();
 | 
						|
      if (styleDisplay->mScrollSnapAlign.inline_ !=
 | 
						|
              StyleScrollSnapAlignKeyword::None ||
 | 
						|
          styleDisplay->mScrollSnapAlign.block !=
 | 
						|
              StyleScrollSnapAlignKeyword::None) {
 | 
						|
        AppendScrollPositionsForSnap(
 | 
						|
            f, aScrolledFrame, aScrolledRect, aScrollPadding, aScrollRange,
 | 
						|
            aWritingModeOnScroller, aSnapInfo, aSnapTargets);
 | 
						|
      }
 | 
						|
      CollectScrollPositionsForSnap(
 | 
						|
          f, aScrolledFrame, aScrolledRect, aScrollPadding, aScrollRange,
 | 
						|
          aWritingModeOnScroller, aSnapInfo, aSnapTargets);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static nscoord ResolveScrollPaddingStyleValue(
 | 
						|
    const StyleRect<mozilla::NonNegativeLengthPercentageOrAuto>&
 | 
						|
        aScrollPaddingStyle,
 | 
						|
    Side aSide, const nsSize& aScrollPortSize) {
 | 
						|
  if (aScrollPaddingStyle.Get(aSide).IsAuto()) {
 | 
						|
    // https://drafts.csswg.org/css-scroll-snap-1/#valdef-scroll-padding-auto
 | 
						|
    return 0;
 | 
						|
  }
 | 
						|
 | 
						|
  nscoord percentageBasis;
 | 
						|
  switch (aSide) {
 | 
						|
    case eSideTop:
 | 
						|
    case eSideBottom:
 | 
						|
      percentageBasis = aScrollPortSize.height;
 | 
						|
      break;
 | 
						|
    case eSideLeft:
 | 
						|
    case eSideRight:
 | 
						|
      percentageBasis = aScrollPortSize.width;
 | 
						|
      break;
 | 
						|
  }
 | 
						|
 | 
						|
  return aScrollPaddingStyle.Get(aSide).AsLengthPercentage().Resolve(
 | 
						|
      percentageBasis);
 | 
						|
}
 | 
						|
 | 
						|
static nsMargin ResolveScrollPaddingStyle(
 | 
						|
    const StyleRect<mozilla::NonNegativeLengthPercentageOrAuto>&
 | 
						|
        aScrollPaddingStyle,
 | 
						|
    const nsSize& aScrollPortSize) {
 | 
						|
  return nsMargin(ResolveScrollPaddingStyleValue(aScrollPaddingStyle, eSideTop,
 | 
						|
                                                 aScrollPortSize),
 | 
						|
                  ResolveScrollPaddingStyleValue(aScrollPaddingStyle,
 | 
						|
                                                 eSideRight, aScrollPortSize),
 | 
						|
                  ResolveScrollPaddingStyleValue(aScrollPaddingStyle,
 | 
						|
                                                 eSideBottom, aScrollPortSize),
 | 
						|
                  ResolveScrollPaddingStyleValue(aScrollPaddingStyle, eSideLeft,
 | 
						|
                                                 aScrollPortSize));
 | 
						|
}
 | 
						|
 | 
						|
nsMargin nsHTMLScrollFrame::GetScrollPadding() const {
 | 
						|
  nsIFrame* styleFrame = GetFrameForStyle();
 | 
						|
  if (!styleFrame) {
 | 
						|
    return nsMargin();
 | 
						|
  }
 | 
						|
 | 
						|
  // The spec says percentage values are relative to the scroll port size.
 | 
						|
  // https://drafts.csswg.org/css-scroll-snap-1/#scroll-padding
 | 
						|
  return ResolveScrollPaddingStyle(styleFrame->StylePadding()->mScrollPadding,
 | 
						|
                                   GetScrollPortRect().Size());
 | 
						|
}
 | 
						|
 | 
						|
ScrollSnapInfo nsHTMLScrollFrame::ComputeScrollSnapInfo() {
 | 
						|
  ScrollSnapInfo result;
 | 
						|
 | 
						|
  nsIFrame* scrollSnapFrame = GetFrameForStyle();
 | 
						|
  if (!scrollSnapFrame) {
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  const nsStyleDisplay* disp = scrollSnapFrame->StyleDisplay();
 | 
						|
  if (disp->mScrollSnapType.strictness == StyleScrollSnapStrictness::None) {
 | 
						|
    // We won't be snapping, short-circuit the computation.
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  WritingMode writingMode = GetWritingMode();
 | 
						|
  result.InitializeScrollSnapStrictness(writingMode, disp);
 | 
						|
 | 
						|
  result.mSnapportSize = GetSnapportSize();
 | 
						|
  CollectScrollPositionsForSnap(
 | 
						|
      mScrolledFrame, mScrolledFrame, GetScrolledRect(), GetScrollPadding(),
 | 
						|
      GetLayoutScrollRange(), writingMode, result, &mSnapTargets);
 | 
						|
  return result;
 | 
						|
}
 | 
						|
 | 
						|
ScrollSnapInfo nsHTMLScrollFrame::GetScrollSnapInfo() {
 | 
						|
  // TODO(botond): Should we cache it?
 | 
						|
  return ComputeScrollSnapInfo();
 | 
						|
}
 | 
						|
 | 
						|
Maybe<SnapDestination> nsHTMLScrollFrame::GetSnapPointForDestination(
 | 
						|
    ScrollUnit aUnit, ScrollSnapFlags aFlags, const nsPoint& aStartPos,
 | 
						|
    const nsPoint& aDestination) {
 | 
						|
  // We can release the strong references for the previous snap target
 | 
						|
  // elements here since calling this ComputeScrollSnapInfo means we are going
 | 
						|
  // to evaluate new snap points, thus there's no chance to generating
 | 
						|
  // nsIContent instances in between this function call and the function call
 | 
						|
  // for the (re-)evaluation.
 | 
						|
  mSnapTargets.Clear();
 | 
						|
  return ScrollSnapUtils::GetSnapPointForDestination(
 | 
						|
      ComputeScrollSnapInfo(), aUnit, aFlags, GetLayoutScrollRange(), aStartPos,
 | 
						|
      aDestination);
 | 
						|
}
 | 
						|
 | 
						|
Maybe<SnapDestination> nsHTMLScrollFrame::GetSnapPointForResnap() {
 | 
						|
  // Same as in GetSnapPointForDestination, We can release the strong references
 | 
						|
  // for the previous snap targets here.
 | 
						|
  mSnapTargets.Clear();
 | 
						|
  nsIContent* focusedContent =
 | 
						|
      GetContent()->GetComposedDoc()->GetUnretargetedFocusedContent();
 | 
						|
  return ScrollSnapUtils::GetSnapPointForResnap(
 | 
						|
      ComputeScrollSnapInfo(), GetLayoutScrollRange(), GetScrollPosition(),
 | 
						|
      mLastSnapTargetIds, focusedContent);
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::NeedsResnap() {
 | 
						|
  nsIContent* focusedContent =
 | 
						|
      GetContent()->GetComposedDoc()->GetUnretargetedFocusedContent();
 | 
						|
  return ScrollSnapUtils::GetSnapPointForResnap(
 | 
						|
             ComputeScrollSnapInfo(), GetLayoutScrollRange(),
 | 
						|
             GetScrollPosition(), mLastSnapTargetIds, focusedContent)
 | 
						|
      .isSome();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::SetLastSnapTargetIds(
 | 
						|
    UniquePtr<ScrollSnapTargetIds> aIds) {
 | 
						|
  if (!aIds) {
 | 
						|
    mLastSnapTargetIds = nullptr;
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // This SetLastSnapTargetIds gets called asynchronously so that there's a race
 | 
						|
  // condition something like;
 | 
						|
  // 1) an async scroll operation triggered snapping to a point on an element
 | 
						|
  // 2) during the async scroll operation, the element got removed from this
 | 
						|
  //    scroll container
 | 
						|
  // 3) re-snapping triggered
 | 
						|
  // 4) this SetLastSnapTargetIds got called
 | 
						|
  // In such case |aIds| are stale, we shouldn't use it.
 | 
						|
  for (const auto* idList : {&aIds->mIdsOnX, &aIds->mIdsOnY}) {
 | 
						|
    for (const auto id : *idList) {
 | 
						|
      if (!mSnapTargets.Contains(reinterpret_cast<nsIContent*>(id))) {
 | 
						|
        mLastSnapTargetIds = nullptr;
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mLastSnapTargetIds = std::move(aIds);
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsLastSnappedTarget(const nsIFrame* aFrame) const {
 | 
						|
  ScrollSnapTargetId id = ScrollSnapUtils::GetTargetIdFor(aFrame);
 | 
						|
  MOZ_ASSERT(id != ScrollSnapTargetId::None,
 | 
						|
             "This function is supposed to be called for contents");
 | 
						|
 | 
						|
  if (!mLastSnapTargetIds) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  return mLastSnapTargetIds->mIdsOnX.Contains(id) ||
 | 
						|
         mLastSnapTargetIds->mIdsOnY.Contains(id);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::TryResnap() {
 | 
						|
  // If there's any async scroll is running or we are processing pan/touch
 | 
						|
  // gestures or scroll thumb dragging,  don't clobber the scroll.
 | 
						|
  if (!ScrollAnimationState().isEmpty() ||
 | 
						|
      mInScrollingGesture == InScrollingGesture::Yes) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (auto snapDestination = GetSnapPointForResnap()) {
 | 
						|
    // We are going to re-snap so that we need to clobber scroll anchoring.
 | 
						|
    mAnchor.UserScrolled();
 | 
						|
 | 
						|
    // Snap to the nearest snap position if exists.
 | 
						|
    ScrollToWithOrigin(
 | 
						|
        snapDestination->mPosition, nullptr /* range */,
 | 
						|
        ScrollOperationParams{
 | 
						|
            IsSmoothScroll(ScrollBehavior::Auto) ? ScrollMode::SmoothMsd
 | 
						|
                                                 : ScrollMode::Instant,
 | 
						|
            ScrollOrigin::Other, std::move(snapDestination->mTargetIds)});
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostPendingResnapIfNeeded(const nsIFrame* aFrame) {
 | 
						|
  if (!IsLastSnappedTarget(aFrame)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  PostPendingResnap();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::PostPendingResnap() {
 | 
						|
  PresShell()->PostPendingScrollResnap(this);
 | 
						|
}
 | 
						|
 | 
						|
nsIScrollableFrame::PhysicalScrollSnapAlign
 | 
						|
nsHTMLScrollFrame::GetScrollSnapAlignFor(const nsIFrame* aFrame) const {
 | 
						|
  StyleScrollSnapAlignKeyword alignForY = StyleScrollSnapAlignKeyword::None;
 | 
						|
  StyleScrollSnapAlignKeyword alignForX = StyleScrollSnapAlignKeyword::None;
 | 
						|
 | 
						|
  nsIFrame* styleFrame = GetFrameForStyle();
 | 
						|
  if (!styleFrame) {
 | 
						|
    return {alignForX, alignForY};
 | 
						|
  }
 | 
						|
 | 
						|
  if (styleFrame->StyleDisplay()->mScrollSnapType.strictness ==
 | 
						|
      StyleScrollSnapStrictness::None) {
 | 
						|
    return {alignForX, alignForY};
 | 
						|
  }
 | 
						|
 | 
						|
  const nsStyleDisplay* styleDisplay = aFrame->StyleDisplay();
 | 
						|
  if (styleDisplay->mScrollSnapAlign.inline_ ==
 | 
						|
          StyleScrollSnapAlignKeyword::None &&
 | 
						|
      styleDisplay->mScrollSnapAlign.block ==
 | 
						|
          StyleScrollSnapAlignKeyword::None) {
 | 
						|
    return {alignForX, alignForY};
 | 
						|
  }
 | 
						|
 | 
						|
  nsSize snapAreaSize =
 | 
						|
      ScrollSnapUtils::GetSnapAreaFor(aFrame, mScrolledFrame, GetScrolledRect())
 | 
						|
          .Size();
 | 
						|
  const WritingMode writingMode =
 | 
						|
      ScrollSnapUtils::NeedsToRespectTargetWritingMode(snapAreaSize,
 | 
						|
                                                       GetSnapportSize())
 | 
						|
          ? aFrame->GetWritingMode()
 | 
						|
          : styleFrame->GetWritingMode();
 | 
						|
 | 
						|
  switch (styleFrame->StyleDisplay()->mScrollSnapType.axis) {
 | 
						|
    case StyleScrollSnapAxis::X:
 | 
						|
      alignForX = writingMode.IsVertical()
 | 
						|
                      ? styleDisplay->mScrollSnapAlign.block
 | 
						|
                      : styleDisplay->mScrollSnapAlign.inline_;
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAxis::Y:
 | 
						|
      alignForY = writingMode.IsVertical()
 | 
						|
                      ? styleDisplay->mScrollSnapAlign.inline_
 | 
						|
                      : styleDisplay->mScrollSnapAlign.block;
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAxis::Block:
 | 
						|
      if (writingMode.IsVertical()) {
 | 
						|
        alignForX = styleDisplay->mScrollSnapAlign.block;
 | 
						|
      } else {
 | 
						|
        alignForY = styleDisplay->mScrollSnapAlign.block;
 | 
						|
      }
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAxis::Inline:
 | 
						|
      if (writingMode.IsVertical()) {
 | 
						|
        alignForY = styleDisplay->mScrollSnapAlign.inline_;
 | 
						|
      } else {
 | 
						|
        alignForX = styleDisplay->mScrollSnapAlign.inline_;
 | 
						|
      }
 | 
						|
      break;
 | 
						|
    case StyleScrollSnapAxis::Both:
 | 
						|
      if (writingMode.IsVertical()) {
 | 
						|
        alignForX = styleDisplay->mScrollSnapAlign.block;
 | 
						|
        alignForY = styleDisplay->mScrollSnapAlign.inline_;
 | 
						|
      } else {
 | 
						|
        alignForX = styleDisplay->mScrollSnapAlign.inline_;
 | 
						|
        alignForY = styleDisplay->mScrollSnapAlign.block;
 | 
						|
      }
 | 
						|
      break;
 | 
						|
  }
 | 
						|
 | 
						|
  return {alignForX, alignForY};
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::UsesOverlayScrollbars() const {
 | 
						|
  return PresContext()->UseOverlayScrollbars();
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::DragScroll(WidgetEvent* aEvent) {
 | 
						|
  // Dragging is allowed while within a 20 pixel border. Note that device pixels
 | 
						|
  // are used so that the same margin is used even when zoomed in or out.
 | 
						|
  nscoord margin = 20 * PresContext()->AppUnitsPerDevPixel();
 | 
						|
 | 
						|
  // Don't drag scroll for small scrollareas.
 | 
						|
  if (mScrollPort.width < margin * 2 || mScrollPort.height < margin * 2) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // If willScroll is computed as false, then the frame is already scrolled as
 | 
						|
  // far as it can go in both directions. Return false so that an ancestor
 | 
						|
  // scrollframe can scroll instead.
 | 
						|
  bool willScroll = false;
 | 
						|
  nsPoint pnt =
 | 
						|
      nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, RelativeTo{this});
 | 
						|
  nsPoint scrollPoint = GetScrollPosition();
 | 
						|
  nsRect rangeRect = GetLayoutScrollRange();
 | 
						|
 | 
						|
  // Only drag scroll when a scrollbar is present.
 | 
						|
  nsPoint offset;
 | 
						|
  if (mHasHorizontalScrollbar) {
 | 
						|
    if (pnt.x >= mScrollPort.x && pnt.x <= mScrollPort.x + margin) {
 | 
						|
      offset.x = -margin;
 | 
						|
      if (scrollPoint.x > 0) {
 | 
						|
        willScroll = true;
 | 
						|
      }
 | 
						|
    } else if (pnt.x >= mScrollPort.XMost() - margin &&
 | 
						|
               pnt.x <= mScrollPort.XMost()) {
 | 
						|
      offset.x = margin;
 | 
						|
      if (scrollPoint.x < rangeRect.width) {
 | 
						|
        willScroll = true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (mHasVerticalScrollbar) {
 | 
						|
    if (pnt.y >= mScrollPort.y && pnt.y <= mScrollPort.y + margin) {
 | 
						|
      offset.y = -margin;
 | 
						|
      if (scrollPoint.y > 0) {
 | 
						|
        willScroll = true;
 | 
						|
      }
 | 
						|
    } else if (pnt.y >= mScrollPort.YMost() - margin &&
 | 
						|
               pnt.y <= mScrollPort.YMost()) {
 | 
						|
      offset.y = margin;
 | 
						|
      if (scrollPoint.y < rangeRect.height) {
 | 
						|
        willScroll = true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (offset.x || offset.y) {
 | 
						|
    ScrollToWithOrigin(
 | 
						|
        GetScrollPosition() + offset, nullptr /* range */,
 | 
						|
        ScrollOperationParams{ScrollMode::Normal, ScrollOrigin::Other});
 | 
						|
  }
 | 
						|
 | 
						|
  return willScroll;
 | 
						|
}
 | 
						|
 | 
						|
static nsSliderFrame* GetSliderFrame(nsIFrame* aScrollbarFrame) {
 | 
						|
  if (!aScrollbarFrame) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  for (const auto& childList : aScrollbarFrame->ChildLists()) {
 | 
						|
    for (nsIFrame* frame : childList.mList) {
 | 
						|
      if (nsSliderFrame* sliderFrame = do_QueryFrame(frame)) {
 | 
						|
        return sliderFrame;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
static void AsyncScrollbarDragInitiated(uint64_t aDragBlockId,
 | 
						|
                                        nsIFrame* aScrollbar) {
 | 
						|
  if (nsSliderFrame* sliderFrame = GetSliderFrame(aScrollbar)) {
 | 
						|
    sliderFrame->AsyncScrollbarDragInitiated(aDragBlockId);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AsyncScrollbarDragInitiated(
 | 
						|
    uint64_t aDragBlockId, ScrollDirection aDirection) {
 | 
						|
  switch (aDirection) {
 | 
						|
    case ScrollDirection::eVertical:
 | 
						|
      ::AsyncScrollbarDragInitiated(aDragBlockId, mVScrollbarBox);
 | 
						|
      break;
 | 
						|
    case ScrollDirection::eHorizontal:
 | 
						|
      ::AsyncScrollbarDragInitiated(aDragBlockId, mHScrollbarBox);
 | 
						|
      break;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static void AsyncScrollbarDragRejected(nsIFrame* aScrollbar) {
 | 
						|
  if (nsSliderFrame* sliderFrame = GetSliderFrame(aScrollbar)) {
 | 
						|
    sliderFrame->AsyncScrollbarDragRejected();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AsyncScrollbarDragRejected() {
 | 
						|
  // We don't get told which scrollbar requested the async drag,
 | 
						|
  // so we notify both.
 | 
						|
  ::AsyncScrollbarDragRejected(mHScrollbarBox);
 | 
						|
  ::AsyncScrollbarDragRejected(mVScrollbarBox);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ApzSmoothScrollTo(
 | 
						|
    const nsPoint& aDestination, ScrollMode aMode, ScrollOrigin aOrigin,
 | 
						|
    ScrollTriggeredByScript aTriggeredByScript,
 | 
						|
    UniquePtr<ScrollSnapTargetIds> aSnapTargetIds) {
 | 
						|
  if (mApzSmoothScrollDestination == Some(aDestination)) {
 | 
						|
    // If we already sent APZ a smooth-scroll request to this
 | 
						|
    // destination (i.e. it was the last request
 | 
						|
    // we sent), then don't send another one because it is redundant.
 | 
						|
    // This is to avoid a scenario where pages do repeated scrollBy
 | 
						|
    // calls, incrementing the generation counter, and blocking APZ from
 | 
						|
    // syncing the scroll offset back to the main thread.
 | 
						|
    // Note that if we get two smooth-scroll requests to the same
 | 
						|
    // destination with some other scroll in between,
 | 
						|
    // mApzSmoothScrollDestination will get reset to Nothing() and so
 | 
						|
    // we shouldn't have the problem where this check discards a
 | 
						|
    // legitimate smooth-scroll.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // The animation will be handled in the compositor, pass the
 | 
						|
  // information needed to start the animation and skip the main-thread
 | 
						|
  // animation for this scroll.
 | 
						|
  MOZ_ASSERT(aOrigin != ScrollOrigin::None);
 | 
						|
  mApzSmoothScrollDestination = Some(aDestination);
 | 
						|
  AppendScrollUpdate(ScrollPositionUpdate::NewSmoothScroll(
 | 
						|
      aMode, aOrigin, aDestination, aTriggeredByScript,
 | 
						|
      std::move(aSnapTargetIds)));
 | 
						|
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
  if (!DisplayPortUtils::HasNonMinimalNonZeroDisplayPort(content)) {
 | 
						|
    // If this frame doesn't have a displayport then there won't be an
 | 
						|
    // APZC instance for it and so there won't be anything to process
 | 
						|
    // this smooth scroll request. We should set a displayport on this
 | 
						|
    // frame to force an APZC which can handle the request.
 | 
						|
    if (MOZ_LOG_TEST(sDisplayportLog, LogLevel::Debug)) {
 | 
						|
      mozilla::layers::ScrollableLayerGuid::ViewID viewID =
 | 
						|
          mozilla::layers::ScrollableLayerGuid::NULL_SCROLL_ID;
 | 
						|
      nsLayoutUtils::FindIDFor(content, &viewID);
 | 
						|
      MOZ_LOG(
 | 
						|
          sDisplayportLog, LogLevel::Debug,
 | 
						|
          ("ApzSmoothScrollTo setting displayport on scrollId=%" PRIu64 "\n",
 | 
						|
           viewID));
 | 
						|
    }
 | 
						|
 | 
						|
    DisplayPortUtils::CalculateAndSetDisplayPortMargins(
 | 
						|
        GetScrollTargetFrame(), DisplayPortUtils::RepaintMode::Repaint);
 | 
						|
    nsIFrame* frame = do_QueryFrame(GetScrollTargetFrame());
 | 
						|
    DisplayPortUtils::SetZeroMarginDisplayPortOnAsyncScrollableAncestors(frame);
 | 
						|
  }
 | 
						|
 | 
						|
  // Schedule a paint to ensure that the frame metrics get updated on
 | 
						|
  // the compositor thread.
 | 
						|
  SchedulePaint();
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::CanApzScrollInTheseDirections(
 | 
						|
    ScrollDirections aDirections) {
 | 
						|
  ScrollStyles styles = GetScrollStyles();
 | 
						|
  if (aDirections.contains(ScrollDirection::eHorizontal) &&
 | 
						|
      styles.mHorizontal == StyleOverflow::Hidden)
 | 
						|
    return false;
 | 
						|
  if (aDirections.contains(ScrollDirection::eVertical) &&
 | 
						|
      styles.mVertical == StyleOverflow::Hidden)
 | 
						|
    return false;
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::SmoothScrollVisual(
 | 
						|
    const nsPoint& aVisualViewportOffset,
 | 
						|
    FrameMetrics::ScrollOffsetUpdateType aUpdateType) {
 | 
						|
  bool canDoApzSmoothScroll =
 | 
						|
      nsLayoutUtils::AsyncPanZoomEnabled(this) && WantAsyncScroll();
 | 
						|
  if (!canDoApzSmoothScroll) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // Clamp the destination to the visual scroll range.
 | 
						|
  // There is a potential issue here, where |mDestination| is usually
 | 
						|
  // clamped to the layout scroll range, and so e.g. a subsequent
 | 
						|
  // window.scrollBy() may have an undesired effect. However, as this function
 | 
						|
  // is only called internally, this should not be a problem in practice.
 | 
						|
  // If it turns out to be, the fix would be:
 | 
						|
  //   - add a new "destination" field that doesn't have to be clamped to
 | 
						|
  //     the layout scroll range
 | 
						|
  //   - clamp mDestination to the layout scroll range here
 | 
						|
  //   - make sure ComputeScrollMetadata() picks up the former as the
 | 
						|
  //     smooth scroll destination to send to APZ.
 | 
						|
  mDestination = GetVisualScrollRange().ClampPoint(aVisualViewportOffset);
 | 
						|
 | 
						|
  UniquePtr<ScrollSnapTargetIds> snapTargetIds;
 | 
						|
  // Perform the scroll.
 | 
						|
  ApzSmoothScrollTo(mDestination, ScrollMode::SmoothMsd,
 | 
						|
                    aUpdateType == FrameMetrics::eRestore
 | 
						|
                        ? ScrollOrigin::Restore
 | 
						|
                        : ScrollOrigin::Other,
 | 
						|
                    ScrollTriggeredByScript::No, std::move(snapTargetIds));
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
bool nsHTMLScrollFrame::IsSmoothScroll(dom::ScrollBehavior aBehavior) const {
 | 
						|
  if (aBehavior == dom::ScrollBehavior::Instant) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // The user smooth scrolling preference should be honored for any requested
 | 
						|
  // smooth scrolls. A requested smooth scroll when smooth scrolling is
 | 
						|
  // disabled should be equivalent to an instant scroll. This is not enforced
 | 
						|
  // for the <scrollbox> XUL element to allow for the browser chrome to
 | 
						|
  // override this behavior when toolkit.scrollbox.smoothScroll is enabled.
 | 
						|
  if (!GetContent()->IsXULElement(nsGkAtoms::scrollbox)) {
 | 
						|
    if (!nsLayoutUtils::IsSmoothScrollingEnabled()) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    if (!StaticPrefs::toolkit_scrollbox_smoothScroll()) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (aBehavior == dom::ScrollBehavior::Smooth) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  nsIFrame* styleFrame = GetFrameForStyle();
 | 
						|
  if (!styleFrame) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  return (aBehavior == dom::ScrollBehavior::Auto &&
 | 
						|
          styleFrame->StyleDisplay()->mScrollBehavior ==
 | 
						|
              StyleScrollBehavior::Smooth);
 | 
						|
}
 | 
						|
 | 
						|
nsTArray<ScrollPositionUpdate> nsHTMLScrollFrame::GetScrollUpdates() const {
 | 
						|
  return mScrollUpdates.Clone();
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::AppendScrollUpdate(
 | 
						|
    const ScrollPositionUpdate& aUpdate) {
 | 
						|
  mScrollGeneration = aUpdate.GetGeneration();
 | 
						|
  mScrollUpdates.AppendElement(aUpdate);
 | 
						|
}
 | 
						|
 | 
						|
void nsHTMLScrollFrame::ScheduleScrollAnimations() {
 | 
						|
  nsIContent* content = GetContent();
 | 
						|
  MOZ_ASSERT(content && content->IsElement(),
 | 
						|
             "The nsIScrollableFrame should have the element.");
 | 
						|
 | 
						|
  const Element* elementOrPseudo = content->AsElement();
 | 
						|
  PseudoStyleType pseudo = elementOrPseudo->GetPseudoElementType();
 | 
						|
  if (pseudo != PseudoStyleType::NotPseudo &&
 | 
						|
      !AnimationUtils::IsSupportedPseudoForAnimations(pseudo)) {
 | 
						|
    // This is not an animatable pseudo element, and so we don't generate
 | 
						|
    // scroll-timeline for it.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  const auto [element, type] =
 | 
						|
      AnimationUtils::GetElementPseudoPair(elementOrPseudo);
 | 
						|
  const auto* scheduler = ProgressTimelineScheduler::Get(element, type);
 | 
						|
  if (!scheduler) {
 | 
						|
    // We don't have scroll timelines associated with this frame.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  scheduler->ScheduleAnimations();
 | 
						|
}
 |