summaryrefslogtreecommitdiffstats
path: root/layout/generic/nsIFrame.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'layout/generic/nsIFrame.cpp')
-rw-r--r--layout/generic/nsIFrame.cpp12724
1 files changed, 12724 insertions, 0 deletions
diff --git a/layout/generic/nsIFrame.cpp b/layout/generic/nsIFrame.cpp
new file mode 100644
index 0000000000..81109151ff
--- /dev/null
+++ b/layout/generic/nsIFrame.cpp
@@ -0,0 +1,12724 @@
+/* -*- 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/. */
+
+/* base class of all rendering objects */
+
+#include "nsIFrame.h"
+
+#include <stdarg.h>
+#include <algorithm>
+
+#include "gfx2DGlue.h"
+#include "gfxUtils.h"
+#include "mozilla/Attributes.h"
+#include "mozilla/CaretAssociationHint.h"
+#include "mozilla/ComputedStyle.h"
+#include "mozilla/DebugOnly.h"
+#include "mozilla/DisplayPortUtils.h"
+#include "mozilla/EventForwards.h"
+#include "mozilla/dom/CSSAnimation.h"
+#include "mozilla/dom/CSSTransition.h"
+#include "mozilla/dom/ContentVisibilityAutoStateChangeEvent.h"
+#include "mozilla/dom/DocumentInlines.h"
+#include "mozilla/dom/AncestorIterator.h"
+#include "mozilla/dom/ElementInlines.h"
+#include "mozilla/dom/ImageTracker.h"
+#include "mozilla/dom/Selection.h"
+#include "mozilla/gfx/2D.h"
+#include "mozilla/gfx/PathHelpers.h"
+#include "mozilla/IntegerRange.h"
+#include "mozilla/intl/BidiEmbeddingLevel.h"
+#include "mozilla/Maybe.h"
+#include "mozilla/PresShell.h"
+#include "mozilla/PresShellInlines.h"
+#include "mozilla/ResultExtensions.h"
+#include "mozilla/SelectionMovementUtils.h"
+#include "mozilla/Sprintf.h"
+#include "mozilla/StaticAnalysisFunctions.h"
+#include "mozilla/StaticPrefs_layout.h"
+#include "mozilla/StaticPrefs_print.h"
+#include "mozilla/StaticPrefs_ui.h"
+#include "mozilla/SVGMaskFrame.h"
+#include "mozilla/SVGObserverUtils.h"
+#include "mozilla/SVGTextFrame.h"
+#include "mozilla/SVGIntegrationUtils.h"
+#include "mozilla/SVGUtils.h"
+#include "mozilla/TextControlElement.h"
+#include "mozilla/ToString.h"
+#include "mozilla/Try.h"
+#include "mozilla/ViewportUtils.h"
+
+#include "nsCOMPtr.h"
+#include "nsFieldSetFrame.h"
+#include "nsFlexContainerFrame.h"
+#include "nsFocusManager.h"
+#include "nsFrameList.h"
+#include "nsPlaceholderFrame.h"
+#include "nsIBaseWindow.h"
+#include "nsIContent.h"
+#include "nsIContentInlines.h"
+#include "nsContentUtils.h"
+#include "nsCSSFrameConstructor.h"
+#include "nsCSSProps.h"
+#include "nsCSSPseudoElements.h"
+#include "nsCSSRendering.h"
+#include "nsAtom.h"
+#include "nsString.h"
+#include "nsReadableUtils.h"
+#include "nsTableWrapperFrame.h"
+#include "nsView.h"
+#include "nsViewManager.h"
+#include "nsIScrollableFrame.h"
+#include "nsPresContext.h"
+#include "nsPresContextInlines.h"
+#include "nsStyleConsts.h"
+#include "mozilla/Logging.h"
+#include "nsLayoutUtils.h"
+#include "LayoutLogging.h"
+#include "mozilla/RestyleManager.h"
+#include "nsImageFrame.h"
+#include "nsInlineFrame.h"
+#include "nsFrameSelection.h"
+#include "nsGkAtoms.h"
+#include "nsGridContainerFrame.h"
+#include "nsGfxScrollFrame.h"
+#include "nsCSSAnonBoxes.h"
+#include "nsCanvasFrame.h"
+
+#include "nsFieldSetFrame.h"
+#include "nsFrameTraversal.h"
+#include "nsRange.h"
+#include "nsITextControlFrame.h"
+#include "nsNameSpaceManager.h"
+#include "nsIPercentBSizeObserver.h"
+#include "nsStyleStructInlines.h"
+
+#include "nsBidiPresUtils.h"
+#include "RubyUtils.h"
+#include "TextOverflow.h"
+#include "nsAnimationManager.h"
+
+// For triple-click pref
+#include "imgIRequest.h"
+#include "nsError.h"
+#include "nsContainerFrame.h"
+#include "nsBlockFrame.h"
+#include "nsDisplayList.h"
+#include "nsChangeHint.h"
+#include "nsSubDocumentFrame.h"
+#include "RetainedDisplayListBuilder.h"
+
+#include "gfxContext.h"
+#include "nsAbsoluteContainingBlock.h"
+#include "ScrollSnap.h"
+#include "StickyScrollContainer.h"
+#include "nsFontInflationData.h"
+#include "nsRegion.h"
+#include "nsIFrameInlines.h"
+#include "nsStyleChangeList.h"
+#include "nsWindowSizes.h"
+
+#ifdef ACCESSIBILITY
+# include "nsAccessibilityService.h"
+#endif
+
+#include "mozilla/AsyncEventDispatcher.h"
+#include "mozilla/CSSClipPathInstance.h"
+#include "mozilla/EffectCompositor.h"
+#include "mozilla/EffectSet.h"
+#include "mozilla/EventListenerManager.h"
+#include "mozilla/EventStateManager.h"
+#include "mozilla/Preferences.h"
+#include "mozilla/LookAndFeel.h"
+#include "mozilla/MouseEvents.h"
+#include "mozilla/ServoStyleSet.h"
+#include "mozilla/ServoStyleSetInlines.h"
+#include "mozilla/css/ImageLoader.h"
+#include "mozilla/dom/HTMLBodyElement.h"
+#include "mozilla/dom/SVGPathData.h"
+#include "mozilla/dom/TouchEvent.h"
+#include "mozilla/gfx/Tools.h"
+#include "mozilla/layers/WebRenderUserData.h"
+#include "mozilla/layout/ScrollAnchorContainer.h"
+#include "nsPrintfCString.h"
+#include "ActiveLayerTracker.h"
+
+#include "nsITheme.h"
+
+using namespace mozilla;
+using namespace mozilla::css;
+using namespace mozilla::dom;
+using namespace mozilla::gfx;
+using namespace mozilla::layers;
+using namespace mozilla::layout;
+typedef nsAbsoluteContainingBlock::AbsPosReflowFlags AbsPosReflowFlags;
+using nsStyleTransformMatrix::TransformReferenceBox;
+
+nsIFrame* nsILineIterator::LineInfo::GetLastFrameOnLine() const {
+ if (!mNumFramesOnLine) {
+ return nullptr; // empty line, not illegal
+ }
+ MOZ_ASSERT(mFirstFrameOnLine);
+ nsIFrame* maybeLastFrame = mFirstFrameOnLine;
+ for ([[maybe_unused]] int32_t i : IntegerRange(mNumFramesOnLine - 1)) {
+ maybeLastFrame = maybeLastFrame->GetNextSibling();
+ if (NS_WARN_IF(!maybeLastFrame)) {
+ return nullptr;
+ }
+ }
+ return maybeLastFrame;
+}
+
+#ifdef HAVE_64BIT_BUILD
+static_assert(sizeof(nsIFrame) == 120, "nsIFrame should remain small");
+#else
+static_assert(sizeof(void*) == 4, "Odd build config?");
+// FIXME(emilio): Investigate why win32 and android-arm32 have bigger sizes (80)
+// than Linux32 (76).
+static_assert(sizeof(nsIFrame) <= 80, "nsIFrame should remain small");
+#endif
+
+const mozilla::LayoutFrameType nsIFrame::sLayoutFrameTypes[kFrameClassCount] = {
+#define FRAME_ID(class_, type_, ...) mozilla::LayoutFrameType::type_,
+#define ABSTRACT_FRAME_ID(...)
+#include "mozilla/FrameIdList.h"
+#undef FRAME_ID
+#undef ABSTRACT_FRAME_ID
+};
+
+const nsIFrame::ClassFlags nsIFrame::sLayoutFrameClassFlags[kFrameClassCount] =
+ {
+#define FRAME_ID(class_, type_, flags_, ...) flags_,
+#define ABSTRACT_FRAME_ID(...)
+#include "mozilla/FrameIdList.h"
+#undef FRAME_ID
+#undef ABSTRACT_FRAME_ID
+};
+
+std::ostream& operator<<(std::ostream& aStream, const nsDirection& aDirection) {
+ return aStream << (aDirection == eDirNext ? "eDirNext" : "eDirPrevious");
+}
+
+struct nsContentAndOffset {
+ nsIContent* mContent = nullptr;
+ int32_t mOffset = 0;
+};
+
+#include "nsILineIterator.h"
+#include "prenv.h"
+
+// Utility function to set a nsRect-valued property table entry on aFrame,
+// reusing the existing storage if the property happens to be already set.
+template <typename T>
+static void SetOrUpdateRectValuedProperty(
+ nsIFrame* aFrame, FrameProperties::Descriptor<T> aProperty,
+ const nsRect& aNewValue) {
+ bool found;
+ nsRect* rectStorage = aFrame->GetProperty(aProperty, &found);
+ if (!found) {
+ rectStorage = new nsRect(aNewValue);
+ aFrame->AddProperty(aProperty, rectStorage);
+ } else {
+ *rectStorage = aNewValue;
+ }
+}
+
+FrameDestroyContext::~FrameDestroyContext() {
+ for (auto& content : mozilla::Reversed(mAnonymousContent)) {
+ mPresShell->NativeAnonymousContentRemoved(content);
+ content->UnbindFromTree();
+ }
+}
+
+// Formerly the nsIFrameDebug interface
+
+std::ostream& operator<<(std::ostream& aStream, const nsReflowStatus& aStatus) {
+ char complete = 'Y';
+ if (aStatus.IsIncomplete()) {
+ complete = 'N';
+ } else if (aStatus.IsOverflowIncomplete()) {
+ complete = 'O';
+ }
+
+ char brk = 'N';
+ if (aStatus.IsInlineBreakBefore()) {
+ brk = 'B';
+ } else if (aStatus.IsInlineBreakAfter()) {
+ brk = 'A';
+ }
+
+ aStream << "["
+ << "Complete=" << complete << ","
+ << "NIF=" << (aStatus.NextInFlowNeedsReflow() ? 'Y' : 'N') << ","
+ << "Break=" << brk << ","
+ << "FirstLetter=" << (aStatus.FirstLetterComplete() ? 'Y' : 'N')
+ << "]";
+ return aStream;
+}
+
+#ifdef DEBUG
+
+/**
+ * Note: the log module is created during library initialization which
+ * means that you cannot perform logging before then.
+ */
+mozilla::LazyLogModule nsIFrame::sFrameLogModule("frame");
+
+#endif
+
+NS_DECLARE_FRAME_PROPERTY_DELETABLE(AbsoluteContainingBlockProperty,
+ nsAbsoluteContainingBlock)
+
+bool nsIFrame::HasAbsolutelyPositionedChildren() const {
+ return IsAbsoluteContainer() &&
+ GetAbsoluteContainingBlock()->HasAbsoluteFrames();
+}
+
+nsAbsoluteContainingBlock* nsIFrame::GetAbsoluteContainingBlock() const {
+ NS_ASSERTION(IsAbsoluteContainer(),
+ "The frame is not marked as an abspos container correctly");
+ nsAbsoluteContainingBlock* absCB =
+ GetProperty(AbsoluteContainingBlockProperty());
+ NS_ASSERTION(absCB,
+ "The frame is marked as an abspos container but doesn't have "
+ "the property");
+ return absCB;
+}
+
+void nsIFrame::MarkAsAbsoluteContainingBlock() {
+ MOZ_ASSERT(HasAnyStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN));
+ NS_ASSERTION(!GetProperty(AbsoluteContainingBlockProperty()),
+ "Already has an abs-pos containing block property?");
+ NS_ASSERTION(!HasAnyStateBits(NS_FRAME_HAS_ABSPOS_CHILDREN),
+ "Already has NS_FRAME_HAS_ABSPOS_CHILDREN state bit?");
+ AddStateBits(NS_FRAME_HAS_ABSPOS_CHILDREN);
+ SetProperty(AbsoluteContainingBlockProperty(),
+ new nsAbsoluteContainingBlock(GetAbsoluteListID()));
+}
+
+void nsIFrame::MarkAsNotAbsoluteContainingBlock() {
+ NS_ASSERTION(!HasAbsolutelyPositionedChildren(), "Think of the children!");
+ NS_ASSERTION(GetProperty(AbsoluteContainingBlockProperty()),
+ "Should have an abs-pos containing block property");
+ NS_ASSERTION(HasAnyStateBits(NS_FRAME_HAS_ABSPOS_CHILDREN),
+ "Should have NS_FRAME_HAS_ABSPOS_CHILDREN state bit");
+ MOZ_ASSERT(HasAnyStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN));
+ RemoveStateBits(NS_FRAME_HAS_ABSPOS_CHILDREN);
+ RemoveProperty(AbsoluteContainingBlockProperty());
+}
+
+bool nsIFrame::CheckAndClearPaintedState() {
+ bool result = HasAnyStateBits(NS_FRAME_PAINTED_THEBES);
+ RemoveStateBits(NS_FRAME_PAINTED_THEBES);
+
+ for (const auto& childList : ChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ if (child->CheckAndClearPaintedState()) {
+ result = true;
+ }
+ }
+ }
+ return result;
+}
+
+bool nsIFrame::CheckAndClearDisplayListState() {
+ bool result = BuiltDisplayList();
+ SetBuiltDisplayList(false);
+
+ for (const auto& childList : ChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ if (child->CheckAndClearDisplayListState()) {
+ result = true;
+ }
+ }
+ }
+ return result;
+}
+
+bool nsIFrame::IsVisibleConsideringAncestors(uint32_t aFlags) const {
+ if (!StyleVisibility()->IsVisible()) {
+ return false;
+ }
+
+ if (PresShell()->IsUnderHiddenEmbedderElement()) {
+ return false;
+ }
+
+ const nsIFrame* frame = this;
+ while (frame) {
+ nsView* view = frame->GetView();
+ if (view && view->GetVisibility() == ViewVisibility::Hide) {
+ return false;
+ }
+
+ if (frame->StyleUIReset()->mMozSubtreeHiddenOnlyVisually) {
+ return false;
+ }
+
+ // This method is used to determine if a frame is focusable, because it's
+ // called by nsIFrame::IsFocusable. `content-visibility: auto` should not
+ // force this frame to be unfocusable, so we only take into account
+ // `content-visibility: hidden` here.
+ if (this != frame &&
+ frame->HidesContent(IncludeContentVisibility::Hidden)) {
+ return false;
+ }
+
+ if (nsIFrame* parent = frame->GetParent()) {
+ frame = parent;
+ } else {
+ parent = nsLayoutUtils::GetCrossDocParentFrameInProcess(frame);
+ if (!parent) break;
+
+ if ((aFlags & nsIFrame::VISIBILITY_CROSS_CHROME_CONTENT_BOUNDARY) == 0 &&
+ parent->PresContext()->IsChrome() &&
+ !frame->PresContext()->IsChrome()) {
+ break;
+ }
+
+ frame = parent;
+ }
+ }
+
+ return true;
+}
+
+void nsIFrame::FindCloserFrameForSelection(
+ const nsPoint& aPoint, FrameWithDistance* aCurrentBestFrame) {
+ if (nsLayoutUtils::PointIsCloserToRect(aPoint, mRect,
+ aCurrentBestFrame->mXDistance,
+ aCurrentBestFrame->mYDistance)) {
+ aCurrentBestFrame->mFrame = this;
+ }
+}
+
+void nsIFrame::ElementStateChanged(mozilla::dom::ElementState aStates) {}
+
+void WeakFrame::Clear(mozilla::PresShell* aPresShell) {
+ if (aPresShell) {
+ aPresShell->RemoveWeakFrame(this);
+ }
+ mFrame = nullptr;
+}
+
+AutoWeakFrame::AutoWeakFrame(const WeakFrame& aOther)
+ : mPrev(nullptr), mFrame(nullptr) {
+ Init(aOther.GetFrame());
+}
+
+void AutoWeakFrame::Clear(mozilla::PresShell* aPresShell) {
+ if (aPresShell) {
+ aPresShell->RemoveAutoWeakFrame(this);
+ }
+ mFrame = nullptr;
+ mPrev = nullptr;
+}
+
+AutoWeakFrame::~AutoWeakFrame() {
+ Clear(mFrame ? mFrame->PresContext()->GetPresShell() : nullptr);
+}
+
+void AutoWeakFrame::Init(nsIFrame* aFrame) {
+ Clear(mFrame ? mFrame->PresContext()->GetPresShell() : nullptr);
+ mFrame = aFrame;
+ if (mFrame) {
+ mozilla::PresShell* presShell = mFrame->PresContext()->GetPresShell();
+ NS_WARNING_ASSERTION(presShell, "Null PresShell in AutoWeakFrame!");
+ if (presShell) {
+ presShell->AddAutoWeakFrame(this);
+ } else {
+ mFrame = nullptr;
+ }
+ }
+}
+
+void WeakFrame::Init(nsIFrame* aFrame) {
+ Clear(mFrame ? mFrame->PresContext()->GetPresShell() : nullptr);
+ mFrame = aFrame;
+ if (mFrame) {
+ mozilla::PresShell* presShell = mFrame->PresContext()->GetPresShell();
+ MOZ_ASSERT(presShell, "Null PresShell in WeakFrame!");
+ if (presShell) {
+ presShell->AddWeakFrame(this);
+ } else {
+ mFrame = nullptr;
+ }
+ }
+}
+
+nsIFrame* NS_NewEmptyFrame(PresShell* aPresShell, ComputedStyle* aStyle) {
+ return new (aPresShell) nsIFrame(aStyle, aPresShell->GetPresContext());
+}
+
+nsIFrame::~nsIFrame() {
+ MOZ_COUNT_DTOR(nsIFrame);
+
+ MOZ_ASSERT(GetVisibility() != Visibility::ApproximatelyVisible,
+ "Visible nsFrame is being destroyed");
+}
+
+NS_IMPL_FRAMEARENA_HELPERS(nsIFrame)
+
+// Dummy operator delete. Will never be called, but must be defined
+// to satisfy some C++ ABIs.
+void nsIFrame::operator delete(void*, size_t) {
+ MOZ_CRASH("nsIFrame::operator delete should never be called");
+}
+
+NS_QUERYFRAME_HEAD(nsIFrame)
+ NS_QUERYFRAME_ENTRY(nsIFrame)
+NS_QUERYFRAME_TAIL_INHERITANCE_ROOT
+
+/////////////////////////////////////////////////////////////////////////////
+// nsIFrame
+
+static bool IsFontSizeInflationContainer(nsIFrame* aFrame,
+ const nsStyleDisplay* aStyleDisplay) {
+ /*
+ * Font size inflation is built around the idea that we're inflating
+ * the fonts for a pan-and-zoom UI so that when the user scales up a
+ * block or other container to fill the width of the device, the fonts
+ * will be readable. To do this, we need to pick what counts as a
+ * container.
+ *
+ * From a code perspective, the only hard requirement is that frames
+ * that are line participants (nsIFrame::IsLineParticipant) are never
+ * containers, since line layout assumes that the inflation is consistent
+ * within a line.
+ *
+ * This is not an imposition, since we obviously want a bunch of text
+ * (possibly with inline elements) flowing within a block to count the
+ * block (or higher) as its container.
+ *
+ * We also want form controls, including the text in the anonymous
+ * content inside of them, to match each other and the text next to
+ * them, so they and their anonymous content should also not be a
+ * container.
+ *
+ * However, because we can't reliably compute sizes across XUL during
+ * reflow, any XUL frame with a XUL parent is always a container.
+ *
+ * There are contexts where it would be nice if some blocks didn't
+ * count as a container, so that, for example, an indented quotation
+ * didn't end up with a smaller font size. However, it's hard to
+ * distinguish these situations where we really do want the indented
+ * thing to count as a container, so we don't try, and blocks are
+ * always containers.
+ */
+
+ // The root frame should always be an inflation container.
+ if (!aFrame->GetParent()) {
+ return true;
+ }
+
+ nsIContent* content = aFrame->GetContent();
+ if (content && content->IsInNativeAnonymousSubtree()) {
+ // Native anonymous content shouldn't be a font inflation root,
+ // except for the canvas custom content container.
+ nsCanvasFrame* canvas = aFrame->PresShell()->GetCanvasFrame();
+ return canvas && canvas->GetCustomContentContainer() == content;
+ }
+
+ LayoutFrameType frameType = aFrame->Type();
+ bool isInline =
+ aFrame->GetDisplay().IsInlineFlow() || RubyUtils::IsRubyBox(frameType) ||
+ (aStyleDisplay->IsFloatingStyle() &&
+ frameType == LayoutFrameType::Letter) ||
+ // Given multiple frames for the same node, only the
+ // outer one should be considered a container.
+ // (Important, e.g., for nsSelectsAreaFrame.)
+ (aFrame->GetParent()->GetContent() == content) ||
+ (content &&
+ // Form controls shouldn't become inflation containers.
+ (content->IsAnyOfHTMLElements(nsGkAtoms::option, nsGkAtoms::optgroup,
+ nsGkAtoms::select, nsGkAtoms::input,
+ nsGkAtoms::button, nsGkAtoms::textarea)));
+ NS_ASSERTION(!aFrame->IsLineParticipant() || isInline ||
+ // br frames and mathml frames report being line
+ // participants even when their position or display is
+ // set
+ aFrame->IsBrFrame() || aFrame->IsMathMLFrame(),
+ "line participants must not be containers");
+ return !isInline;
+}
+
+static void MaybeScheduleReflowSVGNonDisplayText(nsIFrame* aFrame) {
+ if (!aFrame->IsInSVGTextSubtree()) {
+ return;
+ }
+
+ // We need to ensure that any non-display SVGTextFrames get reflowed when a
+ // child text frame gets new style. Thus we need to schedule a reflow in
+ // |DidSetComputedStyle|. We also need to call it from |DestroyFrom|,
+ // because otherwise we won't get notified when style changes to
+ // "display:none".
+ SVGTextFrame* svgTextFrame = static_cast<SVGTextFrame*>(
+ nsLayoutUtils::GetClosestFrameOfType(aFrame, LayoutFrameType::SVGText));
+ nsIFrame* anonBlock = svgTextFrame->PrincipalChildList().FirstChild();
+
+ // Note that we must check NS_FRAME_FIRST_REFLOW on our SVGTextFrame's
+ // anonymous block frame rather than our aFrame, since NS_FRAME_FIRST_REFLOW
+ // may be set on us if we're a new frame that has been inserted after the
+ // document's first reflow. (In which case this DidSetComputedStyle call may
+ // be happening under frame construction under a Reflow() call.)
+ if (!anonBlock || anonBlock->HasAnyStateBits(NS_FRAME_FIRST_REFLOW)) {
+ return;
+ }
+
+ if (!svgTextFrame->HasAnyStateBits(NS_FRAME_IS_NONDISPLAY) ||
+ svgTextFrame->HasAnyStateBits(NS_STATE_SVG_TEXT_IN_REFLOW)) {
+ return;
+ }
+
+ svgTextFrame->ScheduleReflowSVGNonDisplayText(
+ IntrinsicDirty::FrameAncestorsAndDescendants);
+}
+
+bool nsIFrame::IsPrimaryFrameOfRootOrBodyElement() const {
+ if (!IsPrimaryFrame()) {
+ return false;
+ }
+ nsIContent* content = GetContent();
+ Document* document = content->OwnerDoc();
+ return content == document->GetRootElement() ||
+ content == document->GetBodyElement();
+}
+
+bool nsIFrame::IsRenderedLegend() const {
+ if (auto* parent = GetParent(); parent && parent->IsFieldSetFrame()) {
+ return static_cast<nsFieldSetFrame*>(parent)->GetLegend() == this;
+ }
+ return false;
+}
+
+void nsIFrame::Init(nsIContent* aContent, nsContainerFrame* aParent,
+ nsIFrame* aPrevInFlow) {
+ MOZ_ASSERT(nsQueryFrame::FrameIID(mClass) == GetFrameId());
+ MOZ_ASSERT(!mContent, "Double-initing a frame?");
+
+ mContent = aContent;
+ mParent = aParent;
+ MOZ_DIAGNOSTIC_ASSERT(!mParent || PresShell() == mParent->PresShell());
+
+ if (aPrevInFlow) {
+ mWritingMode = aPrevInFlow->GetWritingMode();
+
+ // Copy some state bits from prev-in-flow (the bits that should apply
+ // throughout a continuation chain). The bits are sorted according to their
+ // order in nsFrameStateBits.h.
+
+ // clang-format off
+ AddStateBits(aPrevInFlow->GetStateBits() &
+ (NS_FRAME_GENERATED_CONTENT |
+ NS_FRAME_OUT_OF_FLOW |
+ NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN |
+ NS_FRAME_INDEPENDENT_SELECTION |
+ NS_FRAME_PART_OF_IBSPLIT |
+ NS_FRAME_MAY_BE_TRANSFORMED |
+ NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR));
+ // clang-format on
+
+ // Copy other bits in nsIFrame from prev-in-flow.
+ mHasColumnSpanSiblings = aPrevInFlow->HasColumnSpanSiblings();
+ } else {
+ PresContext()->ConstructedFrame();
+ }
+
+ if (GetParent()) {
+ if (MOZ_UNLIKELY(mContent == PresContext()->Document()->GetRootElement() &&
+ mContent == GetParent()->GetContent())) {
+ // Our content is the root element and we have the same content as our
+ // parent. That is, we are the internal anonymous frame of the root
+ // element. Copy the used mWritingMode from our parent because
+ // mDocElementContainingBlock gets its mWritingMode from <body>.
+ mWritingMode = GetParent()->GetWritingMode();
+ }
+
+ // Copy some state bits from our parent (the bits that should apply
+ // recursively throughout a subtree). The bits are sorted according to their
+ // order in nsFrameStateBits.h.
+
+ // clang-format off
+ AddStateBits(GetParent()->GetStateBits() &
+ (NS_FRAME_GENERATED_CONTENT |
+ NS_FRAME_INDEPENDENT_SELECTION |
+ NS_FRAME_IS_SVG_TEXT |
+ NS_FRAME_IN_POPUP |
+ NS_FRAME_IS_NONDISPLAY));
+ // clang-format on
+
+ if (HasAnyStateBits(NS_FRAME_IN_POPUP) && TrackingVisibility()) {
+ // Assume all frames in popups are visible.
+ IncApproximateVisibleCount();
+ }
+ }
+ if (aPrevInFlow) {
+ mMayHaveOpacityAnimation = aPrevInFlow->MayHaveOpacityAnimation();
+ mMayHaveTransformAnimation = aPrevInFlow->MayHaveTransformAnimation();
+ } else if (mContent) {
+ // It's fine to fetch the EffectSet for the style frame here because in the
+ // following code we take care of the case where animations may target
+ // a different frame.
+ EffectSet* effectSet = EffectSet::GetForStyleFrame(this);
+ if (effectSet) {
+ mMayHaveOpacityAnimation = effectSet->MayHaveOpacityAnimation();
+
+ if (effectSet->MayHaveTransformAnimation()) {
+ // If we are the inner table frame for display:table content, then
+ // transform animations should go on our parent frame (the table wrapper
+ // frame).
+ //
+ // We do this when initializing the child frame (table inner frame),
+ // because when initializng the table wrapper frame, we don't yet have
+ // access to its children so we can't tell if we have transform
+ // animations or not.
+ if (SupportsCSSTransforms()) {
+ mMayHaveTransformAnimation = true;
+ AddStateBits(NS_FRAME_MAY_BE_TRANSFORMED);
+ } else if (aParent && nsLayoutUtils::GetStyleFrame(aParent) == this) {
+ MOZ_ASSERT(
+ aParent->SupportsCSSTransforms(),
+ "Style frames that don't support transforms should have parents"
+ " that do");
+ aParent->mMayHaveTransformAnimation = true;
+ aParent->AddStateBits(NS_FRAME_MAY_BE_TRANSFORMED);
+ }
+ }
+ }
+ }
+
+ const nsStyleDisplay* disp = StyleDisplay();
+ if (disp->HasTransform(this)) {
+ // If 'transform' dynamically changes, RestyleManager takes care of
+ // updating this bit.
+ AddStateBits(NS_FRAME_MAY_BE_TRANSFORMED);
+ }
+
+ if (nsLayoutUtils::FontSizeInflationEnabled(PresContext()) ||
+ !GetParent()
+#ifdef DEBUG
+ // We have assertions that check inflation invariants even when
+ // font size inflation is not enabled.
+ || true
+#endif
+ ) {
+ if (IsFontSizeInflationContainer(this, disp)) {
+ AddStateBits(NS_FRAME_FONT_INFLATION_CONTAINER);
+ if (!GetParent() ||
+ // I'd use NS_FRAME_OUT_OF_FLOW, but it's not set yet.
+ disp->IsFloating(this) || disp->IsAbsolutelyPositioned(this) ||
+ GetParent()->IsFlexContainerFrame() ||
+ GetParent()->IsGridContainerFrame()) {
+ AddStateBits(NS_FRAME_FONT_INFLATION_FLOW_ROOT);
+ }
+ }
+ NS_ASSERTION(
+ GetParent() || HasAnyStateBits(NS_FRAME_FONT_INFLATION_CONTAINER),
+ "root frame should always be a container");
+ }
+
+ if (TrackingVisibility() && PresShell()->AssumeAllFramesVisible()) {
+ IncApproximateVisibleCount();
+ }
+
+ DidSetComputedStyle(nullptr);
+
+ // For a newly created frame, we need to update this frame's visibility state.
+ // Usually we update the state when the frame is restyled and has a
+ // VisibilityChange change hint but we don't generate any change hints for
+ // newly created frames.
+ // Note: We don't need to do this for placeholders since placeholders have
+ // different styles so that the styles don't have visibility:hidden even if
+ // the parent has visibility:hidden style. We also don't need to update the
+ // state when creating continuations because its visibility is the same as its
+ // prev-in-flow, and the animation code cares only primary frames.
+ if (!IsPlaceholderFrame() && !aPrevInFlow) {
+ UpdateVisibleDescendantsState();
+ }
+
+ if (!aPrevInFlow && HasAnyStateBits(NS_FRAME_IS_NONDISPLAY)) {
+ // We aren't going to get a reflow, so nothing else will call
+ // InvalidateRenderingObservers, we have to do it here.
+ SVGObserverUtils::InvalidateRenderingObservers(this);
+ }
+}
+
+void nsIFrame::InitPrimaryFrame() {
+ MOZ_ASSERT(IsPrimaryFrame());
+ HandlePrimaryFrameStyleChange(nullptr);
+}
+
+void nsIFrame::HandlePrimaryFrameStyleChange(ComputedStyle* aOldStyle) {
+ const nsStyleDisplay* disp = StyleDisplay();
+ const nsStyleDisplay* oldDisp =
+ aOldStyle ? aOldStyle->StyleDisplay() : nullptr;
+
+ const bool wasQueryContainer = oldDisp && oldDisp->IsQueryContainer();
+ const bool isQueryContainer = disp->IsQueryContainer();
+ if (wasQueryContainer != isQueryContainer) {
+ auto* pc = PresContext();
+ if (isQueryContainer) {
+ pc->RegisterContainerQueryFrame(this);
+ } else {
+ pc->UnregisterContainerQueryFrame(this);
+ }
+ }
+
+ const auto cv = disp->ContentVisibility(*this);
+ if (!oldDisp || oldDisp->ContentVisibility(*this) != cv) {
+ if (cv == StyleContentVisibility::Auto) {
+ PresShell()->RegisterContentVisibilityAutoFrame(this);
+ } else {
+ if (auto* element = Element::FromNodeOrNull(GetContent())) {
+ element->ClearContentRelevancy();
+ }
+ PresShell()->UnregisterContentVisibilityAutoFrame(this);
+ }
+ PresContext()->SetNeedsToUpdateHiddenByContentVisibilityForAnimations();
+ }
+
+ HandleLastRememberedSize();
+}
+
+void nsIFrame::Destroy(DestroyContext& aContext) {
+ NS_ASSERTION(!nsContentUtils::IsSafeToRunScript(),
+ "destroy called on frame while scripts not blocked");
+ NS_ASSERTION(!GetNextSibling() && !GetPrevSibling(),
+ "Frames should be removed before destruction.");
+ MOZ_ASSERT(!HasAbsolutelyPositionedChildren());
+ MOZ_ASSERT(!HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT),
+ "NS_FRAME_PART_OF_IBSPLIT set on non-nsContainerFrame?");
+
+ MaybeScheduleReflowSVGNonDisplayText(this);
+
+ SVGObserverUtils::InvalidateDirectRenderingObservers(this);
+
+ const auto* disp = StyleDisplay();
+ if (disp->mPosition == StylePositionProperty::Sticky) {
+ if (auto* ssc =
+ StickyScrollContainer::GetStickyScrollContainerForFrame(this)) {
+ ssc->RemoveFrame(this);
+ }
+ }
+
+ if (HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
+ if (nsPlaceholderFrame* placeholder = GetPlaceholderFrame()) {
+ placeholder->SetOutOfFlowFrame(nullptr);
+ }
+ }
+
+ nsPresContext* pc = PresContext();
+ mozilla::PresShell* ps = pc->GetPresShell();
+ if (IsPrimaryFrame()) {
+ if (disp->IsQueryContainer()) {
+ pc->UnregisterContainerQueryFrame(this);
+ }
+ if (disp->ContentVisibility(*this) == StyleContentVisibility::Auto) {
+ ps->UnregisterContentVisibilityAutoFrame(this);
+ }
+ // This needs to happen before we clear our Properties() table.
+ ActiveLayerTracker::TransferActivityToContent(this, mContent);
+ }
+
+ ScrollAnchorContainer* anchor = nullptr;
+ if (IsScrollAnchor(&anchor)) {
+ anchor->InvalidateAnchor();
+ }
+
+ if (HasCSSAnimations() || HasCSSTransitions() ||
+ // It's fine to look up the style frame here since if we're destroying the
+ // frames for display:table content we should be destroying both wrapper
+ // and inner frame.
+ EffectSet::GetForStyleFrame(this)) {
+ // If no new frame for this element is created by the end of the
+ // restyling process, stop animations and transitions for this frame
+ RestyleManager::AnimationsWithDestroyedFrame* adf =
+ pc->RestyleManager()->GetAnimationsWithDestroyedFrame();
+ // AnimationsWithDestroyedFrame only lives during the restyling process.
+ if (adf) {
+ adf->Put(mContent, mComputedStyle);
+ }
+ }
+
+ // Disable visibility tracking. Note that we have to do this before we clear
+ // frame properties and lose track of whether we were previously visible.
+ // XXX(seth): It'd be ideal to assert that we're already marked nonvisible
+ // here, but it's unfortunately tricky to guarantee in the face of things like
+ // frame reconstruction induced by style changes.
+ DisableVisibilityTracking();
+
+ // Ensure that we're not in the approximately visible list anymore.
+ ps->RemoveFrameFromApproximatelyVisibleList(this);
+
+ ps->NotifyDestroyingFrame(this);
+
+ if (HasAnyStateBits(NS_FRAME_EXTERNAL_REFERENCE)) {
+ ps->ClearFrameRefs(this);
+ }
+
+ nsView* view = GetView();
+ if (view) {
+ view->SetFrame(nullptr);
+ view->Destroy();
+ }
+
+ // Make sure that our deleted frame can't be returned from GetPrimaryFrame()
+ if (IsPrimaryFrame()) {
+ mContent->SetPrimaryFrame(nullptr);
+
+ // Pass the root of a generated content subtree (e.g. ::after/::before) to
+ // aPostDestroyData to unbind it after frame destruction is done.
+ if (HasAnyStateBits(NS_FRAME_GENERATED_CONTENT) &&
+ mContent->IsRootOfNativeAnonymousSubtree()) {
+ aContext.AddAnonymousContent(mContent.forget());
+ }
+ }
+
+ // Remove all properties attached to the frame, to ensure any property
+ // destructors that need the frame pointer are handled properly.
+ RemoveAllProperties();
+
+ // Must retrieve the object ID before calling destructors, so the
+ // vtable is still valid.
+ //
+ // Note to future tweakers: having the method that returns the
+ // object size call the destructor will not avoid an indirect call;
+ // the compiler cannot devirtualize the call to the destructor even
+ // if it's from a method defined in the same class.
+
+ nsQueryFrame::FrameIID id = GetFrameId();
+ this->~nsIFrame();
+
+#ifdef DEBUG
+ {
+ nsIFrame* rootFrame = ps->GetRootFrame();
+ MOZ_ASSERT(rootFrame);
+ if (this != rootFrame) {
+ auto* builder = nsLayoutUtils::GetRetainedDisplayListBuilder(rootFrame);
+ auto* data = builder ? builder->Data() : nullptr;
+
+ const bool inData =
+ data && (data->IsModified(this) || data->HasProps(this));
+
+ if (inData) {
+ DL_LOG(LogLevel::Warning, "Frame %p found in retained data", this);
+ }
+
+ MOZ_ASSERT(!inData, "Deleted frame in retained data!");
+ }
+ }
+#endif
+
+ // Now that we're totally cleaned out, we need to add ourselves to
+ // the presshell's recycler.
+ ps->FreeFrame(id, this);
+}
+
+std::pair<int32_t, int32_t> nsIFrame::GetOffsets() const {
+ return std::make_pair(0, 0);
+}
+
+static void CompareLayers(
+ const nsStyleImageLayers* aFirstLayers,
+ const nsStyleImageLayers* aSecondLayers,
+ const std::function<void(imgRequestProxy* aReq)>& aCallback) {
+ NS_FOR_VISIBLE_IMAGE_LAYERS_BACK_TO_FRONT(i, (*aFirstLayers)) {
+ const auto& image = aFirstLayers->mLayers[i].mImage;
+ if (!image.IsImageRequestType() || !image.IsResolved()) {
+ continue;
+ }
+
+ // aCallback is called when the style image in aFirstLayers is thought to
+ // be different with the corresponded one in aSecondLayers
+ if (!aSecondLayers || i >= aSecondLayers->mImageCount ||
+ (!aSecondLayers->mLayers[i].mImage.IsResolved() ||
+ image.GetImageRequest() !=
+ aSecondLayers->mLayers[i].mImage.GetImageRequest())) {
+ if (imgRequestProxy* req = image.GetImageRequest()) {
+ aCallback(req);
+ }
+ }
+ }
+}
+
+static void AddAndRemoveImageAssociations(
+ ImageLoader& aImageLoader, nsIFrame* aFrame,
+ const nsStyleImageLayers* aOldLayers,
+ const nsStyleImageLayers* aNewLayers) {
+ // If the old context had a background-image image, or mask-image image,
+ // and new context does not have the same image, clear the image load
+ // notifier (which keeps the image loading, if it still is) for the frame.
+ // We want to do this conservatively because some frames paint their
+ // backgrounds from some other frame's style data, and we don't want
+ // to clear those notifiers unless we have to. (They'll be reset
+ // when we paint, although we could miss a notification in that
+ // interval.)
+ if (aOldLayers && aFrame->HasImageRequest()) {
+ CompareLayers(aOldLayers, aNewLayers, [&](imgRequestProxy* aReq) {
+ aImageLoader.DisassociateRequestFromFrame(aReq, aFrame);
+ });
+ }
+
+ CompareLayers(aNewLayers, aOldLayers, [&](imgRequestProxy* aReq) {
+ aImageLoader.AssociateRequestToFrame(aReq, aFrame);
+ });
+}
+
+void nsIFrame::AddDisplayItem(nsDisplayItem* aItem) {
+ MOZ_DIAGNOSTIC_ASSERT(!mDisplayItems.Contains(aItem));
+ mDisplayItems.AppendElement(aItem);
+#ifdef ACCESSIBILITY
+ if (nsAccessibilityService* accService = GetAccService()) {
+ accService->NotifyOfPossibleBoundsChange(PresShell(), mContent);
+ }
+#endif
+}
+
+bool nsIFrame::RemoveDisplayItem(nsDisplayItem* aItem) {
+ return mDisplayItems.RemoveElement(aItem);
+}
+
+bool nsIFrame::HasDisplayItems() { return !mDisplayItems.IsEmpty(); }
+
+bool nsIFrame::HasDisplayItem(nsDisplayItem* aItem) {
+ return mDisplayItems.Contains(aItem);
+}
+
+bool nsIFrame::HasDisplayItem(uint32_t aKey) {
+ for (nsDisplayItem* i : mDisplayItems) {
+ if (i->GetPerFrameKey() == aKey) {
+ return true;
+ }
+ }
+ return false;
+}
+
+template <typename Condition>
+static void DiscardDisplayItems(nsIFrame* aFrame, Condition aCondition) {
+ for (nsDisplayItem* i : aFrame->DisplayItems()) {
+ // Only discard items that are invalidated by this frame, as we're only
+ // guaranteed to rebuild those items. Table background items are created by
+ // the relevant table part, but have the cell frame as the primary frame,
+ // and we don't want to remove them if this is the cell.
+ if (aCondition(i) && i->FrameForInvalidation() == aFrame) {
+ i->SetCantBeReused();
+ }
+ }
+}
+
+static void DiscardOldItems(nsIFrame* aFrame) {
+ DiscardDisplayItems(aFrame,
+ [](nsDisplayItem* aItem) { return aItem->IsOldItem(); });
+}
+
+void nsIFrame::RemoveDisplayItemDataForDeletion() {
+ // Destroying a WebRenderUserDataTable can cause destruction of other objects
+ // which can remove frame properties in their destructor. If we delete a frame
+ // property it runs the destructor of the stored object in the middle of
+ // updating the frame property table, so if the destruction of that object
+ // causes another update to the frame property table it would leave the frame
+ // property table in an inconsistent state. So we remove it from the table and
+ // then destroy it. (bug 1530657)
+ WebRenderUserDataTable* userDataTable =
+ TakeProperty(WebRenderUserDataProperty::Key());
+ if (userDataTable) {
+ for (const auto& data : userDataTable->Values()) {
+ data->RemoveFromTable();
+ }
+ delete userDataTable;
+ }
+
+ if (!nsLayoutUtils::AreRetainedDisplayListsEnabled()) {
+ // Retained display lists are disabled, no need to update
+ // RetainedDisplayListData.
+ return;
+ }
+
+ auto* builder = nsLayoutUtils::GetRetainedDisplayListBuilder(this);
+ if (!builder) {
+ MOZ_ASSERT(DisplayItems().IsEmpty());
+ MOZ_ASSERT(!IsFrameModified());
+ return;
+ }
+
+ for (nsDisplayItem* i : DisplayItems()) {
+ if (i->GetDependentFrame() == this && !i->HasDeletedFrame()) {
+ i->Frame()->MarkNeedsDisplayItemRebuild();
+ }
+ i->RemoveFrame(this);
+ }
+
+ DisplayItems().Clear();
+
+ nsAutoString name;
+#ifdef DEBUG_FRAME_DUMP
+ if (DL_LOG_TEST(LogLevel::Debug)) {
+ GetFrameName(name);
+ }
+#endif
+ DL_LOGV("Removing display item data for frame %p (%s)", this,
+ NS_ConvertUTF16toUTF8(name).get());
+
+ auto* data = builder->Data();
+ if (MayHaveWillChangeBudget()) {
+ // Keep the frame in list, so it can be removed from the will-change budget.
+ data->Flags(this) = RetainedDisplayListData::FrameFlag::HadWillChange;
+ } else {
+ data->Remove(this);
+ }
+}
+
+void nsIFrame::MarkNeedsDisplayItemRebuild() {
+ if (!nsLayoutUtils::AreRetainedDisplayListsEnabled() || IsFrameModified() ||
+ HasAnyStateBits(NS_FRAME_IN_POPUP)) {
+ // Skip frames that are already marked modified.
+ return;
+ }
+
+ if (Type() == LayoutFrameType::Placeholder) {
+ nsIFrame* oof = static_cast<nsPlaceholderFrame*>(this)->GetOutOfFlowFrame();
+ if (oof) {
+ oof->MarkNeedsDisplayItemRebuild();
+ }
+ // Do not mark placeholder frames modified.
+ return;
+ }
+
+#ifdef ACCESSIBILITY
+ if (nsAccessibilityService* accService = GetAccService()) {
+ accService->NotifyOfPossibleBoundsChange(PresShell(), mContent);
+ }
+#endif
+
+ nsIFrame* rootFrame = PresShell()->GetRootFrame();
+
+ if (rootFrame->IsFrameModified()) {
+ // The whole frame tree is modified.
+ return;
+ }
+
+ auto* builder = nsLayoutUtils::GetRetainedDisplayListBuilder(this);
+ if (!builder) {
+ MOZ_ASSERT(DisplayItems().IsEmpty());
+ return;
+ }
+
+ RetainedDisplayListData* data = builder->Data();
+ MOZ_ASSERT(data);
+
+ if (data->AtModifiedFrameLimit()) {
+ // This marks the whole frame tree modified.
+ // See |RetainedDisplayListBuilder::ShouldBuildPartial()|.
+ data->AddModifiedFrame(rootFrame);
+ return;
+ }
+
+ nsAutoString name;
+#ifdef DEBUG_FRAME_DUMP
+ if (DL_LOG_TEST(LogLevel::Debug)) {
+ GetFrameName(name);
+ }
+#endif
+
+ DL_LOGV("RDL - Rebuilding display items for frame %p (%s)", this,
+ NS_ConvertUTF16toUTF8(name).get());
+
+ data->AddModifiedFrame(this);
+
+ MOZ_ASSERT(
+ PresContext()->LayoutPhaseCount(nsLayoutPhase::DisplayListBuilding) == 0);
+
+ // Hopefully this is cheap, but we could use a frame state bit to note
+ // the presence of dependencies to speed it up.
+ for (nsDisplayItem* i : DisplayItems()) {
+ if (i->HasDeletedFrame() || i->Frame() == this) {
+ // Ignore the items with deleted frames, and the items with |this| as
+ // the primary frame.
+ continue;
+ }
+
+ if (i->GetDependentFrame() == this) {
+ // For items with |this| as a dependent frame, mark the primary frame
+ // for rebuild.
+ i->Frame()->MarkNeedsDisplayItemRebuild();
+ }
+ }
+}
+
+// Subclass hook for style post processing
+/* virtual */
+void nsIFrame::DidSetComputedStyle(ComputedStyle* aOldComputedStyle) {
+#ifdef ACCESSIBILITY
+ // Don't notify for reconstructed frames here, since the frame is still being
+ // constructed at this point and so LocalAccessible::GetFrame() will return
+ // null. Style changes for reconstructed frames are handled in
+ // DocAccessible::PruneOrInsertSubtree.
+ if (aOldComputedStyle) {
+ if (nsAccessibilityService* accService = GetAccService()) {
+ accService->NotifyOfComputedStyleChange(PresShell(), mContent);
+ }
+ }
+#endif
+
+ MaybeScheduleReflowSVGNonDisplayText(this);
+
+ Document* doc = PresContext()->Document();
+ ImageLoader* loader = doc->StyleImageLoader();
+ // Continuing text frame doesn't initialize its continuation pointer before
+ // reaching here for the first time, so we have to exclude text frames. This
+ // doesn't affect correctness because text can't match selectors.
+ //
+ // FIXME(emilio): We should consider fixing that.
+ //
+ // TODO(emilio): Can we avoid doing some / all of the image stuff when
+ // isNonTextFirstContinuation is false? We should consider doing this just for
+ // primary frames and pseudos, but the first-line reparenting code makes it
+ // all bad, should get around to bug 1465474 eventually :(
+ const bool isNonText = !IsTextFrame();
+ if (isNonText) {
+ mComputedStyle->StartImageLoads(*doc, aOldComputedStyle);
+ }
+
+ const nsStyleImageLayers* oldLayers =
+ aOldComputedStyle ? &aOldComputedStyle->StyleBackground()->mImage
+ : nullptr;
+ const nsStyleImageLayers* newLayers = &StyleBackground()->mImage;
+ AddAndRemoveImageAssociations(*loader, this, oldLayers, newLayers);
+
+ oldLayers =
+ aOldComputedStyle ? &aOldComputedStyle->StyleSVGReset()->mMask : nullptr;
+ newLayers = &StyleSVGReset()->mMask;
+ AddAndRemoveImageAssociations(*loader, this, oldLayers, newLayers);
+
+ const nsStyleDisplay* disp = StyleDisplay();
+ bool handleStickyChange = false;
+ if (aOldComputedStyle) {
+ // Detect style changes that should trigger a scroll anchor adjustment
+ // suppression.
+ // https://drafts.csswg.org/css-scroll-anchoring/#suppression-triggers
+ bool needAnchorSuppression = false;
+
+ const nsStyleMargin* oldMargin = aOldComputedStyle->StyleMargin();
+ if (oldMargin->mMargin != StyleMargin()->mMargin) {
+ needAnchorSuppression = true;
+ }
+
+ const nsStylePadding* oldPadding = aOldComputedStyle->StylePadding();
+ if (oldPadding->mPadding != StylePadding()->mPadding) {
+ SetHasPaddingChange(true);
+ needAnchorSuppression = true;
+ }
+
+ const nsStyleDisplay* oldDisp = aOldComputedStyle->StyleDisplay();
+ if (oldDisp->mOverflowAnchor != disp->mOverflowAnchor) {
+ if (auto* container = ScrollAnchorContainer::FindFor(this)) {
+ container->InvalidateAnchor();
+ }
+ if (nsIScrollableFrame* scrollableFrame = do_QueryFrame(this)) {
+ scrollableFrame->Anchor()->InvalidateAnchor();
+ }
+ }
+
+ if (mInScrollAnchorChain) {
+ const nsStylePosition* pos = StylePosition();
+ const nsStylePosition* oldPos = aOldComputedStyle->StylePosition();
+ if (!needAnchorSuppression &&
+ (oldPos->mOffset != pos->mOffset || oldPos->mWidth != pos->mWidth ||
+ oldPos->mMinWidth != pos->mMinWidth ||
+ oldPos->mMaxWidth != pos->mMaxWidth ||
+ oldPos->mHeight != pos->mHeight ||
+ oldPos->mMinHeight != pos->mMinHeight ||
+ oldPos->mMaxHeight != pos->mMaxHeight ||
+ oldDisp->mPosition != disp->mPosition ||
+ oldDisp->mTransform != disp->mTransform)) {
+ needAnchorSuppression = true;
+ }
+
+ if (needAnchorSuppression &&
+ StaticPrefs::layout_css_scroll_anchoring_suppressions_enabled()) {
+ ScrollAnchorContainer::FindFor(this)->SuppressAdjustments();
+ }
+ }
+
+ if (disp->mPosition != oldDisp->mPosition) {
+ if (!disp->IsRelativelyOrStickyPositionedStyle() &&
+ oldDisp->IsRelativelyOrStickyPositionedStyle()) {
+ RemoveProperty(NormalPositionProperty());
+ }
+
+ handleStickyChange = disp->mPosition == StylePositionProperty::Sticky ||
+ oldDisp->mPosition == StylePositionProperty::Sticky;
+ }
+ if (disp->mScrollSnapAlign != oldDisp->mScrollSnapAlign) {
+ ScrollSnapUtils::PostPendingResnapFor(this);
+ }
+ if (aOldComputedStyle->IsRootElementStyle() &&
+ disp->mScrollSnapType != oldDisp->mScrollSnapType) {
+ if (nsIScrollableFrame* scrollableFrame =
+ PresShell()->GetRootScrollFrameAsScrollable()) {
+ scrollableFrame->PostPendingResnap();
+ }
+ }
+ if (StyleUIReset()->mMozSubtreeHiddenOnlyVisually &&
+ !aOldComputedStyle->StyleUIReset()->mMozSubtreeHiddenOnlyVisually) {
+ PresShell::ClearMouseCapture(this);
+ }
+ } else { // !aOldComputedStyle
+ handleStickyChange = disp->mPosition == StylePositionProperty::Sticky;
+ }
+
+ if (handleStickyChange && !HasAnyStateBits(NS_FRAME_IS_NONDISPLAY) &&
+ !GetPrevInFlow()) {
+ // Note that we only add first continuations, but we really only
+ // want to add first continuation-or-ib-split-siblings. But since we don't
+ // yet know if we're a later part of a block-in-inline split, we'll just
+ // add later members of a block-in-inline split here, and then
+ // StickyScrollContainer will remove them later.
+ if (auto* ssc =
+ StickyScrollContainer::GetStickyScrollContainerForFrame(this)) {
+ if (disp->mPosition == StylePositionProperty::Sticky) {
+ ssc->AddFrame(this);
+ } else {
+ ssc->RemoveFrame(this);
+ }
+ }
+ }
+
+ imgIRequest* oldBorderImage =
+ aOldComputedStyle
+ ? aOldComputedStyle->StyleBorder()->GetBorderImageRequest()
+ : nullptr;
+ imgIRequest* newBorderImage = StyleBorder()->GetBorderImageRequest();
+ // FIXME (Bug 759996): The following is no longer true.
+ // For border-images, we can't be as conservative (we need to set the
+ // new loaders if there has been any change) since the CalcDifference
+ // call depended on the result of GetComputedBorder() and that result
+ // depends on whether the image has loaded, start the image load now
+ // so that we'll get notified when it completes loading and can do a
+ // restyle. Otherwise, the image might finish loading from the
+ // network before we start listening to its notifications, and then
+ // we'll never know that it's finished loading. Likewise, we want to
+ // do this for freshly-created frames to prevent a similar race if the
+ // image loads between reflow (which can depend on whether the image
+ // is loaded) and paint. We also don't really care about any callers who try
+ // to paint borders with a different style, because they won't have the
+ // correct size for the border either.
+ if (oldBorderImage != newBorderImage) {
+ // stop and restart the image loading/notification
+ if (oldBorderImage && HasImageRequest()) {
+ loader->DisassociateRequestFromFrame(oldBorderImage, this);
+ }
+ if (newBorderImage) {
+ loader->AssociateRequestToFrame(newBorderImage, this);
+ }
+ }
+
+ auto GetShapeImageRequest = [](const ComputedStyle* aStyle) -> imgIRequest* {
+ if (!aStyle) {
+ return nullptr;
+ }
+ auto& shape = aStyle->StyleDisplay()->mShapeOutside;
+ if (!shape.IsImage()) {
+ return nullptr;
+ }
+ return shape.AsImage().GetImageRequest();
+ };
+
+ imgIRequest* oldShapeImage = GetShapeImageRequest(aOldComputedStyle);
+ imgIRequest* newShapeImage = GetShapeImageRequest(Style());
+ if (oldShapeImage != newShapeImage) {
+ if (oldShapeImage && HasImageRequest()) {
+ loader->DisassociateRequestFromFrame(oldShapeImage, this);
+ }
+ if (newShapeImage) {
+ loader->AssociateRequestToFrame(
+ newShapeImage, this,
+ ImageLoader::Flags::
+ RequiresReflowOnFirstFrameCompleteAndLoadEventBlocking);
+ }
+ }
+
+ // SVGObserverUtils::GetEffectProperties() asserts that we only invoke it with
+ // the first continuation so we need to check that in advance.
+ const bool isNonTextFirstContinuation = isNonText && !GetPrevContinuation();
+ if (isNonTextFirstContinuation) {
+ // Kick off loading of external SVG resources referenced from properties if
+ // any. This currently includes filter, clip-path, and mask.
+ SVGObserverUtils::InitiateResourceDocLoads(this);
+ }
+
+ // If the page contains markup that overrides text direction, and
+ // does not contain any characters that would activate the Unicode
+ // bidi algorithm, we need to call |SetBidiEnabled| on the pres
+ // context before reflow starts. See bug 115921.
+ if (StyleVisibility()->mDirection == StyleDirection::Rtl) {
+ PresContext()->SetBidiEnabled();
+ }
+
+ // The following part is for caching offset-path:path(). We cache the
+ // flatten gfx path, so we don't have to rebuild and re-flattern it at
+ // each cycle if we have animations on offset-* with a fixed offset-path.
+ const StyleOffsetPath* oldPath =
+ aOldComputedStyle ? &aOldComputedStyle->StyleDisplay()->mOffsetPath
+ : nullptr;
+ const StyleOffsetPath& newPath = StyleDisplay()->mOffsetPath;
+ if (!oldPath || *oldPath != newPath) {
+ // FIXME: Bug 1837042. Cache all basic shapes.
+ if (newPath.IsPath()) {
+ RefPtr<gfx::PathBuilder> builder = MotionPathUtils::GetPathBuilder();
+ RefPtr<gfx::Path> path =
+ MotionPathUtils::BuildSVGPath(newPath.AsSVGPathData(), builder);
+ if (path) {
+ // The newPath could be path('') (i.e. empty path), so its gfx path
+ // could be nullptr, and so we only set property for a non-empty path.
+ SetProperty(nsIFrame::OffsetPathCache(), path.forget().take());
+ } else {
+ // May have an old cached path, so we have to delete it.
+ RemoveProperty(nsIFrame::OffsetPathCache());
+ }
+ } else if (oldPath) {
+ RemoveProperty(nsIFrame::OffsetPathCache());
+ }
+ }
+
+ if (IsPrimaryFrame()) {
+ MOZ_ASSERT(aOldComputedStyle);
+ HandlePrimaryFrameStyleChange(aOldComputedStyle);
+ }
+
+ RemoveStateBits(NS_FRAME_SIMPLE_EVENT_REGIONS | NS_FRAME_SIMPLE_DISPLAYLIST);
+
+ mMayHaveRoundedCorners = true;
+}
+
+void nsIFrame::HandleLastRememberedSize() {
+ MOZ_ASSERT(IsPrimaryFrame());
+ // Storing a last remembered size requires contain-intrinsic-size, and using
+ // a previously stored last remembered size requires content-visibility.
+ if (!StaticPrefs::layout_css_contain_intrinsic_size_enabled() ||
+ !StaticPrefs::layout_css_content_visibility_enabled()) {
+ return;
+ }
+ auto* element = Element::FromNodeOrNull(mContent);
+ if (!element) {
+ return;
+ }
+ const WritingMode wm = GetWritingMode();
+ const nsStylePosition* stylePos = StylePosition();
+ bool canRememberBSize = stylePos->ContainIntrinsicBSize(wm).HasAuto();
+ bool canRememberISize = stylePos->ContainIntrinsicISize(wm).HasAuto();
+ if (!canRememberBSize) {
+ element->RemoveLastRememberedBSize();
+ }
+ if (!canRememberISize) {
+ element->RemoveLastRememberedISize();
+ }
+ if ((canRememberBSize || canRememberISize) && !HidesContent()) {
+ bool isNonReplacedInline = IsLineParticipant() && !IsReplaced();
+ if (!isNonReplacedInline) {
+ PresContext()->Document()->ObserveForLastRememberedSize(*element);
+ return;
+ }
+ }
+ PresContext()->Document()->UnobserveForLastRememberedSize(*element);
+}
+
+#ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
+void nsIFrame::AssertNewStyleIsSane(ComputedStyle& aNewStyle) {
+ MOZ_DIAGNOSTIC_ASSERT(
+ aNewStyle.GetPseudoType() == mComputedStyle->GetPseudoType() ||
+ // ::first-line continuations are weird, this should probably be fixed via
+ // bug 1465474.
+ (mComputedStyle->GetPseudoType() == PseudoStyleType::firstLine &&
+ aNewStyle.GetPseudoType() == PseudoStyleType::mozLineFrame) ||
+ // ::first-letter continuations are broken, in particular floating ones,
+ // see bug 1490281. The construction code tries to fix this up after the
+ // fact, then restyling undoes it...
+ (mComputedStyle->GetPseudoType() == PseudoStyleType::mozText &&
+ aNewStyle.GetPseudoType() == PseudoStyleType::firstLetterContinuation) ||
+ (mComputedStyle->GetPseudoType() ==
+ PseudoStyleType::firstLetterContinuation &&
+ aNewStyle.GetPseudoType() == PseudoStyleType::mozText));
+}
+#endif
+
+void nsIFrame::ReparentFrameViewTo(nsViewManager* aViewManager,
+ nsView* aNewParentView) {
+ if (HasView()) {
+ if (IsMenuPopupFrame()) {
+ // This view must be parented by the root view, don't reparent it.
+ return;
+ }
+ nsView* view = GetView();
+ aViewManager->RemoveChild(view);
+
+ // The view will remember the Z-order and other attributes that have been
+ // set on it.
+ nsView* insertBefore =
+ nsLayoutUtils::FindSiblingViewFor(aNewParentView, this);
+ aViewManager->InsertChild(aNewParentView, view, insertBefore,
+ insertBefore != nullptr);
+ } else if (HasAnyStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW)) {
+ for (const auto& childList : ChildLists()) {
+ // Iterate the child frames, and check each child frame to see if it has
+ // a view
+ for (nsIFrame* child : childList.mList) {
+ child->ReparentFrameViewTo(aViewManager, aNewParentView);
+ }
+ }
+ }
+}
+
+void nsIFrame::SyncFrameViewProperties(nsView* aView) {
+ if (!aView) {
+ aView = GetView();
+ if (!aView) {
+ return;
+ }
+ }
+
+ nsViewManager* vm = aView->GetViewManager();
+
+ // Make sure visibility is correct. This only affects nsSubDocumentFrame.
+ if (!SupportsVisibilityHidden()) {
+ // See if the view should be hidden or visible
+ ComputedStyle* sc = Style();
+ vm->SetViewVisibility(aView, sc->StyleVisibility()->IsVisible()
+ ? ViewVisibility::Show
+ : ViewVisibility::Hide);
+ }
+
+ const auto zIndex = ZIndex();
+ const bool autoZIndex = !zIndex;
+ vm->SetViewZIndex(aView, autoZIndex, zIndex.valueOr(0));
+}
+
+void nsIFrame::CreateView() {
+ MOZ_ASSERT(!HasView());
+
+ nsView* parentView = GetParent()->GetClosestView();
+ MOZ_ASSERT(parentView, "no parent with view");
+
+ nsViewManager* viewManager = parentView->GetViewManager();
+ MOZ_ASSERT(viewManager, "null view manager");
+
+ nsView* view = viewManager->CreateView(GetRect(), parentView);
+ SyncFrameViewProperties(view);
+
+ nsView* insertBefore = nsLayoutUtils::FindSiblingViewFor(parentView, this);
+ // we insert this view 'above' the insertBefore view, unless insertBefore is
+ // null, in which case we want to call with aAbove == false to insert at the
+ // beginning in document order
+ viewManager->InsertChild(parentView, view, insertBefore,
+ insertBefore != nullptr);
+
+ // REVIEW: Don't create a widget for fixed-pos elements anymore.
+ // ComputeRepaintRegionForCopy will calculate the right area to repaint
+ // when we scroll.
+ // Reparent views on any child frames (or their descendants) to this
+ // view. We can just call ReparentFrameViewTo on this frame because
+ // we know this frame has no view, so it will crawl the children. Also,
+ // we know that any descendants with views must have 'parentView' as their
+ // parent view.
+ ReparentFrameViewTo(viewManager, view);
+
+ // Remember our view
+ SetView(view);
+
+ NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
+ ("nsIFrame::CreateView: frame=%p view=%p", this, view));
+}
+
+/* virtual */
+nsMargin nsIFrame::GetUsedMargin() const {
+ nsMargin margin;
+ if (((mState & NS_FRAME_FIRST_REFLOW) && !(mState & NS_FRAME_IN_REFLOW)) ||
+ IsInSVGTextSubtree()) {
+ return margin;
+ }
+
+ if (nsMargin* m = GetProperty(UsedMarginProperty())) {
+ margin = *m;
+ } else if (!StyleMargin()->GetMargin(margin)) {
+ // If we get here, our caller probably shouldn't be calling us...
+ NS_ERROR(
+ "Returning bogus 0-sized margin, because this margin "
+ "depends on layout & isn't cached!");
+ }
+ return margin;
+}
+
+/* virtual */
+nsMargin nsIFrame::GetUsedBorder() const {
+ if (((mState & NS_FRAME_FIRST_REFLOW) && !(mState & NS_FRAME_IN_REFLOW)) ||
+ IsInSVGTextSubtree()) {
+ return {};
+ }
+
+ const nsStyleDisplay* disp = StyleDisplay();
+ if (IsThemed(disp)) {
+ // Theme methods don't use const-ness.
+ auto* mutable_this = const_cast<nsIFrame*>(this);
+ nsPresContext* pc = PresContext();
+ LayoutDeviceIntMargin widgetBorder = pc->Theme()->GetWidgetBorder(
+ pc->DeviceContext(), mutable_this, disp->EffectiveAppearance());
+ return LayoutDevicePixel::ToAppUnits(widgetBorder,
+ pc->AppUnitsPerDevPixel());
+ }
+
+ return StyleBorder()->GetComputedBorder();
+}
+
+/* virtual */
+nsMargin nsIFrame::GetUsedPadding() const {
+ nsMargin padding;
+ if (((mState & NS_FRAME_FIRST_REFLOW) && !(mState & NS_FRAME_IN_REFLOW)) ||
+ IsInSVGTextSubtree()) {
+ return padding;
+ }
+
+ const nsStyleDisplay* disp = StyleDisplay();
+ if (IsThemed(disp)) {
+ // Theme methods don't use const-ness.
+ nsIFrame* mutable_this = const_cast<nsIFrame*>(this);
+ nsPresContext* pc = PresContext();
+ LayoutDeviceIntMargin widgetPadding;
+ if (pc->Theme()->GetWidgetPadding(pc->DeviceContext(), mutable_this,
+ disp->EffectiveAppearance(),
+ &widgetPadding)) {
+ return LayoutDevicePixel::ToAppUnits(widgetPadding,
+ pc->AppUnitsPerDevPixel());
+ }
+ }
+
+ if (nsMargin* p = GetProperty(UsedPaddingProperty())) {
+ padding = *p;
+ } else if (!StylePadding()->GetPadding(padding)) {
+ // If we get here, our caller probably shouldn't be calling us...
+ NS_ERROR(
+ "Returning bogus 0-sized padding, because this padding "
+ "depends on layout & isn't cached!");
+ }
+ return padding;
+}
+
+nsIFrame::Sides nsIFrame::GetSkipSides() const {
+ if (MOZ_UNLIKELY(StyleBorder()->mBoxDecorationBreak ==
+ StyleBoxDecorationBreak::Clone) &&
+ !HasAnyStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER)) {
+ return Sides();
+ }
+
+ // Convert the logical skip sides to physical sides using the frame's
+ // writing mode
+ WritingMode writingMode = GetWritingMode();
+ LogicalSides logicalSkip = GetLogicalSkipSides();
+ Sides skip;
+
+ if (logicalSkip.BStart()) {
+ if (writingMode.IsVertical()) {
+ skip |= writingMode.IsVerticalLR() ? SideBits::eLeft : SideBits::eRight;
+ } else {
+ skip |= SideBits::eTop;
+ }
+ }
+
+ if (logicalSkip.BEnd()) {
+ if (writingMode.IsVertical()) {
+ skip |= writingMode.IsVerticalLR() ? SideBits::eRight : SideBits::eLeft;
+ } else {
+ skip |= SideBits::eBottom;
+ }
+ }
+
+ if (logicalSkip.IStart()) {
+ if (writingMode.IsVertical()) {
+ skip |= SideBits::eTop;
+ } else {
+ skip |= writingMode.IsBidiLTR() ? SideBits::eLeft : SideBits::eRight;
+ }
+ }
+
+ if (logicalSkip.IEnd()) {
+ if (writingMode.IsVertical()) {
+ skip |= SideBits::eBottom;
+ } else {
+ skip |= writingMode.IsBidiLTR() ? SideBits::eRight : SideBits::eLeft;
+ }
+ }
+ return skip;
+}
+
+nsRect nsIFrame::GetPaddingRectRelativeToSelf() const {
+ nsMargin border = GetUsedBorder().ApplySkipSides(GetSkipSides());
+ nsRect r(0, 0, mRect.width, mRect.height);
+ r.Deflate(border);
+ return r;
+}
+
+nsRect nsIFrame::GetPaddingRect() const {
+ return GetPaddingRectRelativeToSelf() + GetPosition();
+}
+
+WritingMode nsIFrame::WritingModeForLine(WritingMode aSelfWM,
+ nsIFrame* aSubFrame) const {
+ MOZ_ASSERT(aSelfWM == GetWritingMode());
+ WritingMode writingMode = aSelfWM;
+
+ if (StyleTextReset()->mUnicodeBidi == StyleUnicodeBidi::Plaintext) {
+ mozilla::intl::BidiEmbeddingLevel frameLevel =
+ nsBidiPresUtils::GetFrameBaseLevel(aSubFrame);
+ writingMode.SetDirectionFromBidiLevel(frameLevel);
+ }
+
+ return writingMode;
+}
+
+nsRect nsIFrame::GetMarginRect() const {
+ return GetMarginRectRelativeToSelf() + GetPosition();
+}
+
+nsRect nsIFrame::GetMarginRectRelativeToSelf() const {
+ nsMargin m = GetUsedMargin().ApplySkipSides(GetSkipSides());
+ nsRect r(0, 0, mRect.width, mRect.height);
+ r.Inflate(m);
+ return r;
+}
+
+bool nsIFrame::IsTransformed() const {
+ if (!HasAnyStateBits(NS_FRAME_MAY_BE_TRANSFORMED)) {
+ MOZ_ASSERT(!IsCSSTransformed());
+ MOZ_ASSERT(!IsSVGTransformed());
+ return false;
+ }
+ return IsCSSTransformed() || IsSVGTransformed();
+}
+
+bool nsIFrame::IsCSSTransformed() const {
+ return HasAnyStateBits(NS_FRAME_MAY_BE_TRANSFORMED) &&
+ (StyleDisplay()->HasTransform(this) || HasAnimationOfTransform());
+}
+
+bool nsIFrame::HasAnimationOfTransform() const {
+ return IsPrimaryFrame() &&
+ nsLayoutUtils::HasAnimationOfTransformAndMotionPath(this) &&
+ SupportsCSSTransforms();
+}
+
+bool nsIFrame::ChildrenHavePerspective(
+ const nsStyleDisplay* aStyleDisplay) const {
+ MOZ_ASSERT(aStyleDisplay == StyleDisplay());
+ return aStyleDisplay->HasPerspective(this);
+}
+
+bool nsIFrame::HasAnimationOfOpacity(EffectSet* aEffectSet) const {
+ return ((nsLayoutUtils::IsPrimaryStyleFrame(this) ||
+ nsLayoutUtils::FirstContinuationOrIBSplitSibling(this)
+ ->IsPrimaryFrame()) &&
+ nsLayoutUtils::HasAnimationOfPropertySet(
+ this, nsCSSPropertyIDSet::OpacityProperties(), aEffectSet));
+}
+
+bool nsIFrame::HasOpacityInternal(float aThreshold,
+ const nsStyleDisplay* aStyleDisplay,
+ const nsStyleEffects* aStyleEffects,
+ EffectSet* aEffectSet) const {
+ MOZ_ASSERT(0.0 <= aThreshold && aThreshold <= 1.0, "Invalid argument");
+ if (aStyleEffects->mOpacity < aThreshold ||
+ aStyleDisplay->mWillChange.bits & StyleWillChangeBits::OPACITY) {
+ return true;
+ }
+
+ if (!mMayHaveOpacityAnimation) {
+ return false;
+ }
+
+ return HasAnimationOfOpacity(aEffectSet);
+}
+
+bool nsIFrame::IsSVGTransformed(gfx::Matrix* aOwnTransforms,
+ gfx::Matrix* aFromParentTransforms) const {
+ return false;
+}
+
+bool nsIFrame::Extend3DContext(const nsStyleDisplay* aStyleDisplay,
+ const nsStyleEffects* aStyleEffects,
+ mozilla::EffectSet* aEffectSetForOpacity) const {
+ if (!HasAnyStateBits(NS_FRAME_MAY_BE_TRANSFORMED)) {
+ return false;
+ }
+ const nsStyleDisplay* disp = StyleDisplayWithOptionalParam(aStyleDisplay);
+ if (disp->mTransformStyle != StyleTransformStyle::Preserve3d ||
+ !SupportsCSSTransforms()) {
+ return false;
+ }
+
+ // If we're all scroll frame, then all descendants will be clipped, so we
+ // can't preserve 3d.
+ if (IsScrollFrame()) {
+ return false;
+ }
+
+ const nsStyleEffects* effects = StyleEffectsWithOptionalParam(aStyleEffects);
+ if (HasOpacity(disp, effects, aEffectSetForOpacity)) {
+ return false;
+ }
+
+ return ShouldApplyOverflowClipping(disp) == PhysicalAxes::None &&
+ !GetClipPropClipRect(disp, effects, GetSize()) &&
+ !SVGIntegrationUtils::UsingEffectsForFrame(this) &&
+ !effects->HasMixBlendMode() &&
+ disp->mIsolation != StyleIsolation::Isolate;
+}
+
+bool nsIFrame::Combines3DTransformWithAncestors() const {
+ // Check these first as they are faster then both calls below and are we are
+ // likely to hit the early return (backface hidden is uncommon and
+ // GetReferenceFrame is a hot caller of this which only calls this if
+ // IsCSSTransformed is false).
+ if (!IsCSSTransformed() && !BackfaceIsHidden()) {
+ return false;
+ }
+ nsIFrame* parent = GetClosestFlattenedTreeAncestorPrimaryFrame();
+ return parent && parent->Extend3DContext();
+}
+
+bool nsIFrame::In3DContextAndBackfaceIsHidden() const {
+ // While both tests fail most of the time, test BackfaceIsHidden()
+ // first since it's likely to fail faster.
+ return BackfaceIsHidden() && Combines3DTransformWithAncestors();
+}
+
+bool nsIFrame::HasPerspective() const {
+ if (!IsCSSTransformed()) {
+ return false;
+ }
+ nsIFrame* parent = GetClosestFlattenedTreeAncestorPrimaryFrame();
+ if (!parent) {
+ return false;
+ }
+ return parent->ChildrenHavePerspective();
+}
+
+nsRect nsIFrame::GetContentRectRelativeToSelf() const {
+ nsMargin bp = GetUsedBorderAndPadding().ApplySkipSides(GetSkipSides());
+ nsRect r(0, 0, mRect.width, mRect.height);
+ r.Deflate(bp);
+ return r;
+}
+
+nsRect nsIFrame::GetContentRect() const {
+ return GetContentRectRelativeToSelf() + GetPosition();
+}
+
+bool nsIFrame::ComputeBorderRadii(const BorderRadius& aBorderRadius,
+ const nsSize& aFrameSize,
+ const nsSize& aBorderArea, Sides aSkipSides,
+ nscoord aRadii[8]) {
+ // Percentages are relative to whichever side they're on.
+ for (const auto i : mozilla::AllPhysicalHalfCorners()) {
+ const LengthPercentage& c = aBorderRadius.Get(i);
+ nscoord axis = HalfCornerIsX(i) ? aFrameSize.width : aFrameSize.height;
+ aRadii[i] = std::max(0, c.Resolve(axis));
+ }
+
+ if (aSkipSides.Top()) {
+ aRadii[eCornerTopLeftX] = 0;
+ aRadii[eCornerTopLeftY] = 0;
+ aRadii[eCornerTopRightX] = 0;
+ aRadii[eCornerTopRightY] = 0;
+ }
+
+ if (aSkipSides.Right()) {
+ aRadii[eCornerTopRightX] = 0;
+ aRadii[eCornerTopRightY] = 0;
+ aRadii[eCornerBottomRightX] = 0;
+ aRadii[eCornerBottomRightY] = 0;
+ }
+
+ if (aSkipSides.Bottom()) {
+ aRadii[eCornerBottomRightX] = 0;
+ aRadii[eCornerBottomRightY] = 0;
+ aRadii[eCornerBottomLeftX] = 0;
+ aRadii[eCornerBottomLeftY] = 0;
+ }
+
+ if (aSkipSides.Left()) {
+ aRadii[eCornerBottomLeftX] = 0;
+ aRadii[eCornerBottomLeftY] = 0;
+ aRadii[eCornerTopLeftX] = 0;
+ aRadii[eCornerTopLeftY] = 0;
+ }
+
+ // css3-background specifies this algorithm for reducing
+ // corner radii when they are too big.
+ bool haveRadius = false;
+ double ratio = 1.0f;
+ for (const auto side : mozilla::AllPhysicalSides()) {
+ uint32_t hc1 = SideToHalfCorner(side, false, true);
+ uint32_t hc2 = SideToHalfCorner(side, true, true);
+ nscoord length =
+ SideIsVertical(side) ? aBorderArea.height : aBorderArea.width;
+ nscoord sum = aRadii[hc1] + aRadii[hc2];
+ if (sum) {
+ haveRadius = true;
+ // avoid floating point division in the normal case
+ if (length < sum) {
+ ratio = std::min(ratio, double(length) / sum);
+ }
+ }
+ }
+ if (ratio < 1.0) {
+ for (const auto corner : mozilla::AllPhysicalHalfCorners()) {
+ aRadii[corner] *= ratio;
+ }
+ }
+
+ return haveRadius;
+}
+
+void nsIFrame::AdjustBorderRadii(nscoord aRadii[8], const nsMargin& aOffsets) {
+ auto AdjustOffset = [](const uint32_t aRadius, const nscoord aOffset) {
+ // Implement the cubic formula to adjust offset when aOffset > 0 and
+ // aRadius / aOffset < 1.
+ // https://drafts.csswg.org/css-shapes/#valdef-shape-box-margin-box
+ if (aOffset > 0) {
+ const double ratio = aRadius / double(aOffset);
+ if (ratio < 1.0) {
+ return nscoord(aOffset * (1.0 + std::pow(ratio - 1, 3)));
+ }
+ }
+ return aOffset;
+ };
+
+ for (const auto side : mozilla::AllPhysicalSides()) {
+ const nscoord offset = aOffsets.Side(side);
+ const uint32_t hc1 = SideToHalfCorner(side, false, false);
+ const uint32_t hc2 = SideToHalfCorner(side, true, false);
+ if (aRadii[hc1] > 0) {
+ const nscoord offset1 = AdjustOffset(aRadii[hc1], offset);
+ aRadii[hc1] = std::max(0, aRadii[hc1] + offset1);
+ }
+ if (aRadii[hc2] > 0) {
+ const nscoord offset2 = AdjustOffset(aRadii[hc2], offset);
+ aRadii[hc2] = std::max(0, aRadii[hc2] + offset2);
+ }
+ }
+}
+
+static inline bool RadiiAreDefinitelyZero(const BorderRadius& aBorderRadius) {
+ for (const auto corner : mozilla::AllPhysicalHalfCorners()) {
+ if (!aBorderRadius.Get(corner).IsDefinitelyZero()) {
+ return false;
+ }
+ }
+ return true;
+}
+
+/* virtual */
+bool nsIFrame::GetBorderRadii(const nsSize& aFrameSize,
+ const nsSize& aBorderArea, Sides aSkipSides,
+ nscoord aRadii[8]) const {
+ if (!mMayHaveRoundedCorners) {
+ memset(aRadii, 0, sizeof(nscoord) * 8);
+ return false;
+ }
+
+ if (IsThemed()) {
+ // When we're themed, the native theme code draws the border and
+ // background, and therefore it doesn't make sense to tell other
+ // code that's interested in border-radius that we have any radii.
+ //
+ // In an ideal world, we might have a way for the them to tell us an
+ // border radius, but since we don't, we're better off assuming
+ // zero.
+ for (const auto corner : mozilla::AllPhysicalHalfCorners()) {
+ aRadii[corner] = 0;
+ }
+ return false;
+ }
+
+ const auto& radii = StyleBorder()->mBorderRadius;
+ const bool hasRadii =
+ ComputeBorderRadii(radii, aFrameSize, aBorderArea, aSkipSides, aRadii);
+ if (!hasRadii) {
+ // TODO(emilio): Maybe we can just remove this bit and do the
+ // IsDefinitelyZero check unconditionally. That should still avoid most of
+ // the work, though maybe not the cache miss of going through the style and
+ // the border struct.
+ const_cast<nsIFrame*>(this)->mMayHaveRoundedCorners =
+ !RadiiAreDefinitelyZero(radii);
+ }
+ return hasRadii;
+}
+
+bool nsIFrame::GetBorderRadii(nscoord aRadii[8]) const {
+ nsSize sz = GetSize();
+ return GetBorderRadii(sz, sz, GetSkipSides(), aRadii);
+}
+
+bool nsIFrame::GetMarginBoxBorderRadii(nscoord aRadii[8]) const {
+ return GetBoxBorderRadii(aRadii, GetUsedMargin());
+}
+
+bool nsIFrame::GetPaddingBoxBorderRadii(nscoord aRadii[8]) const {
+ return GetBoxBorderRadii(aRadii, -GetUsedBorder());
+}
+
+bool nsIFrame::GetContentBoxBorderRadii(nscoord aRadii[8]) const {
+ return GetBoxBorderRadii(aRadii, -GetUsedBorderAndPadding());
+}
+
+bool nsIFrame::GetBoxBorderRadii(nscoord aRadii[8],
+ const nsMargin& aOffsets) const {
+ if (!GetBorderRadii(aRadii)) {
+ return false;
+ }
+ AdjustBorderRadii(aRadii, aOffsets);
+ for (const auto corner : mozilla::AllPhysicalHalfCorners()) {
+ if (aRadii[corner]) {
+ return true;
+ }
+ }
+ return false;
+}
+
+bool nsIFrame::GetShapeBoxBorderRadii(nscoord aRadii[8]) const {
+ using Tag = StyleShapeOutside::Tag;
+ auto& shapeOutside = StyleDisplay()->mShapeOutside;
+ auto box = StyleShapeBox::MarginBox;
+ switch (shapeOutside.tag) {
+ case Tag::Image:
+ case Tag::None:
+ return false;
+ case Tag::Box:
+ box = shapeOutside.AsBox();
+ break;
+ case Tag::Shape:
+ box = shapeOutside.AsShape()._1;
+ break;
+ }
+
+ switch (box) {
+ case StyleShapeBox::ContentBox:
+ return GetContentBoxBorderRadii(aRadii);
+ case StyleShapeBox::PaddingBox:
+ return GetPaddingBoxBorderRadii(aRadii);
+ case StyleShapeBox::BorderBox:
+ return GetBorderRadii(aRadii);
+ case StyleShapeBox::MarginBox:
+ return GetMarginBoxBorderRadii(aRadii);
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unexpected box value");
+ return false;
+ }
+}
+
+ComputedStyle* nsIFrame::GetAdditionalComputedStyle(int32_t aIndex) const {
+ MOZ_ASSERT(aIndex >= 0, "invalid index number");
+ return nullptr;
+}
+
+void nsIFrame::SetAdditionalComputedStyle(int32_t aIndex,
+ ComputedStyle* aComputedStyle) {
+ MOZ_ASSERT(aIndex >= 0, "invalid index number");
+}
+
+nscoord nsIFrame::SynthesizeFallbackBaseline(
+ WritingMode aWM, BaselineSharingGroup aBaselineGroup) const {
+ const auto margin = GetLogicalUsedMargin(aWM);
+ NS_ASSERTION(!IsSubtreeDirty(), "frame must not be dirty");
+ if (aWM.IsCentralBaseline()) {
+ return (BSize(aWM) + GetLogicalUsedMargin(aWM).BEnd(aWM)) / 2;
+ }
+ // Baseline for inverted line content is the top (block-start) margin edge,
+ // as the frame is in effect "flipped" for alignment purposes.
+ if (aWM.IsLineInverted()) {
+ const auto marginStart = margin.BStart(aWM);
+ return aBaselineGroup == BaselineSharingGroup::First
+ ? -marginStart
+ : BSize(aWM) + marginStart;
+ }
+ // Otherwise, the bottom margin edge, per CSS2.1's definition of the
+ // 'baseline' value of 'vertical-align'.
+ const auto marginEnd = margin.BEnd(aWM);
+ return aBaselineGroup == BaselineSharingGroup::First ? BSize(aWM) + marginEnd
+ : -marginEnd;
+}
+
+nscoord nsIFrame::GetLogicalBaseline(WritingMode aWM) const {
+ return GetLogicalBaseline(aWM, GetDefaultBaselineSharingGroup(),
+ BaselineExportContext::LineLayout);
+}
+
+nscoord nsIFrame::GetLogicalBaseline(
+ WritingMode aWM, BaselineSharingGroup aBaselineGroup,
+ BaselineExportContext aExportContext) const {
+ const auto result =
+ GetNaturalBaselineBOffset(aWM, aBaselineGroup, aExportContext)
+ .valueOrFrom([this, aWM, aBaselineGroup]() {
+ return SynthesizeFallbackBaseline(aWM, aBaselineGroup);
+ });
+ if (aBaselineGroup == BaselineSharingGroup::Last) {
+ return BSize(aWM) - result;
+ }
+ return result;
+}
+
+const nsFrameList& nsIFrame::GetChildList(ChildListID aListID) const {
+ if (IsAbsoluteContainer() && aListID == GetAbsoluteListID()) {
+ return GetAbsoluteContainingBlock()->GetChildList();
+ } else {
+ return nsFrameList::EmptyList();
+ }
+}
+
+void nsIFrame::GetChildLists(nsTArray<ChildList>* aLists) const {
+ if (IsAbsoluteContainer()) {
+ const nsFrameList& absoluteList =
+ GetAbsoluteContainingBlock()->GetChildList();
+ absoluteList.AppendIfNonempty(aLists, GetAbsoluteListID());
+ }
+}
+
+AutoTArray<nsIFrame::ChildList, 4> nsIFrame::CrossDocChildLists() {
+ AutoTArray<ChildList, 4> childLists;
+ nsSubDocumentFrame* subdocumentFrame = do_QueryFrame(this);
+ if (subdocumentFrame) {
+ // Descend into the subdocument
+ nsIFrame* root = subdocumentFrame->GetSubdocumentRootFrame();
+ if (root) {
+ childLists.EmplaceBack(
+ nsFrameList(root, nsLayoutUtils::GetLastSibling(root)),
+ FrameChildListID::Principal);
+ }
+ }
+
+ GetChildLists(&childLists);
+ return childLists;
+}
+
+nsIFrame::CaretBlockAxisMetrics nsIFrame::GetCaretBlockAxisMetrics(
+ mozilla::WritingMode aWM, const nsFontMetrics& aFM) const {
+ // Note(dshin): Ultimately, this does something highly similar (But still
+ // different) to `nsLayoutUtils::GetFirstLinePosition`.
+ const auto baseline = GetCaretBaseline();
+ nscoord ascent = 0, descent = 0;
+ ascent = aFM.MaxAscent();
+ descent = aFM.MaxDescent();
+ const nscoord height = ascent + descent;
+ if (aWM.IsVertical() && aWM.IsLineInverted()) {
+ return CaretBlockAxisMetrics{.mOffset = baseline - descent,
+ .mExtent = height};
+ }
+ return CaretBlockAxisMetrics{.mOffset = baseline - ascent, .mExtent = height};
+}
+
+const nsAtom* nsIFrame::ComputePageValue() const {
+ const nsAtom* value = nsGkAtoms::_empty;
+ const nsIFrame* frame = this;
+ // Find what CSS page name value this frame's subtree has, if any.
+ // Starting with this frame, check if a page name other than auto is present,
+ // and record it if so. Then, if the current frame is a container frame, find
+ // the first non-placeholder child and repeat.
+ // This will find the most deeply nested first in-flow child of this frame's
+ // subtree, and return its page name (with auto resolved if applicable, and
+ // subtrees with no page-names returning the empty atom rather than null).
+ do {
+ if (const nsAtom* maybePageName = frame->GetStylePageName()) {
+ value = maybePageName;
+ }
+ // Get the next frame to read from.
+ const nsIFrame* firstNonPlaceholderFrame = nullptr;
+ // If this is a container frame, inspect its in-flow children.
+ if (const nsContainerFrame* containerFrame = do_QueryFrame(frame)) {
+ for (const nsIFrame* childFrame : containerFrame->PrincipalChildList()) {
+ if (!childFrame->IsPlaceholderFrame()) {
+ firstNonPlaceholderFrame = childFrame;
+ break;
+ }
+ }
+ }
+ frame = firstNonPlaceholderFrame;
+ } while (frame);
+ return value;
+}
+
+Visibility nsIFrame::GetVisibility() const {
+ if (!HasAnyStateBits(NS_FRAME_VISIBILITY_IS_TRACKED)) {
+ return Visibility::Untracked;
+ }
+
+ bool isSet = false;
+ uint32_t visibleCount = GetProperty(VisibilityStateProperty(), &isSet);
+
+ MOZ_ASSERT(isSet,
+ "Should have a VisibilityStateProperty value "
+ "if NS_FRAME_VISIBILITY_IS_TRACKED is set");
+
+ return visibleCount > 0 ? Visibility::ApproximatelyVisible
+ : Visibility::ApproximatelyNonVisible;
+}
+
+void nsIFrame::UpdateVisibilitySynchronously() {
+ mozilla::PresShell* presShell = PresShell();
+ if (!presShell) {
+ return;
+ }
+
+ if (presShell->AssumeAllFramesVisible()) {
+ presShell->EnsureFrameInApproximatelyVisibleList(this);
+ return;
+ }
+
+ bool visible = StyleVisibility()->IsVisible();
+ nsIFrame* f = GetParent();
+ nsRect rect = GetRectRelativeToSelf();
+ nsIFrame* rectFrame = this;
+ while (f && visible) {
+ nsIScrollableFrame* sf = do_QueryFrame(f);
+ if (sf) {
+ nsRect transformedRect =
+ nsLayoutUtils::TransformFrameRectToAncestor(rectFrame, rect, f);
+ if (!sf->IsRectNearlyVisible(transformedRect)) {
+ visible = false;
+ break;
+ }
+
+ // In this code we're trying to synchronously update *approximate*
+ // visibility. (In the future we may update precise visibility here as
+ // well, which is why the method name does not contain 'approximate'.) The
+ // IsRectNearlyVisible() check above tells us that the rect we're checking
+ // is approximately visible within the scrollframe, but we still need to
+ // ensure that, even if it was scrolled into view, it'd be visible when we
+ // consider the rest of the document. To do that, we move transformedRect
+ // to be contained in the scrollport as best we can (it might not fit) to
+ // pretend that it was scrolled into view.
+ rect = transformedRect.MoveInsideAndClamp(sf->GetScrollPortRect());
+ rectFrame = f;
+ }
+ nsIFrame* parent = f->GetParent();
+ if (!parent) {
+ parent = nsLayoutUtils::GetCrossDocParentFrameInProcess(f);
+ if (parent && parent->PresContext()->IsChrome()) {
+ break;
+ }
+ }
+ f = parent;
+ }
+
+ if (visible) {
+ presShell->EnsureFrameInApproximatelyVisibleList(this);
+ } else {
+ presShell->RemoveFrameFromApproximatelyVisibleList(this);
+ }
+}
+
+void nsIFrame::EnableVisibilityTracking() {
+ if (HasAnyStateBits(NS_FRAME_VISIBILITY_IS_TRACKED)) {
+ return; // Nothing to do.
+ }
+
+ MOZ_ASSERT(!HasProperty(VisibilityStateProperty()),
+ "Shouldn't have a VisibilityStateProperty value "
+ "if NS_FRAME_VISIBILITY_IS_TRACKED is not set");
+
+ // Add the state bit so we know to track visibility for this frame, and
+ // initialize the frame property.
+ AddStateBits(NS_FRAME_VISIBILITY_IS_TRACKED);
+ SetProperty(VisibilityStateProperty(), 0);
+
+ mozilla::PresShell* presShell = PresShell();
+ if (!presShell) {
+ return;
+ }
+
+ // Schedule a visibility update. This method will virtually always be called
+ // when layout has changed anyway, so it's very unlikely that any additional
+ // visibility updates will be triggered by this, but this way we guarantee
+ // that if this frame is currently visible we'll eventually find out.
+ presShell->ScheduleApproximateFrameVisibilityUpdateSoon();
+}
+
+void nsIFrame::DisableVisibilityTracking() {
+ if (!HasAnyStateBits(NS_FRAME_VISIBILITY_IS_TRACKED)) {
+ return; // Nothing to do.
+ }
+
+ bool isSet = false;
+ uint32_t visibleCount = TakeProperty(VisibilityStateProperty(), &isSet);
+
+ MOZ_ASSERT(isSet,
+ "Should have a VisibilityStateProperty value "
+ "if NS_FRAME_VISIBILITY_IS_TRACKED is set");
+
+ RemoveStateBits(NS_FRAME_VISIBILITY_IS_TRACKED);
+
+ if (visibleCount == 0) {
+ return; // We were nonvisible.
+ }
+
+ // We were visible, so send an OnVisibilityChange() notification.
+ OnVisibilityChange(Visibility::ApproximatelyNonVisible);
+}
+
+void nsIFrame::DecApproximateVisibleCount(
+ const Maybe<OnNonvisible>& aNonvisibleAction
+ /* = Nothing() */) {
+ MOZ_ASSERT(HasAnyStateBits(NS_FRAME_VISIBILITY_IS_TRACKED));
+
+ bool isSet = false;
+ uint32_t visibleCount = GetProperty(VisibilityStateProperty(), &isSet);
+
+ MOZ_ASSERT(isSet,
+ "Should have a VisibilityStateProperty value "
+ "if NS_FRAME_VISIBILITY_IS_TRACKED is set");
+ MOZ_ASSERT(visibleCount > 0,
+ "Frame is already nonvisible and we're "
+ "decrementing its visible count?");
+
+ visibleCount--;
+ SetProperty(VisibilityStateProperty(), visibleCount);
+ if (visibleCount > 0) {
+ return;
+ }
+
+ // We just became nonvisible, so send an OnVisibilityChange() notification.
+ OnVisibilityChange(Visibility::ApproximatelyNonVisible, aNonvisibleAction);
+}
+
+void nsIFrame::IncApproximateVisibleCount() {
+ MOZ_ASSERT(HasAnyStateBits(NS_FRAME_VISIBILITY_IS_TRACKED));
+
+ bool isSet = false;
+ uint32_t visibleCount = GetProperty(VisibilityStateProperty(), &isSet);
+
+ MOZ_ASSERT(isSet,
+ "Should have a VisibilityStateProperty value "
+ "if NS_FRAME_VISIBILITY_IS_TRACKED is set");
+
+ visibleCount++;
+ SetProperty(VisibilityStateProperty(), visibleCount);
+ if (visibleCount > 1) {
+ return;
+ }
+
+ // We just became visible, so send an OnVisibilityChange() notification.
+ OnVisibilityChange(Visibility::ApproximatelyVisible);
+}
+
+void nsIFrame::OnVisibilityChange(Visibility aNewVisibility,
+ const Maybe<OnNonvisible>& aNonvisibleAction
+ /* = Nothing() */) {
+ // XXX(seth): In bug 1218990 we'll implement visibility tracking for CSS
+ // images here.
+}
+
+static nsIFrame* GetActiveSelectionFrame(nsPresContext* aPresContext,
+ nsIFrame* aFrame) {
+ nsIContent* capturingContent = PresShell::GetCapturingContent();
+ if (capturingContent) {
+ nsIFrame* activeFrame = aPresContext->GetPrimaryFrameFor(capturingContent);
+ return activeFrame ? activeFrame : aFrame;
+ }
+
+ return aFrame;
+}
+
+int16_t nsIFrame::DetermineDisplaySelection() {
+ int16_t selType = nsISelectionController::SELECTION_OFF;
+
+ nsCOMPtr<nsISelectionController> selCon;
+ nsresult result =
+ GetSelectionController(PresContext(), getter_AddRefs(selCon));
+ if (NS_SUCCEEDED(result) && selCon) {
+ result = selCon->GetDisplaySelection(&selType);
+ if (NS_SUCCEEDED(result) &&
+ (selType != nsISelectionController::SELECTION_OFF)) {
+ // Check whether style allows selection.
+ if (!IsSelectable(nullptr)) {
+ selType = nsISelectionController::SELECTION_OFF;
+ }
+ }
+ }
+ return selType;
+}
+
+static Element* FindElementAncestorForMozSelection(nsIContent* aContent) {
+ NS_ENSURE_TRUE(aContent, nullptr);
+ while (aContent && aContent->IsInNativeAnonymousSubtree()) {
+ aContent = aContent->GetClosestNativeAnonymousSubtreeRootParentOrHost();
+ }
+ NS_ASSERTION(aContent, "aContent isn't in non-anonymous tree?");
+ return aContent ? aContent->GetAsElementOrParentElement() : nullptr;
+}
+
+already_AddRefed<ComputedStyle> nsIFrame::ComputeSelectionStyle(
+ int16_t aSelectionStatus) const {
+ // Just bail out if not a selection-status that ::selection applies to.
+ if (aSelectionStatus != nsISelectionController::SELECTION_ON &&
+ aSelectionStatus != nsISelectionController::SELECTION_DISABLED) {
+ return nullptr;
+ }
+ Element* element = FindElementAncestorForMozSelection(GetContent());
+ if (!element) {
+ return nullptr;
+ }
+ RefPtr<ComputedStyle> pseudoStyle =
+ PresContext()->StyleSet()->ProbePseudoElementStyle(
+ *element, PseudoStyleType::selection, nullptr, Style());
+ if (!pseudoStyle) {
+ return nullptr;
+ }
+ // When in high-contrast mode, the style system ends up ignoring the color
+ // declarations, which means that the ::selection style becomes the inherited
+ // color, and default background. That's no good.
+ // When force-color-adjust is set to none allow using the color styles,
+ // as they will not be replaced.
+ if (PresContext()->ForcingColors() &&
+ pseudoStyle->StyleText()->mForcedColorAdjust !=
+ StyleForcedColorAdjust::None) {
+ return nullptr;
+ }
+ return do_AddRef(pseudoStyle);
+}
+
+already_AddRefed<ComputedStyle> nsIFrame::ComputeHighlightSelectionStyle(
+ nsAtom* aHighlightName) {
+ Element* element = FindElementAncestorForMozSelection(GetContent());
+ if (!element) {
+ return nullptr;
+ }
+ return PresContext()->StyleSet()->ProbePseudoElementStyle(
+ *element, PseudoStyleType::highlight, aHighlightName, Style());
+}
+
+template <typename SizeOrMaxSize>
+static inline bool IsIntrinsicKeyword(const SizeOrMaxSize& aSize) {
+ // All keywords other than auto/none/-moz-available depend on intrinsic sizes.
+ return aSize.IsMaxContent() || aSize.IsMinContent() || aSize.IsFitContent() ||
+ aSize.IsFitContentFunction();
+}
+
+bool nsIFrame::CanBeDynamicReflowRoot() const {
+ const auto& display = *StyleDisplay();
+ if (IsLineParticipant() || display.mDisplay.IsRuby() ||
+ display.IsInnerTableStyle() ||
+ display.DisplayInside() == StyleDisplayInside::Table) {
+ // We have a display type where 'width' and 'height' don't actually set the
+ // width or height (i.e., the size depends on content).
+ MOZ_ASSERT(!HasAnyStateBits(NS_FRAME_DYNAMIC_REFLOW_ROOT),
+ "should not have dynamic reflow root bit");
+ return false;
+ }
+
+ // In general, frames that have contain:layout+size can be reflow roots.
+ // (One exception: table-wrapper frames don't work well as reflow roots,
+ // because their inner-table ReflowInput init path tries to reuse & deref
+ // the wrapper's containing block's reflow input, which may be null if we
+ // initiate reflow from the table-wrapper itself.)
+ //
+ // Changes to `contain` force frame reconstructions, so we used to use
+ // NS_FRAME_REFLOW_ROOT, this bit could be set for the whole lifetime of
+ // this frame. But after the support of `content-visibility: auto` which
+ // is with contain layout + size when it's not relevant to user, and only
+ // with contain layout when it is relevant. The frame does not reconstruct
+ // when the relevancy changes. So we use NS_FRAME_DYNAMIC_REFLOW_ROOT instead.
+ //
+ // We place it above the pref check on purpose, to make sure it works for
+ // containment even with the pref disabled.
+ if (display.IsContainLayout() && GetContainSizeAxes().IsBoth()) {
+ return true;
+ }
+
+ if (!StaticPrefs::layout_dynamic_reflow_roots_enabled()) {
+ return false;
+ }
+
+ // We can't serve as a dynamic reflow root if our used 'width' and 'height'
+ // might be influenced by content.
+ //
+ // FIXME: For display:block, we should probably optimize inline-size: auto.
+ // FIXME: Other flex and grid cases?
+ const auto& pos = *StylePosition();
+ const auto& width = pos.mWidth;
+ const auto& height = pos.mHeight;
+ if (!width.IsLengthPercentage() || width.HasPercent() ||
+ !height.IsLengthPercentage() || height.HasPercent() ||
+ IsIntrinsicKeyword(pos.mMinWidth) || IsIntrinsicKeyword(pos.mMaxWidth) ||
+ IsIntrinsicKeyword(pos.mMinHeight) ||
+ IsIntrinsicKeyword(pos.mMaxHeight) ||
+ ((pos.mMinWidth.IsAuto() || pos.mMinHeight.IsAuto()) &&
+ IsFlexOrGridItem())) {
+ return false;
+ }
+
+ // If our flex-basis is 'auto', it'll defer to 'width' (or 'height') which
+ // we've already checked. Otherwise, it preempts them, so we need to
+ // perform the same "could-this-value-be-influenced-by-content" checks that
+ // we performed for 'width' and 'height' above.
+ if (IsFlexItem()) {
+ const auto& flexBasis = pos.mFlexBasis;
+ if (!flexBasis.IsAuto()) {
+ if (!flexBasis.IsSize() || !flexBasis.AsSize().IsLengthPercentage() ||
+ flexBasis.AsSize().HasPercent()) {
+ return false;
+ }
+ }
+ }
+
+ if (!IsFixedPosContainingBlock()) {
+ // We can't treat this frame as a reflow root, since dynamic changes
+ // to absolutely-positioned frames inside of it require that we
+ // reflow the placeholder before we reflow the absolutely positioned
+ // frame.
+ // FIXME: Alternatively, we could sort the reflow roots in
+ // PresShell::ProcessReflowCommands by depth in the tree, from
+ // deepest to least deep. However, for performance (FIXME) we
+ // should really be sorting them in the opposite order!
+ return false;
+ }
+
+ // If we participate in a container's block reflow context, or margins
+ // can collapse through us, we can't be a dynamic reflow root.
+ if (IsBlockFrameOrSubclass() && !HasAnyStateBits(NS_BLOCK_BFC_STATE_BITS)) {
+ return false;
+ }
+
+ // Subgrids are never reflow roots, but 'contain:layout/paint' prevents
+ // creating a subgrid in the first place.
+ if (pos.mGridTemplateColumns.IsSubgrid() ||
+ pos.mGridTemplateRows.IsSubgrid()) {
+ // NOTE: we could check that 'display' of our parent's primary frame is
+ // '[inline-]grid' here but that's probably not worth it in practice.
+ if (!display.IsContainLayout() && !display.IsContainPaint()) {
+ return false;
+ }
+ }
+
+ // If we are split, we can't be a dynamic reflow root. Our reflow status may
+ // change after reflow, and our parent is responsible to create or delete our
+ // next-in-flow.
+ if (GetPrevContinuation() || GetNextContinuation()) {
+ return false;
+ }
+
+ return true;
+}
+
+/********************************************************
+ * Refreshes each content's frame
+ *********************************************************/
+
+void nsIFrame::DisplayOutlineUnconditional(nsDisplayListBuilder* aBuilder,
+ const nsDisplayListSet& aLists) {
+ // Per https://drafts.csswg.org/css-tables-3/#global-style-overrides:
+ // "All css properties of table-column and table-column-group boxes are
+ // ignored, except when explicitly specified by this specification."
+ // CSS outlines fall into this category, so we skip them on these boxes.
+ MOZ_ASSERT(!IsTableColGroupFrame() && !IsTableColFrame());
+ const auto& outline = *StyleOutline();
+
+ if (!outline.ShouldPaintOutline()) {
+ return;
+ }
+
+ // Outlines are painted by the table wrapper frame.
+ if (IsTableFrame()) {
+ return;
+ }
+
+ if (HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT) &&
+ ScrollableOverflowRect().IsEmpty()) {
+ // Skip parts of IB-splits with an empty overflow rect, see bug 434301.
+ // We may still want to fix some of the overflow area calculations over in
+ // that bug.
+ return;
+ }
+
+ // We don't display outline-style: auto on themed frames that have their own
+ // focus indicators.
+ if (outline.mOutlineStyle.IsAuto()) {
+ auto* disp = StyleDisplay();
+ if (IsThemed(disp) && PresContext()->Theme()->ThemeDrawsFocusForWidget(
+ this, disp->EffectiveAppearance())) {
+ return;
+ }
+ }
+
+ aLists.Outlines()->AppendNewToTop<nsDisplayOutline>(aBuilder, this);
+}
+
+void nsIFrame::DisplayOutline(nsDisplayListBuilder* aBuilder,
+ const nsDisplayListSet& aLists) {
+ if (!IsVisibleForPainting()) return;
+
+ DisplayOutlineUnconditional(aBuilder, aLists);
+}
+
+void nsIFrame::DisplayInsetBoxShadowUnconditional(
+ nsDisplayListBuilder* aBuilder, nsDisplayList* aList) {
+ // XXXbz should box-shadow for rows/rowgroups/columns/colgroups get painted
+ // just because we're visible? Or should it depend on the cell visibility
+ // when we're not the whole table?
+ const auto* effects = StyleEffects();
+ if (effects->HasBoxShadowWithInset(true)) {
+ aList->AppendNewToTop<nsDisplayBoxShadowInner>(aBuilder, this);
+ }
+}
+
+void nsIFrame::DisplayInsetBoxShadow(nsDisplayListBuilder* aBuilder,
+ nsDisplayList* aList) {
+ if (!IsVisibleForPainting()) return;
+
+ DisplayInsetBoxShadowUnconditional(aBuilder, aList);
+}
+
+void nsIFrame::DisplayOutsetBoxShadowUnconditional(
+ nsDisplayListBuilder* aBuilder, nsDisplayList* aList) {
+ // XXXbz should box-shadow for rows/rowgroups/columns/colgroups get painted
+ // just because we're visible? Or should it depend on the cell visibility
+ // when we're not the whole table?
+ const auto* effects = StyleEffects();
+ if (effects->HasBoxShadowWithInset(false)) {
+ aList->AppendNewToTop<nsDisplayBoxShadowOuter>(aBuilder, this);
+ }
+}
+
+void nsIFrame::DisplayOutsetBoxShadow(nsDisplayListBuilder* aBuilder,
+ nsDisplayList* aList) {
+ if (!IsVisibleForPainting()) return;
+
+ DisplayOutsetBoxShadowUnconditional(aBuilder, aList);
+}
+
+void nsIFrame::DisplayCaret(nsDisplayListBuilder* aBuilder,
+ nsDisplayList* aList) {
+ if (!IsVisibleForPainting()) return;
+
+ aList->AppendNewToTop<nsDisplayCaret>(aBuilder, this);
+}
+
+nscolor nsIFrame::GetCaretColorAt(int32_t aOffset) {
+ return nsLayoutUtils::GetColor(this, &nsStyleUI::mCaretColor);
+}
+
+auto nsIFrame::ComputeShouldPaintBackground() const -> ShouldPaintBackground {
+ nsPresContext* pc = PresContext();
+ ShouldPaintBackground settings{pc->GetBackgroundColorDraw(),
+ pc->GetBackgroundImageDraw()};
+ if (settings.mColor && settings.mImage) {
+ return settings;
+ }
+
+ if (StyleVisibility()->mPrintColorAdjust == StylePrintColorAdjust::Exact) {
+ return {true, true};
+ }
+
+ return settings;
+}
+
+bool nsIFrame::DisplayBackgroundUnconditional(nsDisplayListBuilder* aBuilder,
+ const nsDisplayListSet& aLists) {
+ if (aBuilder->IsForEventDelivery() && !aBuilder->HitTestIsForVisibility()) {
+ // For hit-testing, we generally just need a light-weight data structure
+ // like nsDisplayEventReceiver. But if the hit-testing is for visibility,
+ // then we need to know the opaque region in order to determine whether to
+ // stop or not.
+ aLists.BorderBackground()->AppendNewToTop<nsDisplayEventReceiver>(aBuilder,
+ this);
+ return false;
+ }
+
+ const AppendedBackgroundType result =
+ nsDisplayBackgroundImage::AppendBackgroundItemsToTop(
+ aBuilder, this,
+ GetRectRelativeToSelf() + aBuilder->ToReferenceFrame(this),
+ aLists.BorderBackground());
+
+ if (result == AppendedBackgroundType::None) {
+ aBuilder->BuildCompositorHitTestInfoIfNeeded(this,
+ aLists.BorderBackground());
+ }
+
+ return result == AppendedBackgroundType::ThemedBackground;
+}
+
+void nsIFrame::DisplayBorderBackgroundOutline(nsDisplayListBuilder* aBuilder,
+ const nsDisplayListSet& aLists) {
+ // The visibility check belongs here since child elements have the
+ // opportunity to override the visibility property and display even if
+ // their parent is hidden.
+ if (!IsVisibleForPainting()) {
+ return;
+ }
+
+ DisplayOutsetBoxShadowUnconditional(aBuilder, aLists.BorderBackground());
+
+ bool bgIsThemed = DisplayBackgroundUnconditional(aBuilder, aLists);
+ DisplayInsetBoxShadowUnconditional(aBuilder, aLists.BorderBackground());
+
+ // If there's a themed background, we should not create a border item.
+ // It won't be rendered.
+ // Don't paint borders for tables here, since they paint them in a different
+ // order.
+ if (!bgIsThemed && StyleBorder()->HasBorder() && !IsTableFrame()) {
+ aLists.BorderBackground()->AppendNewToTop<nsDisplayBorder>(aBuilder, this);
+ }
+
+ DisplayOutlineUnconditional(aBuilder, aLists);
+}
+
+inline static bool IsSVGContentWithCSSClip(const nsIFrame* aFrame) {
+ // The CSS spec says that the 'clip' property only applies to absolutely
+ // positioned elements, whereas the SVG spec says that it applies to SVG
+ // elements regardless of the value of the 'position' property. Here we obey
+ // the CSS spec for outer-<svg> (since that's what we generally do), but
+ // obey the SVG spec for other SVG elements to which 'clip' applies.
+ return aFrame->HasAnyStateBits(NS_FRAME_SVG_LAYOUT) &&
+ aFrame->GetContent()->IsAnyOfSVGElements(nsGkAtoms::svg,
+ nsGkAtoms::foreignObject);
+}
+
+Maybe<nsRect> nsIFrame::GetClipPropClipRect(const nsStyleDisplay* aDisp,
+ const nsStyleEffects* aEffects,
+ const nsSize& aSize) const {
+ if (aEffects->mClip.IsAuto() ||
+ !(aDisp->IsAbsolutelyPositioned(this) || IsSVGContentWithCSSClip(this))) {
+ return Nothing();
+ }
+
+ auto& clipRect = aEffects->mClip.AsRect();
+ nsRect rect = clipRect.ToLayoutRect();
+ if (MOZ_LIKELY(StyleBorder()->mBoxDecorationBreak ==
+ StyleBoxDecorationBreak::Slice)) {
+ // The clip applies to the joined boxes so it's relative the first
+ // continuation.
+ nscoord y = 0;
+ for (nsIFrame* f = GetPrevContinuation(); f; f = f->GetPrevContinuation()) {
+ y += f->GetRect().height;
+ }
+ rect.MoveBy(nsPoint(0, -y));
+ }
+
+ if (clipRect.right.IsAuto()) {
+ rect.width = aSize.width - rect.x;
+ }
+ if (clipRect.bottom.IsAuto()) {
+ rect.height = aSize.height - rect.y;
+ }
+ return Some(rect);
+}
+
+/**
+ * If the CSS 'overflow' property applies to this frame, and is not
+ * handled by constructing a dedicated nsHTML/XULScrollFrame, set up clipping
+ * for that overflow in aBuilder->ClipState() to clip all containing-block
+ * descendants.
+ */
+static void ApplyOverflowClipping(
+ nsDisplayListBuilder* aBuilder, const nsIFrame* aFrame,
+ nsIFrame::PhysicalAxes aClipAxes,
+ DisplayListClipState::AutoClipMultiple& aClipState) {
+ // Only 'clip' is handled here (and 'hidden' for table frames, and any
+ // non-'visible' value for blocks in a paginated context).
+ // We allow 'clip' to apply to any kind of frame. This is required by
+ // comboboxes which make their display text (an inline frame) have clipping.
+ MOZ_ASSERT(aClipAxes != nsIFrame::PhysicalAxes::None);
+ MOZ_ASSERT(aFrame->ShouldApplyOverflowClipping(aFrame->StyleDisplay()) ==
+ aClipAxes);
+
+ nsRect clipRect;
+ bool haveRadii = false;
+ nscoord radii[8];
+ auto* disp = aFrame->StyleDisplay();
+ // Only deflate the padding if we clip to the content-box in that axis.
+ auto wm = aFrame->GetWritingMode();
+ bool cbH = (wm.IsVertical() ? disp->mOverflowClipBoxBlock
+ : disp->mOverflowClipBoxInline) ==
+ StyleOverflowClipBox::ContentBox;
+ bool cbV = (wm.IsVertical() ? disp->mOverflowClipBoxInline
+ : disp->mOverflowClipBoxBlock) ==
+ StyleOverflowClipBox::ContentBox;
+
+ nsMargin boxMargin = -aFrame->GetUsedPadding();
+ if (!cbH) {
+ boxMargin.left = boxMargin.right = nscoord(0);
+ }
+ if (!cbV) {
+ boxMargin.top = boxMargin.bottom = nscoord(0);
+ }
+
+ auto clipMargin = aFrame->OverflowClipMargin(aClipAxes);
+
+ boxMargin -= aFrame->GetUsedBorder();
+ boxMargin += nsMargin(clipMargin.height, clipMargin.width, clipMargin.height,
+ clipMargin.width);
+ boxMargin.ApplySkipSides(aFrame->GetSkipSides());
+
+ nsRect rect(nsPoint(0, 0), aFrame->GetSize());
+ rect.Inflate(boxMargin);
+ if (MOZ_UNLIKELY(!(aClipAxes & nsIFrame::PhysicalAxes::Horizontal))) {
+ // NOTE(mats) We shouldn't be clipping at all in this dimension really,
+ // but clipping in just one axis isn't supported by our GFX APIs so we
+ // clip to our visual overflow rect instead.
+ nsRect o = aFrame->InkOverflowRect();
+ rect.x = o.x;
+ rect.width = o.width;
+ }
+ if (MOZ_UNLIKELY(!(aClipAxes & nsIFrame::PhysicalAxes::Vertical))) {
+ // See the note above.
+ nsRect o = aFrame->InkOverflowRect();
+ rect.y = o.y;
+ rect.height = o.height;
+ }
+ clipRect = rect + aBuilder->ToReferenceFrame(aFrame);
+ haveRadii = aFrame->GetBoxBorderRadii(radii, boxMargin);
+ aClipState.ClipContainingBlockDescendantsExtra(clipRect,
+ haveRadii ? radii : nullptr);
+}
+
+nsSize nsIFrame::OverflowClipMargin(PhysicalAxes aClipAxes) const {
+ nsSize result;
+ if (aClipAxes == PhysicalAxes::None) {
+ return result;
+ }
+ const auto& margin = StyleMargin()->mOverflowClipMargin;
+ if (margin.IsZero()) {
+ return result;
+ }
+ nscoord marginAu = margin.ToAppUnits();
+ if (aClipAxes & PhysicalAxes::Horizontal) {
+ result.width = marginAu;
+ }
+ if (aClipAxes & PhysicalAxes::Vertical) {
+ result.height = marginAu;
+ }
+ return result;
+}
+
+/**
+ * Returns whether a display item that gets created with the builder's current
+ * state will have a scrolled clip, i.e. a clip that is scrolled by a scroll
+ * frame which does not move the item itself.
+ */
+static bool BuilderHasScrolledClip(nsDisplayListBuilder* aBuilder) {
+ const DisplayItemClipChain* currentClip =
+ aBuilder->ClipState().GetCurrentCombinedClipChain(aBuilder);
+ if (!currentClip) {
+ return false;
+ }
+
+ const ActiveScrolledRoot* currentClipASR = currentClip->mASR;
+ const ActiveScrolledRoot* currentASR = aBuilder->CurrentActiveScrolledRoot();
+ return ActiveScrolledRoot::PickDescendant(currentClipASR, currentASR) !=
+ currentASR;
+}
+
+class AutoSaveRestoreContainsBlendMode {
+ nsDisplayListBuilder& mBuilder;
+ bool mSavedContainsBlendMode;
+
+ public:
+ explicit AutoSaveRestoreContainsBlendMode(nsDisplayListBuilder& aBuilder)
+ : mBuilder(aBuilder),
+ mSavedContainsBlendMode(aBuilder.ContainsBlendMode()) {}
+
+ ~AutoSaveRestoreContainsBlendMode() {
+ mBuilder.SetContainsBlendMode(mSavedContainsBlendMode);
+ }
+};
+
+static bool IsFrameOrAncestorApzAware(nsIFrame* aFrame) {
+ nsIContent* node = aFrame->GetContent();
+ if (!node) {
+ return false;
+ }
+
+ do {
+ if (node->IsNodeApzAware()) {
+ return true;
+ }
+ nsIContent* shadowRoot = node->GetShadowRoot();
+ if (shadowRoot && shadowRoot->IsNodeApzAware()) {
+ return true;
+ }
+
+ // Even if the node owning aFrame doesn't have apz-aware event listeners
+ // itself, its shadow root or display: contents ancestors (which have no
+ // frames) might, so we need to account for them too.
+ } while ((node = node->GetFlattenedTreeParent()) && node->IsElement() &&
+ node->AsElement()->IsDisplayContents());
+
+ return false;
+}
+
+static void CheckForApzAwareEventHandlers(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aFrame) {
+ if (aBuilder->GetAncestorHasApzAwareEventHandler()) {
+ return;
+ }
+
+ if (IsFrameOrAncestorApzAware(aFrame)) {
+ aBuilder->SetAncestorHasApzAwareEventHandler(true);
+ }
+}
+
+static void UpdateCurrentHitTestInfo(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aFrame) {
+ if (!aBuilder->BuildCompositorHitTestInfo()) {
+ // Compositor hit test info is not used.
+ return;
+ }
+
+ CheckForApzAwareEventHandlers(aBuilder, aFrame);
+
+ const CompositorHitTestInfo info = aFrame->GetCompositorHitTestInfo(aBuilder);
+ aBuilder->SetCompositorHitTestInfo(info);
+}
+
+/**
+ * True if aDescendant participates the context aAncestor participating.
+ */
+static bool FrameParticipatesIn3DContext(nsIFrame* aAncestor,
+ nsIFrame* aDescendant) {
+ MOZ_ASSERT(aAncestor != aDescendant);
+ MOZ_ASSERT(aAncestor->GetContent() != aDescendant->GetContent());
+ MOZ_ASSERT(aAncestor->Extend3DContext());
+
+ nsIFrame* ancestor = aAncestor->FirstContinuation();
+ MOZ_ASSERT(ancestor->IsPrimaryFrame());
+
+ nsIFrame* frame;
+ for (frame = aDescendant->GetClosestFlattenedTreeAncestorPrimaryFrame();
+ frame && ancestor != frame;
+ frame = frame->GetClosestFlattenedTreeAncestorPrimaryFrame()) {
+ if (!frame->Extend3DContext()) {
+ return false;
+ }
+ }
+
+ MOZ_ASSERT(frame == ancestor);
+ return true;
+}
+
+static bool ItemParticipatesIn3DContext(nsIFrame* aAncestor,
+ nsDisplayItem* aItem) {
+ auto type = aItem->GetType();
+ const bool isContainer = type == DisplayItemType::TYPE_WRAP_LIST ||
+ type == DisplayItemType::TYPE_CONTAINER;
+
+ if (isContainer && aItem->GetChildren()->Length() == 1) {
+ // If the wraplist has only one child item, use the type of that item.
+ type = aItem->GetChildren()->GetBottom()->GetType();
+ }
+
+ if (type != DisplayItemType::TYPE_TRANSFORM &&
+ type != DisplayItemType::TYPE_PERSPECTIVE) {
+ return false;
+ }
+ nsIFrame* transformFrame = aItem->Frame();
+ if (aAncestor->GetContent() == transformFrame->GetContent()) {
+ return true;
+ }
+ return FrameParticipatesIn3DContext(aAncestor, transformFrame);
+}
+
+static void WrapSeparatorTransform(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aFrame,
+ nsDisplayList* aNonParticipants,
+ nsDisplayList* aParticipants, int aIndex,
+ nsDisplayItem** aSeparator) {
+ if (aNonParticipants->IsEmpty()) {
+ return;
+ }
+
+ nsDisplayTransform* item = MakeDisplayItemWithIndex<nsDisplayTransform>(
+ aBuilder, aFrame, aIndex, aNonParticipants, aBuilder->GetVisibleRect());
+
+ if (*aSeparator == nullptr && item) {
+ *aSeparator = item;
+ }
+
+ aParticipants->AppendToTop(item);
+}
+
+// Try to compute a clip rect to bound the contents of the mask item
+// that will be built for |aMaskedFrame|. If we're not able to compute
+// one, return an empty Maybe.
+// The returned clip rect, if there is one, is relative to |aMaskedFrame|.
+static Maybe<nsRect> ComputeClipForMaskItem(
+ nsDisplayListBuilder* aBuilder, nsIFrame* aMaskedFrame,
+ const SVGUtils::MaskUsage& aMaskUsage) {
+ const nsStyleSVGReset* svgReset = aMaskedFrame->StyleSVGReset();
+
+ nsPoint offsetToUserSpace =
+ nsLayoutUtils::ComputeOffsetToUserSpace(aBuilder, aMaskedFrame);
+ int32_t devPixelRatio = aMaskedFrame->PresContext()->AppUnitsPerDevPixel();
+ gfxPoint devPixelOffsetToUserSpace =
+ nsLayoutUtils::PointToGfxPoint(offsetToUserSpace, devPixelRatio);
+ CSSToLayoutDeviceScale cssToDevScale =
+ aMaskedFrame->PresContext()->CSSToDevPixelScale();
+
+ nsPoint toReferenceFrame;
+ aBuilder->FindReferenceFrameFor(aMaskedFrame, &toReferenceFrame);
+
+ Maybe<gfxRect> combinedClip;
+ if (aMaskUsage.ShouldApplyBasicShapeOrPath()) {
+ Maybe<Rect> result =
+ CSSClipPathInstance::GetBoundingRectForBasicShapeOrPathClip(
+ aMaskedFrame, svgReset->mClipPath);
+ if (result) {
+ combinedClip = Some(ThebesRect(*result));
+ }
+ } else if (aMaskUsage.ShouldApplyClipPath()) {
+ gfxRect result = SVGUtils::GetBBox(
+ aMaskedFrame,
+ SVGUtils::eBBoxIncludeClipped | SVGUtils::eBBoxIncludeFill |
+ SVGUtils::eBBoxIncludeMarkers | SVGUtils::eBBoxIncludeStroke |
+ SVGUtils::eDoNotClipToBBoxOfContentInsideClipPath);
+ combinedClip = Some(
+ ThebesRect((CSSRect::FromUnknownRect(ToRect(result)) * cssToDevScale)
+ .ToUnknownRect()));
+ } else {
+ // The code for this case is adapted from ComputeMaskGeometry().
+
+ nsRect borderArea(toReferenceFrame, aMaskedFrame->GetSize());
+ borderArea -= offsetToUserSpace;
+
+ // Use an infinite dirty rect to pass into nsCSSRendering::
+ // GetImageLayerClip() because we don't have an actual dirty rect to
+ // pass in. This is fine because the only time GetImageLayerClip() will
+ // not intersect the incoming dirty rect with something is in the "NoClip"
+ // case, and we handle that specially.
+ nsRect dirtyRect(nscoord_MIN / 2, nscoord_MIN / 2, nscoord_MAX,
+ nscoord_MAX);
+
+ nsIFrame* firstFrame =
+ nsLayoutUtils::FirstContinuationOrIBSplitSibling(aMaskedFrame);
+ nsTArray<SVGMaskFrame*> maskFrames;
+ // XXX check return value?
+ SVGObserverUtils::GetAndObserveMasks(firstFrame, &maskFrames);
+
+ for (uint32_t i = 0; i < maskFrames.Length(); ++i) {
+ gfxRect clipArea;
+ if (maskFrames[i]) {
+ clipArea = maskFrames[i]->GetMaskArea(aMaskedFrame);
+ clipArea = ThebesRect(
+ (CSSRect::FromUnknownRect(ToRect(clipArea)) * cssToDevScale)
+ .ToUnknownRect());
+ } else {
+ const auto& layer = svgReset->mMask.mLayers[i];
+ if (layer.mClip == StyleGeometryBox::NoClip) {
+ return Nothing();
+ }
+
+ nsCSSRendering::ImageLayerClipState clipState;
+ nsCSSRendering::GetImageLayerClip(
+ layer, aMaskedFrame, *aMaskedFrame->StyleBorder(), borderArea,
+ dirtyRect, false /* aWillPaintBorder */, devPixelRatio, &clipState);
+ clipArea = clipState.mDirtyRectInDevPx;
+ }
+ combinedClip = UnionMaybeRects(combinedClip, Some(clipArea));
+ }
+ }
+ if (combinedClip) {
+ if (combinedClip->IsEmpty()) {
+ // *clipForMask might be empty if all mask references are not resolvable
+ // or the size of them are empty. We still need to create a transparent
+ // mask before bug 1276834 fixed, so don't clip ctx by an empty rectangle
+ // for for now.
+ return Nothing();
+ }
+
+ // Convert to user space.
+ *combinedClip += devPixelOffsetToUserSpace;
+
+ // Round the clip out. In FrameLayerBuilder we round clips to nearest
+ // pixels, and if we have a really thin clip here, that can cause the
+ // clip to become empty if we didn't round out here.
+ // The rounding happens in coordinates that are relative to the reference
+ // frame, which matches what FrameLayerBuilder does.
+ combinedClip->RoundOut();
+
+ // Convert to app units.
+ nsRect result =
+ nsLayoutUtils::RoundGfxRectToAppRect(*combinedClip, devPixelRatio);
+
+ // The resulting clip is relative to the reference frame, but the caller
+ // expects it to be relative to the masked frame, so adjust it.
+ result -= toReferenceFrame;
+ return Some(result);
+ }
+ return Nothing();
+}
+
+struct AutoCheckBuilder {
+ explicit AutoCheckBuilder(nsDisplayListBuilder* aBuilder)
+ : mBuilder(aBuilder) {
+ aBuilder->Check();
+ }
+
+ ~AutoCheckBuilder() { mBuilder->Check(); }
+
+ nsDisplayListBuilder* mBuilder;
+};
+
+/**
+ * Tries to reuse a top-level stacking context item from the previous paint.
+ * Returns true if an item was reused, otherwise false.
+ */
+bool TryToReuseStackingContextItem(nsDisplayListBuilder* aBuilder,
+ nsDisplayList* aList, nsIFrame* aFrame) {
+ if (!aBuilder->IsForPainting() || !aBuilder->IsPartialUpdate() ||
+ aBuilder->InInvalidSubtree()) {
+ return false;
+ }
+
+ if (aFrame->IsFrameModified() || aFrame->HasModifiedDescendants()) {
+ return false;
+ }
+
+ auto& items = aFrame->DisplayItems();
+ auto* res = std::find_if(
+ items.begin(), items.end(),
+ [](nsDisplayItem* aItem) { return aItem->IsPreProcessed(); });
+
+ if (res == items.end()) {
+ return false;
+ }
+
+ nsDisplayItem* container = *res;
+ MOZ_ASSERT(container->Frame() == aFrame);
+ DL_LOGD("RDL - Found SC item %p (%s) (frame: %p)", container,
+ container->Name(), container->Frame());
+
+ aList->AppendToTop(container);
+ aBuilder->ReuseDisplayItem(container);
+ return true;
+}
+
+void nsIFrame::BuildDisplayListForStackingContext(
+ nsDisplayListBuilder* aBuilder, nsDisplayList* aList,
+ bool* aCreatedContainerItem) {
+#ifdef DEBUG
+ DL_LOGV("BuildDisplayListForStackingContext (%p) <", this);
+ ScopeExit e(
+ [this]() { DL_LOGV("> BuildDisplayListForStackingContext (%p)", this); });
+#endif
+
+ AutoCheckBuilder check(aBuilder);
+
+ if (aBuilder->IsReusingStackingContextItems() &&
+ TryToReuseStackingContextItem(aBuilder, aList, this)) {
+ if (aCreatedContainerItem) {
+ *aCreatedContainerItem = true;
+ }
+ return;
+ }
+
+ if (HasAnyStateBits(NS_FRAME_TOO_DEEP_IN_FRAME_TREE)) {
+ return;
+ }
+
+ const auto& style = *Style();
+ const nsStyleDisplay* disp = style.StyleDisplay();
+ const nsStyleEffects* effects = style.StyleEffects();
+ EffectSet* effectSetForOpacity =
+ EffectSet::GetForFrame(this, nsCSSPropertyIDSet::OpacityProperties());
+ // We can stop right away if this is a zero-opacity stacking context and
+ // we're painting, and we're not animating opacity.
+ bool needHitTestInfo = aBuilder->BuildCompositorHitTestInfo() &&
+ Style()->PointerEvents() != StylePointerEvents::None;
+ bool opacityItemForEventsOnly = false;
+ if (effects->IsTransparent() && aBuilder->IsForPainting() &&
+ !(disp->mWillChange.bits & StyleWillChangeBits::OPACITY) &&
+ !nsLayoutUtils::HasAnimationOfPropertySet(
+ this, nsCSSPropertyIDSet::OpacityProperties(), effectSetForOpacity)) {
+ if (needHitTestInfo) {
+ opacityItemForEventsOnly = true;
+ } else {
+ return;
+ }
+ }
+
+ if (aBuilder->IsForPainting() && disp->mWillChange.bits) {
+ aBuilder->AddToWillChangeBudget(this, GetSize());
+ }
+
+ // For preserves3d, use the dirty rect already installed on the
+ // builder, since aDirtyRect maybe distorted for transforms along
+ // the chain.
+ nsRect visibleRect = aBuilder->GetVisibleRect();
+ nsRect dirtyRect = aBuilder->GetDirtyRect();
+
+ // We build an opacity item if it's not going to be drawn by SVG content.
+ // We could in principle skip creating an nsDisplayOpacity item if
+ // nsDisplayOpacity::NeedsActiveLayer returns false and usingSVGEffects is
+ // true (the nsDisplayFilter/nsDisplayMasksAndClipPaths could handle the
+ // opacity). Since SVG has perf issues where we sometimes spend a lot of
+ // time creating display list items that might be helpful. We'd need to
+ // restore our mechanism to do that (changed in bug 1482403), and we'd
+ // need to invalidate the frame if the value that would be return from
+ // NeedsActiveLayer was to change, which we don't currently do.
+ const bool useOpacity =
+ HasVisualOpacity(disp, effects, effectSetForOpacity) &&
+ !SVGUtils::CanOptimizeOpacity(this);
+
+ const bool isTransformed = IsTransformed();
+ const bool hasPerspective = isTransformed && HasPerspective();
+ const bool extend3DContext =
+ Extend3DContext(disp, effects, effectSetForOpacity);
+ const bool combines3DTransformWithAncestors =
+ (extend3DContext || isTransformed) && Combines3DTransformWithAncestors();
+
+ Maybe<nsDisplayListBuilder::AutoPreserves3DContext> autoPreserves3DContext;
+ if (extend3DContext && !combines3DTransformWithAncestors) {
+ // Start a new preserves3d context to keep informations on
+ // nsDisplayListBuilder.
+ autoPreserves3DContext.emplace(aBuilder);
+ // Save dirty rect on the builder to avoid being distorted for
+ // multiple transforms along the chain.
+ aBuilder->SavePreserves3DRect();
+
+ // We rebuild everything within preserve-3d and don't try
+ // to retain, so override the dirty rect now.
+ if (aBuilder->IsRetainingDisplayList()) {
+ dirtyRect = visibleRect;
+ aBuilder->SetDisablePartialUpdates(true);
+ }
+ }
+
+ const bool useBlendMode = effects->mMixBlendMode != StyleBlend::Normal;
+ if (useBlendMode) {
+ aBuilder->SetContainsBlendMode(true);
+ }
+
+ // reset blend mode so we can keep track if this stacking context needs have
+ // a nsDisplayBlendContainer. Set the blend mode back when the routine exits
+ // so we keep track if the parent stacking context needs a container too.
+ AutoSaveRestoreContainsBlendMode autoRestoreBlendMode(*aBuilder);
+ aBuilder->SetContainsBlendMode(false);
+
+ // NOTE: When changing this condition make sure to tweak nsGfxScrollFrame as
+ // well.
+ bool usingBackdropFilter = effects->HasBackdropFilters() &&
+ IsVisibleForPainting() &&
+ !style.IsRootElementStyle();
+
+ nsRect visibleRectOutsideTransform = visibleRect;
+ nsDisplayTransform::PrerenderInfo prerenderInfo;
+ bool inTransform = aBuilder->IsInTransform();
+ if (isTransformed) {
+ prerenderInfo = nsDisplayTransform::ShouldPrerenderTransformedContent(
+ aBuilder, this, &visibleRect);
+
+ switch (prerenderInfo.mDecision) {
+ case nsDisplayTransform::PrerenderDecision::Full:
+ case nsDisplayTransform::PrerenderDecision::Partial:
+ dirtyRect = visibleRect;
+ break;
+ case nsDisplayTransform::PrerenderDecision::No: {
+ // If we didn't prerender an animated frame in a preserve-3d context,
+ // then we want disable async animations for the rest of the preserve-3d
+ // (especially ancestors).
+ if ((extend3DContext || combines3DTransformWithAncestors) &&
+ prerenderInfo.mHasAnimations) {
+ aBuilder->SavePreserves3DAllowAsyncAnimation(false);
+ }
+
+ const nsRect overflow = InkOverflowRectRelativeToSelf();
+ if (overflow.IsEmpty() && !extend3DContext) {
+ return;
+ }
+
+ // If we're in preserve-3d then grab the dirty rect that was given to
+ // the root and transform using the combined transform.
+ if (combines3DTransformWithAncestors) {
+ visibleRect = dirtyRect = aBuilder->GetPreserves3DRect();
+ }
+
+ float appPerDev = PresContext()->AppUnitsPerDevPixel();
+ auto transform = nsDisplayTransform::GetResultingTransformMatrix(
+ this, nsPoint(), appPerDev,
+ nsDisplayTransform::kTransformRectFlags);
+ nsRect untransformedDirtyRect;
+ if (nsDisplayTransform::UntransformRect(dirtyRect, overflow, transform,
+ appPerDev,
+ &untransformedDirtyRect)) {
+ dirtyRect = untransformedDirtyRect;
+ nsDisplayTransform::UntransformRect(visibleRect, overflow, transform,
+ appPerDev, &visibleRect);
+ } else {
+ // This should only happen if the transform is singular, in which case
+ // nothing is visible anyway
+ dirtyRect.SetEmpty();
+ visibleRect.SetEmpty();
+ }
+ }
+ }
+ inTransform = true;
+ } else if (IsFixedPosContainingBlock()) {
+ // Restict the building area to the overflow rect for these frames, since
+ // RetainedDisplayListBuilder uses it to know if the size of the stacking
+ // context changed.
+ visibleRect.IntersectRect(visibleRect, InkOverflowRect());
+ dirtyRect.IntersectRect(dirtyRect, InkOverflowRect());
+ }
+
+ bool hasOverrideDirtyRect = false;
+ // If we're doing a partial build, we're not invalid and we're capable
+ // of having an override building rect (stacking context and fixed pos
+ // containing block), then we should assume we have one.
+ // Either we have an explicit one, or nothing in our subtree changed and
+ // we have an implicit empty rect.
+ //
+ // These conditions should match |CanStoreDisplayListBuildingRect()| in
+ // RetainedDisplayListBuilder.cpp
+ if (!aBuilder->IsReusingStackingContextItems() &&
+ aBuilder->IsPartialUpdate() && !aBuilder->InInvalidSubtree() &&
+ !IsFrameModified() && IsFixedPosContainingBlock() &&
+ !GetPrevContinuation() && !GetNextContinuation()) {
+ dirtyRect = nsRect();
+ if (HasOverrideDirtyRegion()) {
+ nsDisplayListBuilder::DisplayListBuildingData* data =
+ GetProperty(nsDisplayListBuilder::DisplayListBuildingRect());
+ if (data) {
+ dirtyRect = data->mDirtyRect.Intersect(visibleRect);
+ hasOverrideDirtyRect = true;
+ }
+ }
+ }
+
+ bool usingFilter = effects->HasFilters() && !style.IsRootElementStyle();
+ SVGUtils::MaskUsage maskUsage = SVGUtils::DetermineMaskUsage(this, false);
+ bool usingMask = maskUsage.UsingMaskOrClipPath();
+ bool usingSVGEffects = usingFilter || usingMask;
+
+ nsRect visibleRectOutsideSVGEffects = visibleRect;
+ nsDisplayList hoistedScrollInfoItemsStorage(aBuilder);
+ if (usingSVGEffects) {
+ dirtyRect =
+ SVGIntegrationUtils::GetRequiredSourceForInvalidArea(this, dirtyRect);
+ visibleRect =
+ SVGIntegrationUtils::GetRequiredSourceForInvalidArea(this, visibleRect);
+ aBuilder->EnterSVGEffectsContents(this, &hoistedScrollInfoItemsStorage);
+ }
+
+ bool useStickyPosition = disp->mPosition == StylePositionProperty::Sticky;
+
+ bool useFixedPosition =
+ disp->mPosition == StylePositionProperty::Fixed &&
+ (DisplayPortUtils::IsFixedPosFrameInDisplayPort(this) ||
+ BuilderHasScrolledClip(aBuilder));
+
+ nsDisplayListBuilder::AutoBuildingDisplayList buildingDisplayList(
+ aBuilder, this, visibleRect, dirtyRect, isTransformed);
+
+ UpdateCurrentHitTestInfo(aBuilder, this);
+
+ // Depending on the effects that are applied to this frame, we can create
+ // multiple container display items and wrap them around our contents.
+ // This enum lists all the potential container display items, in the order
+ // outside to inside.
+ enum class ContainerItemType : uint8_t {
+ None = 0,
+ OwnLayerIfNeeded,
+ BlendMode,
+ FixedPosition,
+ OwnLayerForTransformWithRoundedClip,
+ Perspective,
+ Transform,
+ SeparatorTransforms,
+ Opacity,
+ Filter,
+ BlendContainer
+ };
+
+ nsDisplayListBuilder::AutoContainerASRTracker contASRTracker(aBuilder);
+
+ auto cssClip = GetClipPropClipRect(disp, effects, GetSize());
+ auto ApplyClipProp = [&](DisplayListClipState::AutoSaveRestore& aClipState) {
+ if (!cssClip) {
+ return;
+ }
+ nsPoint offset = aBuilder->GetCurrentFrameOffsetToReferenceFrame();
+ aBuilder->IntersectDirtyRect(*cssClip);
+ aBuilder->IntersectVisibleRect(*cssClip);
+ aClipState.ClipContentDescendants(*cssClip + offset);
+ };
+
+ // The CSS clip property is effectively inside the transform, but outside the
+ // filters. So if we're not transformed we can apply it just here for
+ // simplicity, instead of on each of the places that handle clipCapturedBy.
+ DisplayListClipState::AutoSaveRestore untransformedCssClip(aBuilder);
+ if (!isTransformed) {
+ ApplyClipProp(untransformedCssClip);
+ }
+
+ // If there is a current clip, then depending on the container items we
+ // create, different things can happen to it. Some container items simply
+ // propagate the clip to their children and aren't clipped themselves.
+ // But other container items, especially those that establish a different
+ // geometry for their contents (e.g. transforms), capture the clip on
+ // themselves and unset the clip for their contents. If we create more than
+ // one of those container items, the clip will be captured on the outermost
+ // one and the inner container items will be unclipped.
+ ContainerItemType clipCapturedBy = ContainerItemType::None;
+ if (useFixedPosition) {
+ clipCapturedBy = ContainerItemType::FixedPosition;
+ } else if (isTransformed) {
+ const DisplayItemClipChain* currentClip =
+ aBuilder->ClipState().GetCurrentCombinedClipChain(aBuilder);
+ if ((hasPerspective || extend3DContext) &&
+ (currentClip && currentClip->HasRoundedCorners())) {
+ // If we're creating an nsDisplayTransform item that is going to combine
+ // its transform with its children (preserve-3d or perspective), then we
+ // can't have an intermediate surface. Mask layers force an intermediate
+ // surface, so if we're going to need both then create a separate
+ // wrapping layer for the mask.
+ clipCapturedBy = ContainerItemType::OwnLayerForTransformWithRoundedClip;
+ } else if (hasPerspective) {
+ clipCapturedBy = ContainerItemType::Perspective;
+ } else {
+ clipCapturedBy = ContainerItemType::Transform;
+ }
+ } else if (usingFilter) {
+ clipCapturedBy = ContainerItemType::Filter;
+ }
+
+ DisplayListClipState::AutoSaveRestore clipState(aBuilder);
+ if (clipCapturedBy != ContainerItemType::None) {
+ clipState.Clear();
+ }
+
+ DisplayListClipState::AutoSaveRestore transformedCssClip(aBuilder);
+ if (isTransformed) {
+ // FIXME(emilio, bug 1525159): In the case we have a both a transform _and_
+ // filters, this clips the input to the filters as well, which is not
+ // correct (clipping by the `clip` property is supposed to happen after
+ // applying the filter effects, per [1].
+ //
+ // This is not a regression though, since we used to do that anyway before
+ // bug 1514384, and even without the transform we get it wrong.
+ //
+ // [1]: https://drafts.fxtf.org/css-masking/#placement
+ ApplyClipProp(transformedCssClip);
+ }
+
+ nsDisplayListCollection set(aBuilder);
+ Maybe<nsRect> clipForMask;
+ {
+ DisplayListClipState::AutoSaveRestore nestedClipState(aBuilder);
+ nsDisplayListBuilder::AutoInTransformSetter inTransformSetter(aBuilder,
+ inTransform);
+ nsDisplayListBuilder::AutoEnterFilter filterASRSetter(aBuilder,
+ usingFilter);
+ nsDisplayListBuilder::AutoInEventsOnly inEventsSetter(
+ aBuilder, opacityItemForEventsOnly);
+
+ // If we have a mask, compute a clip to bound the masked content.
+ // This is necessary in case the content moves with an ancestor
+ // ASR of the mask.
+ // Don't do this if we also have a filter, because then the clip
+ // would be applied before the filter, violating
+ // https://www.w3.org/TR/filter-effects-1/#placement.
+ // Filters are a containing block for fixed and absolute descendants,
+ // so the masked content cannot move with an ancestor ASR.
+ if (usingMask && !usingFilter) {
+ clipForMask = ComputeClipForMaskItem(aBuilder, this, maskUsage);
+ if (clipForMask) {
+ aBuilder->IntersectDirtyRect(*clipForMask);
+ aBuilder->IntersectVisibleRect(*clipForMask);
+ nestedClipState.ClipContentDescendants(
+ *clipForMask + aBuilder->GetCurrentFrameOffsetToReferenceFrame());
+ }
+ }
+
+ // extend3DContext also guarantees that applyAbsPosClipping and
+ // usingSVGEffects are false We only modify the preserve-3d rect if we are
+ // the top of a preserve-3d heirarchy
+ if (extend3DContext) {
+ // Mark these first so MarkAbsoluteFramesForDisplayList knows if we are
+ // going to be forced to descend into frames.
+ aBuilder->MarkPreserve3DFramesForDisplayList(this);
+ }
+
+ aBuilder->AdjustWindowDraggingRegion(this);
+
+ MarkAbsoluteFramesForDisplayList(aBuilder);
+ aBuilder->Check();
+ BuildDisplayList(aBuilder, set);
+ SetBuiltDisplayList(true);
+ aBuilder->Check();
+ aBuilder->DisplayCaret(this, set.Outlines());
+
+ // Blend modes are a real pain for retained display lists. We build a blend
+ // container item if the built list contains any blend mode items within
+ // the current stacking context. This can change without an invalidation
+ // to the stacking context frame, or the blend mode frame (e.g. by moving
+ // an intermediate frame).
+ // When we gain/remove a blend container item, we need to mark this frame
+ // as invalid and have the full display list for merging to track
+ // the change correctly.
+ // It seems really hard to track this in advance, as the bookkeeping
+ // required to note which stacking contexts have blend descendants
+ // is complex and likely to be buggy.
+ // Instead we're doing the sad thing, detecting it afterwards, and just
+ // repeating display list building if it changed.
+ // We have to repeat building for the entire display list (or at least
+ // the outer stacking context), since we need to mark this frame as invalid
+ // to remove any existing content that isn't wrapped in the blend container,
+ // and then we need to build content infront/behind the blend container
+ // to get correct positioning during merging.
+ if (aBuilder->ContainsBlendMode() && aBuilder->IsRetainingDisplayList()) {
+ if (aBuilder->IsPartialUpdate()) {
+ aBuilder->SetPartialBuildFailed(true);
+ } else {
+ aBuilder->SetDisablePartialUpdates(true);
+ }
+ }
+ }
+
+ if (aBuilder->IsBackgroundOnly()) {
+ set.BlockBorderBackgrounds()->DeleteAll(aBuilder);
+ set.Floats()->DeleteAll(aBuilder);
+ set.Content()->DeleteAll(aBuilder);
+ set.PositionedDescendants()->DeleteAll(aBuilder);
+ set.Outlines()->DeleteAll(aBuilder);
+ }
+
+ if (hasOverrideDirtyRect &&
+ StaticPrefs::layout_display_list_show_rebuild_area()) {
+ nsDisplaySolidColor* color = MakeDisplayItem<nsDisplaySolidColor>(
+ aBuilder, this,
+ dirtyRect + aBuilder->GetCurrentFrameOffsetToReferenceFrame(),
+ NS_RGBA(255, 0, 0, 64), false);
+ if (color) {
+ color->SetOverrideZIndex(INT32_MAX);
+ set.PositionedDescendants()->AppendToTop(color);
+ }
+ }
+
+ nsIContent* content = GetContent();
+ if (!content) {
+ content = PresContext()->Document()->GetRootElement();
+ }
+
+ nsDisplayList resultList(aBuilder);
+ set.SerializeWithCorrectZOrder(&resultList, content);
+
+ // Get the ASR to use for the container items that we create here.
+ const ActiveScrolledRoot* containerItemASR = contASRTracker.GetContainerASR();
+
+ bool createdContainer = false;
+
+ // If adding both a nsDisplayBlendContainer and a nsDisplayBlendMode to the
+ // same list, the nsDisplayBlendContainer should be added first. This only
+ // happens when the element creating this stacking context has mix-blend-mode
+ // and also contains a child which has mix-blend-mode.
+ // The nsDisplayBlendContainer must be added to the list first, so it does not
+ // isolate the containing element blending as well.
+ if (aBuilder->ContainsBlendMode()) {
+ resultList.AppendToTop(nsDisplayBlendContainer::CreateForMixBlendMode(
+ aBuilder, this, &resultList, containerItemASR));
+ createdContainer = true;
+ }
+
+ if (usingBackdropFilter) {
+ nsRect backdropRect =
+ GetRectRelativeToSelf() + aBuilder->ToReferenceFrame(this);
+ resultList.AppendNewToTop<nsDisplayBackdropFilters>(
+ aBuilder, this, &resultList, backdropRect, this);
+ createdContainer = true;
+ }
+
+ // If there are any SVG effects, wrap the list up in an SVG effects item
+ // (which also handles CSS group opacity). Note that we create an SVG effects
+ // item even if resultList is empty, since a filter can produce graphical
+ // output even if the element being filtered wouldn't otherwise do so.
+ if (usingSVGEffects) {
+ MOZ_ASSERT(usingFilter || usingMask,
+ "Beside filter & mask/clip-path, what else effect do we have?");
+
+ if (clipCapturedBy == ContainerItemType::Filter) {
+ clipState.Restore();
+ }
+ // Revert to the post-filter dirty rect.
+ aBuilder->SetVisibleRect(visibleRectOutsideSVGEffects);
+
+ // Skip all filter effects while generating glyph mask.
+ if (usingFilter && !aBuilder->IsForGenerateGlyphMask()) {
+ /* List now emptied, so add the new list to the top. */
+ resultList.AppendNewToTop<nsDisplayFilters>(aBuilder, this, &resultList,
+ this, usingBackdropFilter);
+ createdContainer = true;
+ }
+
+ if (usingMask) {
+ // The mask should move with aBuilder->CurrentActiveScrolledRoot(), so
+ // that's the ASR we prefer to use for the mask item. However, we can
+ // only do this if the mask if clipped with respect to that ASR, because
+ // an item always needs to have finite bounds with respect to its ASR.
+ // If we weren't able to compute a clip for the mask, we fall back to
+ // using containerItemASR, which is the lowest common ancestor clip of
+ // the mask's contents. That's not entirely correct, but it satisfies
+ // the base requirement of the ASR system (that items have finite bounds
+ // wrt. their ASR).
+ const ActiveScrolledRoot* maskASR =
+ clipForMask.isSome() ? aBuilder->CurrentActiveScrolledRoot()
+ : containerItemASR;
+ /* List now emptied, so add the new list to the top. */
+ resultList.AppendNewToTop<nsDisplayMasksAndClipPaths>(
+ aBuilder, this, &resultList, maskASR, usingBackdropFilter);
+ createdContainer = true;
+ }
+
+ // TODO(miko): We could probably create a wraplist here and avoid creating
+ // it later in |BuildDisplayListForChild()|.
+ createdContainer = false;
+
+ // Also add the hoisted scroll info items. We need those for APZ scrolling
+ // because nsDisplayMasksAndClipPaths items can't build active layers.
+ aBuilder->ExitSVGEffectsContents();
+ resultList.AppendToTop(&hoistedScrollInfoItemsStorage);
+ }
+
+ // If the list is non-empty and there is CSS group opacity without SVG
+ // effects, wrap it up in an opacity item.
+ if (useOpacity) {
+ const bool needsActiveOpacityLayer =
+ nsDisplayOpacity::NeedsActiveLayer(aBuilder, this);
+ resultList.AppendNewToTop<nsDisplayOpacity>(
+ aBuilder, this, &resultList, containerItemASR, opacityItemForEventsOnly,
+ needsActiveOpacityLayer, usingBackdropFilter);
+ createdContainer = true;
+ }
+
+ // If we're going to apply a transformation and don't have preserve-3d set,
+ // wrap everything in an nsDisplayTransform. If there's nothing in the list,
+ // don't add anything.
+ //
+ // For the preserve-3d case we want to individually wrap every child in the
+ // list with a separate nsDisplayTransform instead. When the child is already
+ // an nsDisplayTransform, we can skip this step, as the computed transform
+ // will already include our own.
+ //
+ // We also traverse into sublists created by nsDisplayWrapList, so that we
+ // find all the correct children.
+ if (isTransformed && extend3DContext) {
+ // Install dummy nsDisplayTransform as a leaf containing
+ // descendants not participating this 3D rendering context.
+ nsDisplayList nonparticipants(aBuilder);
+ nsDisplayList participants(aBuilder);
+ int index = 1;
+
+ nsDisplayItem* separator = nullptr;
+
+ // TODO: This can be simplified: |participants| is just |resultList|.
+ for (nsDisplayItem* item : resultList.TakeItems()) {
+ if (ItemParticipatesIn3DContext(this, item) &&
+ !item->GetClip().HasClip()) {
+ // The frame of this item participates the same 3D context.
+ WrapSeparatorTransform(aBuilder, this, &nonparticipants, &participants,
+ index++, &separator);
+
+ participants.AppendToTop(item);
+ } else {
+ // The frame of the item doesn't participate the current
+ // context, or has no transform.
+ //
+ // For items participating but not transformed, they are add
+ // to nonparticipants to get a separator layer for handling
+ // clips, if there is, on an intermediate surface.
+ // \see ContainerLayer::DefaultComputeEffectiveTransforms().
+ nonparticipants.AppendToTop(item);
+ }
+ }
+ WrapSeparatorTransform(aBuilder, this, &nonparticipants, &participants,
+ index++, &separator);
+
+ if (separator) {
+ createdContainer = true;
+ }
+
+ resultList.AppendToTop(&participants);
+ }
+
+ if (isTransformed) {
+ transformedCssClip.Restore();
+ if (clipCapturedBy == ContainerItemType::Transform) {
+ // Restore clip state now so nsDisplayTransform is clipped properly.
+ clipState.Restore();
+ }
+ // Revert to the dirtyrect coming in from the parent, without our transform
+ // taken into account.
+ aBuilder->SetVisibleRect(visibleRectOutsideTransform);
+
+ if (this != aBuilder->RootReferenceFrame()) {
+ // Revert to the outer reference frame and offset because all display
+ // items we create from now on are outside the transform.
+ nsPoint toOuterReferenceFrame;
+ const nsIFrame* outerReferenceFrame =
+ aBuilder->FindReferenceFrameFor(GetParent(), &toOuterReferenceFrame);
+ toOuterReferenceFrame += GetPosition();
+
+ buildingDisplayList.SetReferenceFrameAndCurrentOffset(
+ outerReferenceFrame, toOuterReferenceFrame);
+ }
+
+ // We would like to block async animations for ancestors of ones not
+ // prerendered in the preserve-3d tree. Now that we've finished processing
+ // all descendants, update allowAsyncAnimation to take their prerender
+ // state into account
+ // FIXME: We don't block async animations for previous siblings because
+ // their prerender decisions have been made. We may have to figure out a
+ // better way to rollback their prerender decisions.
+ // Alternatively we could not block animations for later siblings, and only
+ // block them for ancestors of a blocked one.
+ if ((extend3DContext || combines3DTransformWithAncestors) &&
+ prerenderInfo.CanUseAsyncAnimations() &&
+ !aBuilder->GetPreserves3DAllowAsyncAnimation()) {
+ // aBuilder->GetPreserves3DAllowAsyncAnimation() means the inner or
+ // previous silbing frames are allowed/disallowed for async animations.
+ prerenderInfo.mDecision = nsDisplayTransform::PrerenderDecision::No;
+ }
+
+ nsDisplayTransform* transformItem = MakeDisplayItem<nsDisplayTransform>(
+ aBuilder, this, &resultList, visibleRect, prerenderInfo.mDecision);
+ if (transformItem) {
+ resultList.AppendToTop(transformItem);
+ createdContainer = true;
+ }
+
+ if (hasPerspective) {
+ transformItem->MarkWithAssociatedPerspective();
+
+ if (clipCapturedBy == ContainerItemType::Perspective) {
+ clipState.Restore();
+ }
+ resultList.AppendNewToTop<nsDisplayPerspective>(aBuilder, this,
+ &resultList);
+ createdContainer = true;
+ }
+ }
+
+ if (clipCapturedBy ==
+ ContainerItemType::OwnLayerForTransformWithRoundedClip) {
+ clipState.Restore();
+ resultList.AppendNewToTopWithIndex<nsDisplayOwnLayer>(
+ aBuilder, this,
+ /* aIndex = */ nsDisplayOwnLayer::OwnLayerForTransformWithRoundedClip,
+ &resultList, aBuilder->CurrentActiveScrolledRoot(),
+ nsDisplayOwnLayerFlags::None, ScrollbarData{},
+ /* aForceActive = */ false, false);
+ createdContainer = true;
+ }
+
+ // If we have sticky positioning, wrap it in a sticky position item.
+ if (useFixedPosition) {
+ if (clipCapturedBy == ContainerItemType::FixedPosition) {
+ clipState.Restore();
+ }
+ // The ASR for the fixed item should be the ASR of our containing block,
+ // which has been set as the builder's current ASR, unless this frame is
+ // invisible and we hadn't saved display item data for it. In that case,
+ // we need to take the containerItemASR since we might have fixed children.
+ // For WebRender, we want to the know what |containerItemASR| is for the
+ // case where the fixed-pos item is not a "real" fixed-pos item (e.g. it's
+ // nested inside a scrolling transform), so we stash that on the display
+ // item as well.
+ const ActiveScrolledRoot* fixedASR = ActiveScrolledRoot::PickAncestor(
+ containerItemASR, aBuilder->CurrentActiveScrolledRoot());
+ resultList.AppendNewToTop<nsDisplayFixedPosition>(
+ aBuilder, this, &resultList, fixedASR, containerItemASR);
+ createdContainer = true;
+ } else if (useStickyPosition) {
+ // For position:sticky, the clip needs to be applied both to the sticky
+ // container item and to the contents. The container item needs the clip
+ // because a scrolled clip needs to move independently from the sticky
+ // contents, and the contents need the clip so that they have finite
+ // clipped bounds with respect to the container item's ASR. The latter is
+ // a little tricky in the case where the sticky item has both fixed and
+ // non-fixed descendants, because that means that the sticky container
+ // item's ASR is the ASR of the fixed descendant.
+ // For WebRender display list building, though, we still want to know the
+ // the ASR that the sticky container item would normally have, so we stash
+ // that on the display item as the "container ASR" (i.e. the normal ASR of
+ // the container item, excluding the special behaviour induced by fixed
+ // descendants).
+ const ActiveScrolledRoot* stickyASR = ActiveScrolledRoot::PickAncestor(
+ containerItemASR, aBuilder->CurrentActiveScrolledRoot());
+
+ auto* stickyItem = MakeDisplayItem<nsDisplayStickyPosition>(
+ aBuilder, this, &resultList, stickyASR,
+ aBuilder->CurrentActiveScrolledRoot(),
+ clipState.IsClippedToDisplayPort());
+
+ bool shouldFlatten = true;
+
+ StickyScrollContainer* stickyScrollContainer =
+ StickyScrollContainer::GetStickyScrollContainerForFrame(this);
+ if (stickyScrollContainer &&
+ stickyScrollContainer->ScrollFrame()->IsMaybeAsynchronouslyScrolled()) {
+ shouldFlatten = false;
+ }
+
+ stickyItem->SetShouldFlatten(shouldFlatten);
+
+ resultList.AppendToTop(stickyItem);
+ createdContainer = true;
+
+ // If the sticky element is inside a filter, annotate the scroll frame that
+ // scrolls the filter as having out-of-flow content inside a filter (this
+ // inhibits paint skipping).
+ if (aBuilder->GetFilterASR() && aBuilder->GetFilterASR() == stickyASR) {
+ aBuilder->GetFilterASR()
+ ->mScrollableFrame->SetHasOutOfFlowContentInsideFilter();
+ }
+ }
+
+ // If there's blending, wrap up the list in a blend-mode item. Note that
+ // opacity can be applied before blending as the blend color is not affected
+ // by foreground opacity (only background alpha).
+ if (useBlendMode) {
+ DisplayListClipState::AutoSaveRestore blendModeClipState(aBuilder);
+ resultList.AppendNewToTop<nsDisplayBlendMode>(aBuilder, this, &resultList,
+ effects->mMixBlendMode,
+ containerItemASR, false);
+ createdContainer = true;
+ }
+
+ if (aBuilder->IsReusingStackingContextItems()) {
+ if (resultList.IsEmpty()) {
+ return;
+ }
+
+ nsDisplayItem* container = resultList.GetBottom();
+ if (resultList.Length() > 1 || container->Frame() != this) {
+ container = MakeDisplayItem<nsDisplayContainer>(
+ aBuilder, this, containerItemASR, &resultList);
+ } else {
+ MOZ_ASSERT(resultList.Length() == 1);
+ resultList.Clear();
+ }
+
+ // Mark the outermost display item as reusable. These display items and
+ // their chidren can be reused during the next paint if no ancestor or
+ // descendant frames have been modified.
+ if (!container->IsReusedItem()) {
+ container->SetReusable();
+ }
+ aList->AppendToTop(container);
+ createdContainer = true;
+ } else {
+ aList->AppendToTop(&resultList);
+ }
+
+ if (aCreatedContainerItem) {
+ *aCreatedContainerItem = createdContainer;
+ }
+}
+
+static nsDisplayItem* WrapInWrapList(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aFrame, nsDisplayList* aList,
+ const ActiveScrolledRoot* aContainerASR,
+ bool aBuiltContainerItem = false) {
+ nsDisplayItem* item = aList->GetBottom();
+ if (!item) {
+ return nullptr;
+ }
+
+ // We need a wrap list if there are multiple items, or if the single
+ // item has a different frame. This can change in a partial build depending
+ // on which items we build, so we need to ensure that we don't transition
+ // to/from a wrap list without invalidating correctly.
+ bool needsWrapList =
+ aList->Length() > 1 || item->Frame() != aFrame || item->GetChildren();
+
+ // If we have an explicit container item (that can't change without an
+ // invalidation) or we're doing a full build and don't need a wrap list, then
+ // we can skip adding one.
+ if (aBuiltContainerItem || (!aBuilder->IsPartialUpdate() && !needsWrapList)) {
+ MOZ_ASSERT(aList->Length() == 1);
+ aList->Clear();
+ return item;
+ }
+
+ // If we're doing a partial build and we didn't need a wrap list
+ // previously then we can try to work from there.
+ if (aBuilder->IsPartialUpdate() &&
+ !aFrame->HasDisplayItem(uint32_t(DisplayItemType::TYPE_CONTAINER))) {
+ // If we now need a wrap list, we must previously have had no display items
+ // or a single one belonging to this frame. Mark the item itself as
+ // discarded so that RetainedDisplayListBuilder uses the ones we just built.
+ // We don't want to mark the frame as modified as that would invalidate
+ // positioned descendants that might be outside of this list, and might not
+ // have been rebuilt this time.
+ if (needsWrapList) {
+ DiscardOldItems(aFrame);
+ } else {
+ MOZ_ASSERT(aList->Length() == 1);
+ aList->Clear();
+ return item;
+ }
+ }
+
+ // The last case we could try to handle is when we previously had a wrap list,
+ // but no longer need it. Unfortunately we can't differentiate this case from
+ // a partial build where other children exist but we just didn't build them
+ // this time.
+ // TODO:RetainedDisplayListBuilder's merge phase has the full list and
+ // could strip them out.
+
+ return MakeDisplayItem<nsDisplayContainer>(aBuilder, aFrame, aContainerASR,
+ aList);
+}
+
+/**
+ * Check if a frame should be visited for building display list.
+ */
+static bool DescendIntoChild(nsDisplayListBuilder* aBuilder,
+ const nsIFrame* aChild, const nsRect& aVisible,
+ const nsRect& aDirty) {
+ if (aChild->HasAnyStateBits(NS_FRAME_FORCE_DISPLAY_LIST_DESCEND_INTO)) {
+ return true;
+ }
+
+ // If the child is a scrollframe that we want to ignore, then we need
+ // to descend into it because its scrolled child may intersect the dirty
+ // area even if the scrollframe itself doesn't.
+ if (aChild == aBuilder->GetIgnoreScrollFrame()) {
+ return true;
+ }
+
+ // There are cases where the "ignore scroll frame" on the builder is not set
+ // correctly, and so we additionally want to catch cases where the child is
+ // a root scrollframe and we are ignoring scrolling on the viewport.
+ if (aChild == aBuilder->GetPresShellIgnoreScrollFrame()) {
+ return true;
+ }
+
+ nsRect overflow = aChild->InkOverflowRect();
+
+ // On mobile, there may be a dynamic toolbar. The root content document's
+ // root scroll frame's ink overflow rect does not include the toolbar
+ // height, but if the toolbar is hidden, we still want to be able to target
+ // content underneath the toolbar, so expand the overflow rect here to
+ // allow display list building to descend into the scroll frame.
+ if (aBuilder->IsForEventDelivery() &&
+ aChild == aChild->PresShell()->GetRootScrollFrame() &&
+ aChild->PresContext()->IsRootContentDocumentCrossProcess() &&
+ aChild->PresContext()->HasDynamicToolbar()) {
+ overflow.SizeTo(nsLayoutUtils::ExpandHeightForDynamicToolbar(
+ aChild->PresContext(), overflow.Size()));
+ }
+
+ if (aDirty.Intersects(overflow)) {
+ return true;
+ }
+
+ if (aChild->ForceDescendIntoIfVisible() && aVisible.Intersects(overflow)) {
+ return true;
+ }
+
+ if (aChild->IsTablePart()) {
+ // Relative positioning and transforms can cause table parts to move, but we
+ // will still paint the backgrounds for their ancestor parts under them at
+ // their 'normal' position. That means that we must consider the overflow
+ // rects at both positions.
+
+ // We convert the overflow rect into the nsTableFrame's coordinate
+ // space, applying the normal position offset at each step. Then we
+ // compare that against the builder's cached dirty rect in table
+ // coordinate space.
+ const nsIFrame* f = aChild;
+ nsRect normalPositionOverflowRelativeToTable = overflow;
+
+ while (f->IsTablePart()) {
+ normalPositionOverflowRelativeToTable += f->GetNormalPosition();
+ f = f->GetParent();
+ }
+
+ nsDisplayTableBackgroundSet* tableBGs = aBuilder->GetTableBackgroundSet();
+ if (tableBGs && tableBGs->GetDirtyRect().Intersects(
+ normalPositionOverflowRelativeToTable)) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+void nsIFrame::BuildDisplayListForSimpleChild(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aChild,
+ const nsDisplayListSet& aLists) {
+ // This is the shortcut for frames been handled along the common
+ // path, the most common one of THE COMMON CASE mentioned later.
+ MOZ_ASSERT(aChild->Type() != LayoutFrameType::Placeholder);
+ MOZ_ASSERT(!aBuilder->GetSelectedFramesOnly() &&
+ !aBuilder->GetIncludeAllOutOfFlows(),
+ "It should be held for painting to window");
+ MOZ_ASSERT(aChild->HasAnyStateBits(NS_FRAME_SIMPLE_DISPLAYLIST));
+
+ const nsPoint offset = aChild->GetOffsetTo(this);
+ const nsRect visible = aBuilder->GetVisibleRect() - offset;
+ const nsRect dirty = aBuilder->GetDirtyRect() - offset;
+
+ if (!DescendIntoChild(aBuilder, aChild, visible, dirty)) {
+ DL_LOGV("Skipped frame %p", aChild);
+ return;
+ }
+
+ // Child cannot be transformed since it is not a stacking context.
+ nsDisplayListBuilder::AutoBuildingDisplayList buildingForChild(
+ aBuilder, aChild, visible, dirty, false);
+
+ UpdateCurrentHitTestInfo(aBuilder, aChild);
+
+ aChild->MarkAbsoluteFramesForDisplayList(aBuilder);
+ aBuilder->AdjustWindowDraggingRegion(aChild);
+ aBuilder->Check();
+ aChild->BuildDisplayList(aBuilder, aLists);
+ aChild->SetBuiltDisplayList(true);
+ aBuilder->Check();
+ aBuilder->DisplayCaret(aChild, aLists.Outlines());
+}
+
+static bool ShouldSkipFrame(nsDisplayListBuilder* aBuilder,
+ const nsIFrame* aFrame) {
+ // If painting is restricted to just the background of the top level frame,
+ // then we have nothing to do here.
+ if (aBuilder->IsBackgroundOnly()) {
+ return true;
+ }
+ if (aBuilder->IsForGenerateGlyphMask() &&
+ (!aFrame->IsTextFrame() && aFrame->IsLeaf())) {
+ return true;
+ }
+ // The placeholder frame should have the same content as the OOF frame.
+ if (aBuilder->GetSelectedFramesOnly() &&
+ (aFrame->IsLeaf() && !aFrame->IsSelected())) {
+ return true;
+ }
+ static const nsFrameState skipFlags =
+ (NS_FRAME_TOO_DEEP_IN_FRAME_TREE | NS_FRAME_IS_NONDISPLAY);
+ if (aFrame->HasAnyStateBits(skipFlags)) {
+ return true;
+ }
+ return aFrame->StyleUIReset()->mMozSubtreeHiddenOnlyVisually;
+}
+
+void nsIFrame::BuildDisplayListForChild(nsDisplayListBuilder* aBuilder,
+ nsIFrame* aChild,
+ const nsDisplayListSet& aLists,
+ DisplayChildFlags aFlags) {
+ AutoCheckBuilder check(aBuilder);
+#ifdef DEBUG
+ DL_LOGV("BuildDisplayListForChild (%p) <", aChild);
+ ScopeExit e(
+ [aChild]() { DL_LOGV("> BuildDisplayListForChild (%p)", aChild); });
+#endif
+
+ if (ShouldSkipFrame(aBuilder, aChild)) {
+ return;
+ }
+
+ if (HidesContent()) {
+ return;
+ }
+
+ // If we're generating a display list for printing, include Link items for
+ // frames that correspond to HTML link elements so that we can have active
+ // links in saved PDF output. Note that the state of "within a link" is
+ // set on the display-list builder, such that all descendants of the link
+ // element will generate display-list links.
+ // TODO: we should be able to optimize this so as to avoid creating links
+ // for the same destination that entirely overlap each other, which adds
+ // nothing useful to the final PDF.
+ Maybe<nsDisplayListBuilder::Linkifier> linkifier;
+ if (StaticPrefs::print_save_as_pdf_links_enabled() &&
+ aBuilder->IsForPrinting()) {
+ linkifier.emplace(aBuilder, aChild, aLists.Content());
+ linkifier->MaybeAppendLink(aBuilder, aChild);
+ }
+
+ nsIFrame* child = aChild;
+ auto* placeholder = child->IsPlaceholderFrame()
+ ? static_cast<nsPlaceholderFrame*>(child)
+ : nullptr;
+ nsIFrame* childOrOutOfFlow =
+ placeholder ? placeholder->GetOutOfFlowFrame() : child;
+
+ nsIFrame* parent = childOrOutOfFlow->GetParent();
+ const auto* parentDisplay = parent->StyleDisplay();
+ const auto overflowClipAxes =
+ parent->ShouldApplyOverflowClipping(parentDisplay);
+
+ const bool isPaintingToWindow = aBuilder->IsPaintingToWindow();
+ const bool doingShortcut =
+ isPaintingToWindow &&
+ child->HasAnyStateBits(NS_FRAME_SIMPLE_DISPLAYLIST) &&
+ // Animations may change the stacking context state.
+ // ShouldApplyOverflowClipping is affected by the parent style, which does
+ // not invalidate the NS_FRAME_SIMPLE_DISPLAYLIST bit.
+ !(overflowClipAxes != PhysicalAxes::None ||
+ child->MayHaveTransformAnimation() || child->MayHaveOpacityAnimation());
+
+ if (aBuilder->IsForPainting()) {
+ aBuilder->ClearWillChangeBudgetStatus(child);
+ }
+
+ if (StaticPrefs::layout_css_scroll_anchoring_highlight()) {
+ if (child->FirstContinuation()->IsScrollAnchor()) {
+ nsRect bounds = child->GetContentRectRelativeToSelf() +
+ aBuilder->ToReferenceFrame(child);
+ nsDisplaySolidColor* color = MakeDisplayItem<nsDisplaySolidColor>(
+ aBuilder, child, bounds, NS_RGBA(255, 0, 255, 64));
+ if (color) {
+ color->SetOverrideZIndex(INT32_MAX);
+ aLists.PositionedDescendants()->AppendToTop(color);
+ }
+ }
+ }
+
+ if (doingShortcut) {
+ BuildDisplayListForSimpleChild(aBuilder, child, aLists);
+ return;
+ }
+
+ // dirty rect in child-relative coordinates
+ NS_ASSERTION(aBuilder->GetCurrentFrame() == this, "Wrong coord space!");
+ const nsPoint offset = child->GetOffsetTo(this);
+ nsRect visible = aBuilder->GetVisibleRect() - offset;
+ nsRect dirty = aBuilder->GetDirtyRect() - offset;
+
+ nsDisplayListBuilder::OutOfFlowDisplayData* savedOutOfFlowData = nullptr;
+ if (placeholder) {
+ if (placeholder->HasAnyStateBits(PLACEHOLDER_FOR_TOPLAYER)) {
+ // If the out-of-flow frame is in the top layer, the viewport frame
+ // will paint it. Skip it here. Note that, only out-of-flow frames
+ // with this property should be skipped, because non-HTML elements
+ // may stop their children from being out-of-flow. Those frames
+ // should still be handled in the normal in-flow path.
+ return;
+ }
+
+ child = childOrOutOfFlow;
+ if (aBuilder->IsForPainting()) {
+ aBuilder->ClearWillChangeBudgetStatus(child);
+ }
+
+ // If 'child' is a pushed float then it's owned by a block that's not an
+ // ancestor of the placeholder, and it will be painted by that block and
+ // should not be painted through the placeholder. Also recheck
+ // NS_FRAME_TOO_DEEP_IN_FRAME_TREE and NS_FRAME_IS_NONDISPLAY.
+ static const nsFrameState skipFlags =
+ (NS_FRAME_IS_PUSHED_FLOAT | NS_FRAME_TOO_DEEP_IN_FRAME_TREE |
+ NS_FRAME_IS_NONDISPLAY);
+ if (child->HasAnyStateBits(skipFlags) || nsLayoutUtils::IsPopup(child)) {
+ return;
+ }
+
+ MOZ_ASSERT(child->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW));
+ savedOutOfFlowData = nsDisplayListBuilder::GetOutOfFlowData(child);
+
+ if (aBuilder->GetIncludeAllOutOfFlows()) {
+ visible = child->InkOverflowRect();
+ dirty = child->InkOverflowRect();
+ } else if (savedOutOfFlowData) {
+ visible =
+ savedOutOfFlowData->GetVisibleRectForFrame(aBuilder, child, &dirty);
+ } else {
+ // The out-of-flow frame did not intersect the dirty area. We may still
+ // need to traverse into it, since it may contain placeholders we need
+ // to enter to reach other out-of-flow frames that are visible.
+ visible.SetEmpty();
+ dirty.SetEmpty();
+ }
+ }
+
+ NS_ASSERTION(!child->IsPlaceholderFrame(),
+ "Should have dealt with placeholders already");
+
+ if (!DescendIntoChild(aBuilder, child, visible, dirty)) {
+ DL_LOGV("Skipped frame %p", child);
+ return;
+ }
+
+ const bool isSVG = child->HasAnyStateBits(NS_FRAME_SVG_LAYOUT);
+
+ // This flag is raised if the control flow strays off the common path.
+ // The common path is the most common one of THE COMMON CASE mentioned later.
+ bool awayFromCommonPath = !isPaintingToWindow;
+
+ // true if this is a real or pseudo stacking context
+ bool pseudoStackingContext =
+ aFlags.contains(DisplayChildFlag::ForcePseudoStackingContext);
+
+ if (!pseudoStackingContext && !isSVG &&
+ aFlags.contains(DisplayChildFlag::Inline) &&
+ !child->IsLineParticipant()) {
+ // child is a non-inline frame in an inline context, i.e.,
+ // it acts like inline-block or inline-table. Therefore it is a
+ // pseudo-stacking-context.
+ pseudoStackingContext = true;
+ }
+
+ const nsStyleDisplay* ourDisp = StyleDisplay();
+ // Don't paint our children if the theme object is a leaf.
+ if (IsThemed(ourDisp) && !PresContext()->Theme()->WidgetIsContainer(
+ ourDisp->EffectiveAppearance())) {
+ return;
+ }
+
+ // Since we're now sure that we're adding this frame to the display list
+ // (which means we're painting it, modulo occlusion), mark it as visible
+ // within the displayport.
+ if (isPaintingToWindow && child->TrackingVisibility() &&
+ child->IsVisibleForPainting()) {
+ child->PresShell()->EnsureFrameInApproximatelyVisibleList(child);
+ awayFromCommonPath = true;
+ }
+
+ // Child is composited if it's transformed, partially transparent, or has
+ // SVG effects or a blend mode..
+ const nsStyleDisplay* disp = child->StyleDisplay();
+ const nsStyleEffects* effects = child->StyleEffects();
+
+ const bool isPositioned = disp->IsPositionedStyle();
+ const bool isStackingContext =
+ aFlags.contains(DisplayChildFlag::ForceStackingContext) ||
+ child->IsStackingContext(disp, effects);
+
+ if (pseudoStackingContext || isStackingContext || isPositioned ||
+ placeholder || (!isSVG && disp->IsFloating(child)) ||
+ (isSVG && effects->mClip.IsRect() && IsSVGContentWithCSSClip(child))) {
+ pseudoStackingContext = true;
+ awayFromCommonPath = true;
+ }
+
+ NS_ASSERTION(!isStackingContext || pseudoStackingContext,
+ "Stacking contexts must also be pseudo-stacking-contexts");
+
+ nsDisplayListBuilder::AutoBuildingDisplayList buildingForChild(
+ aBuilder, child, visible, dirty);
+
+ UpdateCurrentHitTestInfo(aBuilder, child);
+
+ DisplayListClipState::AutoClipMultiple clipState(aBuilder);
+ nsDisplayListBuilder::AutoCurrentActiveScrolledRootSetter asrSetter(aBuilder);
+
+ if (savedOutOfFlowData) {
+ aBuilder->SetBuildingInvisibleItems(false);
+
+ clipState.SetClipChainForContainingBlockDescendants(
+ savedOutOfFlowData->mContainingBlockClipChain);
+ asrSetter.SetCurrentActiveScrolledRoot(
+ savedOutOfFlowData->mContainingBlockActiveScrolledRoot);
+ asrSetter.SetCurrentScrollParentId(savedOutOfFlowData->mScrollParentId);
+ MOZ_ASSERT(awayFromCommonPath,
+ "It is impossible when savedOutOfFlowData is true");
+ } else if (HasAnyStateBits(NS_FRAME_FORCE_DISPLAY_LIST_DESCEND_INTO) &&
+ placeholder) {
+ NS_ASSERTION(visible.IsEmpty(), "should have empty visible rect");
+ // Every item we build from now until we descent into an out of flow that
+ // does have saved out of flow data should be invisible. This state gets
+ // restored when AutoBuildingDisplayList gets out of scope.
+ aBuilder->SetBuildingInvisibleItems(true);
+
+ // If we have nested out-of-flow frames and the outer one isn't visible
+ // then we won't have stored clip data for it. We can just clear the clip
+ // instead since we know we won't render anything, and the inner out-of-flow
+ // frame will setup the correct clip for itself.
+ clipState.SetClipChainForContainingBlockDescendants(nullptr);
+ }
+
+ // Setup clipping for the parent's overflow:clip,
+ // or overflow:hidden on elements that don't support scrolling (and therefore
+ // don't create nsHTML/XULScrollFrame). This clipping needs to not clip
+ // anything directly rendered by the parent, only the rendering of its
+ // children.
+ // Don't use overflowClip to restrict the dirty rect, since some of the
+ // descendants may not be clipped by it. Even if we end up with unnecessary
+ // display items, they'll be pruned during ComputeVisibility.
+ //
+ // FIXME(emilio): Why can't we handle this more similarly to `clip` (on the
+ // parent, rather than on the children)? Would ClipContentDescendants do what
+ // we want?
+ if (overflowClipAxes != PhysicalAxes::None) {
+ ApplyOverflowClipping(aBuilder, parent, overflowClipAxes, clipState);
+ awayFromCommonPath = true;
+ }
+
+ nsDisplayList list(aBuilder);
+ nsDisplayList extraPositionedDescendants(aBuilder);
+ const ActiveScrolledRoot* wrapListASR;
+ bool builtContainerItem = false;
+ if (isStackingContext) {
+ // True stacking context.
+ // For stacking contexts, BuildDisplayListForStackingContext handles
+ // clipping and MarkAbsoluteFramesForDisplayList.
+ nsDisplayListBuilder::AutoContainerASRTracker contASRTracker(aBuilder);
+ child->BuildDisplayListForStackingContext(aBuilder, &list,
+ &builtContainerItem);
+ wrapListASR = contASRTracker.GetContainerASR();
+ if (!aBuilder->IsReusingStackingContextItems() &&
+ aBuilder->GetCaretFrame() == child) {
+ builtContainerItem = false;
+ }
+ } else {
+ Maybe<nsRect> clipPropClip =
+ child->GetClipPropClipRect(disp, effects, child->GetSize());
+ if (clipPropClip) {
+ aBuilder->IntersectVisibleRect(*clipPropClip);
+ aBuilder->IntersectDirtyRect(*clipPropClip);
+ clipState.ClipContentDescendants(*clipPropClip +
+ aBuilder->ToReferenceFrame(child));
+ awayFromCommonPath = true;
+ }
+
+ child->MarkAbsoluteFramesForDisplayList(aBuilder);
+ child->SetBuiltDisplayList(true);
+
+ // Some SVG frames might change opacity without invalidating the frame, so
+ // exclude them from the fast-path.
+ if (!awayFromCommonPath && !child->IsSVGFrame()) {
+ // The shortcut is available for the child for next time.
+ child->AddStateBits(NS_FRAME_SIMPLE_DISPLAYLIST);
+ }
+
+ if (!pseudoStackingContext) {
+ // THIS IS THE COMMON CASE.
+ // Not a pseudo or real stacking context. Do the simple thing and
+ // return early.
+ aBuilder->AdjustWindowDraggingRegion(child);
+ aBuilder->Check();
+ child->BuildDisplayList(aBuilder, aLists);
+ aBuilder->Check();
+ aBuilder->DisplayCaret(child, aLists.Outlines());
+ return;
+ }
+
+ // A pseudo-stacking context (e.g., a positioned element with z-index auto).
+ // We allow positioned descendants of the child to escape to our parent
+ // stacking context's positioned descendant list, because they might be
+ // z-index:non-auto
+ nsDisplayListCollection pseudoStack(aBuilder);
+
+ aBuilder->AdjustWindowDraggingRegion(child);
+ nsDisplayListBuilder::AutoContainerASRTracker contASRTracker(aBuilder);
+ aBuilder->Check();
+ child->BuildDisplayList(aBuilder, pseudoStack);
+ aBuilder->Check();
+ if (aBuilder->DisplayCaret(child, pseudoStack.Outlines())) {
+ builtContainerItem = false;
+ }
+ wrapListASR = contASRTracker.GetContainerASR();
+
+ list.AppendToTop(pseudoStack.BorderBackground());
+ list.AppendToTop(pseudoStack.BlockBorderBackgrounds());
+ list.AppendToTop(pseudoStack.Floats());
+ list.AppendToTop(pseudoStack.Content());
+ list.AppendToTop(pseudoStack.Outlines());
+ extraPositionedDescendants.AppendToTop(pseudoStack.PositionedDescendants());
+ }
+
+ buildingForChild.RestoreBuildingInvisibleItemsValue();
+
+ if (!list.IsEmpty()) {
+ if (isPositioned || isStackingContext) {
+ // Genuine stacking contexts, and positioned pseudo-stacking-contexts,
+ // go in this level.
+ nsDisplayItem* item = WrapInWrapList(aBuilder, child, &list, wrapListASR,
+ builtContainerItem);
+ if (isSVG) {
+ aLists.Content()->AppendToTop(item);
+ } else {
+ aLists.PositionedDescendants()->AppendToTop(item);
+ }
+ } else if (!isSVG && disp->IsFloating(child)) {
+ aLists.Floats()->AppendToTop(
+ WrapInWrapList(aBuilder, child, &list, wrapListASR));
+ } else {
+ aLists.Content()->AppendToTop(&list);
+ }
+ }
+ // We delay placing the positioned descendants of positioned frames to here,
+ // because in the absence of z-index this is the correct order for them.
+ // This doesn't affect correctness because the positioned descendants list
+ // is sorted by z-order and content in BuildDisplayListForStackingContext,
+ // but it means that sort routine needs to do less work.
+ aLists.PositionedDescendants()->AppendToTop(&extraPositionedDescendants);
+}
+
+void nsIFrame::MarkAbsoluteFramesForDisplayList(
+ nsDisplayListBuilder* aBuilder) {
+ if (IsAbsoluteContainer()) {
+ aBuilder->MarkFramesForDisplayList(
+ this, GetAbsoluteContainingBlock()->GetChildList());
+ }
+}
+
+nsresult nsIFrame::GetContentForEvent(const WidgetEvent* aEvent,
+ nsIContent** aContent) {
+ nsIFrame* f = nsLayoutUtils::GetNonGeneratedAncestor(this);
+ *aContent = f->GetContent();
+ NS_IF_ADDREF(*aContent);
+ return NS_OK;
+}
+
+void nsIFrame::FireDOMEvent(const nsAString& aDOMEventName,
+ nsIContent* aContent) {
+ nsIContent* target = aContent ? aContent : GetContent();
+
+ if (target) {
+ RefPtr<AsyncEventDispatcher> asyncDispatcher = new AsyncEventDispatcher(
+ target, aDOMEventName, CanBubble::eYes, ChromeOnlyDispatch::eNo);
+ DebugOnly<nsresult> rv = asyncDispatcher->PostDOMEvent();
+ NS_ASSERTION(NS_SUCCEEDED(rv), "AsyncEventDispatcher failed to dispatch");
+ }
+}
+
+nsresult nsIFrame::HandleEvent(nsPresContext* aPresContext,
+ WidgetGUIEvent* aEvent,
+ nsEventStatus* aEventStatus) {
+ if (aEvent->mMessage == eMouseMove) {
+ // XXX If the second argument of HandleDrag() is WidgetMouseEvent,
+ // the implementation becomes simpler.
+ return HandleDrag(aPresContext, aEvent, aEventStatus);
+ }
+
+ if ((aEvent->mClass == eMouseEventClass &&
+ aEvent->AsMouseEvent()->mButton == MouseButton::ePrimary) ||
+ aEvent->mClass == eTouchEventClass) {
+ if (aEvent->mMessage == eMouseDown || aEvent->mMessage == eTouchStart) {
+ HandlePress(aPresContext, aEvent, aEventStatus);
+ } else if (aEvent->mMessage == eMouseUp || aEvent->mMessage == eTouchEnd) {
+ HandleRelease(aPresContext, aEvent, aEventStatus);
+ }
+ return NS_OK;
+ }
+
+ // When secondary buttion is down, we need to move selection to make users
+ // possible to paste something at click point quickly.
+ // When middle button is down, we need to just move selection and focus at
+ // the clicked point. Note that even if middle click paste is not enabled,
+ // Chrome moves selection at middle mouse button down. So, we should follow
+ // the behavior for the compatibility.
+ if (aEvent->mMessage == eMouseDown) {
+ WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent();
+ if (mouseEvent && (mouseEvent->mButton == MouseButton::eSecondary ||
+ mouseEvent->mButton == MouseButton::eMiddle)) {
+ if (*aEventStatus == nsEventStatus_eConsumeNoDefault) {
+ return NS_OK;
+ }
+ return MoveCaretToEventPoint(aPresContext, mouseEvent, aEventStatus);
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult nsIFrame::GetDataForTableSelection(
+ const nsFrameSelection* aFrameSelection, mozilla::PresShell* aPresShell,
+ WidgetMouseEvent* aMouseEvent, nsIContent** aParentContent,
+ int32_t* aContentOffset, TableSelectionMode* aTarget) {
+ if (!aFrameSelection || !aPresShell || !aMouseEvent || !aParentContent ||
+ !aContentOffset || !aTarget)
+ return NS_ERROR_NULL_POINTER;
+
+ *aParentContent = nullptr;
+ *aContentOffset = 0;
+ *aTarget = TableSelectionMode::None;
+
+ int16_t displaySelection = aPresShell->GetSelectionFlags();
+
+ bool selectingTableCells = aFrameSelection->IsInTableSelectionMode();
+
+ // DISPLAY_ALL means we're in an editor.
+ // If already in cell selection mode,
+ // continue selecting with mouse drag or end on mouse up,
+ // or when using shift key to extend block of cells
+ // (Mouse down does normal selection unless Ctrl/Cmd is pressed)
+ bool doTableSelection =
+ displaySelection == nsISelectionDisplay::DISPLAY_ALL &&
+ selectingTableCells &&
+ (aMouseEvent->mMessage == eMouseMove ||
+ (aMouseEvent->mMessage == eMouseUp &&
+ aMouseEvent->mButton == MouseButton::ePrimary) ||
+ aMouseEvent->IsShift());
+
+ if (!doTableSelection) {
+ // In Browser, special 'table selection' key must be pressed for table
+ // selection or when just Shift is pressed and we're already in table/cell
+ // selection mode
+#ifdef XP_MACOSX
+ doTableSelection = aMouseEvent->IsMeta() ||
+ (aMouseEvent->IsShift() && selectingTableCells);
+#else
+ doTableSelection = aMouseEvent->IsControl() ||
+ (aMouseEvent->IsShift() && selectingTableCells);
+#endif
+ }
+ if (!doTableSelection) return NS_OK;
+
+ // Get the cell frame or table frame (or parent) of the current content node
+ nsIFrame* frame = this;
+ bool foundCell = false;
+ bool foundTable = false;
+
+ // Get the limiting node to stop parent frame search
+ nsIContent* limiter = aFrameSelection->GetLimiter();
+
+ // If our content node is an ancestor of the limiting node,
+ // we should stop the search right now.
+ if (limiter && limiter->IsInclusiveDescendantOf(GetContent())) return NS_OK;
+
+ // We don't initiate row/col selection from here now,
+ // but we may in future
+ // bool selectColumn = false;
+ // bool selectRow = false;
+
+ while (frame) {
+ // Check for a table cell by querying to a known CellFrame interface
+ nsITableCellLayout* cellElement = do_QueryFrame(frame);
+ if (cellElement) {
+ foundCell = true;
+ // TODO: If we want to use proximity to top or left border
+ // for row and column selection, this is the place to do it
+ break;
+ } else {
+ // If not a cell, check for table
+ // This will happen when starting frame is the table or child of a table,
+ // such as a row (we were inbetween cells or in table border)
+ nsTableWrapperFrame* tableFrame = do_QueryFrame(frame);
+ if (tableFrame) {
+ foundTable = true;
+ // TODO: How can we select row when along left table edge
+ // or select column when along top edge?
+ break;
+ } else {
+ frame = frame->GetParent();
+ // Stop if we have hit the selection's limiting content node
+ if (frame && frame->GetContent() == limiter) break;
+ }
+ }
+ }
+ // We aren't in a cell or table
+ if (!foundCell && !foundTable) return NS_OK;
+
+ nsIContent* tableOrCellContent = frame->GetContent();
+ if (!tableOrCellContent) return NS_ERROR_FAILURE;
+
+ nsCOMPtr<nsIContent> parentContent = tableOrCellContent->GetParent();
+ if (!parentContent) return NS_ERROR_FAILURE;
+
+ const int32_t offset =
+ parentContent->ComputeIndexOf_Deprecated(tableOrCellContent);
+ // Not likely?
+ if (offset < 0) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Everything is OK -- set the return values
+ parentContent.forget(aParentContent);
+
+ *aContentOffset = offset;
+
+#if 0
+ if (selectRow)
+ *aTarget = TableSelectionMode::Row;
+ else if (selectColumn)
+ *aTarget = TableSelectionMode::Column;
+ else
+#endif
+ if (foundCell) {
+ *aTarget = TableSelectionMode::Cell;
+ } else if (foundTable) {
+ *aTarget = TableSelectionMode::Table;
+ }
+
+ return NS_OK;
+}
+
+static bool IsEditingHost(const nsIFrame* aFrame) {
+ nsIContent* content = aFrame->GetContent();
+ return content && content->IsEditingHost();
+}
+
+static StyleUserSelect UsedUserSelect(const nsIFrame* aFrame) {
+ if (aFrame->IsGeneratedContentFrame()) {
+ return StyleUserSelect::None;
+ }
+
+ // Per https://drafts.csswg.org/css-ui-4/#content-selection:
+ //
+ // The used value is the same as the computed value, except:
+ //
+ // 1 - on editable elements where the used value is always 'contain'
+ // regardless of the computed value
+ // 2 - when the computed value is auto, in which case the used value is one
+ // of the other values...
+ //
+ // See https://github.com/w3c/csswg-drafts/issues/3344 to see why we do this
+ // at used-value time instead of at computed-value time.
+
+ if (aFrame->IsTextInputFrame() || IsEditingHost(aFrame)) {
+ // We don't implement 'contain' itself, but we make 'text' behave as
+ // 'contain' for contenteditable and <input> / <textarea> elements anyway so
+ // this is ok.
+ return StyleUserSelect::Text;
+ }
+
+ auto style = aFrame->Style()->UserSelect();
+ if (style != StyleUserSelect::Auto) {
+ return style;
+ }
+
+ auto* parent = nsLayoutUtils::GetParentOrPlaceholderFor(aFrame);
+ return parent ? UsedUserSelect(parent) : StyleUserSelect::Text;
+}
+
+bool nsIFrame::IsSelectable(StyleUserSelect* aSelectStyle) const {
+ auto style = UsedUserSelect(this);
+ if (aSelectStyle) {
+ *aSelectStyle = style;
+ }
+ return style != StyleUserSelect::None;
+}
+
+bool nsIFrame::ShouldHaveLineIfEmpty() const {
+ if (Style()->IsPseudoOrAnonBox() &&
+ Style()->GetPseudoType() != PseudoStyleType::scrolledContent) {
+ return false;
+ }
+ return IsEditingHost(this);
+}
+
+/**
+ * Handles the Mouse Press Event for the frame
+ */
+NS_IMETHODIMP
+nsIFrame::HandlePress(nsPresContext* aPresContext, WidgetGUIEvent* aEvent,
+ nsEventStatus* aEventStatus) {
+ NS_ENSURE_ARG_POINTER(aEventStatus);
+ if (nsEventStatus_eConsumeNoDefault == *aEventStatus) {
+ return NS_OK;
+ }
+
+ NS_ENSURE_ARG_POINTER(aEvent);
+ if (aEvent->mClass == eTouchEventClass) {
+ return NS_OK;
+ }
+
+ return MoveCaretToEventPoint(aPresContext, aEvent->AsMouseEvent(),
+ aEventStatus);
+}
+
+nsresult nsIFrame::MoveCaretToEventPoint(nsPresContext* aPresContext,
+ WidgetMouseEvent* aMouseEvent,
+ nsEventStatus* aEventStatus) {
+ MOZ_ASSERT(aPresContext);
+ MOZ_ASSERT(aMouseEvent);
+ MOZ_ASSERT(aMouseEvent->mMessage == eMouseDown);
+ MOZ_ASSERT(aEventStatus);
+ MOZ_ASSERT(nsEventStatus_eConsumeNoDefault != *aEventStatus);
+
+ mozilla::PresShell* presShell = aPresContext->GetPresShell();
+ if (!presShell) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // We often get out of sync state issues with mousedown events that
+ // get interrupted by alerts/dialogs.
+ // Check with the ESM to see if we should process this one
+ if (!aPresContext->EventStateManager()->EventStatusOK(aMouseEvent)) {
+ return NS_OK;
+ }
+
+ const nsPoint pt = nsLayoutUtils::GetEventCoordinatesRelativeTo(
+ aMouseEvent, RelativeTo{this});
+
+ // When not using `alt`, and clicking on a draggable, but non-editable
+ // element, don't do anything, and let d&d handle the event.
+ //
+ // See bug 48876, bug 388659 and bug 55921 for context here.
+ //
+ // FIXME(emilio): The .Contains(pt) check looks a bit fishy. When would it be
+ // false given we're the event target? If it is needed, why not checking the
+ // actual draggable node rect instead?
+ if (!aMouseEvent->IsAlt() && GetRectRelativeToSelf().Contains(pt)) {
+ for (nsIContent* content = mContent; content;
+ content = content->GetFlattenedTreeParent()) {
+ if (nsContentUtils::ContentIsDraggable(content) &&
+ !content->IsEditable()) {
+ return NS_OK;
+ }
+ }
+ }
+
+ // If we are in Navigator and the click is in a draggable node, we don't want
+ // to start selection because we don't want to interfere with a potential
+ // drag of said node and steal all its glory.
+ const bool isEditor =
+ presShell->GetSelectionFlags() == nsISelectionDisplay::DISPLAY_ALL;
+
+ // Don't do something if it's middle button down event.
+ const bool isPrimaryButtonDown =
+ aMouseEvent->mButton == MouseButton::ePrimary;
+
+ // check whether style allows selection
+ // if not, don't tell selection the mouse event even occurred.
+ StyleUserSelect selectStyle;
+ // check for select: none
+ if (!IsSelectable(&selectStyle)) {
+ return NS_OK;
+ }
+
+ if (isPrimaryButtonDown) {
+ // If the mouse is dragged outside the nearest enclosing scrollable area
+ // while making a selection, the area will be scrolled. To do this, capture
+ // the mouse on the nearest scrollable frame. If there isn't a scrollable
+ // frame, or something else is already capturing the mouse, there's no
+ // reason to capture.
+ if (!PresShell::GetCapturingContent()) {
+ nsIScrollableFrame* scrollFrame =
+ nsLayoutUtils::GetNearestScrollableFrame(
+ this, nsLayoutUtils::SCROLLABLE_SAME_DOC |
+ nsLayoutUtils::SCROLLABLE_INCLUDE_HIDDEN);
+ if (scrollFrame) {
+ nsIFrame* capturingFrame = do_QueryFrame(scrollFrame);
+ PresShell::SetCapturingContent(capturingFrame->GetContent(),
+ CaptureFlags::IgnoreAllowedState);
+ }
+ }
+ }
+
+ // XXX This is screwy; it really should use the selection frame, not the
+ // event frame
+ const nsFrameSelection* frameselection =
+ selectStyle == StyleUserSelect::Text ? GetConstFrameSelection()
+ : presShell->ConstFrameSelection();
+
+ if (!frameselection || frameselection->GetDisplaySelection() ==
+ nsISelectionController::SELECTION_OFF) {
+ return NS_OK; // nothing to do we cannot affect selection from here
+ }
+
+#ifdef XP_MACOSX
+ // If Control key is pressed on macOS, it should be treated as right click.
+ // So, don't change selection.
+ if (aMouseEvent->IsControl()) {
+ return NS_OK;
+ }
+ const bool control = aMouseEvent->IsMeta();
+#else
+ const bool control = aMouseEvent->IsControl();
+#endif
+
+ RefPtr<nsFrameSelection> fc = const_cast<nsFrameSelection*>(frameselection);
+ if (isPrimaryButtonDown && aMouseEvent->mClickCount > 1) {
+ // These methods aren't const but can't actually delete anything,
+ // so no need for AutoWeakFrame.
+ fc->SetDragState(true);
+ return HandleMultiplePress(aPresContext, aMouseEvent, aEventStatus,
+ control);
+ }
+
+ ContentOffsets offsets = GetContentOffsetsFromPoint(pt, SKIP_HIDDEN);
+
+ if (!offsets.content) {
+ return NS_ERROR_FAILURE;
+ }
+
+ const bool isSecondaryButton =
+ aMouseEvent->mButton == MouseButton::eSecondary;
+ if (isSecondaryButton &&
+ !MovingCaretToEventPointAllowedIfSecondaryButtonEvent(
+ *frameselection, *aMouseEvent, *offsets.content,
+ // When we collapse selection in nsFrameSelection::TakeFocus,
+ // we always collapse selection to the start offset. Therefore,
+ // we can ignore the end offset here. E.g., when an <img> is clicked,
+ // set the primary offset to after it, but the the secondary offset
+ // may be before it, see OffsetsForSingleFrame for the detail.
+ offsets.StartOffset())) {
+ return NS_OK;
+ }
+
+ if (aMouseEvent->mMessage == eMouseDown &&
+ aMouseEvent->mButton == MouseButton::eMiddle &&
+ !offsets.content->IsEditable()) {
+ // However, some users don't like the Chrome compatible behavior of
+ // middle mouse click. They want to keep selection after starting
+ // autoscroll. However, the selection change is important for middle
+ // mouse past. Therefore, we should allow users to take the traditional
+ // behavior back by themselves unless middle click paste is enabled or
+ // autoscrolling is disabled.
+ if (!Preferences::GetBool("middlemouse.paste", false) &&
+ Preferences::GetBool("general.autoScroll", false) &&
+ Preferences::GetBool("general.autoscroll.prevent_to_collapse_selection_"
+ "by_middle_mouse_down",
+ false)) {
+ return NS_OK;
+ }
+ }
+
+ if (isPrimaryButtonDown) {
+ // Let Ctrl/Cmd + left mouse down do table selection instead of drag
+ // initiation.
+ nsCOMPtr<nsIContent> parentContent;
+ int32_t contentOffset;
+ TableSelectionMode target;
+ nsresult rv = GetDataForTableSelection(
+ frameselection, presShell, aMouseEvent, getter_AddRefs(parentContent),
+ &contentOffset, &target);
+ if (NS_SUCCEEDED(rv) && parentContent) {
+ fc->SetDragState(true);
+ return fc->HandleTableSelection(parentContent, contentOffset, target,
+ aMouseEvent);
+ }
+ }
+
+ fc->SetDelayedCaretData(0);
+
+ if (isPrimaryButtonDown) {
+ // Check if any part of this frame is selected, and if the user clicked
+ // inside the selected region, and if it's the left button. If so, we delay
+ // starting a new selection since the user may be trying to drag the
+ // selected region to some other app.
+
+ if (GetContent() && GetContent()->IsMaybeSelected()) {
+ bool inSelection = false;
+ UniquePtr<SelectionDetails> details = frameselection->LookUpSelection(
+ offsets.content, 0, offsets.EndOffset(), false);
+
+ //
+ // If there are any details, check to see if the user clicked
+ // within any selected region of the frame.
+ //
+
+ for (SelectionDetails* curDetail = details.get(); curDetail;
+ curDetail = curDetail->mNext.get()) {
+ //
+ // If the user clicked inside a selection, then just
+ // return without doing anything. We will handle placing
+ // the caret later on when the mouse is released. We ignore
+ // the spellcheck, find and url formatting selections.
+ //
+ if (curDetail->mSelectionType != SelectionType::eSpellCheck &&
+ curDetail->mSelectionType != SelectionType::eFind &&
+ curDetail->mSelectionType != SelectionType::eURLSecondary &&
+ curDetail->mSelectionType != SelectionType::eURLStrikeout &&
+ curDetail->mSelectionType != SelectionType::eHighlight &&
+ curDetail->mStart <= offsets.StartOffset() &&
+ offsets.EndOffset() <= curDetail->mEnd) {
+ inSelection = true;
+ }
+ }
+
+ if (inSelection) {
+ fc->SetDragState(false);
+ fc->SetDelayedCaretData(aMouseEvent);
+ return NS_OK;
+ }
+ }
+
+ fc->SetDragState(true);
+ }
+
+ // Do not touch any nsFrame members after this point without adding
+ // weakFrame checks.
+ const nsFrameSelection::FocusMode focusMode = [&]() {
+ // If "Shift" and "Ctrl" are both pressed, "Shift" is given precedence. This
+ // mimics the old behaviour.
+ const bool isShift =
+ aMouseEvent->IsShift() &&
+ // If Shift + secondary button press shoud open context menu without a
+ // contextmenu event, user wants to open context menu like as a
+ // secondary button press without Shift key.
+ !(isSecondaryButton &&
+ StaticPrefs::dom_event_contextmenu_shift_suppresses_event());
+ if (isShift) {
+ // If clicked in a link when focused content is editable, we should
+ // collapse selection in the link for compatibility with Blink.
+ if (isEditor) {
+ for (Element* element : mContent->InclusiveAncestorsOfType<Element>()) {
+ if (element->IsLink()) {
+ return nsFrameSelection::FocusMode::kCollapseToNewPoint;
+ }
+ }
+ }
+ return nsFrameSelection::FocusMode::kExtendSelection;
+ }
+
+ if (isPrimaryButtonDown && control) {
+ return nsFrameSelection::FocusMode::kMultiRangeSelection;
+ }
+
+ return nsFrameSelection::FocusMode::kCollapseToNewPoint;
+ }();
+
+ nsresult rv = fc->HandleClick(
+ MOZ_KnownLive(offsets.content) /* bug 1636889 */, offsets.StartOffset(),
+ offsets.EndOffset(), focusMode, offsets.associate);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // We don't handle mouse button up if it's middle button.
+ if (isPrimaryButtonDown && offsets.offset != offsets.secondaryOffset) {
+ fc->MaintainSelection();
+ }
+
+ if (isPrimaryButtonDown && isEditor && !aMouseEvent->IsShift() &&
+ (offsets.EndOffset() - offsets.StartOffset()) == 1) {
+ // A single node is selected and we aren't extending an existing selection,
+ // which means the user clicked directly on an object (either
+ // `user-select: all` or a non-text node without children). Therefore,
+ // disable selection extension during mouse moves.
+ // XXX This is a bit hacky; shouldn't editor be able to deal with this?
+ fc->SetDragState(false);
+ }
+
+ return NS_OK;
+}
+
+bool nsIFrame::MovingCaretToEventPointAllowedIfSecondaryButtonEvent(
+ const nsFrameSelection& aFrameSelection,
+ WidgetMouseEvent& aSecondaryButtonEvent,
+ const nsIContent& aContentAtEventPoint, int32_t aOffsetAtEventPoint) const {
+ MOZ_ASSERT(aSecondaryButtonEvent.mButton == MouseButton::eSecondary);
+
+ if (NS_WARN_IF(aOffsetAtEventPoint < 0)) {
+ return false;
+ }
+
+ const bool contentIsEditable = aContentAtEventPoint.IsEditable();
+ const TextControlElement* const contentAsTextControl =
+ TextControlElement::FromNodeOrNull(
+ aContentAtEventPoint.IsTextControlElement()
+ ? &aContentAtEventPoint
+ : aContentAtEventPoint.GetClosestNativeAnonymousSubtreeRoot());
+ if (Selection* selection =
+ aFrameSelection.GetSelection(SelectionType::eNormal)) {
+ const bool selectionIsCollapsed = selection->IsCollapsed();
+ // If right click in a selection range, we should not collapse selection.
+ if (!selectionIsCollapsed &&
+ nsContentUtils::IsPointInSelection(
+ *selection, aContentAtEventPoint,
+ static_cast<uint32_t>(aOffsetAtEventPoint))) {
+ return false;
+ }
+ const bool wantToPreventMoveCaret =
+ StaticPrefs::
+ ui_mouse_right_click_move_caret_stop_if_in_focused_editable_node() &&
+ selectionIsCollapsed && (contentIsEditable || contentAsTextControl);
+ const bool wantToPreventCollapseSelection =
+ StaticPrefs::
+ ui_mouse_right_click_collapse_selection_stop_if_non_collapsed_selection() &&
+ !selectionIsCollapsed;
+ if (wantToPreventMoveCaret || wantToPreventCollapseSelection) {
+ // If currently selection is limited in an editing host, we should not
+ // collapse selection nor move caret if the clicked point is in the
+ // ancestor limiter. Otherwise, this mouse click moves focus from the
+ // editing host to different one or blur the editing host. In this case,
+ // we need to update selection because keeping current selection in the
+ // editing host looks like it's not blurred.
+ // FIXME: If the active editing host is the document element, editor
+ // does not set ancestor limiter properly. Fix it in the editor side.
+ if (nsIContent* ancestorLimiter = selection->GetAncestorLimiter()) {
+ MOZ_ASSERT(ancestorLimiter->IsEditable());
+ return !aContentAtEventPoint.IsInclusiveDescendantOf(ancestorLimiter);
+ }
+ }
+ // If selection is editable and `stop_if_in_focused_editable_node` pref is
+ // set to true, user does not want to move caret to right click place if
+ // clicked in the focused text control element.
+ if (wantToPreventMoveCaret && contentAsTextControl &&
+ contentAsTextControl == nsFocusManager::GetFocusedElementStatic()) {
+ return false;
+ }
+ // If currently selection is not limited in an editing host, we should
+ // collapse selection only when this click moves focus to an editing
+ // host because we need to update selection in this case.
+ if (wantToPreventCollapseSelection && !contentIsEditable) {
+ return false;
+ }
+ }
+
+ return !StaticPrefs::
+ ui_mouse_right_click_collapse_selection_stop_if_non_editable_node() ||
+ // The user does not want to collapse selection into non-editable
+ // content by a right button click.
+ contentIsEditable ||
+ // Treat clicking in a text control as always clicked on editable
+ // content because we want a hack only for clicking in normal text
+ // nodes which is outside any editing hosts.
+ contentAsTextControl;
+}
+
+nsresult nsIFrame::SelectByTypeAtPoint(nsPresContext* aPresContext,
+ const nsPoint& aPoint,
+ nsSelectionAmount aBeginAmountType,
+ nsSelectionAmount aEndAmountType,
+ uint32_t aSelectFlags) {
+ NS_ENSURE_ARG_POINTER(aPresContext);
+
+ // No point in selecting if selection is turned off
+ if (DetermineDisplaySelection() == nsISelectionController::SELECTION_OFF) {
+ return NS_OK;
+ }
+
+ ContentOffsets offsets = GetContentOffsetsFromPoint(
+ aPoint, SKIP_HIDDEN | IGNORE_NATIVE_ANONYMOUS_SUBTREE);
+ if (!offsets.content) {
+ return NS_ERROR_FAILURE;
+ }
+
+ uint32_t offset;
+ nsIFrame* frame = SelectionMovementUtils::GetFrameForNodeOffset(
+ offsets.content, offsets.offset, offsets.associate, &offset);
+ if (!frame) {
+ return NS_ERROR_FAILURE;
+ }
+ return frame->PeekBackwardAndForward(
+ aBeginAmountType, aEndAmountType, static_cast<int32_t>(offset),
+ aBeginAmountType != eSelectWord, aSelectFlags);
+}
+
+/**
+ * Multiple Mouse Press -- line or paragraph selection -- for the frame.
+ * Wouldn't it be nice if this didn't have to be hardwired into Frame code?
+ */
+NS_IMETHODIMP
+nsIFrame::HandleMultiplePress(nsPresContext* aPresContext,
+ WidgetGUIEvent* aEvent,
+ nsEventStatus* aEventStatus, bool aControlHeld) {
+ NS_ENSURE_ARG_POINTER(aEvent);
+ NS_ENSURE_ARG_POINTER(aEventStatus);
+
+ if (nsEventStatus_eConsumeNoDefault == *aEventStatus ||
+ DetermineDisplaySelection() == nsISelectionController::SELECTION_OFF) {
+ return NS_OK;
+ }
+
+ // Find out whether we're doing line or paragraph selection.
+ // If browser.triple_click_selects_paragraph is true, triple-click selects
+ // paragraph. Otherwise, triple-click selects line, and quadruple-click
+ // selects paragraph (on platforms that support quadruple-click).
+ nsSelectionAmount beginAmount, endAmount;
+ WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent();
+ if (!mouseEvent) {
+ return NS_OK;
+ }
+
+ if (mouseEvent->mClickCount == 4) {
+ beginAmount = endAmount = eSelectParagraph;
+ } else if (mouseEvent->mClickCount == 3) {
+ if (Preferences::GetBool("browser.triple_click_selects_paragraph")) {
+ beginAmount = endAmount = eSelectParagraph;
+ } else {
+ beginAmount = eSelectBeginLine;
+ endAmount = eSelectEndLine;
+ }
+ } else if (mouseEvent->mClickCount == 2) {
+ // We only want inline frames; PeekBackwardAndForward dislikes blocks
+ beginAmount = endAmount = eSelectWord;
+ } else {
+ return NS_OK;
+ }
+
+ nsPoint relPoint = nsLayoutUtils::GetEventCoordinatesRelativeTo(
+ mouseEvent, RelativeTo{this});
+ return SelectByTypeAtPoint(aPresContext, relPoint, beginAmount, endAmount,
+ (aControlHeld ? SELECT_ACCUMULATE : 0));
+}
+
+nsresult nsIFrame::PeekBackwardAndForward(nsSelectionAmount aAmountBack,
+ nsSelectionAmount aAmountForward,
+ int32_t aStartPos, bool aJumpLines,
+ uint32_t aSelectFlags) {
+ nsIFrame* baseFrame = this;
+ int32_t baseOffset = aStartPos;
+ nsresult rv;
+
+ PeekOffsetOptions peekOffsetOptions{PeekOffsetOption::StopAtScroller};
+ if (aJumpLines) {
+ peekOffsetOptions += PeekOffsetOption::JumpLines;
+ }
+
+ if (aAmountBack == eSelectWord) {
+ // To avoid selecting the previous word when at start of word,
+ // first move one character forward.
+ PeekOffsetStruct pos(eSelectCharacter, eDirNext, aStartPos, nsPoint(0, 0),
+ peekOffsetOptions);
+ rv = PeekOffset(&pos);
+ if (NS_SUCCEEDED(rv)) {
+ baseFrame = pos.mResultFrame;
+ baseOffset = pos.mContentOffset;
+ }
+ }
+
+ // Search backward for a boundary.
+ PeekOffsetStruct startpos(aAmountBack, eDirPrevious, baseOffset,
+ nsPoint(0, 0), peekOffsetOptions);
+ rv = baseFrame->PeekOffset(&startpos);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // If the backward search stayed within the same frame, search forward from
+ // that position for the end boundary; but if it crossed out to a sibling or
+ // ancestor, start from the original position.
+ if (startpos.mResultFrame == baseFrame) {
+ baseOffset = startpos.mContentOffset;
+ } else {
+ baseFrame = this;
+ baseOffset = aStartPos;
+ }
+
+ PeekOffsetStruct endpos(aAmountForward, eDirNext, baseOffset, nsPoint(0, 0),
+ peekOffsetOptions);
+ rv = baseFrame->PeekOffset(&endpos);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // Keep frameSelection alive.
+ RefPtr<nsFrameSelection> frameSelection = GetFrameSelection();
+
+ const nsFrameSelection::FocusMode focusMode =
+ (aSelectFlags & SELECT_ACCUMULATE)
+ ? nsFrameSelection::FocusMode::kMultiRangeSelection
+ : nsFrameSelection::FocusMode::kCollapseToNewPoint;
+ rv = frameSelection->HandleClick(
+ MOZ_KnownLive(startpos.mResultContent) /* bug 1636889 */,
+ startpos.mContentOffset, startpos.mContentOffset, focusMode,
+ CaretAssociationHint::After);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ rv = frameSelection->HandleClick(
+ MOZ_KnownLive(endpos.mResultContent) /* bug 1636889 */,
+ endpos.mContentOffset, endpos.mContentOffset,
+ nsFrameSelection::FocusMode::kExtendSelection,
+ CaretAssociationHint::Before);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (aAmountBack == eSelectWord) {
+ frameSelection->SetIsDoubleClickSelection(true);
+ }
+
+ // maintain selection
+ return frameSelection->MaintainSelection(aAmountBack);
+}
+
+NS_IMETHODIMP nsIFrame::HandleDrag(nsPresContext* aPresContext,
+ WidgetGUIEvent* aEvent,
+ nsEventStatus* aEventStatus) {
+ MOZ_ASSERT(aEvent->mClass == eMouseEventClass,
+ "HandleDrag can only handle mouse event");
+
+ NS_ENSURE_ARG_POINTER(aEventStatus);
+
+ RefPtr<nsFrameSelection> frameselection = GetFrameSelection();
+ if (!frameselection) {
+ return NS_OK;
+ }
+
+ bool mouseDown = frameselection->GetDragState();
+ if (!mouseDown) {
+ return NS_OK;
+ }
+
+ nsIFrame* scrollbar =
+ nsLayoutUtils::GetClosestFrameOfType(this, LayoutFrameType::Scrollbar);
+ if (!scrollbar) {
+ // XXX Do we really need to exclude non-selectable content here?
+ // GetContentOffsetsFromPoint can handle it just fine, although some
+ // other stuff might not like it.
+ // NOTE: DetermineDisplaySelection() returns SELECTION_OFF for
+ // non-selectable frames.
+ if (DetermineDisplaySelection() == nsISelectionController::SELECTION_OFF) {
+ return NS_OK;
+ }
+ }
+
+ frameselection->StopAutoScrollTimer();
+
+ // Check if we are dragging in a table cell
+ nsCOMPtr<nsIContent> parentContent;
+ int32_t contentOffset;
+ TableSelectionMode target;
+ WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent();
+ mozilla::PresShell* presShell = aPresContext->PresShell();
+ nsresult result;
+ result = GetDataForTableSelection(frameselection, presShell, mouseEvent,
+ getter_AddRefs(parentContent),
+ &contentOffset, &target);
+
+ AutoWeakFrame weakThis = this;
+ if (NS_SUCCEEDED(result) && parentContent) {
+ result = frameselection->HandleTableSelection(parentContent, contentOffset,
+ target, mouseEvent);
+ if (NS_WARN_IF(NS_FAILED(result))) {
+ return result;
+ }
+ } else {
+ nsPoint pt = nsLayoutUtils::GetEventCoordinatesRelativeTo(mouseEvent,
+ RelativeTo{this});
+ frameselection->HandleDrag(this, pt);
+ }
+
+ // The frameselection object notifies selection listeners synchronously above
+ // which might have killed us.
+ if (!weakThis.IsAlive()) {
+ return NS_OK;
+ }
+
+ // get the nearest scrollframe
+ nsIScrollableFrame* scrollFrame = nsLayoutUtils::GetNearestScrollableFrame(
+ this, nsLayoutUtils::SCROLLABLE_SAME_DOC |
+ nsLayoutUtils::SCROLLABLE_INCLUDE_HIDDEN);
+
+ if (scrollFrame) {
+ nsIFrame* capturingFrame = scrollFrame->GetScrolledFrame();
+ if (capturingFrame) {
+ nsPoint pt = nsLayoutUtils::GetEventCoordinatesRelativeTo(
+ mouseEvent, RelativeTo{capturingFrame});
+ frameselection->StartAutoScrollTimer(capturingFrame, pt, 30);
+ }
+ }
+
+ return NS_OK;
+}
+
+/**
+ * This static method handles part of the nsIFrame::HandleRelease in a way
+ * which doesn't rely on the nsFrame object to stay alive.
+ */
+MOZ_CAN_RUN_SCRIPT_BOUNDARY static nsresult HandleFrameSelection(
+ nsFrameSelection* aFrameSelection, nsIFrame::ContentOffsets& aOffsets,
+ bool aHandleTableSel, int32_t aContentOffsetForTableSel,
+ TableSelectionMode aTargetForTableSel,
+ nsIContent* aParentContentForTableSel, WidgetGUIEvent* aEvent,
+ const nsEventStatus* aEventStatus) {
+ if (!aFrameSelection) {
+ return NS_OK;
+ }
+
+ nsresult rv = NS_OK;
+
+ if (nsEventStatus_eConsumeNoDefault != *aEventStatus) {
+ if (!aHandleTableSel) {
+ if (!aOffsets.content || !aFrameSelection->HasDelayedCaretData()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // We are doing this to simulate what we would have done on HandlePress.
+ // We didn't do it there to give the user an opportunity to drag
+ // the text, but since they didn't drag, we want to place the
+ // caret.
+ // However, we'll use the mouse position from the release, since:
+ // * it's easier
+ // * that's the normal click position to use (although really, in
+ // the normal case, small movements that don't count as a drag
+ // can do selection)
+ aFrameSelection->SetDragState(true);
+
+ const nsFrameSelection::FocusMode focusMode =
+ aFrameSelection->IsShiftDownInDelayedCaretData()
+ ? nsFrameSelection::FocusMode::kExtendSelection
+ : nsFrameSelection::FocusMode::kCollapseToNewPoint;
+ rv = aFrameSelection->HandleClick(
+ MOZ_KnownLive(aOffsets.content) /* bug 1636889 */,
+ aOffsets.StartOffset(), aOffsets.EndOffset(), focusMode,
+ aOffsets.associate);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ } else if (aParentContentForTableSel) {
+ aFrameSelection->SetDragState(false);
+ rv = aFrameSelection->HandleTableSelection(
+ aParentContentForTableSel, aContentOffsetForTableSel,
+ aTargetForTableSel, aEvent->AsMouseEvent());
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+ aFrameSelection->SetDelayedCaretData(0);
+ }
+
+ aFrameSelection->SetDragState(false);
+ aFrameSelection->StopAutoScrollTimer();
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP nsIFrame::HandleRelease(nsPresContext* aPresContext,
+ WidgetGUIEvent* aEvent,
+ nsEventStatus* aEventStatus) {
+ if (aEvent->mClass != eMouseEventClass) {
+ return NS_OK;
+ }
+
+ nsIFrame* activeFrame = GetActiveSelectionFrame(aPresContext, this);
+
+ nsCOMPtr<nsIContent> captureContent = PresShell::GetCapturingContent();
+
+ bool selectionOff =
+ (DetermineDisplaySelection() == nsISelectionController::SELECTION_OFF);
+
+ RefPtr<nsFrameSelection> frameselection;
+ ContentOffsets offsets;
+ nsCOMPtr<nsIContent> parentContent;
+ int32_t contentOffsetForTableSel = 0;
+ TableSelectionMode targetForTableSel = TableSelectionMode::None;
+ bool handleTableSelection = true;
+
+ if (!selectionOff) {
+ frameselection = GetFrameSelection();
+ if (nsEventStatus_eConsumeNoDefault != *aEventStatus && frameselection) {
+ // Check if the frameselection recorded the mouse going down.
+ // If not, the user must have clicked in a part of the selection.
+ // Place the caret before continuing!
+
+ if (frameselection->MouseDownRecorded()) {
+ nsPoint pt = nsLayoutUtils::GetEventCoordinatesRelativeTo(
+ aEvent, RelativeTo{this});
+ offsets = GetContentOffsetsFromPoint(pt, SKIP_HIDDEN);
+ handleTableSelection = false;
+ } else {
+ GetDataForTableSelection(frameselection, PresShell(),
+ aEvent->AsMouseEvent(),
+ getter_AddRefs(parentContent),
+ &contentOffsetForTableSel, &targetForTableSel);
+ }
+ }
+ }
+
+ // We might be capturing in some other document and the event just happened to
+ // trickle down here. Make sure that document's frame selection is notified.
+ // Note, this may cause the current nsFrame object to be deleted, bug 336592.
+ RefPtr<nsFrameSelection> frameSelection;
+ if (activeFrame != this && activeFrame->DetermineDisplaySelection() !=
+ nsISelectionController::SELECTION_OFF) {
+ frameSelection = activeFrame->GetFrameSelection();
+ }
+
+ // Also check the selection of the capturing content which might be in a
+ // different document.
+ if (!frameSelection && captureContent) {
+ if (Document* doc = captureContent->GetComposedDoc()) {
+ mozilla::PresShell* capturingPresShell = doc->GetPresShell();
+ if (capturingPresShell &&
+ capturingPresShell != PresContext()->GetPresShell()) {
+ frameSelection = capturingPresShell->FrameSelection();
+ }
+ }
+ }
+
+ if (frameSelection) {
+ AutoWeakFrame wf(this);
+ frameSelection->SetDragState(false);
+ frameSelection->StopAutoScrollTimer();
+ if (wf.IsAlive()) {
+ nsIScrollableFrame* scrollFrame =
+ nsLayoutUtils::GetNearestScrollableFrame(
+ this, nsLayoutUtils::SCROLLABLE_SAME_DOC |
+ nsLayoutUtils::SCROLLABLE_INCLUDE_HIDDEN);
+ if (scrollFrame) {
+ // Perform any additional scrolling needed to maintain CSS snap point
+ // requirements when autoscrolling is over.
+ scrollFrame->ScrollSnap();
+ }
+ }
+ }
+
+ // Do not call any methods of the current object after this point!!!
+ // The object is perhaps dead!
+
+ return selectionOff ? NS_OK
+ : HandleFrameSelection(
+ frameselection, offsets, handleTableSelection,
+ contentOffsetForTableSel, targetForTableSel,
+ parentContent, aEvent, aEventStatus);
+}
+
+struct MOZ_STACK_CLASS FrameContentRange {
+ FrameContentRange(nsIContent* aContent, int32_t aStart, int32_t aEnd)
+ : content(aContent), start(aStart), end(aEnd) {}
+ nsCOMPtr<nsIContent> content;
+ int32_t start;
+ int32_t end;
+};
+
+// Retrieve the content offsets of a frame
+static FrameContentRange GetRangeForFrame(const nsIFrame* aFrame) {
+ nsIContent* content = aFrame->GetContent();
+ if (!content) {
+ NS_WARNING("Frame has no content");
+ return FrameContentRange(nullptr, -1, -1);
+ }
+
+ LayoutFrameType type = aFrame->Type();
+ if (type == LayoutFrameType::Text) {
+ auto [offset, offsetEnd] = aFrame->GetOffsets();
+ return FrameContentRange(content, offset, offsetEnd);
+ }
+
+ if (type == LayoutFrameType::Br) {
+ nsIContent* parent = content->GetParent();
+ const int32_t beginOffset = parent->ComputeIndexOf_Deprecated(content);
+ return FrameContentRange(parent, beginOffset, beginOffset);
+ }
+
+ while (content->IsRootOfNativeAnonymousSubtree()) {
+ content = content->GetParent();
+ }
+
+ nsIContent* parent = content->GetParent();
+ if (aFrame->IsBlockOutside() || !parent) {
+ return FrameContentRange(content, 0, content->GetChildCount());
+ }
+
+ // TODO(emilio): Revise this in presence of Shadow DOM / display: contents,
+ // it's likely that we don't want to just walk the light tree, and we need to
+ // change the representation of FrameContentRange.
+ const int32_t index = parent->ComputeIndexOf_Deprecated(content);
+ MOZ_ASSERT(index >= 0);
+ return FrameContentRange(parent, index, index + 1);
+}
+
+// The FrameTarget represents the closest frame to a point that can be selected
+// The frame is the frame represented, frameEdge says whether one end of the
+// frame is the result (in which case different handling is needed), and
+// afterFrame says which end is represented if frameEdge is true
+struct FrameTarget {
+ explicit operator bool() const { return !!frame; }
+
+ nsIFrame* frame = nullptr;
+ bool frameEdge = false;
+ bool afterFrame = false;
+};
+
+// See function implementation for information
+static FrameTarget GetSelectionClosestFrame(nsIFrame* aFrame,
+ const nsPoint& aPoint,
+ uint32_t aFlags);
+
+static bool SelfIsSelectable(nsIFrame* aFrame, uint32_t aFlags) {
+ if ((aFlags & nsIFrame::SKIP_HIDDEN) &&
+ !aFrame->StyleVisibility()->IsVisible()) {
+ return false;
+ }
+ return !aFrame->IsGeneratedContentFrame() &&
+ aFrame->Style()->UserSelect() != StyleUserSelect::None;
+}
+
+static bool FrameContentCanHaveParentSelectionRange(nsIFrame* aFrame) {
+ // If we are only near (not directly over) then don't traverse
+ // frames with independent selection (e.g. text and list controls, see bug
+ // 268497). Note that this prevents any of the users of this method from
+ // entering form controls.
+ // XXX We might want some way to allow using the up-arrow to go into a form
+ // control, but the focus didn't work right anyway; it'd probably be enough
+ // if the left and right arrows could enter textboxes (which I don't believe
+ // they can at the moment)
+ if (aFrame->IsTextInputFrame() || aFrame->IsListControlFrame()) {
+ MOZ_ASSERT(aFrame->HasAnyStateBits(NS_FRAME_INDEPENDENT_SELECTION));
+ return false;
+ }
+
+ // Failure in this assertion means a new type of frame forms the root of an
+ // NS_FRAME_INDEPENDENT_SELECTION subtree. In such case, the condition above
+ // should be changed to handle it.
+ MOZ_ASSERT_IF(
+ aFrame->HasAnyStateBits(NS_FRAME_INDEPENDENT_SELECTION),
+ aFrame->GetParent()->HasAnyStateBits(NS_FRAME_INDEPENDENT_SELECTION));
+
+ return !aFrame->IsGeneratedContentFrame();
+}
+
+static bool SelectionDescendToKids(nsIFrame* aFrame) {
+ if (!FrameContentCanHaveParentSelectionRange(aFrame)) {
+ return false;
+ }
+ auto style = aFrame->Style()->UserSelect();
+ return style != StyleUserSelect::All && style != StyleUserSelect::None;
+}
+
+static FrameTarget GetSelectionClosestFrameForChild(nsIFrame* aChild,
+ const nsPoint& aPoint,
+ uint32_t aFlags) {
+ nsIFrame* parent = aChild->GetParent();
+ if (SelectionDescendToKids(aChild)) {
+ nsPoint pt = aPoint - aChild->GetOffsetTo(parent);
+ return GetSelectionClosestFrame(aChild, pt, aFlags);
+ }
+ return FrameTarget{aChild, false, false};
+}
+
+// When the cursor needs to be at the beginning of a block, it shouldn't be
+// before the first child. A click on a block whose first child is a block
+// should put the cursor in the child. The cursor shouldn't be between the
+// blocks, because that's not where it's expected.
+// Note that this method is guaranteed to succeed.
+static FrameTarget DrillDownToSelectionFrame(nsIFrame* aFrame, bool aEndFrame,
+ uint32_t aFlags) {
+ if (SelectionDescendToKids(aFrame)) {
+ nsIFrame* result = nullptr;
+ nsIFrame* frame = aFrame->PrincipalChildList().FirstChild();
+ if (!aEndFrame) {
+ while (frame && (!SelfIsSelectable(frame, aFlags) || frame->IsEmpty()))
+ frame = frame->GetNextSibling();
+ if (frame) result = frame;
+ } else {
+ // Because the frame tree is singly linked, to find the last frame,
+ // we have to iterate through all the frames
+ // XXX I have a feeling this could be slow for long blocks, although
+ // I can't find any slowdowns
+ while (frame) {
+ if (!frame->IsEmpty() && SelfIsSelectable(frame, aFlags))
+ result = frame;
+ frame = frame->GetNextSibling();
+ }
+ }
+ if (result) return DrillDownToSelectionFrame(result, aEndFrame, aFlags);
+ }
+ // If the current frame has no targetable children, target the current frame
+ return FrameTarget{aFrame, true, aEndFrame};
+}
+
+// This method finds the closest valid FrameTarget on a given line; if there is
+// no valid FrameTarget on the line, it returns a null FrameTarget
+static FrameTarget GetSelectionClosestFrameForLine(
+ nsBlockFrame* aParent, nsBlockFrame::LineIterator aLine,
+ const nsPoint& aPoint, uint32_t aFlags) {
+ // Account for end of lines (any iterator from the block is valid)
+ if (aLine == aParent->LinesEnd())
+ return DrillDownToSelectionFrame(aParent, true, aFlags);
+ nsIFrame* frame = aLine->mFirstChild;
+ nsIFrame* closestFromIStart = nullptr;
+ nsIFrame* closestFromIEnd = nullptr;
+ nscoord closestIStart = aLine->IStart(), closestIEnd = aLine->IEnd();
+ WritingMode wm = aLine->mWritingMode;
+ LogicalPoint pt(wm, aPoint, aLine->mContainerSize);
+ bool canSkipBr = false;
+ bool lastFrameWasEditable = false;
+ for (int32_t n = aLine->GetChildCount(); n;
+ --n, frame = frame->GetNextSibling()) {
+ // Skip brFrames. Can only skip if the line contains at least
+ // one selectable and non-empty frame before. Also, avoid skipping brs if
+ // the previous thing had a different editableness than us, since then we
+ // may end up not being able to select after it if the br is the last thing
+ // on the line.
+ if (!SelfIsSelectable(frame, aFlags) || frame->IsEmpty() ||
+ (canSkipBr && frame->IsBrFrame() &&
+ lastFrameWasEditable == frame->GetContent()->IsEditable())) {
+ continue;
+ }
+ canSkipBr = true;
+ lastFrameWasEditable =
+ frame->GetContent() && frame->GetContent()->IsEditable();
+ LogicalRect frameRect =
+ LogicalRect(wm, frame->GetRect(), aLine->mContainerSize);
+ if (pt.I(wm) >= frameRect.IStart(wm)) {
+ if (pt.I(wm) < frameRect.IEnd(wm)) {
+ return GetSelectionClosestFrameForChild(frame, aPoint, aFlags);
+ }
+ if (frameRect.IEnd(wm) >= closestIStart) {
+ closestFromIStart = frame;
+ closestIStart = frameRect.IEnd(wm);
+ }
+ } else {
+ if (frameRect.IStart(wm) <= closestIEnd) {
+ closestFromIEnd = frame;
+ closestIEnd = frameRect.IStart(wm);
+ }
+ }
+ }
+ if (!closestFromIStart && !closestFromIEnd) {
+ // We should only get here if there are no selectable frames on a line
+ // XXX Do we need more elaborate handling here?
+ return FrameTarget();
+ }
+ if (closestFromIStart &&
+ (!closestFromIEnd ||
+ (abs(pt.I(wm) - closestIStart) <= abs(pt.I(wm) - closestIEnd)))) {
+ return GetSelectionClosestFrameForChild(closestFromIStart, aPoint, aFlags);
+ }
+ return GetSelectionClosestFrameForChild(closestFromIEnd, aPoint, aFlags);
+}
+
+// This method is for the special handling we do for block frames; they're
+// special because they represent paragraphs and because they are organized
+// into lines, which have bounds that are not stored elsewhere in the
+// frame tree. Returns a null FrameTarget for frames which are not
+// blocks or blocks with no lines except editable one.
+static FrameTarget GetSelectionClosestFrameForBlock(nsIFrame* aFrame,
+ const nsPoint& aPoint,
+ uint32_t aFlags) {
+ nsBlockFrame* bf = do_QueryFrame(aFrame);
+ if (!bf) {
+ return FrameTarget();
+ }
+
+ // This code searches for the correct line
+ nsBlockFrame::LineIterator end = bf->LinesEnd();
+ nsBlockFrame::LineIterator curLine = bf->LinesBegin();
+ nsBlockFrame::LineIterator closestLine = end;
+
+ if (curLine != end) {
+ // Convert aPoint into a LogicalPoint in the writing-mode of this block
+ WritingMode wm = curLine->mWritingMode;
+ LogicalPoint pt(wm, aPoint, curLine->mContainerSize);
+ do {
+ // Check to see if our point lies within the line's block-direction bounds
+ nscoord BCoord = pt.B(wm) - curLine->BStart();
+ nscoord BSize = curLine->BSize();
+ if (BCoord >= 0 && BCoord < BSize) {
+ closestLine = curLine;
+ break; // We found the line; stop looking
+ }
+ if (BCoord < 0) break;
+ ++curLine;
+ } while (curLine != end);
+
+ if (closestLine == end) {
+ nsBlockFrame::LineIterator prevLine = curLine.prev();
+ nsBlockFrame::LineIterator nextLine = curLine;
+ // Avoid empty lines
+ while (nextLine != end && nextLine->IsEmpty()) ++nextLine;
+ while (prevLine != end && prevLine->IsEmpty()) --prevLine;
+
+ // This hidden pref dictates whether a point above or below all lines
+ // comes up with a line or the beginning or end of the frame; 0 on
+ // Windows, 1 on other platforms by default at the writing of this code
+ int32_t dragOutOfFrame =
+ Preferences::GetInt("browser.drag_out_of_frame_style");
+
+ if (prevLine == end) {
+ if (dragOutOfFrame == 1 || nextLine == end)
+ return DrillDownToSelectionFrame(aFrame, false, aFlags);
+ closestLine = nextLine;
+ } else if (nextLine == end) {
+ if (dragOutOfFrame == 1)
+ return DrillDownToSelectionFrame(aFrame, true, aFlags);
+ closestLine = prevLine;
+ } else { // Figure out which line is closer
+ if (pt.B(wm) - prevLine->BEnd() < nextLine->BStart() - pt.B(wm))
+ closestLine = prevLine;
+ else
+ closestLine = nextLine;
+ }
+ }
+ }
+
+ do {
+ if (auto target =
+ GetSelectionClosestFrameForLine(bf, closestLine, aPoint, aFlags)) {
+ return target;
+ }
+ ++closestLine;
+ } while (closestLine != end);
+
+ // Fall back to just targeting the last targetable place
+ return DrillDownToSelectionFrame(aFrame, true, aFlags);
+}
+
+// Use frame edge for grid, flex, table, and non-editable images. Choose the
+// edge based on the point position past the frame rect. If past the middle,
+// caret should be at end, otherwise at start. This behavior matches Blink.
+//
+// TODO(emilio): Can we use this code path for other replaced elements other
+// than images? Or even all other frames? We only get there when we didn't find
+// selectable children... At least one XUL test fails if we make this apply to
+// XUL labels. Also, editable images need _not_ to use the frame edge, see
+// below.
+static bool UseFrameEdge(nsIFrame* aFrame) {
+ if (aFrame->IsFlexOrGridContainer() || aFrame->IsTableFrame()) {
+ return true;
+ }
+ const nsImageFrame* image = do_QueryFrame(aFrame);
+ if (image && !aFrame->GetContent()->IsEditable()) {
+ // Editable images are a special-case because editing relies on clicking on
+ // an editable image selecting it, for it to show resizers.
+ return true;
+ }
+ return false;
+}
+
+static FrameTarget LastResortFrameTargetForFrame(nsIFrame* aFrame,
+ const nsPoint& aPoint) {
+ if (!UseFrameEdge(aFrame)) {
+ return {aFrame, false, false};
+ }
+ const auto& rect = aFrame->GetRectRelativeToSelf();
+ nscoord reference;
+ nscoord middle;
+ if (aFrame->GetWritingMode().IsVertical()) {
+ reference = aPoint.y;
+ middle = rect.Height() / 2;
+ } else {
+ reference = aPoint.x;
+ middle = rect.Width() / 2;
+ }
+ const bool afterFrame = reference > middle;
+ return {aFrame, true, afterFrame};
+}
+
+// GetSelectionClosestFrame is the helper function that calculates the closest
+// frame to the given point.
+// It doesn't completely account for offset styles, so needs to be used in
+// restricted environments.
+// Cannot handle overlapping frames correctly, so it should receive the output
+// of GetFrameForPoint
+// Guaranteed to return a valid FrameTarget.
+// aPoint is relative to aFrame.
+static FrameTarget GetSelectionClosestFrame(nsIFrame* aFrame,
+ const nsPoint& aPoint,
+ uint32_t aFlags) {
+ // Handle blocks; if the frame isn't a block, the method fails
+ if (auto target = GetSelectionClosestFrameForBlock(aFrame, aPoint, aFlags)) {
+ return target;
+ }
+
+ if (aFlags & nsIFrame::IGNORE_NATIVE_ANONYMOUS_SUBTREE &&
+ !FrameContentCanHaveParentSelectionRange(aFrame)) {
+ return LastResortFrameTargetForFrame(aFrame, aPoint);
+ }
+
+ if (nsIFrame* kid = aFrame->PrincipalChildList().FirstChild()) {
+ // Go through all the child frames to find the closest one
+ nsIFrame::FrameWithDistance closest = {nullptr, nscoord_MAX, nscoord_MAX};
+ for (; kid; kid = kid->GetNextSibling()) {
+ if (!SelfIsSelectable(kid, aFlags) || kid->IsEmpty()) {
+ continue;
+ }
+
+ kid->FindCloserFrameForSelection(aPoint, &closest);
+ }
+ if (closest.mFrame) {
+ if (closest.mFrame->IsInSVGTextSubtree())
+ return FrameTarget{closest.mFrame, false, false};
+ return GetSelectionClosestFrameForChild(closest.mFrame, aPoint, aFlags);
+ }
+ }
+
+ return LastResortFrameTargetForFrame(aFrame, aPoint);
+}
+
+static nsIFrame::ContentOffsets OffsetsForSingleFrame(nsIFrame* aFrame,
+ const nsPoint& aPoint) {
+ nsIFrame::ContentOffsets offsets;
+ FrameContentRange range = GetRangeForFrame(aFrame);
+ offsets.content = range.content;
+ // If there are continuations (meaning it's not one rectangle), this is the
+ // best this function can do
+ if (aFrame->GetNextContinuation() || aFrame->GetPrevContinuation()) {
+ offsets.offset = range.start;
+ offsets.secondaryOffset = range.end;
+ offsets.associate = CaretAssociationHint::After;
+ return offsets;
+ }
+
+ // Figure out whether the offsets should be over, after, or before the frame
+ nsRect rect(nsPoint(0, 0), aFrame->GetSize());
+
+ bool isBlock = !aFrame->StyleDisplay()->IsInlineFlow();
+ bool isRtl = (aFrame->StyleVisibility()->mDirection == StyleDirection::Rtl);
+ if ((isBlock && rect.y < aPoint.y) ||
+ (!isBlock && ((isRtl && rect.x + rect.width / 2 > aPoint.x) ||
+ (!isRtl && rect.x + rect.width / 2 < aPoint.x)))) {
+ offsets.offset = range.end;
+ if (rect.Contains(aPoint))
+ offsets.secondaryOffset = range.start;
+ else
+ offsets.secondaryOffset = range.end;
+ } else {
+ offsets.offset = range.start;
+ if (rect.Contains(aPoint))
+ offsets.secondaryOffset = range.end;
+ else
+ offsets.secondaryOffset = range.start;
+ }
+ offsets.associate = offsets.offset == range.start
+ ? CaretAssociationHint::After
+ : CaretAssociationHint::Before;
+ return offsets;
+}
+
+static nsIFrame* AdjustFrameForSelectionStyles(nsIFrame* aFrame) {
+ nsIFrame* adjustedFrame = aFrame;
+ for (nsIFrame* frame = aFrame; frame; frame = frame->GetParent()) {
+ // These are the conditions that make all children not able to handle
+ // a cursor.
+ auto userSelect = frame->Style()->UserSelect();
+ if (userSelect != StyleUserSelect::Auto &&
+ userSelect != StyleUserSelect::All) {
+ break;
+ }
+ if (userSelect == StyleUserSelect::All ||
+ frame->IsGeneratedContentFrame()) {
+ adjustedFrame = frame;
+ }
+ }
+ return adjustedFrame;
+}
+
+nsIFrame::ContentOffsets nsIFrame::GetContentOffsetsFromPoint(
+ const nsPoint& aPoint, uint32_t aFlags) {
+ nsIFrame* adjustedFrame;
+ if (aFlags & IGNORE_SELECTION_STYLE) {
+ adjustedFrame = this;
+ } else {
+ // This section of code deals with special selection styles. Note that
+ // -moz-all exists, even though it doesn't need to be explicitly handled.
+ //
+ // The offset is forced not to end up in generated content; content offsets
+ // cannot represent content outside of the document's content tree.
+
+ adjustedFrame = AdjustFrameForSelectionStyles(this);
+
+ // `user-select: all` needs special handling, because clicking on it should
+ // lead to the whole frame being selected.
+ if (adjustedFrame->Style()->UserSelect() == StyleUserSelect::All) {
+ nsPoint adjustedPoint = aPoint + GetOffsetTo(adjustedFrame);
+ return OffsetsForSingleFrame(adjustedFrame, adjustedPoint);
+ }
+
+ // For other cases, try to find a closest frame starting from the parent of
+ // the unselectable frame
+ if (adjustedFrame != this) {
+ adjustedFrame = adjustedFrame->GetParent();
+ }
+ }
+
+ nsPoint adjustedPoint = aPoint + GetOffsetTo(adjustedFrame);
+
+ FrameTarget closest =
+ GetSelectionClosestFrame(adjustedFrame, adjustedPoint, aFlags);
+
+ // If the correct offset is at one end of a frame, use offset-based
+ // calculation method
+ if (closest.frameEdge) {
+ ContentOffsets offsets;
+ FrameContentRange range = GetRangeForFrame(closest.frame);
+ offsets.content = range.content;
+ if (closest.afterFrame)
+ offsets.offset = range.end;
+ else
+ offsets.offset = range.start;
+ offsets.secondaryOffset = offsets.offset;
+ offsets.associate = offsets.offset == range.start
+ ? CaretAssociationHint::After
+ : CaretAssociationHint::Before;
+ return offsets;
+ }
+
+ nsPoint pt;
+ if (closest.frame != this) {
+ if (closest.frame->IsInSVGTextSubtree()) {
+ pt = nsLayoutUtils::TransformAncestorPointToFrame(
+ RelativeTo{closest.frame}, aPoint, RelativeTo{this});
+ } else {
+ pt = aPoint - closest.frame->GetOffsetTo(this);
+ }
+ } else {
+ pt = aPoint;
+ }
+ return closest.frame->CalcContentOffsetsFromFramePoint(pt);
+
+ // XXX should I add some kind of offset standardization?
+ // consider <b>xxxxx</b><i>zzzzz</i>; should any click between the last
+ // x and first z put the cursor in the same logical position in addition
+ // to the same visual position?
+}
+
+nsIFrame::ContentOffsets nsIFrame::CalcContentOffsetsFromFramePoint(
+ const nsPoint& aPoint) {
+ return OffsetsForSingleFrame(this, aPoint);
+}
+
+bool nsIFrame::AssociateImage(const StyleImage& aImage) {
+ imgRequestProxy* req = aImage.GetImageRequest();
+ if (!req) {
+ return false;
+ }
+
+ mozilla::css::ImageLoader* loader =
+ PresContext()->Document()->StyleImageLoader();
+
+ loader->AssociateRequestToFrame(req, this);
+ return true;
+}
+
+void nsIFrame::DisassociateImage(const StyleImage& aImage) {
+ imgRequestProxy* req = aImage.GetImageRequest();
+ if (!req) {
+ return;
+ }
+
+ mozilla::css::ImageLoader* loader =
+ PresContext()->Document()->StyleImageLoader();
+
+ loader->DisassociateRequestFromFrame(req, this);
+}
+
+StyleImageRendering nsIFrame::UsedImageRendering() const {
+ ComputedStyle* style;
+ if (IsCanvasFrame()) {
+ // XXXdholbert Maybe we should use FindCanvasBackground here (instead of
+ // FindBackground), since we're inside an IsCanvasFrame check? Though then
+ // we'd also have to copypaste or abstract-away the multi-part root-frame
+ // lookup that the canvas-flavored API requires.
+ style = nsCSSRendering::FindBackground(this);
+ } else {
+ style = Style();
+ }
+ return style->StyleVisibility()->mImageRendering;
+}
+
+// The touch-action CSS property applies to: all elements except: non-replaced
+// inline elements, table rows, row groups, table columns, and column groups.
+StyleTouchAction nsIFrame::UsedTouchAction() const {
+ if (IsLineParticipant()) {
+ return StyleTouchAction::AUTO;
+ }
+ auto& disp = *StyleDisplay();
+ if (disp.IsInternalTableStyleExceptCell()) {
+ return StyleTouchAction::AUTO;
+ }
+ return disp.mTouchAction;
+}
+
+nsIFrame::Cursor nsIFrame::GetCursor(const nsPoint&) {
+ StyleCursorKind kind = StyleUI()->Cursor().keyword;
+ if (kind == StyleCursorKind::Auto) {
+ // If this is editable, I-beam cursor is better for most elements.
+ kind = (mContent && mContent->IsEditable()) ? StyleCursorKind::Text
+ : StyleCursorKind::Default;
+ }
+ if (kind == StyleCursorKind::Text && GetWritingMode().IsVertical()) {
+ // Per CSS UI spec, UA may treat value 'text' as
+ // 'vertical-text' for vertical text.
+ kind = StyleCursorKind::VerticalText;
+ }
+
+ return Cursor{kind, AllowCustomCursorImage::Yes};
+}
+
+// Resize and incremental reflow
+
+/* virtual */
+void nsIFrame::MarkIntrinsicISizesDirty() {
+ // If we're a flex item, clear our flex-item-specific cached measurements
+ // (which likely depended on our now-stale intrinsic isize).
+ if (IsFlexItem()) {
+ nsFlexContainerFrame::MarkCachedFlexMeasurementsDirty(this);
+ }
+
+ if (HasAnyStateBits(NS_FRAME_FONT_INFLATION_FLOW_ROOT)) {
+ nsFontInflationData::MarkFontInflationDataTextDirty(this);
+ }
+
+ RemoveProperty(nsGridContainerFrame::CachedBAxisMeasurement::Prop());
+}
+
+void nsIFrame::MarkSubtreeDirty() {
+ if (HasAnyStateBits(NS_FRAME_IS_DIRTY)) {
+ return;
+ }
+ // Unconditionally mark given frame dirty.
+ AddStateBits(NS_FRAME_IS_DIRTY);
+
+ // Mark all descendants dirty, unless:
+ // - Already dirty.
+ // - TableColGroup
+ AutoTArray<nsIFrame*, 32> stack;
+ for (const auto& childLists : ChildLists()) {
+ for (nsIFrame* kid : childLists.mList) {
+ stack.AppendElement(kid);
+ }
+ }
+ while (!stack.IsEmpty()) {
+ nsIFrame* f = stack.PopLastElement();
+ if (f->HasAnyStateBits(NS_FRAME_IS_DIRTY) || f->IsTableColGroupFrame()) {
+ continue;
+ }
+
+ f->AddStateBits(NS_FRAME_IS_DIRTY);
+
+ for (const auto& childLists : f->ChildLists()) {
+ for (nsIFrame* kid : childLists.mList) {
+ stack.AppendElement(kid);
+ }
+ }
+ }
+}
+
+/* virtual */
+nscoord nsIFrame::GetMinISize(gfxContext* aRenderingContext) {
+ nscoord result = 0;
+ DISPLAY_MIN_INLINE_SIZE(this, result);
+ return result;
+}
+
+/* virtual */
+nscoord nsIFrame::GetPrefISize(gfxContext* aRenderingContext) {
+ nscoord result = 0;
+ DISPLAY_PREF_INLINE_SIZE(this, result);
+ return result;
+}
+
+/* virtual */
+void nsIFrame::AddInlineMinISize(gfxContext* aRenderingContext,
+ nsIFrame::InlineMinISizeData* aData) {
+ nscoord isize = nsLayoutUtils::IntrinsicForContainer(
+ aRenderingContext, this, IntrinsicISizeType::MinISize);
+ aData->DefaultAddInlineMinISize(this, isize);
+}
+
+/* virtual */
+void nsIFrame::AddInlinePrefISize(gfxContext* aRenderingContext,
+ nsIFrame::InlinePrefISizeData* aData) {
+ nscoord isize = nsLayoutUtils::IntrinsicForContainer(
+ aRenderingContext, this, IntrinsicISizeType::PrefISize);
+ aData->DefaultAddInlinePrefISize(isize);
+}
+
+void nsIFrame::InlineMinISizeData::DefaultAddInlineMinISize(nsIFrame* aFrame,
+ nscoord aISize,
+ bool aAllowBreak) {
+ auto parent = aFrame->GetParent();
+ MOZ_ASSERT(parent, "Must have a parent if we get here!");
+ const bool mayBreak = aAllowBreak && !aFrame->CanContinueTextRun() &&
+ !parent->Style()->ShouldSuppressLineBreak() &&
+ parent->StyleText()->WhiteSpaceCanWrap(parent);
+ if (mayBreak) {
+ OptionallyBreak();
+ }
+ mTrailingWhitespace = 0;
+ mSkipWhitespace = false;
+ mCurrentLine += aISize;
+ mAtStartOfLine = false;
+ if (mayBreak) {
+ OptionallyBreak();
+ }
+}
+
+void nsIFrame::InlinePrefISizeData::DefaultAddInlinePrefISize(nscoord aISize) {
+ mCurrentLine = NSCoordSaturatingAdd(mCurrentLine, aISize);
+ mTrailingWhitespace = 0;
+ mSkipWhitespace = false;
+ mLineIsEmpty = false;
+}
+
+void nsIFrame::InlineMinISizeData::ForceBreak() {
+ mCurrentLine -= mTrailingWhitespace;
+ mPrevLines = std::max(mPrevLines, mCurrentLine);
+ mCurrentLine = mTrailingWhitespace = 0;
+
+ for (uint32_t i = 0, i_end = mFloats.Length(); i != i_end; ++i) {
+ nscoord float_min = mFloats[i].Width();
+ if (float_min > mPrevLines) mPrevLines = float_min;
+ }
+ mFloats.Clear();
+ mSkipWhitespace = true;
+}
+
+void nsIFrame::InlineMinISizeData::OptionallyBreak(nscoord aHyphenWidth) {
+ // If we can fit more content into a smaller width by staying on this
+ // line (because we're still at a negative offset due to negative
+ // text-indent or negative margin), don't break. Otherwise, do the
+ // same as ForceBreak. it doesn't really matter when we accumulate
+ // floats.
+ if (mCurrentLine + aHyphenWidth < 0 || mAtStartOfLine) return;
+ mCurrentLine += aHyphenWidth;
+ ForceBreak();
+}
+
+void nsIFrame::InlinePrefISizeData::ForceBreak(StyleClear aClearType) {
+ // If this force break is not clearing any float, we can leave all the
+ // floats to the next force break.
+ if (!mFloats.IsEmpty() && aClearType != StyleClear::None) {
+ // preferred widths accumulated for floats that have already
+ // been cleared past
+ nscoord floats_done = 0,
+ // preferred widths accumulated for floats that have not yet
+ // been cleared past
+ floats_cur_left = 0, floats_cur_right = 0;
+
+ for (const FloatInfo& floatInfo : mFloats) {
+ const nsStyleDisplay* floatDisp = floatInfo.Frame()->StyleDisplay();
+ StyleClear clearType = floatDisp->mClear;
+ if (clearType == StyleClear::Left || clearType == StyleClear::Right ||
+ clearType == StyleClear::Both) {
+ nscoord floats_cur =
+ NSCoordSaturatingAdd(floats_cur_left, floats_cur_right);
+ if (floats_cur > floats_done) {
+ floats_done = floats_cur;
+ }
+ if (clearType != StyleClear::Right) {
+ floats_cur_left = 0;
+ }
+ if (clearType != StyleClear::Left) {
+ floats_cur_right = 0;
+ }
+ }
+
+ StyleFloat floatStyle = floatDisp->mFloat;
+ nscoord& floats_cur =
+ floatStyle == StyleFloat::Left ? floats_cur_left : floats_cur_right;
+ nscoord floatWidth = floatInfo.Width();
+ // Negative-width floats don't change the available space so they
+ // shouldn't change our intrinsic line width either.
+ floats_cur = NSCoordSaturatingAdd(floats_cur, std::max(0, floatWidth));
+ }
+
+ nscoord floats_cur =
+ NSCoordSaturatingAdd(floats_cur_left, floats_cur_right);
+ if (floats_cur > floats_done) floats_done = floats_cur;
+
+ mCurrentLine = NSCoordSaturatingAdd(mCurrentLine, floats_done);
+
+ if (aClearType == StyleClear::Both) {
+ mFloats.Clear();
+ } else {
+ // If the break type does not clear all floats, it means there may
+ // be some floats whose isize should contribute to the intrinsic
+ // isize of the next line. The code here scans the current mFloats
+ // and keeps floats which are not cleared by this break. Note that
+ // floats may be cleared directly or indirectly. See below.
+ nsTArray<FloatInfo> newFloats;
+ MOZ_ASSERT(
+ aClearType == StyleClear::Left || aClearType == StyleClear::Right,
+ "Other values should have been handled in other branches");
+ StyleFloat clearFloatType =
+ aClearType == StyleClear::Left ? StyleFloat::Left : StyleFloat::Right;
+ // Iterate the array in reverse so that we can stop when there are
+ // no longer any floats we need to keep. See below.
+ for (FloatInfo& floatInfo : Reversed(mFloats)) {
+ const nsStyleDisplay* floatDisp = floatInfo.Frame()->StyleDisplay();
+ if (floatDisp->mFloat != clearFloatType) {
+ newFloats.AppendElement(floatInfo);
+ } else {
+ // This is a float on the side that this break directly clears
+ // which means we're not keeping it in mFloats. However, if
+ // this float clears floats on the opposite side (via a value
+ // of either 'both' or one of 'left'/'right'), any remaining
+ // (earlier) floats on that side would be indirectly cleared
+ // as well. Thus, we should break out of this loop and stop
+ // considering earlier floats to be kept in mFloats.
+ StyleClear clearType = floatDisp->mClear;
+ if (clearType != aClearType && clearType != StyleClear::None) {
+ break;
+ }
+ }
+ }
+ newFloats.Reverse();
+ mFloats = std::move(newFloats);
+ }
+ }
+
+ mCurrentLine =
+ NSCoordSaturatingSubtract(mCurrentLine, mTrailingWhitespace, nscoord_MAX);
+ mPrevLines = std::max(mPrevLines, mCurrentLine);
+ mCurrentLine = mTrailingWhitespace = 0;
+ mSkipWhitespace = true;
+ mLineIsEmpty = true;
+}
+
+static nscoord ResolveMargin(const LengthPercentageOrAuto& aStyle,
+ nscoord aPercentageBasis) {
+ if (aStyle.IsAuto()) {
+ return nscoord(0);
+ }
+ return nsLayoutUtils::ResolveToLength<false>(aStyle.AsLengthPercentage(),
+ aPercentageBasis);
+}
+
+static nscoord ResolvePadding(const LengthPercentage& aStyle,
+ nscoord aPercentageBasis) {
+ return nsLayoutUtils::ResolveToLength<true>(aStyle, aPercentageBasis);
+}
+
+static nsIFrame::IntrinsicSizeOffsetData IntrinsicSizeOffsets(
+ nsIFrame* aFrame, nscoord aPercentageBasis, bool aForISize) {
+ nsIFrame::IntrinsicSizeOffsetData result;
+ WritingMode wm = aFrame->GetWritingMode();
+ const auto& margin = aFrame->StyleMargin()->mMargin;
+ bool verticalAxis = aForISize == wm.IsVertical();
+ if (verticalAxis) {
+ result.margin += ResolveMargin(margin.Get(eSideTop), aPercentageBasis);
+ result.margin += ResolveMargin(margin.Get(eSideBottom), aPercentageBasis);
+ } else {
+ result.margin += ResolveMargin(margin.Get(eSideLeft), aPercentageBasis);
+ result.margin += ResolveMargin(margin.Get(eSideRight), aPercentageBasis);
+ }
+
+ const auto& padding = aFrame->StylePadding()->mPadding;
+ if (verticalAxis) {
+ result.padding += ResolvePadding(padding.Get(eSideTop), aPercentageBasis);
+ result.padding +=
+ ResolvePadding(padding.Get(eSideBottom), aPercentageBasis);
+ } else {
+ result.padding += ResolvePadding(padding.Get(eSideLeft), aPercentageBasis);
+ result.padding += ResolvePadding(padding.Get(eSideRight), aPercentageBasis);
+ }
+
+ const nsStyleBorder* styleBorder = aFrame->StyleBorder();
+ if (verticalAxis) {
+ result.border += styleBorder->GetComputedBorderWidth(eSideTop);
+ result.border += styleBorder->GetComputedBorderWidth(eSideBottom);
+ } else {
+ result.border += styleBorder->GetComputedBorderWidth(eSideLeft);
+ result.border += styleBorder->GetComputedBorderWidth(eSideRight);
+ }
+
+ const nsStyleDisplay* disp = aFrame->StyleDisplay();
+ if (aFrame->IsThemed(disp)) {
+ nsPresContext* presContext = aFrame->PresContext();
+
+ LayoutDeviceIntMargin border = presContext->Theme()->GetWidgetBorder(
+ presContext->DeviceContext(), aFrame, disp->EffectiveAppearance());
+ result.border = presContext->DevPixelsToAppUnits(
+ verticalAxis ? border.TopBottom() : border.LeftRight());
+
+ LayoutDeviceIntMargin padding;
+ if (presContext->Theme()->GetWidgetPadding(
+ presContext->DeviceContext(), aFrame, disp->EffectiveAppearance(),
+ &padding)) {
+ result.padding = presContext->DevPixelsToAppUnits(
+ verticalAxis ? padding.TopBottom() : padding.LeftRight());
+ }
+ }
+ return result;
+}
+
+/* virtual */ nsIFrame::IntrinsicSizeOffsetData nsIFrame::IntrinsicISizeOffsets(
+ nscoord aPercentageBasis) {
+ return IntrinsicSizeOffsets(this, aPercentageBasis, true);
+}
+
+nsIFrame::IntrinsicSizeOffsetData nsIFrame::IntrinsicBSizeOffsets(
+ nscoord aPercentageBasis) {
+ return IntrinsicSizeOffsets(this, aPercentageBasis, false);
+}
+
+/* virtual */
+IntrinsicSize nsIFrame::GetIntrinsicSize() {
+ return IntrinsicSize(); // default is width/height set to eStyleUnit_None
+}
+
+AspectRatio nsIFrame::GetAspectRatio() const {
+ // Per spec, 'aspect-ratio' property applies to all elements except inline
+ // boxes and internal ruby or table boxes.
+ // https://drafts.csswg.org/css-sizing-4/#aspect-ratio
+ // For those frame types that don't support aspect-ratio, they must not have
+ // the natural ratio, so this early return is fine.
+ if (!SupportsAspectRatio()) {
+ return AspectRatio();
+ }
+
+ const StyleAspectRatio& aspectRatio = StylePosition()->mAspectRatio;
+ // If aspect-ratio is zero or infinite, it's a degenerate ratio and behaves
+ // as auto.
+ // https://drafts.csswg.org/css-sizing-4/#valdef-aspect-ratio-ratio
+ if (!aspectRatio.BehavesAsAuto()) {
+ // Non-auto. Return the preferred aspect ratio from the aspect-ratio style.
+ return aspectRatio.ratio.AsRatio().ToLayoutRatio(UseBoxSizing::Yes);
+ }
+
+ // The rest of the cases are when aspect-ratio has 'auto'.
+ if (auto intrinsicRatio = GetIntrinsicRatio()) {
+ return intrinsicRatio;
+ }
+
+ if (aspectRatio.HasRatio()) {
+ // If it's a degenerate ratio, this returns 0. Just the same as the auto
+ // case.
+ return aspectRatio.ratio.AsRatio().ToLayoutRatio(UseBoxSizing::No);
+ }
+
+ return AspectRatio();
+}
+
+/* virtual */
+AspectRatio nsIFrame::GetIntrinsicRatio() const { return AspectRatio(); }
+
+static bool ShouldApplyAutomaticMinimumOnInlineAxis(
+ WritingMode aWM, const nsStyleDisplay* aDisplay,
+ const nsStylePosition* aPosition) {
+ // Apply the automatic minimum size for aspect ratio:
+ // Note: The replaced elements shouldn't be here, so we only check the scroll
+ // container.
+ // https://drafts.csswg.org/css-sizing-4/#aspect-ratio-minimum
+ return !aDisplay->IsScrollableOverflow() && aPosition->MinISize(aWM).IsAuto();
+}
+
+struct MinMaxSize {
+ nscoord mMinSize = 0;
+ nscoord mMaxSize = NS_UNCONSTRAINEDSIZE;
+
+ nscoord ClampSizeToMinAndMax(nscoord aSize) const {
+ return NS_CSS_MINMAX(aSize, mMinSize, mMaxSize);
+ }
+};
+static MinMaxSize ComputeTransferredMinMaxInlineSize(
+ const WritingMode aWM, const AspectRatio& aAspectRatio,
+ const MinMaxSize& aMinMaxBSize, const LogicalSize& aBoxSizingAdjustment) {
+ // Note: the spec mentions that
+ // 1. This transferred minimum is capped by any definite preferred or maximum
+ // size in the destination axis.
+ // 2. This transferred maximum is floored by any definite preferred or minimum
+ // size in the destination axis
+ //
+ // https://drafts.csswg.org/css-sizing-4/#aspect-ratio
+ //
+ // The spec requires us to clamp these by the specified size (it calls it the
+ // preferred size). However, we actually don't need to worry about that,
+ // because we only use this if the inline size is indefinite.
+ //
+ // We do not need to clamp the transferred minimum and maximum as long as we
+ // always apply the transferred min/max size before the explicit min/max size,
+ // the result will be identical.
+
+ MinMaxSize transferredISize;
+
+ if (aMinMaxBSize.mMinSize > 0) {
+ transferredISize.mMinSize = aAspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisInline, aWM, aMinMaxBSize.mMinSize,
+ aBoxSizingAdjustment);
+ }
+
+ if (aMinMaxBSize.mMaxSize != NS_UNCONSTRAINEDSIZE) {
+ transferredISize.mMaxSize = aAspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisInline, aWM, aMinMaxBSize.mMaxSize,
+ aBoxSizingAdjustment);
+ }
+
+ // Minimum size wins over maximum size.
+ transferredISize.mMaxSize =
+ std::max(transferredISize.mMinSize, transferredISize.mMaxSize);
+ return transferredISize;
+}
+
+/* virtual */
+nsIFrame::SizeComputationResult nsIFrame::ComputeSize(
+ gfxContext* aRenderingContext, WritingMode aWM, const LogicalSize& aCBSize,
+ nscoord aAvailableISize, const LogicalSize& aMargin,
+ const LogicalSize& aBorderPadding, const StyleSizeOverrides& aSizeOverrides,
+ ComputeSizeFlags aFlags) {
+ MOZ_ASSERT(!GetIntrinsicRatio(),
+ "Please override this method and call "
+ "nsContainerFrame::ComputeSizeWithIntrinsicDimensions instead.");
+ LogicalSize result =
+ ComputeAutoSize(aRenderingContext, aWM, aCBSize, aAvailableISize, aMargin,
+ aBorderPadding, aSizeOverrides, aFlags);
+ const nsStylePosition* stylePos = StylePosition();
+ const nsStyleDisplay* disp = StyleDisplay();
+ auto aspectRatioUsage = AspectRatioUsage::None;
+
+ const auto boxSizingAdjust = stylePos->mBoxSizing == StyleBoxSizing::Border
+ ? aBorderPadding
+ : LogicalSize(aWM);
+ nscoord boxSizingToMarginEdgeISize = aMargin.ISize(aWM) +
+ aBorderPadding.ISize(aWM) -
+ boxSizingAdjust.ISize(aWM);
+
+ const auto& styleISize = aSizeOverrides.mStyleISize
+ ? *aSizeOverrides.mStyleISize
+ : stylePos->ISize(aWM);
+ const auto& styleBSize = aSizeOverrides.mStyleBSize
+ ? *aSizeOverrides.mStyleBSize
+ : stylePos->BSize(aWM);
+ const auto& aspectRatio = aSizeOverrides.mAspectRatio
+ ? *aSizeOverrides.mAspectRatio
+ : GetAspectRatio();
+
+ auto parentFrame = GetParent();
+ auto alignCB = parentFrame;
+ bool isGridItem = IsGridItem();
+ const bool isSubgrid = IsSubgrid();
+ if (parentFrame && parentFrame->IsTableWrapperFrame() && IsTableFrame()) {
+ // An inner table frame is sized as a grid item if its table wrapper is,
+ // because they actually have the same CB (the wrapper's CB).
+ // @see ReflowInput::InitCBReflowInput
+ auto tableWrapper = GetParent();
+ auto grandParent = tableWrapper->GetParent();
+ isGridItem = grandParent->IsGridContainerFrame() &&
+ !tableWrapper->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW);
+ if (isGridItem) {
+ // When resolving justify/align-self below, we want to use the grid
+ // container's justify/align-items value and WritingMode.
+ alignCB = grandParent;
+ }
+ }
+ const bool isFlexItem =
+ IsFlexItem() && !parentFrame->HasAnyStateBits(
+ NS_STATE_FLEX_IS_EMULATING_LEGACY_WEBKIT_BOX);
+ // This variable only gets set (and used) if isFlexItem is true. It
+ // indicates which axis (in this frame's own WM) corresponds to its
+ // flex container's main axis.
+ LogicalAxis flexMainAxis =
+ eLogicalAxisInline; // (init to make valgrind happy)
+ if (isFlexItem) {
+ flexMainAxis = nsFlexContainerFrame::IsItemInlineAxisMainAxis(this)
+ ? eLogicalAxisInline
+ : eLogicalAxisBlock;
+ }
+
+ const bool isOrthogonal = aWM.IsOrthogonalTo(alignCB->GetWritingMode());
+ const bool isAutoISize = styleISize.IsAuto();
+ const bool isAutoBSize =
+ nsLayoutUtils::IsAutoBSize(styleBSize, aCBSize.BSize(aWM));
+
+ // Compute inline-axis size
+ const bool isSubgriddedInInlineAxis =
+ isSubgrid && static_cast<nsGridContainerFrame*>(this)->IsColSubgrid();
+
+ // Per https://drafts.csswg.org/css-grid/#subgrid-box-alignment, if we are
+ // subgridded in the inline-axis, ignore our style inline-size, and stretch to
+ // fill the CB.
+ const bool shouldComputeISize = !isAutoISize && !isSubgriddedInInlineAxis;
+ if (shouldComputeISize) {
+ auto iSizeResult = ComputeISizeValue(
+ aRenderingContext, aWM, aCBSize, boxSizingAdjust,
+ boxSizingToMarginEdgeISize, styleISize, aSizeOverrides, aFlags);
+ result.ISize(aWM) = iSizeResult.mISize;
+ aspectRatioUsage = iSizeResult.mAspectRatioUsage;
+ } else if (MOZ_UNLIKELY(isGridItem) && !IsTrueOverflowContainer()) {
+ // 'auto' inline-size for grid-level box - fill the CB for 'stretch' /
+ // 'normal' and clamp it to the CB if requested:
+ bool stretch = false;
+ bool mayUseAspectRatio = aspectRatio && !isAutoBSize;
+ if (!aFlags.contains(ComputeSizeFlag::ShrinkWrap) &&
+ !StyleMargin()->HasInlineAxisAuto(aWM) &&
+ !alignCB->IsMasonry(isOrthogonal ? eLogicalAxisBlock
+ : eLogicalAxisInline)) {
+ auto inlineAxisAlignment =
+ isOrthogonal ? StylePosition()->UsedAlignSelf(alignCB->Style())._0
+ : StylePosition()->UsedJustifySelf(alignCB->Style())._0;
+ stretch = inlineAxisAlignment == StyleAlignFlags::STRETCH ||
+ (inlineAxisAlignment == StyleAlignFlags::NORMAL &&
+ !mayUseAspectRatio);
+ }
+
+ // Apply the preferred aspect ratio for alignments other than *stretch* and
+ // *normal without aspect ratio*.
+ // The spec says all other values should size the items as fit-content, and
+ // the intrinsic size should respect the preferred aspect ratio, so we also
+ // apply aspect ratio for all other values.
+ // https://drafts.csswg.org/css-grid/#grid-item-sizing
+ if (!stretch && mayUseAspectRatio) {
+ // Note: we don't need to handle aspect ratio for inline axis if both
+ // width/height are auto. The default ratio-dependent axis is block axis
+ // in this case, so we can simply get the block size from the non-auto
+ // |styleBSize|.
+ auto bSize = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ styleBSize.AsLengthPercentage());
+ result.ISize(aWM) = aspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisInline, aWM, bSize, boxSizingAdjust);
+ aspectRatioUsage = AspectRatioUsage::ToComputeISize;
+ }
+
+ if (stretch || aFlags.contains(ComputeSizeFlag::IClampMarginBoxMinSize)) {
+ auto iSizeToFillCB =
+ std::max(nscoord(0), aCBSize.ISize(aWM) - aBorderPadding.ISize(aWM) -
+ aMargin.ISize(aWM));
+ if (stretch || result.ISize(aWM) > iSizeToFillCB) {
+ result.ISize(aWM) = iSizeToFillCB;
+ }
+ }
+ } else if (aspectRatio && !isAutoBSize) {
+ auto bSize = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ styleBSize.AsLengthPercentage());
+ result.ISize(aWM) = aspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisInline, aWM, bSize, boxSizingAdjust);
+ aspectRatioUsage = AspectRatioUsage::ToComputeISize;
+ }
+
+ // Calculate and apply transferred min & max size contraints.
+ // https://drafts.csswg.org/css-sizing-4/#aspect-ratio-size-transfers
+ //
+ // Note: The basic principle is that sizing constraints transfer through the
+ // aspect-ratio to the other side to preserve the aspect ratio to the extent
+ // that they can without violating any sizes specified explicitly on that
+ // affected axis.
+ //
+ // FIXME: The spec words may not be correct, so we may have to update this
+ // tentative solution once this spec issue gets resolved. Here, we clamp the
+ // flex base size by the transferred min and max sizes, and don't include
+ // the transferred min & max sizes into its used min & max sizes. So this
+ // lets us match other browsers' current behaviors.
+ // https://github.com/w3c/csswg-drafts/issues/6071
+ //
+ // Note: This may make more sense if we clamp the flex base size in
+ // FlexItem::ResolveFlexBaseSizeFromAspectRatio(). However, the result should
+ // be identical. FlexItem::ResolveFlexBaseSizeFromAspectRatio() only handles
+ // the case of the definite cross size, and the definite cross size is clamped
+ // by the min & max cross sizes below in this function. This means its flex
+ // base size has been clamped by the transferred min & max size already after
+ // generating the flex items. So here we make the code more general for both
+ // definite cross size and indefinite cross size.
+ const bool isDefiniteISize = styleISize.IsLengthPercentage();
+ const auto& minBSizeCoord = stylePos->MinBSize(aWM);
+ const auto& maxBSizeCoord = stylePos->MaxBSize(aWM);
+ const bool isAutoMinBSize =
+ nsLayoutUtils::IsAutoBSize(minBSizeCoord, aCBSize.BSize(aWM));
+ const bool isAutoMaxBSize =
+ nsLayoutUtils::IsAutoBSize(maxBSizeCoord, aCBSize.BSize(aWM));
+ if (aspectRatio && !isDefiniteISize) {
+ const MinMaxSize minMaxBSize{
+ isAutoMinBSize ? 0
+ : nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ minBSizeCoord.AsLengthPercentage()),
+ isAutoMaxBSize ? NS_UNCONSTRAINEDSIZE
+ : nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ maxBSizeCoord.AsLengthPercentage())};
+ MinMaxSize transferredMinMaxISize = ComputeTransferredMinMaxInlineSize(
+ aWM, aspectRatio, minMaxBSize, boxSizingAdjust);
+
+ result.ISize(aWM) =
+ transferredMinMaxISize.ClampSizeToMinAndMax(result.ISize(aWM));
+ }
+
+ // Flex items ignore their min & max sizing properties in their
+ // flex container's main-axis. (Those properties get applied later in
+ // the flexbox algorithm.)
+ const bool isFlexItemInlineAxisMainAxis =
+ isFlexItem && flexMainAxis == eLogicalAxisInline;
+ // Grid items that are subgridded in inline-axis also ignore their min & max
+ // sizing properties in that axis.
+ const bool shouldIgnoreMinMaxISize =
+ isFlexItemInlineAxisMainAxis || isSubgriddedInInlineAxis;
+ const auto& maxISizeCoord = stylePos->MaxISize(aWM);
+ nscoord maxISize = NS_UNCONSTRAINEDSIZE;
+ if (!maxISizeCoord.IsNone() && !shouldIgnoreMinMaxISize) {
+ maxISize = ComputeISizeValue(aRenderingContext, aWM, aCBSize,
+ boxSizingAdjust, boxSizingToMarginEdgeISize,
+ maxISizeCoord, aSizeOverrides, aFlags)
+ .mISize;
+ result.ISize(aWM) = std::min(maxISize, result.ISize(aWM));
+ }
+
+ const auto& minISizeCoord = stylePos->MinISize(aWM);
+ nscoord minISize;
+ if (!minISizeCoord.IsAuto() && !shouldIgnoreMinMaxISize) {
+ minISize = ComputeISizeValue(aRenderingContext, aWM, aCBSize,
+ boxSizingAdjust, boxSizingToMarginEdgeISize,
+ minISizeCoord, aSizeOverrides, aFlags)
+ .mISize;
+ } else if (MOZ_UNLIKELY(
+ aFlags.contains(ComputeSizeFlag::IApplyAutoMinSize))) {
+ // This implements "Implied Minimum Size of Grid Items".
+ // https://drafts.csswg.org/css-grid/#min-size-auto
+ minISize = std::min(maxISize, GetMinISize(aRenderingContext));
+ if (styleISize.IsLengthPercentage()) {
+ minISize = std::min(minISize, result.ISize(aWM));
+ } else if (aFlags.contains(ComputeSizeFlag::IClampMarginBoxMinSize)) {
+ // "if the grid item spans only grid tracks that have a fixed max track
+ // sizing function, its automatic minimum size in that dimension is
+ // further clamped to less than or equal to the size necessary to fit
+ // its margin box within the resulting grid area (flooring at zero)"
+ // https://drafts.csswg.org/css-grid/#min-size-auto
+ auto maxMinISize =
+ std::max(nscoord(0), aCBSize.ISize(aWM) - aBorderPadding.ISize(aWM) -
+ aMargin.ISize(aWM));
+ minISize = std::min(minISize, maxMinISize);
+ }
+ } else if (aspectRatioUsage == AspectRatioUsage::ToComputeISize &&
+ ShouldApplyAutomaticMinimumOnInlineAxis(aWM, disp, stylePos)) {
+ // This means we successfully applied aspect-ratio and now need to check
+ // if we need to apply the implied minimum size:
+ // https://drafts.csswg.org/css-sizing-4/#aspect-ratio-minimum
+ MOZ_ASSERT(!HasReplacedSizing(),
+ "aspect-ratio minimums should not apply to replaced elements");
+ // The inline size computed by aspect-ratio shouldn't less than the content
+ // size.
+ minISize = GetMinISize(aRenderingContext);
+ } else {
+ // Treat "min-width: auto" as 0.
+ // NOTE: Technically, "auto" is supposed to behave like "min-content" on
+ // flex items. However, we don't need to worry about that here, because
+ // flex items' min-sizes are intentionally ignored until the flex
+ // container explicitly considers them during space distribution.
+ minISize = 0;
+ }
+ result.ISize(aWM) = std::max(minISize, result.ISize(aWM));
+
+ // Compute block-axis size
+ // (but not if we have auto bsize -- then, we'll just stick with the bsize
+ // that we already calculated in the initial ComputeAutoSize() call. However,
+ // if we have a valid preferred aspect ratio, we still have to compute the
+ // block size because aspect ratio affects the intrinsic content size.)
+ const bool isSubgriddedInBlockAxis =
+ isSubgrid && static_cast<nsGridContainerFrame*>(this)->IsRowSubgrid();
+
+ // Per https://drafts.csswg.org/css-grid/#subgrid-box-alignment, if we are
+ // subgridded in the block-axis, ignore our style block-size, and stretch to
+ // fill the CB.
+ const bool shouldComputeBSize = !isAutoBSize && !isSubgriddedInBlockAxis;
+ if (shouldComputeBSize) {
+ result.BSize(aWM) = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ styleBSize.AsLengthPercentage());
+ } else if (MOZ_UNLIKELY(isGridItem) && styleBSize.IsAuto() &&
+ !aFlags.contains(ComputeSizeFlag::IsGridMeasuringReflow) &&
+ !IsTrueOverflowContainer() &&
+ !alignCB->IsMasonry(isOrthogonal ? eLogicalAxisInline
+ : eLogicalAxisBlock)) {
+ auto cbSize = aCBSize.BSize(aWM);
+ if (cbSize != NS_UNCONSTRAINEDSIZE) {
+ // 'auto' block-size for grid-level box - fill the CB for 'stretch' /
+ // 'normal' and clamp it to the CB if requested:
+ bool stretch = false;
+ bool mayUseAspectRatio =
+ aspectRatio && result.ISize(aWM) != NS_UNCONSTRAINEDSIZE;
+ if (!StyleMargin()->HasBlockAxisAuto(aWM)) {
+ auto blockAxisAlignment =
+ isOrthogonal ? StylePosition()->UsedJustifySelf(alignCB->Style())._0
+ : StylePosition()->UsedAlignSelf(alignCB->Style())._0;
+ stretch = blockAxisAlignment == StyleAlignFlags::STRETCH ||
+ (blockAxisAlignment == StyleAlignFlags::NORMAL &&
+ !mayUseAspectRatio);
+ }
+
+ // Apply the preferred aspect ratio for alignments other than *stretch*
+ // and *normal without aspect ratio*.
+ // The spec says all other values should size the items as fit-content,
+ // and the intrinsic size should respect the preferred aspect ratio, so
+ // we also apply aspect ratio for all other values.
+ // https://drafts.csswg.org/css-grid/#grid-item-sizing
+ if (!stretch && mayUseAspectRatio) {
+ result.BSize(aWM) = aspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisBlock, aWM, result.ISize(aWM),
+ boxSizingAdjust);
+ MOZ_ASSERT(aspectRatioUsage == AspectRatioUsage::None);
+ aspectRatioUsage = AspectRatioUsage::ToComputeBSize;
+ }
+
+ if (stretch || aFlags.contains(ComputeSizeFlag::BClampMarginBoxMinSize)) {
+ auto bSizeToFillCB =
+ std::max(nscoord(0),
+ cbSize - aBorderPadding.BSize(aWM) - aMargin.BSize(aWM));
+ if (stretch || (result.BSize(aWM) != NS_UNCONSTRAINEDSIZE &&
+ result.BSize(aWM) > bSizeToFillCB)) {
+ result.BSize(aWM) = bSizeToFillCB;
+ }
+ }
+ }
+ } else if (aspectRatio) {
+ // If both inline and block dimensions are auto, the block axis is the
+ // ratio-dependent axis by default.
+ // If we have a super large inline size, aspect-ratio should still be
+ // applied (so aspectRatioUsage flag is set as expected). That's why we
+ // apply aspect-ratio unconditionally for auto block size here.
+ result.BSize(aWM) = aspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisBlock, aWM, result.ISize(aWM),
+ boxSizingAdjust);
+ MOZ_ASSERT(aspectRatioUsage == AspectRatioUsage::None);
+ aspectRatioUsage = AspectRatioUsage::ToComputeBSize;
+ }
+
+ if (result.BSize(aWM) != NS_UNCONSTRAINEDSIZE) {
+ // Flex items ignore their min & max sizing properties in their flex
+ // container's main-axis. (Those properties get applied later in the flexbox
+ // algorithm.)
+ const bool isFlexItemBlockAxisMainAxis =
+ isFlexItem && flexMainAxis == eLogicalAxisBlock;
+ // Grid items that are subgridded in block-axis also ignore their min & max
+ // sizing properties in that axis.
+ const bool shouldIgnoreMinMaxBSize =
+ isFlexItemBlockAxisMainAxis || isSubgriddedInBlockAxis;
+ if (!isAutoMaxBSize && !shouldIgnoreMinMaxBSize) {
+ nscoord maxBSize = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ maxBSizeCoord.AsLengthPercentage());
+ result.BSize(aWM) = std::min(maxBSize, result.BSize(aWM));
+ }
+
+ if (!isAutoMinBSize && !shouldIgnoreMinMaxBSize) {
+ nscoord minBSize = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), boxSizingAdjust.BSize(aWM),
+ minBSizeCoord.AsLengthPercentage());
+ result.BSize(aWM) = std::max(minBSize, result.BSize(aWM));
+ }
+ }
+
+ if (IsThemed(disp)) {
+ nsPresContext* pc = PresContext();
+ const LayoutDeviceIntSize widget = pc->Theme()->GetMinimumWidgetSize(
+ pc, this, disp->EffectiveAppearance());
+
+ // Convert themed widget's physical dimensions to logical coords
+ LogicalSize size(aWM, LayoutDeviceIntSize::ToAppUnits(
+ widget, pc->AppUnitsPerDevPixel()));
+
+ // GetMinimumWidgetSize() returns border-box; we need content-box.
+ size -= aBorderPadding;
+
+ if (size.BSize(aWM) > result.BSize(aWM)) {
+ result.BSize(aWM) = size.BSize(aWM);
+ }
+ if (size.ISize(aWM) > result.ISize(aWM)) {
+ result.ISize(aWM) = size.ISize(aWM);
+ }
+ }
+
+ result.ISize(aWM) = std::max(0, result.ISize(aWM));
+ result.BSize(aWM) = std::max(0, result.BSize(aWM));
+
+ return {result, aspectRatioUsage};
+}
+
+nsRect nsIFrame::ComputeTightBounds(DrawTarget* aDrawTarget) const {
+ return InkOverflowRect();
+}
+
+/* virtual */
+nsresult nsIFrame::GetPrefWidthTightBounds(gfxContext* aContext, nscoord* aX,
+ nscoord* aXMost) {
+ return NS_ERROR_NOT_IMPLEMENTED;
+}
+
+/* virtual */
+LogicalSize nsIFrame::ComputeAutoSize(
+ gfxContext* aRenderingContext, WritingMode aWM,
+ const mozilla::LogicalSize& aCBSize, nscoord aAvailableISize,
+ const mozilla::LogicalSize& aMargin,
+ const mozilla::LogicalSize& aBorderPadding,
+ const StyleSizeOverrides& aSizeOverrides, ComputeSizeFlags aFlags) {
+ // Use basic shrink-wrapping as a default implementation.
+ LogicalSize result(aWM, 0xdeadbeef, NS_UNCONSTRAINEDSIZE);
+
+ // don't bother setting it if the result won't be used
+ const auto& styleISize = aSizeOverrides.mStyleISize
+ ? *aSizeOverrides.mStyleISize
+ : StylePosition()->ISize(aWM);
+ if (styleISize.IsAuto()) {
+ nscoord availBased =
+ aAvailableISize - aMargin.ISize(aWM) - aBorderPadding.ISize(aWM);
+ result.ISize(aWM) = ShrinkISizeToFit(aRenderingContext, availBased, aFlags);
+ }
+ return result;
+}
+
+nscoord nsIFrame::ShrinkISizeToFit(gfxContext* aRenderingContext,
+ nscoord aISizeInCB,
+ ComputeSizeFlags aFlags) {
+ // If we're a container for font size inflation, then shrink
+ // wrapping inside of us should not apply font size inflation.
+ AutoMaybeDisableFontInflation an(this);
+
+ nscoord result;
+ nscoord minISize = GetMinISize(aRenderingContext);
+ if (minISize > aISizeInCB) {
+ const bool clamp = aFlags.contains(ComputeSizeFlag::IClampMarginBoxMinSize);
+ result = MOZ_UNLIKELY(clamp) ? aISizeInCB : minISize;
+ } else {
+ nscoord prefISize = GetPrefISize(aRenderingContext);
+ if (prefISize > aISizeInCB) {
+ result = aISizeInCB;
+ } else {
+ result = prefISize;
+ }
+ }
+ return result;
+}
+
+Maybe<nscoord> nsIFrame::ComputeInlineSizeFromAspectRatio(
+ WritingMode aWM, const LogicalSize& aCBSize,
+ const LogicalSize& aContentEdgeToBoxSizing,
+ const StyleSizeOverrides& aSizeOverrides, ComputeSizeFlags aFlags) const {
+ // FIXME: Bug 1670151: Use GetAspectRatio() to cover replaced elements (and
+ // then we can drop the check of eSupportsAspectRatio).
+ const AspectRatio aspectRatio =
+ aSizeOverrides.mAspectRatio
+ ? *aSizeOverrides.mAspectRatio
+ : StylePosition()->mAspectRatio.ToLayoutRatio();
+ if (!SupportsAspectRatio() || !aspectRatio) {
+ return Nothing();
+ }
+
+ const StyleSize& styleBSize = aSizeOverrides.mStyleBSize
+ ? *aSizeOverrides.mStyleBSize
+ : StylePosition()->BSize(aWM);
+ if (nsLayoutUtils::IsAutoBSize(styleBSize, aCBSize.BSize(aWM))) {
+ return Nothing();
+ }
+
+ MOZ_ASSERT(styleBSize.IsLengthPercentage());
+ nscoord bSize = nsLayoutUtils::ComputeBSizeValue(
+ aCBSize.BSize(aWM), aContentEdgeToBoxSizing.BSize(aWM),
+ styleBSize.AsLengthPercentage());
+ return Some(aspectRatio.ComputeRatioDependentSize(
+ LogicalAxis::eLogicalAxisInline, aWM, bSize, aContentEdgeToBoxSizing));
+}
+
+nsIFrame::ISizeComputationResult nsIFrame::ComputeISizeValue(
+ gfxContext* aRenderingContext, const WritingMode aWM,
+ const LogicalSize& aContainingBlockSize,
+ const LogicalSize& aContentEdgeToBoxSizing, nscoord aBoxSizingToMarginEdge,
+ ExtremumLength aSize, Maybe<nscoord> aAvailableISizeOverride,
+ const StyleSizeOverrides& aSizeOverrides, ComputeSizeFlags aFlags) {
+ // If 'this' is a container for font size inflation, then shrink
+ // wrapping inside of it should not apply font size inflation.
+ AutoMaybeDisableFontInflation an(this);
+ // If we have an aspect-ratio and a definite block size, we resolve the
+ // min-content and max-content size by the aspect-ratio and the block size.
+ // https://github.com/w3c/csswg-drafts/issues/5032
+ Maybe<nscoord> intrinsicSizeFromAspectRatio =
+ aSize == ExtremumLength::MozAvailable
+ ? Nothing()
+ : ComputeInlineSizeFromAspectRatio(aWM, aContainingBlockSize,
+ aContentEdgeToBoxSizing,
+ aSizeOverrides, aFlags);
+ nscoord result;
+ switch (aSize) {
+ case ExtremumLength::MaxContent:
+ result = intrinsicSizeFromAspectRatio ? *intrinsicSizeFromAspectRatio
+ : GetPrefISize(aRenderingContext);
+ NS_ASSERTION(result >= 0, "inline-size less than zero");
+ return {result, intrinsicSizeFromAspectRatio
+ ? AspectRatioUsage::ToComputeISize
+ : AspectRatioUsage::None};
+ case ExtremumLength::MinContent:
+ result = intrinsicSizeFromAspectRatio ? *intrinsicSizeFromAspectRatio
+ : GetMinISize(aRenderingContext);
+ NS_ASSERTION(result >= 0, "inline-size less than zero");
+ if (MOZ_UNLIKELY(
+ aFlags.contains(ComputeSizeFlag::IClampMarginBoxMinSize))) {
+ auto available =
+ aContainingBlockSize.ISize(aWM) -
+ (aBoxSizingToMarginEdge + aContentEdgeToBoxSizing.ISize(aWM));
+ result = std::min(available, result);
+ }
+ return {result, intrinsicSizeFromAspectRatio
+ ? AspectRatioUsage::ToComputeISize
+ : AspectRatioUsage::None};
+ case ExtremumLength::FitContentFunction:
+ case ExtremumLength::FitContent: {
+ nscoord pref = NS_UNCONSTRAINEDSIZE;
+ nscoord min = 0;
+ if (intrinsicSizeFromAspectRatio) {
+ // The min-content and max-content size are identical and equal to the
+ // size computed from the block size and the aspect ratio.
+ pref = min = *intrinsicSizeFromAspectRatio;
+ } else {
+ pref = GetPrefISize(aRenderingContext);
+ min = GetMinISize(aRenderingContext);
+ }
+
+ nscoord fill = aAvailableISizeOverride
+ ? *aAvailableISizeOverride
+ : aContainingBlockSize.ISize(aWM) -
+ (aBoxSizingToMarginEdge +
+ aContentEdgeToBoxSizing.ISize(aWM));
+
+ if (MOZ_UNLIKELY(
+ aFlags.contains(ComputeSizeFlag::IClampMarginBoxMinSize))) {
+ min = std::min(min, fill);
+ }
+ result = std::max(min, std::min(pref, fill));
+ NS_ASSERTION(result >= 0, "inline-size less than zero");
+ return {result};
+ }
+ case ExtremumLength::MozAvailable:
+ return {aContainingBlockSize.ISize(aWM) -
+ (aBoxSizingToMarginEdge + aContentEdgeToBoxSizing.ISize(aWM))};
+ }
+ MOZ_ASSERT_UNREACHABLE("Unknown extremum length?");
+ return {};
+}
+
+nscoord nsIFrame::ComputeISizeValue(const WritingMode aWM,
+ const LogicalSize& aContainingBlockSize,
+ const LogicalSize& aContentEdgeToBoxSizing,
+ const LengthPercentage& aSize) {
+ LAYOUT_WARN_IF_FALSE(
+ aContainingBlockSize.ISize(aWM) != NS_UNCONSTRAINEDSIZE,
+ "have unconstrained inline-size; this should only result from "
+ "very large sizes, not attempts at intrinsic inline-size "
+ "calculation");
+ NS_ASSERTION(aContainingBlockSize.ISize(aWM) >= 0,
+ "inline-size less than zero");
+
+ nscoord result = aSize.Resolve(aContainingBlockSize.ISize(aWM));
+ // The result of a calc() expression might be less than 0; we
+ // should clamp at runtime (below). (Percentages and coords that
+ // are less than 0 have already been dropped by the parser.)
+ result -= aContentEdgeToBoxSizing.ISize(aWM);
+ return std::max(0, result);
+}
+
+void nsIFrame::DidReflow(nsPresContext* aPresContext,
+ const ReflowInput* aReflowInput) {
+ NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS, ("nsIFrame::DidReflow"));
+
+ if (IsHiddenByContentVisibilityOfInFlowParentForLayout()) {
+ RemoveStateBits(NS_FRAME_IN_REFLOW);
+ return;
+ }
+
+ SVGObserverUtils::InvalidateDirectRenderingObservers(
+ this, SVGObserverUtils::INVALIDATE_REFLOW);
+
+ RemoveStateBits(NS_FRAME_IN_REFLOW | NS_FRAME_FIRST_REFLOW |
+ NS_FRAME_IS_DIRTY | NS_FRAME_HAS_DIRTY_CHILDREN);
+
+ // Clear bits that were used in ReflowInput::InitResizeFlags (see
+ // comment there for why we can't clear it there).
+ SetHasBSizeChange(false);
+ SetHasPaddingChange(false);
+
+ // Notify the percent bsize observer if there is a percent bsize.
+ // The observer may be able to initiate another reflow with a computed
+ // bsize. This happens in the case where a table cell has no computed
+ // bsize but can fabricate one when the cell bsize is known.
+ if (aReflowInput && aReflowInput->mPercentBSizeObserver && !GetPrevInFlow()) {
+ const auto& bsize =
+ aReflowInput->mStylePosition->BSize(aReflowInput->GetWritingMode());
+ if (bsize.HasPercent()) {
+ aReflowInput->mPercentBSizeObserver->NotifyPercentBSize(*aReflowInput);
+ }
+ }
+
+ aPresContext->ReflowedFrame();
+}
+
+void nsIFrame::FinishReflowWithAbsoluteFrames(nsPresContext* aPresContext,
+ ReflowOutput& aDesiredSize,
+ const ReflowInput& aReflowInput,
+ nsReflowStatus& aStatus,
+ bool aConstrainBSize) {
+ ReflowAbsoluteFrames(aPresContext, aDesiredSize, aReflowInput, aStatus,
+ aConstrainBSize);
+
+ FinishAndStoreOverflow(&aDesiredSize, aReflowInput.mStyleDisplay);
+}
+
+void nsIFrame::ReflowAbsoluteFrames(nsPresContext* aPresContext,
+ ReflowOutput& aDesiredSize,
+ const ReflowInput& aReflowInput,
+ nsReflowStatus& aStatus,
+ bool aConstrainBSize) {
+ if (HasAbsolutelyPositionedChildren()) {
+ nsAbsoluteContainingBlock* absoluteContainer = GetAbsoluteContainingBlock();
+
+ // Let the absolutely positioned container reflow any absolutely positioned
+ // child frames that need to be reflowed
+
+ // The containing block for the abs pos kids is formed by our padding edge.
+ nsMargin usedBorder = GetUsedBorder();
+ nscoord containingBlockWidth =
+ std::max(0, aDesiredSize.Width() - usedBorder.LeftRight());
+ nscoord containingBlockHeight =
+ std::max(0, aDesiredSize.Height() - usedBorder.TopBottom());
+ nsContainerFrame* container = do_QueryFrame(this);
+ NS_ASSERTION(container,
+ "Abs-pos children only supported on container frames for now");
+
+ nsRect containingBlock(0, 0, containingBlockWidth, containingBlockHeight);
+ AbsPosReflowFlags flags =
+ AbsPosReflowFlags::CBWidthAndHeightChanged; // XXX could be optimized
+ if (aConstrainBSize) {
+ flags |= AbsPosReflowFlags::ConstrainHeight;
+ }
+ absoluteContainer->Reflow(container, aPresContext, aReflowInput, aStatus,
+ containingBlock, flags,
+ &aDesiredSize.mOverflowAreas);
+ }
+}
+
+/* virtual */
+bool nsIFrame::CanContinueTextRun() const {
+ // By default, a frame will *not* allow a text run to be continued
+ // through it.
+ return false;
+}
+
+void nsIFrame::Reflow(nsPresContext* aPresContext, ReflowOutput& aDesiredSize,
+ const ReflowInput& aReflowInput,
+ nsReflowStatus& aStatus) {
+ MarkInReflow();
+ DO_GLOBAL_REFLOW_COUNT("nsFrame");
+ MOZ_ASSERT(aStatus.IsEmpty(), "Caller should pass a fresh reflow status!");
+ aDesiredSize.ClearSize();
+}
+
+bool nsIFrame::IsContentDisabled() const {
+ // FIXME(emilio): Doing this via CSS means callers must ensure the style is up
+ // to date, and they don't!
+ if (StyleUI()->UserInput() == StyleUserInput::None) {
+ return true;
+ }
+
+ auto* element = nsGenericHTMLElement::FromNodeOrNull(GetContent());
+ return element && element->IsDisabled();
+}
+
+bool nsIFrame::IsContentRelevant() const {
+ MOZ_ASSERT(StyleDisplay()->ContentVisibility(*this) ==
+ StyleContentVisibility::Auto);
+
+ auto* element = Element::FromNodeOrNull(GetContent());
+ MOZ_ASSERT(element);
+
+ Maybe<ContentRelevancy> relevancy = element->GetContentRelevancy();
+ return relevancy.isSome() && !relevancy->isEmpty();
+}
+
+bool nsIFrame::HidesContent(
+ const EnumSet<IncludeContentVisibility>& aInclude) const {
+ auto effectiveContentVisibility = StyleDisplay()->ContentVisibility(*this);
+ if (aInclude.contains(IncludeContentVisibility::Hidden) &&
+ effectiveContentVisibility == StyleContentVisibility::Hidden) {
+ return true;
+ }
+
+ if (aInclude.contains(IncludeContentVisibility::Auto) &&
+ effectiveContentVisibility == StyleContentVisibility::Auto) {
+ return !IsContentRelevant();
+ }
+
+ return false;
+}
+
+bool nsIFrame::HidesContentForLayout() const {
+ return HidesContent() && !PresShell()->IsForcingLayoutForHiddenContent(this);
+}
+
+bool nsIFrame::IsHiddenByContentVisibilityOfInFlowParentForLayout() const {
+ const auto* parent = GetInFlowParent();
+ // The anonymous children owned by parent are important for properly sizing
+ // their parents.
+ return parent && parent->HidesContentForLayout() &&
+ !(parent->HasAnyStateBits(NS_FRAME_OWNS_ANON_BOXES) &&
+ Style()->IsAnonBox());
+}
+
+nsIFrame* nsIFrame::GetClosestContentVisibilityAncestor(
+ const EnumSet<IncludeContentVisibility>& aInclude) const {
+ if (!StaticPrefs::layout_css_content_visibility_enabled()) {
+ return nullptr;
+ }
+
+ auto* parent = GetInFlowParent();
+ bool isAnonymousBlock = Style()->IsAnonBox() && parent &&
+ parent->HasAnyStateBits(NS_FRAME_OWNS_ANON_BOXES);
+ for (nsIFrame* cur = parent; cur; cur = cur->GetInFlowParent()) {
+ if (!isAnonymousBlock && cur->HidesContent(aInclude)) {
+ return cur;
+ }
+
+ // Anonymous boxes are not hidden by the content-visibility of their first
+ // non-anonymous ancestor, but can be hidden by ancestors further up the
+ // tree.
+ isAnonymousBlock = false;
+ }
+
+ return nullptr;
+}
+
+bool nsIFrame::IsHiddenByContentVisibilityOnAnyAncestor(
+ const EnumSet<IncludeContentVisibility>& aInclude) const {
+ return !!GetClosestContentVisibilityAncestor(aInclude);
+}
+
+bool nsIFrame::HasSelectionInSubtree() {
+ if (IsSelected()) {
+ return true;
+ }
+
+ RefPtr<nsFrameSelection> frameSelection = GetFrameSelection();
+ if (!frameSelection) {
+ return false;
+ }
+
+ const Selection* selection =
+ frameSelection->GetSelection(SelectionType::eNormal);
+ if (!selection) {
+ return false;
+ }
+
+ for (uint32_t i = 0; i < selection->RangeCount(); i++) {
+ auto* range = selection->GetRangeAt(i);
+ MOZ_ASSERT(range);
+
+ const auto* commonAncestorNode =
+ range->GetRegisteredClosestCommonInclusiveAncestor();
+ if (commonAncestorNode &&
+ commonAncestorNode->IsInclusiveDescendantOf(GetContent())) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+bool nsIFrame::UpdateIsRelevantContent(
+ const ContentRelevancy& aRelevancyToUpdate) {
+ MOZ_ASSERT(StyleDisplay()->ContentVisibility(*this) ==
+ StyleContentVisibility::Auto);
+
+ auto* element = Element::FromNodeOrNull(GetContent());
+ MOZ_ASSERT(element);
+
+ ContentRelevancy newRelevancy;
+ Maybe<ContentRelevancy> oldRelevancy = element->GetContentRelevancy();
+ if (oldRelevancy.isSome()) {
+ newRelevancy = *oldRelevancy;
+ }
+
+ auto setRelevancyValue = [&](ContentRelevancyReason reason, bool value) {
+ if (value) {
+ newRelevancy += reason;
+ } else {
+ newRelevancy -= reason;
+ }
+ };
+
+ if (!oldRelevancy ||
+ aRelevancyToUpdate.contains(ContentRelevancyReason::Visible)) {
+ Maybe<bool> visible = element->GetVisibleForContentVisibility();
+ if (visible.isSome()) {
+ setRelevancyValue(ContentRelevancyReason::Visible, *visible);
+ }
+ }
+
+ if (!oldRelevancy ||
+ aRelevancyToUpdate.contains(ContentRelevancyReason::FocusInSubtree)) {
+ setRelevancyValue(ContentRelevancyReason::FocusInSubtree,
+ element->State().HasAtLeastOneOfStates(
+ ElementState::FOCUS_WITHIN | ElementState::FOCUS));
+ }
+
+ if (!oldRelevancy ||
+ aRelevancyToUpdate.contains(ContentRelevancyReason::Selected)) {
+ setRelevancyValue(ContentRelevancyReason::Selected,
+ HasSelectionInSubtree());
+ }
+
+ // If the proximity to the viewport has not been determined yet,
+ // and neither the element nor its contents are focused or selected,
+ // we should wait for the determination of the proximity. Otherwise,
+ // there might be a redundant contentvisibilityautostatechange event.
+ // See https://github.com/w3c/csswg-drafts/issues/9803
+ bool isProximityToViewportDetermined =
+ oldRelevancy ? true : element->GetVisibleForContentVisibility().isSome();
+ if (!isProximityToViewportDetermined && newRelevancy.isEmpty()) {
+ return false;
+ }
+
+ bool overallRelevancyChanged =
+ !oldRelevancy || oldRelevancy->isEmpty() != newRelevancy.isEmpty();
+ if (!oldRelevancy || *oldRelevancy != newRelevancy) {
+ element->SetContentRelevancy(newRelevancy);
+ }
+
+ if (!overallRelevancyChanged) {
+ return false;
+ }
+
+ HandleLastRememberedSize();
+ PresContext()->SetNeedsToUpdateHiddenByContentVisibilityForAnimations();
+ PresShell()->FrameNeedsReflow(
+ this, IntrinsicDirty::FrameAncestorsAndDescendants, NS_FRAME_IS_DIRTY);
+ InvalidateFrame();
+
+ ContentVisibilityAutoStateChangeEventInit init;
+ init.mSkipped = newRelevancy.isEmpty();
+ RefPtr<ContentVisibilityAutoStateChangeEvent> event =
+ ContentVisibilityAutoStateChangeEvent::Constructor(
+ element, u"contentvisibilityautostatechange"_ns, init);
+
+ // Per
+ // https://drafts.csswg.org/css-contain/#content-visibility-auto-state-changed
+ // "This event is dispatched by posting a task at the time when the state
+ // change occurs."
+ RefPtr<AsyncEventDispatcher> asyncDispatcher =
+ new AsyncEventDispatcher(element, event.forget());
+ DebugOnly<nsresult> rv = asyncDispatcher->PostDOMEvent();
+ NS_ASSERTION(NS_SUCCEEDED(rv), "AsyncEventDispatcher failed to dispatch");
+ return true;
+}
+
+nsresult nsIFrame::CharacterDataChanged(const CharacterDataChangeInfo&) {
+ MOZ_ASSERT_UNREACHABLE("should only be called for text frames");
+ return NS_OK;
+}
+
+nsresult nsIFrame::AttributeChanged(int32_t aNameSpaceID, nsAtom* aAttribute,
+ int32_t aModType) {
+ return NS_OK;
+}
+
+// Flow member functions
+
+nsIFrame* nsIFrame::GetPrevContinuation() const { return nullptr; }
+
+void nsIFrame::SetPrevContinuation(nsIFrame* aPrevContinuation) {
+ MOZ_ASSERT(false, "not splittable");
+}
+
+nsIFrame* nsIFrame::GetNextContinuation() const { return nullptr; }
+
+void nsIFrame::SetNextContinuation(nsIFrame*) {
+ MOZ_ASSERT(false, "not splittable");
+}
+
+nsIFrame* nsIFrame::GetPrevInFlow() const { return nullptr; }
+
+void nsIFrame::SetPrevInFlow(nsIFrame* aPrevInFlow) {
+ MOZ_ASSERT(false, "not splittable");
+}
+
+nsIFrame* nsIFrame::GetNextInFlow() const { return nullptr; }
+
+void nsIFrame::SetNextInFlow(nsIFrame*) { MOZ_ASSERT(false, "not splittable"); }
+
+nsIFrame* nsIFrame::GetTailContinuation() {
+ nsIFrame* frame = this;
+ while (frame->HasAnyStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER)) {
+ frame = frame->GetPrevContinuation();
+ NS_ASSERTION(frame, "first continuation can't be overflow container");
+ }
+ for (nsIFrame* next = frame->GetNextContinuation();
+ next && !next->HasAnyStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER);
+ next = frame->GetNextContinuation()) {
+ frame = next;
+ }
+
+ MOZ_ASSERT(frame, "illegal state in continuation chain.");
+ return frame;
+}
+
+// Associated view object
+void nsIFrame::SetView(nsView* aView) {
+ if (aView) {
+ aView->SetFrame(this);
+
+#ifdef DEBUG
+ LayoutFrameType frameType = Type();
+ NS_ASSERTION(frameType == LayoutFrameType::SubDocument ||
+ frameType == LayoutFrameType::ListControl ||
+ frameType == LayoutFrameType::Viewport ||
+ frameType == LayoutFrameType::MenuPopup,
+ "Only specific frame types can have an nsView");
+#endif
+
+ // Store the view on the frame.
+ SetViewInternal(aView);
+
+ // Set the frame state bit that says the frame has a view
+ AddStateBits(NS_FRAME_HAS_VIEW);
+
+ // Let all of the ancestors know they have a descendant with a view.
+ for (nsIFrame* f = GetParent();
+ f && !f->HasAnyStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW);
+ f = f->GetParent())
+ f->AddStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW);
+ } else {
+ MOZ_ASSERT_UNREACHABLE("Destroying a view while the frame is alive?");
+ RemoveStateBits(NS_FRAME_HAS_VIEW);
+ SetViewInternal(nullptr);
+ }
+}
+
+// Find the first geometric parent that has a view
+nsIFrame* nsIFrame::GetAncestorWithView() const {
+ for (nsIFrame* f = GetParent(); nullptr != f; f = f->GetParent()) {
+ if (f->HasView()) {
+ return f;
+ }
+ }
+ return nullptr;
+}
+
+template <nsPoint (nsIFrame::*PositionGetter)() const>
+static nsPoint OffsetCalculator(const nsIFrame* aThis, const nsIFrame* aOther) {
+ MOZ_ASSERT(aOther, "Must have frame for destination coordinate system!");
+
+ NS_ASSERTION(aThis->PresContext() == aOther->PresContext(),
+ "GetOffsetTo called on frames in different documents");
+
+ nsPoint offset(0, 0);
+ const nsIFrame* f;
+ for (f = aThis; f != aOther && f; f = f->GetParent()) {
+ offset += (f->*PositionGetter)();
+ }
+
+ if (f != aOther) {
+ // Looks like aOther wasn't an ancestor of |this|. So now we have
+ // the root-frame-relative position of |this| in |offset|. Convert back
+ // to the coordinates of aOther
+ while (aOther) {
+ offset -= (aOther->*PositionGetter)();
+ aOther = aOther->GetParent();
+ }
+ }
+
+ return offset;
+}
+
+nsPoint nsIFrame::GetOffsetTo(const nsIFrame* aOther) const {
+ return OffsetCalculator<&nsIFrame::GetPosition>(this, aOther);
+}
+
+nsPoint nsIFrame::GetOffsetToIgnoringScrolling(const nsIFrame* aOther) const {
+ return OffsetCalculator<&nsIFrame::GetPositionIgnoringScrolling>(this,
+ aOther);
+}
+
+nsPoint nsIFrame::GetOffsetToCrossDoc(const nsIFrame* aOther) const {
+ return GetOffsetToCrossDoc(aOther, PresContext()->AppUnitsPerDevPixel());
+}
+
+nsPoint nsIFrame::GetOffsetToCrossDoc(const nsIFrame* aOther,
+ const int32_t aAPD) const {
+ MOZ_ASSERT(aOther, "Must have frame for destination coordinate system!");
+ NS_ASSERTION(PresContext()->GetRootPresContext() ==
+ aOther->PresContext()->GetRootPresContext(),
+ "trying to get the offset between frames in different document "
+ "hierarchies?");
+ if (PresContext()->GetRootPresContext() !=
+ aOther->PresContext()->GetRootPresContext()) {
+ // crash right away, we are almost certainly going to crash anyway.
+ MOZ_CRASH(
+ "trying to get the offset between frames in different "
+ "document hierarchies?");
+ }
+
+ const nsIFrame* root = nullptr;
+ // offset will hold the final offset
+ // docOffset holds the currently accumulated offset at the current APD, it
+ // will be converted and added to offset when the current APD changes.
+ nsPoint offset(0, 0), docOffset(0, 0);
+ const nsIFrame* f = this;
+ int32_t currAPD = PresContext()->AppUnitsPerDevPixel();
+ while (f && f != aOther) {
+ docOffset += f->GetPosition();
+ nsIFrame* parent = f->GetParent();
+ if (parent) {
+ f = parent;
+ } else {
+ nsPoint newOffset(0, 0);
+ root = f;
+ f = nsLayoutUtils::GetCrossDocParentFrameInProcess(f, &newOffset);
+ int32_t newAPD = f ? f->PresContext()->AppUnitsPerDevPixel() : 0;
+ if (!f || newAPD != currAPD) {
+ // Convert docOffset to the right APD and add it to offset.
+ offset += docOffset.ScaleToOtherAppUnits(currAPD, aAPD);
+ docOffset.x = docOffset.y = 0;
+ }
+ currAPD = newAPD;
+ docOffset += newOffset;
+ }
+ }
+ if (f == aOther) {
+ offset += docOffset.ScaleToOtherAppUnits(currAPD, aAPD);
+ } else {
+ // Looks like aOther wasn't an ancestor of |this|. So now we have
+ // the root-document-relative position of |this| in |offset|. Subtract the
+ // root-document-relative position of |aOther| from |offset|.
+ // This call won't try to recurse again because root is an ancestor of
+ // aOther.
+ nsPoint negOffset = aOther->GetOffsetToCrossDoc(root, aAPD);
+ offset -= negOffset;
+ }
+
+ return offset;
+}
+
+CSSIntRect nsIFrame::GetScreenRect() const {
+ return CSSIntRect::FromAppUnitsToNearest(GetScreenRectInAppUnits());
+}
+
+nsRect nsIFrame::GetScreenRectInAppUnits() const {
+ nsPresContext* presContext = PresContext();
+ nsIFrame* rootFrame = presContext->PresShell()->GetRootFrame();
+ nsPoint rootScreenPos(0, 0);
+ nsPoint rootFrameOffsetInParent(0, 0);
+ nsIFrame* rootFrameParent = nsLayoutUtils::GetCrossDocParentFrameInProcess(
+ rootFrame, &rootFrameOffsetInParent);
+ if (rootFrameParent) {
+ nsRect parentScreenRectAppUnits =
+ rootFrameParent->GetScreenRectInAppUnits();
+ nsPresContext* parentPresContext = rootFrameParent->PresContext();
+ double parentScale = double(presContext->AppUnitsPerDevPixel()) /
+ parentPresContext->AppUnitsPerDevPixel();
+ nsPoint rootPt =
+ parentScreenRectAppUnits.TopLeft() + rootFrameOffsetInParent;
+ rootScreenPos.x = NS_round(parentScale * rootPt.x);
+ rootScreenPos.y = NS_round(parentScale * rootPt.y);
+ } else {
+ nsCOMPtr<nsIWidget> rootWidget =
+ presContext->PresShell()->GetViewManager()->GetRootWidget();
+ if (rootWidget) {
+ LayoutDeviceIntPoint rootDevPx = rootWidget->WidgetToScreenOffset();
+ rootScreenPos.x = presContext->DevPixelsToAppUnits(rootDevPx.x);
+ rootScreenPos.y = presContext->DevPixelsToAppUnits(rootDevPx.y);
+ }
+ }
+
+ return nsRect(rootScreenPos + GetOffsetTo(rootFrame), GetSize());
+}
+
+// Returns the offset from this frame to the closest geometric parent that
+// has a view. Also returns the containing view or null in case of error
+void nsIFrame::GetOffsetFromView(nsPoint& aOffset, nsView** aView) const {
+ MOZ_ASSERT(nullptr != aView, "null OUT parameter pointer");
+ nsIFrame* frame = const_cast<nsIFrame*>(this);
+
+ *aView = nullptr;
+ aOffset.MoveTo(0, 0);
+ do {
+ aOffset += frame->GetPosition();
+ frame = frame->GetParent();
+ } while (frame && !frame->HasView());
+
+ if (frame) {
+ *aView = frame->GetView();
+ }
+}
+
+nsIWidget* nsIFrame::GetNearestWidget() const {
+ return GetClosestView()->GetNearestWidget(nullptr);
+}
+
+nsIWidget* nsIFrame::GetNearestWidget(nsPoint& aOffset) const {
+ nsPoint offsetToView;
+ nsPoint offsetToWidget;
+ nsIWidget* widget =
+ GetClosestView(&offsetToView)->GetNearestWidget(&offsetToWidget);
+ aOffset = offsetToView + offsetToWidget;
+ return widget;
+}
+
+Matrix4x4Flagged nsIFrame::GetTransformMatrix(ViewportType aViewportType,
+ RelativeTo aStopAtAncestor,
+ nsIFrame** aOutAncestor,
+ uint32_t aFlags) const {
+ MOZ_ASSERT(aOutAncestor, "Need a place to put the ancestor!");
+
+ /* If we're transformed, we want to hand back the combination
+ * transform/translate matrix that will apply our current transform, then
+ * shift us to our parent.
+ */
+ const bool isTransformed = IsTransformed();
+ const nsIFrame* zoomedContentRoot = nullptr;
+ if (aStopAtAncestor.mViewportType == ViewportType::Visual) {
+ zoomedContentRoot = ViewportUtils::IsZoomedContentRoot(this);
+ if (zoomedContentRoot) {
+ MOZ_ASSERT(aViewportType != ViewportType::Visual);
+ }
+ }
+
+ if (isTransformed || zoomedContentRoot) {
+ Matrix4x4 result;
+ int32_t scaleFactor =
+ ((aFlags & IN_CSS_UNITS) ? AppUnitsPerCSSPixel()
+ : PresContext()->AppUnitsPerDevPixel());
+
+ /* Compute the delta to the parent, which we need because we are converting
+ * coordinates to our parent.
+ */
+ if (isTransformed) {
+ NS_ASSERTION(nsLayoutUtils::GetCrossDocParentFrameInProcess(this),
+ "Cannot transform the viewport frame!");
+
+ result = result * nsDisplayTransform::GetResultingTransformMatrix(
+ this, nsPoint(0, 0), scaleFactor,
+ nsDisplayTransform::INCLUDE_PERSPECTIVE |
+ nsDisplayTransform::OFFSET_BY_ORIGIN);
+ }
+
+ // The offset from a zoomed content root to its parent (e.g. from
+ // a canvas frame to a scroll frame) is in layout coordinates, so
+ // apply it before applying any layout-to-visual transform.
+ *aOutAncestor = nsLayoutUtils::GetCrossDocParentFrameInProcess(this);
+ nsPoint delta = GetOffsetToCrossDoc(*aOutAncestor);
+ /* Combine the raw transform with a translation to our parent. */
+ result.PostTranslate(NSAppUnitsToFloatPixels(delta.x, scaleFactor),
+ NSAppUnitsToFloatPixels(delta.y, scaleFactor), 0.0f);
+
+ if (zoomedContentRoot) {
+ Matrix4x4 layoutToVisual;
+ ScrollableLayerGuid::ViewID targetScrollId =
+ nsLayoutUtils::FindOrCreateIDFor(zoomedContentRoot->GetContent());
+ if (aFlags & nsIFrame::IN_CSS_UNITS) {
+ layoutToVisual =
+ ViewportUtils::GetVisualToLayoutTransform(targetScrollId)
+ .Inverse()
+ .ToUnknownMatrix();
+ } else {
+ layoutToVisual =
+ ViewportUtils::GetVisualToLayoutTransform<LayoutDevicePixel>(
+ targetScrollId)
+ .Inverse()
+ .ToUnknownMatrix();
+ }
+ result = result * layoutToVisual;
+ }
+
+ return result;
+ }
+
+ *aOutAncestor = nsLayoutUtils::GetCrossDocParentFrameInProcess(this);
+
+ /* Otherwise, we're not transformed. In that case, we'll walk up the frame
+ * tree until we either hit the root frame or something that may be
+ * transformed. We'll then change coordinates into that frame, since we're
+ * guaranteed that nothing in-between can be transformed. First, however,
+ * we have to check to see if we have a parent. If not, we'll set the
+ * outparam to null (indicating that there's nothing left) and will hand back
+ * the identity matrix.
+ */
+ if (!*aOutAncestor) return Matrix4x4();
+
+ /* Keep iterating while the frame can't possibly be transformed. */
+ const nsIFrame* current = this;
+ auto shouldStopAt = [](const nsIFrame* aCurrent, nsIFrame* aAncestor,
+ uint32_t aFlags) {
+ return aAncestor->IsTransformed() || nsLayoutUtils::IsPopup(aAncestor) ||
+ ViewportUtils::IsZoomedContentRoot(aAncestor) ||
+ ((aFlags & STOP_AT_STACKING_CONTEXT_AND_DISPLAY_PORT) &&
+ (aAncestor->IsStackingContext() ||
+ DisplayPortUtils::FrameHasDisplayPort(aAncestor, aCurrent)));
+ };
+ while (*aOutAncestor != aStopAtAncestor.mFrame &&
+ !shouldStopAt(current, *aOutAncestor, aFlags)) {
+ /* If no parent, stop iterating. Otherwise, update the ancestor. */
+ nsIFrame* parent =
+ nsLayoutUtils::GetCrossDocParentFrameInProcess(*aOutAncestor);
+ if (!parent) break;
+
+ current = *aOutAncestor;
+ *aOutAncestor = parent;
+ }
+
+ NS_ASSERTION(*aOutAncestor, "Somehow ended up with a null ancestor...?");
+
+ /* Translate from this frame to our ancestor, if it exists. That's the
+ * entire transform, so we're done.
+ */
+ nsPoint delta = GetOffsetToCrossDoc(*aOutAncestor);
+ int32_t scaleFactor =
+ ((aFlags & IN_CSS_UNITS) ? AppUnitsPerCSSPixel()
+ : PresContext()->AppUnitsPerDevPixel());
+ return Matrix4x4::Translation(NSAppUnitsToFloatPixels(delta.x, scaleFactor),
+ NSAppUnitsToFloatPixels(delta.y, scaleFactor),
+ 0.0f);
+}
+
+static void InvalidateRenderingObservers(nsIFrame* aDisplayRoot,
+ nsIFrame* aFrame,
+ bool aFrameChanged = true) {
+ MOZ_ASSERT(aDisplayRoot == nsLayoutUtils::GetDisplayRootFrame(aFrame));
+ SVGObserverUtils::InvalidateDirectRenderingObservers(aFrame);
+ nsIFrame* parent = aFrame;
+ while (parent != aDisplayRoot &&
+ (parent = nsLayoutUtils::GetCrossDocParentFrameInProcess(parent)) &&
+ !parent->HasAnyStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT)) {
+ SVGObserverUtils::InvalidateDirectRenderingObservers(parent);
+ }
+
+ if (!aFrameChanged) {
+ return;
+ }
+
+ aFrame->MarkNeedsDisplayItemRebuild();
+}
+
+static void SchedulePaintInternal(
+ nsIFrame* aDisplayRoot, nsIFrame* aFrame,
+ nsIFrame::PaintType aType = nsIFrame::PAINT_DEFAULT) {
+ MOZ_ASSERT(aDisplayRoot == nsLayoutUtils::GetDisplayRootFrame(aFrame));
+ nsPresContext* pres = aDisplayRoot->PresContext()->GetRootPresContext();
+
+ // No need to schedule a paint for an external document since they aren't
+ // painted directly.
+ if (!pres || (pres->Document() && pres->Document()->IsResourceDoc())) {
+ return;
+ }
+ if (!pres->GetContainerWeak()) {
+ NS_WARNING("Shouldn't call SchedulePaint in a detached pres context");
+ return;
+ }
+
+ pres->PresShell()->ScheduleViewManagerFlush();
+
+ if (aType == nsIFrame::PAINT_DEFAULT) {
+ aDisplayRoot->AddStateBits(NS_FRAME_UPDATE_LAYER_TREE);
+ }
+}
+
+static void InvalidateFrameInternal(nsIFrame* aFrame, bool aHasDisplayItem,
+ bool aRebuildDisplayItems) {
+ if (aHasDisplayItem) {
+ aFrame->AddStateBits(NS_FRAME_NEEDS_PAINT);
+ }
+
+ if (aRebuildDisplayItems) {
+ aFrame->MarkNeedsDisplayItemRebuild();
+ }
+ SVGObserverUtils::InvalidateDirectRenderingObservers(aFrame);
+ bool needsSchedulePaint = false;
+ if (nsLayoutUtils::IsPopup(aFrame)) {
+ needsSchedulePaint = true;
+ } else {
+ nsIFrame* parent = nsLayoutUtils::GetCrossDocParentFrameInProcess(aFrame);
+ while (parent &&
+ !parent->HasAnyStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT)) {
+ if (aHasDisplayItem && !parent->HasAnyStateBits(NS_FRAME_IS_NONDISPLAY)) {
+ parent->AddStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT);
+ }
+ SVGObserverUtils::InvalidateDirectRenderingObservers(parent);
+
+ // If we're inside a popup, then we need to make sure that we
+ // call schedule paint so that the NS_FRAME_UPDATE_LAYER_TREE
+ // flag gets added to the popup display root frame.
+ if (nsLayoutUtils::IsPopup(parent)) {
+ needsSchedulePaint = true;
+ break;
+ }
+ parent = nsLayoutUtils::GetCrossDocParentFrameInProcess(parent);
+ }
+ if (!parent) {
+ needsSchedulePaint = true;
+ }
+ }
+ if (!aHasDisplayItem) {
+ return;
+ }
+ if (needsSchedulePaint) {
+ nsIFrame* displayRoot = nsLayoutUtils::GetDisplayRootFrame(aFrame);
+ SchedulePaintInternal(displayRoot, aFrame);
+ }
+ if (aFrame->HasAnyStateBits(NS_FRAME_HAS_INVALID_RECT)) {
+ aFrame->RemoveProperty(nsIFrame::InvalidationRect());
+ aFrame->RemoveStateBits(NS_FRAME_HAS_INVALID_RECT);
+ }
+}
+
+void nsIFrame::InvalidateFrameSubtree(bool aRebuildDisplayItems /* = true */) {
+ InvalidateFrame(0, aRebuildDisplayItems);
+
+ if (HasAnyStateBits(NS_FRAME_ALL_DESCENDANTS_NEED_PAINT)) {
+ return;
+ }
+
+ AddStateBits(NS_FRAME_ALL_DESCENDANTS_NEED_PAINT);
+
+ for (const auto& childList : CrossDocChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ // Don't explicitly rebuild display items for our descendants,
+ // since we should be marked and it implicitly includes all
+ // descendants.
+ child->InvalidateFrameSubtree(false);
+ }
+ }
+}
+
+void nsIFrame::ClearInvalidationStateBits() {
+ if (HasAnyStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT)) {
+ for (const auto& childList : CrossDocChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ child->ClearInvalidationStateBits();
+ }
+ }
+ }
+
+ RemoveStateBits(NS_FRAME_NEEDS_PAINT | NS_FRAME_DESCENDANT_NEEDS_PAINT |
+ NS_FRAME_ALL_DESCENDANTS_NEED_PAINT);
+}
+
+bool HasRetainedDataFor(const nsIFrame* aFrame, uint32_t aDisplayItemKey) {
+ if (RefPtr<WebRenderUserData> data =
+ GetWebRenderUserData<WebRenderFallbackData>(aFrame,
+ aDisplayItemKey)) {
+ return true;
+ }
+
+ return false;
+}
+
+void nsIFrame::InvalidateFrame(uint32_t aDisplayItemKey,
+ bool aRebuildDisplayItems /* = true */) {
+ bool hasDisplayItem =
+ !aDisplayItemKey || HasRetainedDataFor(this, aDisplayItemKey);
+ InvalidateFrameInternal(this, hasDisplayItem, aRebuildDisplayItems);
+}
+
+void nsIFrame::InvalidateFrameWithRect(const nsRect& aRect,
+ uint32_t aDisplayItemKey,
+ bool aRebuildDisplayItems /* = true */) {
+ if (aRect.IsEmpty()) {
+ return;
+ }
+ bool hasDisplayItem =
+ !aDisplayItemKey || HasRetainedDataFor(this, aDisplayItemKey);
+ bool alreadyInvalid = false;
+ if (!HasAnyStateBits(NS_FRAME_NEEDS_PAINT)) {
+ InvalidateFrameInternal(this, hasDisplayItem, aRebuildDisplayItems);
+ } else {
+ alreadyInvalid = true;
+ }
+
+ if (!hasDisplayItem) {
+ return;
+ }
+
+ nsRect* rect;
+ if (HasAnyStateBits(NS_FRAME_HAS_INVALID_RECT)) {
+ rect = GetProperty(InvalidationRect());
+ MOZ_ASSERT(rect);
+ } else {
+ if (alreadyInvalid) {
+ return;
+ }
+ rect = new nsRect();
+ AddProperty(InvalidationRect(), rect);
+ AddStateBits(NS_FRAME_HAS_INVALID_RECT);
+ }
+
+ *rect = rect->Union(aRect);
+}
+
+/*static*/
+uint8_t nsIFrame::sLayerIsPrerenderedDataKey;
+
+bool nsIFrame::IsInvalid(nsRect& aRect) {
+ if (!HasAnyStateBits(NS_FRAME_NEEDS_PAINT)) {
+ return false;
+ }
+
+ if (HasAnyStateBits(NS_FRAME_HAS_INVALID_RECT)) {
+ nsRect* rect = GetProperty(InvalidationRect());
+ NS_ASSERTION(
+ rect, "Must have an invalid rect if NS_FRAME_HAS_INVALID_RECT is set!");
+ aRect = *rect;
+ } else {
+ aRect.SetEmpty();
+ }
+ return true;
+}
+
+void nsIFrame::SchedulePaint(PaintType aType, bool aFrameChanged) {
+ if (PresShell()->IsPaintingSuppressed()) {
+ // We can't have any display items yet, and when we unsuppress we will
+ // invalidate the root frame.
+ return;
+ }
+ nsIFrame* displayRoot = nsLayoutUtils::GetDisplayRootFrame(this);
+ InvalidateRenderingObservers(displayRoot, this, aFrameChanged);
+ SchedulePaintInternal(displayRoot, this, aType);
+}
+
+void nsIFrame::SchedulePaintWithoutInvalidatingObservers(PaintType aType) {
+ nsIFrame* displayRoot = nsLayoutUtils::GetDisplayRootFrame(this);
+ SchedulePaintInternal(displayRoot, this, aType);
+}
+
+void nsIFrame::InvalidateLayer(DisplayItemType aDisplayItemKey,
+ const nsIntRect* aDamageRect,
+ const nsRect* aFrameDamageRect,
+ uint32_t aFlags /* = 0 */) {
+ NS_ASSERTION(aDisplayItemKey > DisplayItemType::TYPE_ZERO, "Need a key");
+
+ nsIFrame* displayRoot = nsLayoutUtils::GetDisplayRootFrame(this);
+ InvalidateRenderingObservers(displayRoot, this, false);
+
+ // Check if frame supports WebRender's async update
+ if ((aFlags & UPDATE_IS_ASYNC) &&
+ WebRenderUserData::SupportsAsyncUpdate(this)) {
+ // WebRender does not use layer, then return nullptr.
+ return;
+ }
+
+ if (aFrameDamageRect && aFrameDamageRect->IsEmpty()) {
+ return;
+ }
+
+ // In the bug 930056, dialer app startup but not shown on the
+ // screen because sometimes we don't have any retainned data
+ // for remote type displayitem and thus Repaint event is not
+ // triggered. So, always invalidate in this case.
+ DisplayItemType displayItemKey = aDisplayItemKey;
+ if (aDisplayItemKey == DisplayItemType::TYPE_REMOTE) {
+ displayItemKey = DisplayItemType::TYPE_ZERO;
+ }
+
+ if (aFrameDamageRect) {
+ InvalidateFrameWithRect(*aFrameDamageRect,
+ static_cast<uint32_t>(displayItemKey));
+ } else {
+ InvalidateFrame(static_cast<uint32_t>(displayItemKey));
+ }
+}
+
+static nsRect ComputeEffectsRect(nsIFrame* aFrame, const nsRect& aOverflowRect,
+ const nsSize& aNewSize) {
+ nsRect r = aOverflowRect;
+
+ if (aFrame->HasAnyStateBits(NS_FRAME_SVG_LAYOUT)) {
+ // For SVG frames, we only need to account for filters.
+ // TODO: We could also take account of clipPath and mask to reduce the
+ // ink overflow, but that's not essential.
+ if (aFrame->StyleEffects()->HasFilters()) {
+ SetOrUpdateRectValuedProperty(aFrame, nsIFrame::PreEffectsBBoxProperty(),
+ r);
+ r = SVGUtils::GetPostFilterInkOverflowRect(aFrame, aOverflowRect);
+ }
+ return r;
+ }
+
+ // box-shadow
+ r.UnionRect(r, nsLayoutUtils::GetBoxShadowRectForFrame(aFrame, aNewSize));
+
+ // border-image-outset.
+ // We need to include border-image-outset because it can cause the
+ // border image to be drawn beyond the border box.
+
+ // (1) It's important we not check whether there's a border-image
+ // since the style hint for a change in border image doesn't cause
+ // reflow, and that's probably more important than optimizing the
+ // overflow areas for the silly case of border-image-outset without
+ // border-image
+ // (2) It's important that we not check whether the border-image
+ // is actually loaded, since that would require us to reflow when
+ // the image loads.
+ const nsStyleBorder* styleBorder = aFrame->StyleBorder();
+ nsMargin outsetMargin = styleBorder->GetImageOutset();
+
+ if (outsetMargin != nsMargin(0, 0, 0, 0)) {
+ nsRect outsetRect(nsPoint(0, 0), aNewSize);
+ outsetRect.Inflate(outsetMargin);
+ r.UnionRect(r, outsetRect);
+ }
+
+ // Note that we don't remove the outlineInnerRect if a frame loses outline
+ // style. That would require an extra property lookup for every frame,
+ // or a new frame state bit to track whether a property had been stored,
+ // or something like that. It's not worth doing that here. At most it's
+ // only one heap-allocated rect per frame and it will be cleaned up when
+ // the frame dies.
+
+ if (SVGIntegrationUtils::UsingOverflowAffectingEffects(aFrame)) {
+ SetOrUpdateRectValuedProperty(aFrame, nsIFrame::PreEffectsBBoxProperty(),
+ r);
+ r = SVGIntegrationUtils::ComputePostEffectsInkOverflowRect(aFrame, r);
+ }
+
+ return r;
+}
+
+void nsIFrame::SetPosition(const nsPoint& aPt) {
+ if (mRect.TopLeft() == aPt) {
+ return;
+ }
+ mRect.MoveTo(aPt);
+ MarkNeedsDisplayItemRebuild();
+}
+
+void nsIFrame::MovePositionBy(const nsPoint& aTranslation) {
+ nsPoint position = GetNormalPosition() + aTranslation;
+
+ const nsMargin* computedOffsets = nullptr;
+ if (IsRelativelyOrStickyPositioned()) {
+ computedOffsets = GetProperty(nsIFrame::ComputedOffsetProperty());
+ }
+ ReflowInput::ApplyRelativePositioning(
+ this, computedOffsets ? *computedOffsets : nsMargin(), &position);
+ SetPosition(position);
+}
+
+nsRect nsIFrame::GetNormalRect() const {
+ // It might be faster to first check
+ // StyleDisplay()->IsRelativelyPositionedStyle().
+ bool hasProperty;
+ nsPoint normalPosition = GetProperty(NormalPositionProperty(), &hasProperty);
+ if (hasProperty) {
+ return nsRect(normalPosition, GetSize());
+ }
+ return GetRect();
+}
+
+nsRect nsIFrame::GetBoundingClientRect() {
+ return nsLayoutUtils::GetAllInFlowRectsUnion(
+ this, nsLayoutUtils::GetContainingBlockForClientRect(this),
+ nsLayoutUtils::RECTS_ACCOUNT_FOR_TRANSFORMS);
+}
+
+nsPoint nsIFrame::GetPositionIgnoringScrolling() const {
+ return GetParent() ? GetParent()->GetPositionOfChildIgnoringScrolling(this)
+ : GetPosition();
+}
+
+nsRect nsIFrame::GetOverflowRect(OverflowType aType) const {
+ // Note that in some cases the overflow area might not have been
+ // updated (yet) to reflect any outline set on the frame or the area
+ // of child frames. That's OK because any reflow that updates these
+ // areas will invalidate the appropriate area, so any (mis)uses of
+ // this method will be fixed up.
+
+ if (mOverflow.mType == OverflowStorageType::Large) {
+ // there is an overflow rect, and it's not stored as deltas but as
+ // a separately-allocated rect
+ return GetOverflowAreasProperty()->Overflow(aType);
+ }
+
+ if (aType == OverflowType::Ink &&
+ mOverflow.mType != OverflowStorageType::None) {
+ return InkOverflowFromDeltas();
+ }
+
+ return GetRectRelativeToSelf();
+}
+
+OverflowAreas nsIFrame::GetOverflowAreas() const {
+ if (mOverflow.mType == OverflowStorageType::Large) {
+ // there is an overflow rect, and it's not stored as deltas but as
+ // a separately-allocated rect
+ return *GetOverflowAreasProperty();
+ }
+
+ return OverflowAreas(InkOverflowFromDeltas(),
+ nsRect(nsPoint(0, 0), GetSize()));
+}
+
+OverflowAreas nsIFrame::GetOverflowAreasRelativeToSelf() const {
+ if (IsTransformed()) {
+ if (OverflowAreas* preTransformOverflows =
+ GetProperty(PreTransformOverflowAreasProperty())) {
+ return *preTransformOverflows;
+ }
+ }
+ return GetOverflowAreas();
+}
+
+OverflowAreas nsIFrame::GetOverflowAreasRelativeToParent() const {
+ return GetOverflowAreas() + GetPosition();
+}
+
+OverflowAreas nsIFrame::GetActualAndNormalOverflowAreasRelativeToParent()
+ const {
+ if (MOZ_LIKELY(!IsRelativelyOrStickyPositioned())) {
+ return GetOverflowAreasRelativeToParent();
+ }
+
+ const OverflowAreas overflows = GetOverflowAreas();
+ OverflowAreas actualAndNormalOverflows = overflows + GetPosition();
+ actualAndNormalOverflows.UnionWith(overflows + GetNormalPosition());
+ return actualAndNormalOverflows;
+}
+
+nsRect nsIFrame::ScrollableOverflowRectRelativeToParent() const {
+ return ScrollableOverflowRect() + GetPosition();
+}
+
+nsRect nsIFrame::InkOverflowRectRelativeToParent() const {
+ return InkOverflowRect() + GetPosition();
+}
+
+nsRect nsIFrame::ScrollableOverflowRectRelativeToSelf() const {
+ if (IsTransformed()) {
+ if (OverflowAreas* preTransformOverflows =
+ GetProperty(PreTransformOverflowAreasProperty())) {
+ return preTransformOverflows->ScrollableOverflow();
+ }
+ }
+ return ScrollableOverflowRect();
+}
+
+nsRect nsIFrame::InkOverflowRectRelativeToSelf() const {
+ if (IsTransformed()) {
+ if (OverflowAreas* preTransformOverflows =
+ GetProperty(PreTransformOverflowAreasProperty())) {
+ return preTransformOverflows->InkOverflow();
+ }
+ }
+ return InkOverflowRect();
+}
+
+nsRect nsIFrame::PreEffectsInkOverflowRect() const {
+ nsRect* r = GetProperty(nsIFrame::PreEffectsBBoxProperty());
+ return r ? *r : InkOverflowRectRelativeToSelf();
+}
+
+bool nsIFrame::UpdateOverflow() {
+ MOZ_ASSERT(FrameMaintainsOverflow(),
+ "Non-display SVG do not maintain ink overflow rects");
+
+ nsRect rect(nsPoint(0, 0), GetSize());
+ OverflowAreas overflowAreas(rect, rect);
+
+ if (!ComputeCustomOverflow(overflowAreas)) {
+ // If updating overflow wasn't supported by this frame, then it should
+ // have scheduled any necessary reflows. We can return false to say nothing
+ // changed, and wait for reflow to correct it.
+ return false;
+ }
+
+ UnionChildOverflow(overflowAreas);
+
+ if (FinishAndStoreOverflow(overflowAreas, GetSize())) {
+ if (nsView* view = GetView()) {
+ // Make sure the frame's view is properly sized.
+ nsViewManager* vm = view->GetViewManager();
+ vm->ResizeView(view, overflowAreas.InkOverflow(), true);
+ }
+
+ return true;
+ }
+
+ // Frames that combine their 3d transform with their ancestors
+ // only compute a pre-transform overflow rect, and then contribute
+ // to the normal overflow rect of the preserve-3d root. Always return
+ // true here so that we propagate changes up to the root for final
+ // calculation.
+ return Combines3DTransformWithAncestors();
+}
+
+/* virtual */
+bool nsIFrame::ComputeCustomOverflow(OverflowAreas& aOverflowAreas) {
+ return true;
+}
+
+bool nsIFrame::DoesClipChildrenInBothAxes() const {
+ if (IsScrollContainer()) {
+ return true;
+ }
+ const nsStyleDisplay* display = StyleDisplay();
+ if (display->IsContainPaint() && SupportsContainLayoutAndPaint()) {
+ return true;
+ }
+ return display->mOverflowX == StyleOverflow::Clip &&
+ display->mOverflowY == StyleOverflow::Clip;
+}
+
+/* virtual */
+void nsIFrame::UnionChildOverflow(OverflowAreas& aOverflowAreas) {
+ if (!DoesClipChildrenInBothAxes()) {
+ nsLayoutUtils::UnionChildOverflow(this, aOverflowAreas);
+ }
+}
+
+// Return true if this form control element's preferred size property (but not
+// percentage max size property) contains a percentage value that should be
+// resolved against zero when calculating its min-content contribution in the
+// corresponding axis.
+//
+// For proper replaced elements, the percentage value in both their max size
+// property or preferred size property should be resolved against zero. This is
+// handled in IsPercentageResolvedAgainstZero().
+inline static bool FormControlShrinksForPercentSize(const nsIFrame* aFrame) {
+ if (!aFrame->IsReplaced()) {
+ // Quick test to reject most frames.
+ return false;
+ }
+
+ LayoutFrameType fType = aFrame->Type();
+ if (fType == LayoutFrameType::Meter || fType == LayoutFrameType::Progress ||
+ fType == LayoutFrameType::Range) {
+ // progress, meter and range do have this shrinking behavior
+ // FIXME: Maybe these should be nsIFormControlFrame?
+ return true;
+ }
+
+ if (!static_cast<nsIFormControlFrame*>(do_QueryFrame(aFrame))) {
+ // Not a form control. This includes fieldsets, which do not
+ // shrink.
+ return false;
+ }
+
+ if (fType == LayoutFrameType::GfxButtonControl ||
+ fType == LayoutFrameType::HTMLButtonControl) {
+ // Buttons don't have this shrinking behavior. (Note that color
+ // inputs do, even though they inherit from button, so we can't use
+ // do_QueryFrame here.)
+ return false;
+ }
+
+ return true;
+}
+
+bool nsIFrame::IsPercentageResolvedAgainstZero(
+ const StyleSize& aStyleSize, const StyleMaxSize& aStyleMaxSize) const {
+ const bool sizeHasPercent = aStyleSize.HasPercent();
+ return ((sizeHasPercent || aStyleMaxSize.HasPercent()) &&
+ HasReplacedSizing()) ||
+ (sizeHasPercent && FormControlShrinksForPercentSize(this));
+}
+
+// Summary of the Cyclic-Percentage Intrinsic Size Contribution Rules:
+//
+// Element Type | Replaced | Non-replaced
+// Contribution Type | min-content max-content | min-content max-content
+// ---------------------------------------------------------------------------
+// min size | zero zero | zero zero
+// max & preferred size | zero initial | initial initial
+//
+// https://drafts.csswg.org/css-sizing-3/#cyclic-percentage-contribution
+bool nsIFrame::IsPercentageResolvedAgainstZero(const LengthPercentage& aSize,
+ SizeProperty aProperty) const {
+ // Early return to avoid calling the virtual function, IsFrameOfType().
+ if (aProperty == SizeProperty::MinSize) {
+ return true;
+ }
+
+ const bool hasPercentOnReplaced = aSize.HasPercent() && HasReplacedSizing();
+ if (aProperty == SizeProperty::MaxSize) {
+ return hasPercentOnReplaced;
+ }
+
+ MOZ_ASSERT(aProperty == SizeProperty::Size);
+ return hasPercentOnReplaced ||
+ (aSize.HasPercent() && FormControlShrinksForPercentSize(this));
+}
+
+bool nsIFrame::IsBlockWrapper() const {
+ auto pseudoType = Style()->GetPseudoType();
+ return pseudoType == PseudoStyleType::mozBlockInsideInlineWrapper ||
+ pseudoType == PseudoStyleType::buttonContent ||
+ pseudoType == PseudoStyleType::cellContent ||
+ pseudoType == PseudoStyleType::columnSpanWrapper;
+}
+
+bool nsIFrame::IsBlockFrameOrSubclass() const {
+ const nsBlockFrame* thisAsBlock = do_QueryFrame(this);
+ return !!thisAsBlock;
+}
+
+bool nsIFrame::IsImageFrameOrSubclass() const {
+ const nsImageFrame* asImage = do_QueryFrame(this);
+ return !!asImage;
+}
+
+bool nsIFrame::IsSubgrid() const {
+ return IsGridContainerFrame() &&
+ static_cast<const nsGridContainerFrame*>(this)->IsSubgrid();
+}
+
+static nsIFrame* GetNearestBlockContainer(nsIFrame* frame) {
+ while (!frame->IsBlockContainer()) {
+ frame = frame->GetParent();
+ NS_ASSERTION(
+ frame,
+ "How come we got to the root frame without seeing a containing block?");
+ }
+ return frame;
+}
+
+bool nsIFrame::IsBlockContainer() const {
+ // The block wrappers we use to wrap blocks inside inlines aren't
+ // described in the CSS spec. We need to make them not be containing
+ // blocks.
+ // Since the parent of such a block is either a normal block or
+ // another such pseudo, this shouldn't cause anything bad to happen.
+ // Also the anonymous blocks inside table cells are not containing blocks.
+ //
+ // If we ever start skipping table row groups from being containing blocks,
+ // you need to remove the StickyScrollContainer hack referencing bug 1421660.
+ return !IsLineParticipant() && !IsBlockWrapper() && !IsSubgrid() &&
+ // Table rows are not containing blocks either
+ !IsTableRowFrame();
+}
+
+nsIFrame* nsIFrame::GetContainingBlock(
+ uint32_t aFlags, const nsStyleDisplay* aStyleDisplay) const {
+ MOZ_ASSERT(aStyleDisplay == StyleDisplay());
+
+ // Keep this in sync with MightBeContainingBlockFor in ReflowInput.cpp.
+
+ if (!GetParent()) {
+ return nullptr;
+ }
+ // MathML frames might have absolute positioning style, but they would
+ // still be in-flow. So we have to check to make sure that the frame
+ // is really out-of-flow too.
+ nsIFrame* f;
+ if (IsAbsolutelyPositioned(aStyleDisplay)) {
+ f = GetParent(); // the parent is always the containing block
+ } else {
+ f = GetNearestBlockContainer(GetParent());
+ }
+
+ if (aFlags & SKIP_SCROLLED_FRAME && f &&
+ f->Style()->GetPseudoType() == PseudoStyleType::scrolledContent) {
+ f = f->GetParent();
+ }
+ return f;
+}
+
+#ifdef DEBUG_FRAME_DUMP
+
+Maybe<uint32_t> nsIFrame::ContentIndexInContainer(const nsIFrame* aFrame) {
+ if (nsIContent* content = aFrame->GetContent()) {
+ return content->ComputeIndexInParentContent();
+ }
+ return Nothing();
+}
+
+nsAutoCString nsIFrame::ListTag() const {
+ nsAutoString tmp;
+ GetFrameName(tmp);
+
+ nsAutoCString tag;
+ tag += NS_ConvertUTF16toUTF8(tmp);
+ tag += nsPrintfCString("@%p", static_cast<const void*>(this));
+ return tag;
+}
+
+std::string nsIFrame::ConvertToString(const LogicalRect& aRect,
+ const WritingMode aWM, ListFlags aFlags) {
+ if (aFlags.contains(ListFlag::DisplayInCSSPixels)) {
+ // Abuse CSSRect to store all LogicalRect's dimensions in CSS pixels.
+ return ToString(mozilla::CSSRect(CSSPixel::FromAppUnits(aRect.IStart(aWM)),
+ CSSPixel::FromAppUnits(aRect.BStart(aWM)),
+ CSSPixel::FromAppUnits(aRect.ISize(aWM)),
+ CSSPixel::FromAppUnits(aRect.BSize(aWM))));
+ }
+ return ToString(aRect);
+}
+
+std::string nsIFrame::ConvertToString(const LogicalSize& aSize,
+ const WritingMode aWM, ListFlags aFlags) {
+ if (aFlags.contains(ListFlag::DisplayInCSSPixels)) {
+ // Abuse CSSSize to store all LogicalSize's dimensions in CSS pixels.
+ return ToString(CSSSize(CSSPixel::FromAppUnits(aSize.ISize(aWM)),
+ CSSPixel::FromAppUnits(aSize.BSize(aWM))));
+ }
+ return ToString(aSize);
+}
+
+// Debugging
+void nsIFrame::ListGeneric(nsACString& aTo, const char* aPrefix,
+ ListFlags aFlags) const {
+ aTo += aPrefix;
+ aTo += ListTag();
+ if (HasView()) {
+ aTo += nsPrintfCString(" [view=%p]", static_cast<void*>(GetView()));
+ }
+ if (GetParent()) {
+ aTo += nsPrintfCString(" parent=%p", static_cast<void*>(GetParent()));
+ }
+ if (GetNextSibling()) {
+ aTo += nsPrintfCString(" next=%p", static_cast<void*>(GetNextSibling()));
+ }
+ if (GetPrevContinuation()) {
+ bool fluid = GetPrevInFlow() == GetPrevContinuation();
+ aTo += nsPrintfCString(" prev-%s=%p", fluid ? "in-flow" : "continuation",
+ static_cast<void*>(GetPrevContinuation()));
+ }
+ if (GetNextContinuation()) {
+ bool fluid = GetNextInFlow() == GetNextContinuation();
+ aTo += nsPrintfCString(" next-%s=%p", fluid ? "in-flow" : "continuation",
+ static_cast<void*>(GetNextContinuation()));
+ }
+ if (const nsAtom* const autoPageValue =
+ GetProperty(AutoPageValueProperty())) {
+ aTo += " AutoPage=";
+ aTo += nsAtomCString(autoPageValue);
+ }
+ if (const nsIFrame::PageValues* const pageValues =
+ GetProperty(PageValuesProperty())) {
+ aTo += " PageValues={";
+ if (pageValues->mStartPageValue) {
+ aTo += nsAtomCString(pageValues->mStartPageValue);
+ } else {
+ aTo += "<null>";
+ }
+ aTo += ", ";
+ if (pageValues->mEndPageValue) {
+ aTo += nsAtomCString(pageValues->mEndPageValue);
+ } else {
+ aTo += "<null>";
+ }
+ aTo += "}";
+ }
+ void* IBsibling = GetProperty(IBSplitSibling());
+ if (IBsibling) {
+ aTo += nsPrintfCString(" IBSplitSibling=%p", IBsibling);
+ }
+ void* IBprevsibling = GetProperty(IBSplitPrevSibling());
+ if (IBprevsibling) {
+ aTo += nsPrintfCString(" IBSplitPrevSibling=%p", IBprevsibling);
+ }
+ if (nsLayoutUtils::FontSizeInflationEnabled(PresContext())) {
+ if (HasAnyStateBits(NS_FRAME_FONT_INFLATION_FLOW_ROOT)) {
+ aTo += nsPrintfCString(" FFR");
+ if (nsFontInflationData* data =
+ nsFontInflationData::FindFontInflationDataFor(this)) {
+ aTo += nsPrintfCString(
+ ",enabled=%s,UIS=%s", data->InflationEnabled() ? "yes" : "no",
+ ConvertToString(data->UsableISize(), aFlags).c_str());
+ }
+ }
+ if (HasAnyStateBits(NS_FRAME_FONT_INFLATION_CONTAINER)) {
+ aTo += nsPrintfCString(" FIC");
+ }
+ aTo += nsPrintfCString(" FI=%f", nsLayoutUtils::FontSizeInflationFor(this));
+ }
+ aTo += nsPrintfCString(" %s", ConvertToString(mRect, aFlags).c_str());
+
+ mozilla::WritingMode wm = GetWritingMode();
+ if (wm.IsVertical() || wm.IsBidiRTL()) {
+ aTo +=
+ nsPrintfCString(" wm=%s logical-size=(%s)", ToString(wm).c_str(),
+ ConvertToString(GetLogicalSize(), wm, aFlags).c_str());
+ }
+
+ nsIFrame* parent = GetParent();
+ if (parent) {
+ WritingMode pWM = parent->GetWritingMode();
+ if (pWM.IsVertical() || pWM.IsBidiRTL()) {
+ nsSize containerSize = parent->mRect.Size();
+ LogicalRect lr(pWM, mRect, containerSize);
+ aTo += nsPrintfCString(" parent-wm=%s cs=(%s) logical-rect=%s",
+ ToString(pWM).c_str(),
+ ConvertToString(containerSize, aFlags).c_str(),
+ ConvertToString(lr, pWM, aFlags).c_str());
+ }
+ }
+ nsIFrame* f = const_cast<nsIFrame*>(this);
+ if (f->HasOverflowAreas()) {
+ nsRect io = f->InkOverflowRect();
+ if (!io.IsEqualEdges(mRect)) {
+ aTo += nsPrintfCString(" ink-overflow=%s",
+ ConvertToString(io, aFlags).c_str());
+ }
+ nsRect so = f->ScrollableOverflowRect();
+ if (!so.IsEqualEdges(mRect)) {
+ aTo += nsPrintfCString(" scr-overflow=%s",
+ ConvertToString(so, aFlags).c_str());
+ }
+ }
+ if (OverflowAreas* preTransformOverflows =
+ f->GetProperty(PreTransformOverflowAreasProperty())) {
+ nsRect io = preTransformOverflows->InkOverflow();
+ if (!io.IsEqualEdges(mRect) &&
+ (!f->HasOverflowAreas() || !io.IsEqualEdges(f->InkOverflowRect()))) {
+ aTo += nsPrintfCString(" pre-transform-ink-overflow=%s",
+ ConvertToString(io, aFlags).c_str());
+ }
+ nsRect so = preTransformOverflows->ScrollableOverflow();
+ if (!so.IsEqualEdges(mRect) &&
+ (!f->HasOverflowAreas() ||
+ !so.IsEqualEdges(f->ScrollableOverflowRect()))) {
+ aTo += nsPrintfCString(" pre-transform-scr-overflow=%s",
+ ConvertToString(so, aFlags).c_str());
+ }
+ }
+ bool hasNormalPosition;
+ nsPoint normalPosition = GetNormalPosition(&hasNormalPosition);
+ if (hasNormalPosition) {
+ aTo += nsPrintfCString(" normal-position=%s",
+ ConvertToString(normalPosition, aFlags).c_str());
+ }
+ if (HasProperty(BidiDataProperty())) {
+ FrameBidiData bidi = GetBidiData();
+ aTo += nsPrintfCString(" bidi(%d,%d,%d)", bidi.baseLevel.Value(),
+ bidi.embeddingLevel.Value(),
+ bidi.precedingControl.Value());
+ }
+ if (IsTransformed()) {
+ aTo += nsPrintfCString(" transformed");
+ }
+ if (ChildrenHavePerspective()) {
+ aTo += nsPrintfCString(" perspective");
+ }
+ if (Extend3DContext()) {
+ aTo += nsPrintfCString(" extend-3d");
+ }
+ if (Combines3DTransformWithAncestors()) {
+ aTo += nsPrintfCString(" combines-3d-transform-with-ancestors");
+ }
+ if (mContent) {
+ aTo += nsPrintfCString(" [content=%p]", static_cast<void*>(mContent));
+ }
+ aTo += nsPrintfCString(" [cs=%p", static_cast<void*>(mComputedStyle));
+ if (mComputedStyle) {
+ auto pseudoType = mComputedStyle->GetPseudoType();
+ aTo += ToString(pseudoType).c_str();
+ }
+ aTo += "]";
+
+ auto contentVisibility = StyleDisplay()->ContentVisibility(*this);
+ if (contentVisibility != StyleContentVisibility::Visible) {
+ aTo += nsPrintfCString(" [content-visibility=");
+ if (contentVisibility == StyleContentVisibility::Auto) {
+ aTo += "auto, "_ns;
+ } else if (contentVisibility == StyleContentVisibility::Hidden) {
+ aTo += "hiden, "_ns;
+ }
+
+ if (HidesContent()) {
+ aTo += "HidesContent=hidden"_ns;
+ } else {
+ aTo += "HidesContent=visibile"_ns;
+ }
+ aTo += "]";
+ }
+
+ if (IsFrameModified()) {
+ aTo += nsPrintfCString(" modified");
+ }
+
+ if (HasModifiedDescendants()) {
+ aTo += nsPrintfCString(" has-modified-descendants");
+ }
+}
+
+void nsIFrame::List(FILE* out, const char* aPrefix, ListFlags aFlags) const {
+ nsCString str;
+ ListGeneric(str, aPrefix, aFlags);
+ fprintf_stderr(out, "%s\n", str.get());
+}
+
+void nsIFrame::ListTextRuns(FILE* out) const {
+ nsTHashSet<const void*> seen;
+ ListTextRuns(out, seen);
+}
+
+void nsIFrame::ListTextRuns(FILE* out, nsTHashSet<const void*>& aSeen) const {
+ for (const auto& childList : ChildLists()) {
+ for (const nsIFrame* kid : childList.mList) {
+ kid->ListTextRuns(out, aSeen);
+ }
+ }
+}
+
+void nsIFrame::ListMatchedRules(FILE* out, const char* aPrefix) const {
+ nsTArray<const StyleLockedStyleRule*> rawRuleList;
+ Servo_ComputedValues_GetStyleRuleList(mComputedStyle, &rawRuleList);
+ for (const StyleLockedStyleRule* rawRule : rawRuleList) {
+ nsAutoCString ruleText;
+ Servo_StyleRule_GetCssText(rawRule, &ruleText);
+ fprintf_stderr(out, "%s%s\n", aPrefix, ruleText.get());
+ }
+}
+
+void nsIFrame::ListWithMatchedRules(FILE* out, const char* aPrefix) const {
+ fprintf_stderr(out, "%s%s\n", aPrefix, ListTag().get());
+
+ nsCString rulePrefix;
+ rulePrefix += aPrefix;
+ rulePrefix += " ";
+ ListMatchedRules(out, rulePrefix.get());
+}
+
+nsresult nsIFrame::GetFrameName(nsAString& aResult) const {
+ return MakeFrameName(u"Frame"_ns, aResult);
+}
+
+nsresult nsIFrame::MakeFrameName(const nsAString& aType,
+ nsAString& aResult) const {
+ aResult = aType;
+ if (mContent && !mContent->IsText()) {
+ nsAutoString buf;
+ mContent->NodeInfo()->NameAtom()->ToString(buf);
+ if (nsAtom* id = mContent->GetID()) {
+ buf.AppendLiteral(" id=");
+ buf.Append(nsDependentAtomString(id));
+ }
+ if (IsSubDocumentFrame()) {
+ nsAutoString src;
+ mContent->AsElement()->GetAttr(nsGkAtoms::src, src);
+ buf.AppendLiteral(" src=");
+ buf.Append(src);
+ }
+ aResult.Append('(');
+ aResult.Append(buf);
+ aResult.Append(')');
+ }
+ aResult.Append('(');
+ Maybe<uint32_t> index = ContentIndexInContainer(this);
+ if (index.isSome()) {
+ aResult.AppendInt(*index);
+ } else {
+ aResult.AppendInt(-1);
+ }
+ aResult.Append(')');
+ return NS_OK;
+}
+
+void nsIFrame::DumpFrameTree() const {
+ PresShell()->GetRootFrame()->List(stderr);
+}
+
+void nsIFrame::DumpFrameTreeInCSSPixels() const {
+ PresShell()->GetRootFrame()->List(stderr, "", ListFlag::DisplayInCSSPixels);
+}
+
+void nsIFrame::DumpFrameTreeLimited() const { List(stderr); }
+void nsIFrame::DumpFrameTreeLimitedInCSSPixels() const {
+ List(stderr, "", ListFlag::DisplayInCSSPixels);
+}
+
+#endif
+
+bool nsIFrame::IsVisibleForPainting() const {
+ return StyleVisibility()->IsVisible();
+}
+
+bool nsIFrame::IsVisibleOrCollapsedForPainting() const {
+ return StyleVisibility()->IsVisibleOrCollapsed();
+}
+
+/* virtual */
+bool nsIFrame::IsEmpty() {
+ return IsHiddenByContentVisibilityOfInFlowParentForLayout();
+}
+
+bool nsIFrame::CachedIsEmpty() {
+ MOZ_ASSERT(!HasAnyStateBits(NS_FRAME_IS_DIRTY) ||
+ IsHiddenByContentVisibilityOfInFlowParentForLayout(),
+ "Must only be called on reflowed lines or those hidden by "
+ "content-visibility.");
+ return IsEmpty();
+}
+
+/* virtual */
+bool nsIFrame::IsSelfEmpty() {
+ return IsHiddenByContentVisibilityOfInFlowParentForLayout();
+}
+
+nsresult nsIFrame::GetSelectionController(nsPresContext* aPresContext,
+ nsISelectionController** aSelCon) {
+ if (!aPresContext || !aSelCon) return NS_ERROR_INVALID_ARG;
+
+ nsIFrame* frame = this;
+ while (frame && frame->HasAnyStateBits(NS_FRAME_INDEPENDENT_SELECTION)) {
+ nsITextControlFrame* tcf = do_QueryFrame(frame);
+ if (tcf) {
+ return tcf->GetOwnedSelectionController(aSelCon);
+ }
+ frame = frame->GetParent();
+ }
+
+ *aSelCon = do_AddRef(aPresContext->PresShell()).take();
+ return NS_OK;
+}
+
+already_AddRefed<nsFrameSelection> nsIFrame::GetFrameSelection() {
+ RefPtr<nsFrameSelection> fs =
+ const_cast<nsFrameSelection*>(GetConstFrameSelection());
+ return fs.forget();
+}
+
+const nsFrameSelection* nsIFrame::GetConstFrameSelection() const {
+ nsIFrame* frame = const_cast<nsIFrame*>(this);
+ while (frame && frame->HasAnyStateBits(NS_FRAME_INDEPENDENT_SELECTION)) {
+ nsITextControlFrame* tcf = do_QueryFrame(frame);
+ if (tcf) {
+ return tcf->GetOwnedFrameSelection();
+ }
+ frame = frame->GetParent();
+ }
+
+ return PresShell()->ConstFrameSelection();
+}
+
+bool nsIFrame::IsFrameSelected() const {
+ NS_ASSERTION(!GetContent() || GetContent()->IsMaybeSelected(),
+ "use the public IsSelected() instead");
+ return GetContent()->IsSelected(0, GetContent()->GetChildCount());
+}
+
+nsresult nsIFrame::GetPointFromOffset(int32_t inOffset, nsPoint* outPoint) {
+ MOZ_ASSERT(outPoint != nullptr, "Null parameter");
+ nsRect contentRect = GetContentRectRelativeToSelf();
+ nsPoint pt = contentRect.TopLeft();
+ if (mContent) {
+ nsIContent* newContent = mContent->GetParent();
+ if (newContent) {
+ const int32_t newOffset = newContent->ComputeIndexOf_Deprecated(mContent);
+
+ // Find the direction of the frame from the EmbeddingLevelProperty,
+ // which is the resolved bidi level set in
+ // nsBidiPresUtils::ResolveParagraph (odd levels = right-to-left).
+ // If the embedding level isn't set, just use the CSS direction
+ // property.
+ bool hasBidiData;
+ FrameBidiData bidiData = GetProperty(BidiDataProperty(), &hasBidiData);
+ bool isRTL = hasBidiData
+ ? bidiData.embeddingLevel.IsRTL()
+ : StyleVisibility()->mDirection == StyleDirection::Rtl;
+ if ((!isRTL && inOffset > newOffset) ||
+ (isRTL && inOffset <= newOffset)) {
+ pt = contentRect.TopRight();
+ }
+ }
+ }
+ *outPoint = pt;
+ return NS_OK;
+}
+
+nsresult nsIFrame::GetCharacterRectsInRange(int32_t aInOffset, int32_t aLength,
+ nsTArray<nsRect>& aOutRect) {
+ /* no text */
+ return NS_ERROR_FAILURE;
+}
+
+nsresult nsIFrame::GetChildFrameContainingOffset(int32_t inContentOffset,
+ bool inHint,
+ int32_t* outFrameContentOffset,
+ nsIFrame** outChildFrame) {
+ MOZ_ASSERT(outChildFrame && outFrameContentOffset, "Null parameter");
+ *outFrameContentOffset = (int32_t)inHint;
+ // the best frame to reflect any given offset would be a visible frame if
+ // possible i.e. we are looking for a valid frame to place the blinking caret
+ nsRect rect = GetRect();
+ if (!rect.width || !rect.height) {
+ // if we have a 0 width or height then lets look for another frame that
+ // possibly has the same content. If we have no frames in flow then just
+ // let us return 'this' frame
+ nsIFrame* nextFlow = GetNextInFlow();
+ if (nextFlow)
+ return nextFlow->GetChildFrameContainingOffset(
+ inContentOffset, inHint, outFrameContentOffset, outChildFrame);
+ }
+ *outChildFrame = this;
+ return NS_OK;
+}
+
+//
+// What I've pieced together about this routine:
+// Starting with a block frame (from which a line frame can be gotten)
+// and a line number, drill down and get the first/last selectable
+// frame on that line, depending on aPos->mDirection.
+// aOutSideLimit != 0 means ignore aLineStart, instead work from
+// the end (if > 0) or beginning (if < 0).
+//
+static nsresult GetNextPrevLineFromBlockFrame(PeekOffsetStruct* aPos,
+ nsIFrame* aBlockFrame,
+ int32_t aLineStart,
+ int8_t aOutSideLimit) {
+ MOZ_ASSERT(aPos);
+ MOZ_ASSERT(aBlockFrame);
+
+ nsPresContext* pc = aBlockFrame->PresContext();
+
+ // magic numbers: aLineStart will be -1 for end of block, 0 will be start of
+ // block.
+
+ aPos->mResultFrame = nullptr;
+ aPos->mResultContent = nullptr;
+ aPos->mAttach = aPos->mDirection == eDirNext ? CaretAssociationHint::After
+ : CaretAssociationHint::Before;
+
+ AutoAssertNoDomMutations guard;
+ nsILineIterator* it = aBlockFrame->GetLineIterator();
+ if (!it) {
+ return NS_ERROR_FAILURE;
+ }
+ int32_t searchingLine = aLineStart;
+ int32_t countLines = it->GetNumLines();
+ if (aOutSideLimit > 0) { // start at end
+ searchingLine = countLines;
+ } else if (aOutSideLimit < 0) { // start at beginning
+ searchingLine = -1; //"next" will be 0
+ } else if ((aPos->mDirection == eDirPrevious && searchingLine == 0) ||
+ (aPos->mDirection == eDirNext &&
+ searchingLine >= (countLines - 1))) {
+ // Not found.
+ return NS_ERROR_FAILURE;
+ }
+ nsIFrame* resultFrame = nullptr;
+ nsIFrame* farStoppingFrame = nullptr; // we keep searching until we find a
+ // "this" frame then we go to next line
+ nsIFrame* nearStoppingFrame = nullptr; // if we are backing up from edge,
+ // stop here
+ nsIFrame* firstFrame;
+ nsIFrame* lastFrame;
+ bool isBeforeFirstFrame, isAfterLastFrame;
+ bool found = false;
+
+ while (!found) {
+ if (aPos->mDirection == eDirPrevious)
+ searchingLine--;
+ else
+ searchingLine++;
+ if ((aPos->mDirection == eDirPrevious && searchingLine < 0) ||
+ (aPos->mDirection == eDirNext && searchingLine >= countLines)) {
+ // we need to jump to new block frame.
+ return NS_ERROR_FAILURE;
+ }
+ auto line = it->GetLine(searchingLine).unwrap();
+ if (!line.mNumFramesOnLine) {
+ continue;
+ }
+ lastFrame = firstFrame = line.mFirstFrameOnLine;
+ for (int32_t lineFrameCount = line.mNumFramesOnLine; lineFrameCount > 1;
+ lineFrameCount--) {
+ lastFrame = lastFrame->GetNextSibling();
+ if (!lastFrame) {
+ NS_ERROR("GetLine promised more frames than could be found");
+ return NS_ERROR_FAILURE;
+ }
+ }
+ nsIFrame::GetLastLeaf(&lastFrame);
+
+ if (aPos->mDirection == eDirNext) {
+ nearStoppingFrame = firstFrame;
+ farStoppingFrame = lastFrame;
+ } else {
+ nearStoppingFrame = lastFrame;
+ farStoppingFrame = firstFrame;
+ }
+ nsPoint offset;
+ nsView* view; // used for call of get offset from view
+ aBlockFrame->GetOffsetFromView(offset, &view);
+ nsPoint newDesiredPos =
+ aPos->mDesiredCaretPos -
+ offset; // get desired position into blockframe coords
+ nsresult rv = it->FindFrameAt(searchingLine, newDesiredPos, &resultFrame,
+ &isBeforeFirstFrame, &isAfterLastFrame);
+ if (NS_FAILED(rv)) {
+ continue;
+ }
+
+ if (resultFrame) {
+ // check to see if this is ANOTHER blockframe inside the other one if so
+ // then call into its lines
+ if (resultFrame->CanProvideLineIterator()) {
+ aPos->mResultFrame = resultFrame;
+ return NS_OK;
+ }
+ // resultFrame is not a block frame
+ Maybe<nsFrameIterator> frameIterator;
+ frameIterator.emplace(
+ pc, resultFrame, nsFrameIterator::Type::PostOrder,
+ false, // aVisual
+ aPos->mOptions.contains(PeekOffsetOption::StopAtScroller),
+ false, // aFollowOOFs
+ false // aSkipPopupChecks
+ );
+
+ auto FoundValidFrame = [aPos](const nsIFrame::ContentOffsets& aOffsets,
+ const nsIFrame* aFrame) {
+ if (!aOffsets.content) {
+ return false;
+ }
+ if (!aFrame->IsSelectable(nullptr)) {
+ return false;
+ }
+ if (aPos->mOptions.contains(PeekOffsetOption::ForceEditableRegion) &&
+ !aOffsets.content->IsEditable()) {
+ return false;
+ }
+ return true;
+ };
+
+ nsIFrame* storeOldResultFrame = resultFrame;
+ while (!found) {
+ nsPoint point;
+ nsRect tempRect = resultFrame->GetRect();
+ nsPoint offset;
+ nsView* view; // used for call of get offset from view
+ resultFrame->GetOffsetFromView(offset, &view);
+ if (!view) {
+ return NS_ERROR_FAILURE;
+ }
+ if (resultFrame->GetWritingMode().IsVertical()) {
+ point.y = aPos->mDesiredCaretPos.y;
+ point.x = tempRect.width + offset.x;
+ } else {
+ point.y = tempRect.height + offset.y;
+ point.x = aPos->mDesiredCaretPos.x;
+ }
+
+ if (!resultFrame->HasView()) {
+ nsView* view;
+ nsPoint offset;
+ resultFrame->GetOffsetFromView(offset, &view);
+ nsIFrame::ContentOffsets offsets =
+ resultFrame->GetContentOffsetsFromPoint(
+ point - offset, nsIFrame::IGNORE_NATIVE_ANONYMOUS_SUBTREE);
+ aPos->mResultContent = offsets.content;
+ aPos->mContentOffset = offsets.offset;
+ aPos->mAttach = offsets.associate;
+ if (FoundValidFrame(offsets, resultFrame)) {
+ found = true;
+ break;
+ }
+ }
+
+ if (aPos->mDirection == eDirPrevious &&
+ resultFrame == farStoppingFrame) {
+ break;
+ }
+ if (aPos->mDirection == eDirNext && resultFrame == nearStoppingFrame) {
+ break;
+ }
+ // always try previous on THAT line if that fails go the other way
+ resultFrame = frameIterator->Traverse(/* aForward = */ false);
+ if (!resultFrame) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ if (!found) {
+ resultFrame = storeOldResultFrame;
+ frameIterator.reset();
+ frameIterator.emplace(
+ pc, resultFrame, nsFrameIterator::Type::Leaf,
+ false, // aVisual
+ aPos->mOptions.contains(PeekOffsetOption::StopAtScroller),
+ false, // aFollowOOFs
+ false // aSkipPopupChecks
+ );
+ MOZ_ASSERT(frameIterator);
+ }
+ while (!found) {
+ nsPoint point = aPos->mDesiredCaretPos;
+ nsView* view;
+ nsPoint offset;
+ resultFrame->GetOffsetFromView(offset, &view);
+ nsIFrame::ContentOffsets offsets =
+ resultFrame->GetContentOffsetsFromPoint(
+ point - offset, nsIFrame::IGNORE_NATIVE_ANONYMOUS_SUBTREE);
+ aPos->mResultContent = offsets.content;
+ aPos->mContentOffset = offsets.offset;
+ aPos->mAttach = offsets.associate;
+ if (FoundValidFrame(offsets, resultFrame)) {
+ found = true;
+ aPos->mAttach = resultFrame == farStoppingFrame
+ ? CaretAssociationHint::Before
+ : CaretAssociationHint::After;
+ break;
+ }
+ if (aPos->mDirection == eDirPrevious &&
+ (resultFrame == nearStoppingFrame))
+ break;
+ if (aPos->mDirection == eDirNext && (resultFrame == farStoppingFrame))
+ break;
+ // previous didnt work now we try "next"
+ nsIFrame* tempFrame = frameIterator->Traverse(/* aForward = */ true);
+ if (!tempFrame) break;
+ resultFrame = tempFrame;
+ }
+ aPos->mResultFrame = resultFrame;
+ } else {
+ // we need to jump to new block frame.
+ aPos->mAmount = eSelectLine;
+ aPos->mStartOffset = 0;
+ aPos->mAttach = aPos->mDirection == eDirNext
+ ? CaretAssociationHint::Before
+ : CaretAssociationHint::After;
+ if (aPos->mDirection == eDirPrevious)
+ aPos->mStartOffset = -1; // start from end
+ return aBlockFrame->PeekOffset(aPos);
+ }
+ }
+ return NS_OK;
+}
+
+nsIFrame::CaretPosition nsIFrame::GetExtremeCaretPosition(bool aStart) {
+ CaretPosition result;
+
+ FrameTarget targetFrame = DrillDownToSelectionFrame(this, !aStart, 0);
+ FrameContentRange range = GetRangeForFrame(targetFrame.frame);
+ result.mResultContent = range.content;
+ result.mContentOffset = aStart ? range.start : range.end;
+ return result;
+}
+
+// If this is a preformatted text frame, see if it ends with a newline
+static nsContentAndOffset FindLineBreakInText(nsIFrame* aFrame,
+ nsDirection aDirection) {
+ nsContentAndOffset result;
+
+ if (aFrame->IsGeneratedContentFrame() ||
+ !aFrame->HasSignificantTerminalNewline()) {
+ return result;
+ }
+
+ int32_t endOffset = aFrame->GetOffsets().second;
+ result.mContent = aFrame->GetContent();
+ result.mOffset = endOffset - (aDirection == eDirPrevious ? 0 : 1);
+ return result;
+}
+
+// Find the first (or last) descendant of the given frame
+// which is either a block-level frame or a BRFrame, or some other kind of break
+// which stops the line.
+static nsContentAndOffset FindLineBreakingFrame(nsIFrame* aFrame,
+ nsDirection aDirection) {
+ nsContentAndOffset result;
+
+ if (aFrame->IsGeneratedContentFrame()) {
+ return result;
+ }
+
+ // Treat form controls as inline leaves
+ // XXX we really need a way to determine whether a frame is inline-level
+ if (static_cast<nsIFormControlFrame*>(do_QueryFrame(aFrame))) {
+ return result;
+ }
+
+ // Check the frame itself
+ // Fall through block-in-inline split frames because their mContent is
+ // the content of the inline frames they were created from. The
+ // first/last child of such frames is the real block frame we're
+ // looking for.
+ if ((aFrame->IsBlockOutside() &&
+ !aFrame->HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT)) ||
+ aFrame->IsBrFrame()) {
+ nsIContent* content = aFrame->GetContent();
+ result.mContent = content->GetParent();
+ // In some cases (bug 310589, bug 370174) we end up here with a null
+ // content. This probably shouldn't ever happen, but since it sometimes
+ // does, we want to avoid crashing here.
+ NS_ASSERTION(result.mContent, "Unexpected orphan content");
+ if (result.mContent) {
+ result.mOffset = result.mContent->ComputeIndexOf_Deprecated(content) +
+ (aDirection == eDirPrevious ? 1 : 0);
+ }
+ return result;
+ }
+
+ result = FindLineBreakInText(aFrame, aDirection);
+ if (result.mContent) {
+ return result;
+ }
+
+ // Iterate over children and call ourselves recursively
+ if (aDirection == eDirPrevious) {
+ nsIFrame* child = aFrame->PrincipalChildList().LastChild();
+ while (child && !result.mContent) {
+ result = FindLineBreakingFrame(child, aDirection);
+ child = child->GetPrevSibling();
+ }
+ } else { // eDirNext
+ nsIFrame* child = aFrame->PrincipalChildList().FirstChild();
+ while (child && !result.mContent) {
+ result = FindLineBreakingFrame(child, aDirection);
+ child = child->GetNextSibling();
+ }
+ }
+ return result;
+}
+
+nsresult nsIFrame::PeekOffsetForParagraph(PeekOffsetStruct* aPos) {
+ nsIFrame* frame = this;
+ nsContentAndOffset blockFrameOrBR;
+ blockFrameOrBR.mContent = nullptr;
+ bool reachedLimit = frame->IsBlockOutside() || IsEditingHost(frame);
+
+ auto traverse = [&aPos](nsIFrame* current) {
+ return aPos->mDirection == eDirPrevious ? current->GetPrevSibling()
+ : current->GetNextSibling();
+ };
+
+ // Go through containing frames until reaching a block frame.
+ // In each step, search the previous (or next) siblings for the closest
+ // "stop frame" (a block frame or a BRFrame).
+ // If found, set it to be the selection boundary and abort.
+ while (!reachedLimit) {
+ nsIFrame* parent = frame->GetParent();
+ // Treat a frame associated with the root content as if it were a block
+ // frame.
+ if (!frame->mContent || !frame->mContent->GetParent()) {
+ reachedLimit = true;
+ break;
+ }
+
+ if (aPos->mDirection == eDirNext) {
+ // Try to find our own line-break before looking at our siblings.
+ blockFrameOrBR = FindLineBreakInText(frame, eDirNext);
+ }
+
+ nsIFrame* sibling = traverse(frame);
+ while (sibling && !blockFrameOrBR.mContent) {
+ blockFrameOrBR = FindLineBreakingFrame(sibling, aPos->mDirection);
+ sibling = traverse(sibling);
+ }
+ if (blockFrameOrBR.mContent) {
+ aPos->mResultContent = blockFrameOrBR.mContent;
+ aPos->mContentOffset = blockFrameOrBR.mOffset;
+ break;
+ }
+ frame = parent;
+ reachedLimit = frame && (frame->IsBlockOutside() || IsEditingHost(frame));
+ }
+
+ if (reachedLimit) { // no "stop frame" found
+ aPos->mResultContent = frame->GetContent();
+ if (aPos->mDirection == eDirPrevious) {
+ aPos->mContentOffset = 0;
+ } else if (aPos->mResultContent) {
+ aPos->mContentOffset = aPos->mResultContent->GetChildCount();
+ }
+ }
+ return NS_OK;
+}
+
+// Determine movement direction relative to frame
+static bool IsMovingInFrameDirection(const nsIFrame* frame,
+ nsDirection aDirection, bool aVisual) {
+ bool isReverseDirection =
+ aVisual && nsBidiPresUtils::IsReversedDirectionFrame(frame);
+ return aDirection == (isReverseDirection ? eDirPrevious : eDirNext);
+}
+
+// Determines "are we looking for a boundary between whitespace and
+// non-whitespace (in the direction we're moving in)". It is true when moving
+// forward and looking for a beginning of a word, or when moving backwards and
+// looking for an end of a word.
+static bool ShouldWordSelectionEatSpace(const PeekOffsetStruct& aPos) {
+ if (aPos.mWordMovementType != eDefaultBehavior) {
+ // aPos->mWordMovementType possible values:
+ // eEndWord: eat the space if we're moving backwards
+ // eStartWord: eat the space if we're moving forwards
+ return (aPos.mWordMovementType == eEndWord) ==
+ (aPos.mDirection == eDirPrevious);
+ }
+ // Use the hidden preference which is based on operating system
+ // behavior. This pref only affects whether moving forward by word
+ // should go to the end of this word or start of the next word. When
+ // going backwards, the start of the word is always used, on every
+ // operating system.
+ return aPos.mDirection == eDirNext &&
+ StaticPrefs::layout_word_select_eat_space_to_next_word();
+}
+
+enum class OffsetIsAtLineEdge : bool { No, Yes };
+
+static void SetPeekResultFromFrame(PeekOffsetStruct& aPos, nsIFrame* aFrame,
+ int32_t aOffset,
+ OffsetIsAtLineEdge aAtLineEdge) {
+ FrameContentRange range = GetRangeForFrame(aFrame);
+ aPos.mResultFrame = aFrame;
+ aPos.mResultContent = range.content;
+ // Output offset is relative to content, not frame
+ aPos.mContentOffset =
+ aOffset < 0 ? range.end + aOffset + 1 : range.start + aOffset;
+ if (aAtLineEdge == OffsetIsAtLineEdge::Yes) {
+ aPos.mAttach = aPos.mContentOffset == range.start
+ ? CaretAssociationHint::After
+ : CaretAssociationHint::Before;
+ }
+}
+
+void nsIFrame::SelectablePeekReport::TransferTo(PeekOffsetStruct& aPos) const {
+ return SetPeekResultFromFrame(aPos, mFrame, mOffset, OffsetIsAtLineEdge::No);
+}
+
+nsIFrame::SelectablePeekReport::SelectablePeekReport(
+ const mozilla::GenericErrorResult<nsresult>&& aErr) {
+ MOZ_ASSERT(NS_FAILED(aErr.operator nsresult()));
+ // Return an empty report
+}
+
+nsresult nsIFrame::PeekOffsetForCharacter(PeekOffsetStruct* aPos,
+ int32_t aOffset) {
+ SelectablePeekReport current{this, aOffset};
+
+ nsIFrame::FrameSearchResult peekSearchState = CONTINUE;
+
+ while (peekSearchState != FOUND) {
+ const bool movingInFrameDirection = IsMovingInFrameDirection(
+ current.mFrame, aPos->mDirection,
+ aPos->mOptions.contains(PeekOffsetOption::Visual));
+
+ if (current.mJumpedLine) {
+ // If we jumped lines, it's as if we found a character, but we still need
+ // to eat non-renderable content on the new line.
+ peekSearchState = current.PeekOffsetNoAmount(movingInFrameDirection);
+ } else {
+ PeekOffsetCharacterOptions options;
+ options.mRespectClusters = aPos->mAmount == eSelectCluster;
+ peekSearchState =
+ current.PeekOffsetCharacter(movingInFrameDirection, options);
+ }
+
+ current.mMovedOverNonSelectableText |=
+ peekSearchState == CONTINUE_UNSELECTABLE;
+
+ if (peekSearchState != FOUND) {
+ SelectablePeekReport next = current.mFrame->GetFrameFromDirection(*aPos);
+ if (next.Failed()) {
+ return NS_ERROR_FAILURE;
+ }
+ next.mJumpedLine |= current.mJumpedLine;
+ next.mMovedOverNonSelectableText |= current.mMovedOverNonSelectableText;
+ next.mHasSelectableFrame |= current.mHasSelectableFrame;
+ current = next;
+ }
+
+ // Found frame, but because we moved over non selectable text we want
+ // the offset to be at the frame edge. Note that if we are extending the
+ // selection, this doesn't matter.
+ if (peekSearchState == FOUND && current.mMovedOverNonSelectableText &&
+ (!aPos->mOptions.contains(PeekOffsetOption::Extend) ||
+ current.mHasSelectableFrame)) {
+ auto [start, end] = current.mFrame->GetOffsets();
+ current.mOffset = aPos->mDirection == eDirNext ? 0 : end - start;
+ }
+ }
+
+ // Set outputs
+ current.TransferTo(*aPos);
+ // If we're dealing with a text frame and moving backward positions us at
+ // the end of that line, decrease the offset by one to make sure that
+ // we're placed before the linefeed character on the previous line.
+ if (current.mOffset < 0 && current.mJumpedLine &&
+ aPos->mDirection == eDirPrevious &&
+ current.mFrame->HasSignificantTerminalNewline() &&
+ !current.mIgnoredBrFrame) {
+ --aPos->mContentOffset;
+ }
+ return NS_OK;
+}
+
+nsresult nsIFrame::PeekOffsetForWord(PeekOffsetStruct* aPos, int32_t aOffset) {
+ SelectablePeekReport current{this, aOffset};
+ bool shouldStopAtHardBreak =
+ aPos->mWordMovementType == eDefaultBehavior &&
+ StaticPrefs::layout_word_select_eat_space_to_next_word();
+ bool wordSelectEatSpace = ShouldWordSelectionEatSpace(*aPos);
+
+ PeekWordState state;
+ while (true) {
+ bool movingInFrameDirection = IsMovingInFrameDirection(
+ current.mFrame, aPos->mDirection,
+ aPos->mOptions.contains(PeekOffsetOption::Visual));
+
+ FrameSearchResult searchResult = current.mFrame->PeekOffsetWord(
+ movingInFrameDirection, wordSelectEatSpace,
+ aPos->mOptions.contains(PeekOffsetOption::IsKeyboardSelect),
+ &current.mOffset, &state,
+ !aPos->mOptions.contains(PeekOffsetOption::PreserveSpaces));
+ if (searchResult == FOUND) {
+ break;
+ }
+
+ SelectablePeekReport next = [&]() {
+ PeekOffsetOptions options = aPos->mOptions;
+ if (state.mSawInlineCharacter) {
+ // If we've already found a character, we don't want to stop at
+ // placeholder frame boundary if there is in the word.
+ options += PeekOffsetOption::StopAtPlaceholder;
+ }
+ return current.mFrame->GetFrameFromDirection(aPos->mDirection, options);
+ }();
+ if (next.Failed()) {
+ // If we've crossed the line boundary, check to make sure that we
+ // have not consumed a trailing newline as whitespace if it's
+ // significant.
+ if (next.mJumpedLine && wordSelectEatSpace &&
+ current.mFrame->HasSignificantTerminalNewline() &&
+ current.mFrame->StyleText()->mWhiteSpaceCollapse !=
+ StyleWhiteSpaceCollapse::PreserveBreaks) {
+ current.mOffset -= 1;
+ }
+ break;
+ }
+
+ if ((next.mJumpedLine || next.mFoundPlaceholder) && !wordSelectEatSpace &&
+ state.mSawBeforeType) {
+ // We can't jump lines if we're looking for whitespace following
+ // non-whitespace, and we already encountered non-whitespace.
+ break;
+ }
+
+ if (shouldStopAtHardBreak && next.mJumpedHardBreak) {
+ /**
+ * Prev, always: Jump and stop right there
+ * Next, saw inline: just stop
+ * Next, no inline: Jump and consume whitespaces
+ */
+ if (aPos->mDirection == eDirPrevious) {
+ // Try moving to the previous line if exists
+ current.TransferTo(*aPos);
+ current.mFrame->PeekOffsetForCharacter(aPos, current.mOffset);
+ return NS_OK;
+ }
+ if (state.mSawInlineCharacter || current.mJumpedHardBreak) {
+ if (current.mFrame->HasSignificantTerminalNewline()) {
+ current.mOffset -= 1;
+ }
+ current.TransferTo(*aPos);
+ return NS_OK;
+ }
+ // Mark the state as whitespace and continue
+ state.Update(false, true);
+ }
+
+ if (next.mJumpedLine) {
+ state.mContext.Truncate();
+ }
+ current = next;
+ // Jumping a line is equivalent to encountering whitespace
+ // This affects only when it already met an actual character
+ if (wordSelectEatSpace && next.mJumpedLine) {
+ state.SetSawBeforeType();
+ }
+ }
+
+ // Set outputs
+ current.TransferTo(*aPos);
+ return NS_OK;
+}
+
+static nsIFrame* GetFirstSelectableDescendantWithLineIterator(
+ nsIFrame* aParentFrame, bool aForceEditableRegion) {
+ auto FoundValidFrame = [aForceEditableRegion](const nsIFrame* aFrame) {
+ if (!aFrame->IsSelectable(nullptr)) {
+ return false;
+ }
+ if (aForceEditableRegion && !aFrame->GetContent()->IsEditable()) {
+ return false;
+ }
+ return true;
+ };
+
+ for (nsIFrame* child : aParentFrame->PrincipalChildList()) {
+ // some children may not be selectable, e.g. :before / :after pseudoelements
+ // content with user-select: none, or contenteditable="false"
+ // we need to skip them
+ if (child->CanProvideLineIterator() && FoundValidFrame(child)) {
+ return child;
+ }
+ if (nsIFrame* nested = GetFirstSelectableDescendantWithLineIterator(
+ child, aForceEditableRegion)) {
+ return nested;
+ }
+ }
+ return nullptr;
+}
+
+nsresult nsIFrame::PeekOffsetForLine(PeekOffsetStruct* aPos) {
+ nsIFrame* blockFrame = this;
+ nsresult result = NS_ERROR_FAILURE;
+
+ // outer loop
+ // moving to a next block when no more blocks are available in a subtree
+ AutoAssertNoDomMutations guard;
+ while (NS_FAILED(result)) {
+ auto [newBlock, lineFrame] = blockFrame->GetContainingBlockForLine(
+ aPos->mOptions.contains(PeekOffsetOption::StopAtScroller));
+ if (!newBlock) {
+ return NS_ERROR_FAILURE;
+ }
+ blockFrame = newBlock;
+ nsILineIterator* iter = blockFrame->GetLineIterator();
+ int32_t thisLine = iter->FindLineContaining(lineFrame);
+ if (NS_WARN_IF(thisLine < 0)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ int8_t edgeCase = 0; // no edge case. This should look at thisLine
+
+ // this part will find a frame or a block frame. If it's a block frame
+ // it will "drill down" to find a viable frame or it will return an
+ // error.
+ nsIFrame* lastFrame = this;
+
+ // inner loop - crawling the frames within a specific block subtree
+ while (true) {
+ result =
+ GetNextPrevLineFromBlockFrame(aPos, blockFrame, thisLine, edgeCase);
+ // we came back to same spot! keep going
+ if (NS_SUCCEEDED(result) &&
+ (!aPos->mResultFrame || aPos->mResultFrame == lastFrame)) {
+ aPos->mResultFrame = nullptr;
+ lastFrame = nullptr;
+ if (aPos->mDirection == eDirPrevious) {
+ thisLine--;
+ } else {
+ thisLine++;
+ }
+ continue;
+ }
+
+ if (NS_FAILED(result)) {
+ break;
+ }
+
+ lastFrame = aPos->mResultFrame; // set last frame
+ /* SPECIAL CHECK FOR NAVIGATION INTO TABLES
+ * when we hit a frame which doesn't have line iterator, we need to
+ * drill down and find a child with the line iterator to prevent the
+ * crawling process to prematurely finish. Note that this is only sound if
+ * we're guaranteed to not have multiple children implementing
+ * LineIterator.
+ *
+ * So far known cases are:
+ * 1) table wrapper (drill down into table row group)
+ * 2) table cell (drill down into its only anon child)
+ */
+ const bool shouldDrillIntoChildren =
+ aPos->mResultFrame->IsTableWrapperFrame() ||
+ aPos->mResultFrame->IsTableCellFrame();
+
+ if (shouldDrillIntoChildren) {
+ nsIFrame* child = GetFirstSelectableDescendantWithLineIterator(
+ aPos->mResultFrame,
+ aPos->mOptions.contains(PeekOffsetOption::ForceEditableRegion));
+ if (child) {
+ aPos->mResultFrame = child;
+ }
+ }
+
+ if (!aPos->mResultFrame->CanProvideLineIterator()) {
+ // no more selectable content at this level
+ break;
+ }
+
+ if (aPos->mResultFrame == blockFrame) {
+ // Make sure block element is not the same as the one we had before.
+ break;
+ }
+
+ // we've struck another block element with selectable content!
+ if (aPos->mDirection == eDirPrevious) {
+ edgeCase = 1; // far edge, search from end backwards
+ } else {
+ edgeCase = -1; // near edge search from beginning onwards
+ }
+ thisLine = 0; // this line means nothing now.
+ // everything else means something so keep looking "inside" the
+ // block
+ blockFrame = aPos->mResultFrame;
+ }
+ }
+ return result;
+}
+
+nsresult nsIFrame::PeekOffsetForLineEdge(PeekOffsetStruct* aPos) {
+ // Adjusted so that the caret can't get confused when content changes
+ nsIFrame* frame = AdjustFrameForSelectionStyles(this);
+ Element* editingHost = frame->GetContent()->GetEditingHost();
+
+ auto [blockFrame, lineFrame] = frame->GetContainingBlockForLine(
+ aPos->mOptions.contains(PeekOffsetOption::StopAtScroller));
+ if (!blockFrame) {
+ return NS_ERROR_FAILURE;
+ }
+ AutoAssertNoDomMutations guard;
+ nsILineIterator* it = blockFrame->GetLineIterator();
+ int32_t thisLine = it->FindLineContaining(lineFrame);
+ if (thisLine < 0) {
+ return NS_ERROR_FAILURE;
+ }
+
+ nsIFrame* baseFrame = nullptr;
+ bool endOfLine = eSelectEndLine == aPos->mAmount;
+
+ if (aPos->mOptions.contains(PeekOffsetOption::Visual) &&
+ PresContext()->BidiEnabled()) {
+ nsIFrame* firstFrame;
+ bool isReordered;
+ nsIFrame* lastFrame;
+ MOZ_TRY(
+ it->CheckLineOrder(thisLine, &isReordered, &firstFrame, &lastFrame));
+ baseFrame = endOfLine ? lastFrame : firstFrame;
+ } else {
+ auto line = it->GetLine(thisLine).unwrap();
+
+ nsIFrame* frame = line.mFirstFrameOnLine;
+ bool lastFrameWasEditable = false;
+ for (int32_t count = line.mNumFramesOnLine; count;
+ --count, frame = frame->GetNextSibling()) {
+ if (frame->IsGeneratedContentFrame()) {
+ continue;
+ }
+ // When jumping to the end of the line with the "end" key,
+ // try to skip over brFrames
+ if (endOfLine && line.mNumFramesOnLine > 1 && frame->IsBrFrame() &&
+ lastFrameWasEditable == frame->GetContent()->IsEditable()) {
+ continue;
+ }
+ lastFrameWasEditable =
+ frame->GetContent() && frame->GetContent()->IsEditable();
+ baseFrame = frame;
+ if (!endOfLine) {
+ break;
+ }
+ }
+ }
+ if (!baseFrame) {
+ return NS_ERROR_FAILURE;
+ }
+ // Make sure we are not leaving our inline editing host if exists
+ if (editingHost) {
+ if (nsIFrame* frame = editingHost->GetPrimaryFrame()) {
+ if (frame->IsInlineOutside() &&
+ !editingHost->Contains(baseFrame->GetContent())) {
+ baseFrame = frame;
+ if (endOfLine) {
+ baseFrame = baseFrame->LastContinuation();
+ }
+ }
+ }
+ }
+ FrameTarget targetFrame = DrillDownToSelectionFrame(baseFrame, endOfLine, 0);
+ SetPeekResultFromFrame(*aPos, targetFrame.frame, endOfLine ? -1 : 0,
+ OffsetIsAtLineEdge::Yes);
+ if (endOfLine && targetFrame.frame->HasSignificantTerminalNewline()) {
+ // Do not position the caret after the terminating newline if we're
+ // trying to move to the end of line (see bug 596506)
+ --aPos->mContentOffset;
+ }
+ if (!aPos->mResultContent) {
+ return NS_ERROR_FAILURE;
+ }
+ return NS_OK;
+}
+
+nsresult nsIFrame::PeekOffset(PeekOffsetStruct* aPos) {
+ MOZ_ASSERT(aPos);
+
+ if (NS_WARN_IF(HasAnyStateBits(NS_FRAME_IS_DIRTY))) {
+ // FIXME(Bug 1654362): <caption> currently can remain dirty.
+ return NS_ERROR_UNEXPECTED;
+ }
+
+ // Translate content offset to be relative to frame
+ int32_t offset = aPos->mStartOffset - GetRangeForFrame(this).start;
+
+ switch (aPos->mAmount) {
+ case eSelectCharacter:
+ case eSelectCluster:
+ return PeekOffsetForCharacter(aPos, offset);
+ case eSelectWordNoSpace:
+ // eSelectWordNoSpace means that we should not be eating any whitespace
+ // when moving to the adjacent word. This means that we should set aPos->
+ // mWordMovementType to eEndWord if we're moving forwards, and to
+ // eStartWord if we're moving backwards.
+ if (aPos->mDirection == eDirPrevious) {
+ aPos->mWordMovementType = eStartWord;
+ } else {
+ aPos->mWordMovementType = eEndWord;
+ }
+ // Intentionally fall through the eSelectWord case.
+ [[fallthrough]];
+ case eSelectWord:
+ return PeekOffsetForWord(aPos, offset);
+ case eSelectLine:
+ return PeekOffsetForLine(aPos);
+ case eSelectBeginLine:
+ case eSelectEndLine:
+ return PeekOffsetForLineEdge(aPos);
+ case eSelectParagraph:
+ return PeekOffsetForParagraph(aPos);
+ default: {
+ NS_ASSERTION(false, "Invalid amount");
+ return NS_ERROR_FAILURE;
+ }
+ }
+}
+
+nsIFrame::FrameSearchResult nsIFrame::PeekOffsetNoAmount(bool aForward,
+ int32_t* aOffset) {
+ NS_ASSERTION(aOffset && *aOffset <= 1, "aOffset out of range");
+ // Sure, we can stop right here.
+ return FOUND;
+}
+
+nsIFrame::FrameSearchResult nsIFrame::PeekOffsetCharacter(
+ bool aForward, int32_t* aOffset, PeekOffsetCharacterOptions aOptions) {
+ NS_ASSERTION(aOffset && *aOffset <= 1, "aOffset out of range");
+ int32_t startOffset = *aOffset;
+ // A negative offset means "end of frame", which in our case means offset 1.
+ if (startOffset < 0) startOffset = 1;
+ if (aForward == (startOffset == 0)) {
+ // We're before the frame and moving forward, or after it and moving
+ // backwards: skip to the other side and we're done.
+ *aOffset = 1 - startOffset;
+ return FOUND;
+ }
+ return CONTINUE;
+}
+
+nsIFrame::FrameSearchResult nsIFrame::PeekOffsetWord(
+ bool aForward, bool aWordSelectEatSpace, bool aIsKeyboardSelect,
+ int32_t* aOffset, PeekWordState* aState, bool /*aTrimSpaces*/) {
+ NS_ASSERTION(aOffset && *aOffset <= 1, "aOffset out of range");
+ int32_t startOffset = *aOffset;
+ // This isn't text, so truncate the context
+ aState->mContext.Truncate();
+ if (startOffset < 0) startOffset = 1;
+ if (aForward == (startOffset == 0)) {
+ // We're before the frame and moving forward, or after it and moving
+ // backwards. If we're looking for non-whitespace, we found it (without
+ // skipping this frame).
+ if (!aState->mAtStart) {
+ if (aState->mLastCharWasPunctuation) {
+ // We're not punctuation, so this is a punctuation boundary.
+ if (BreakWordBetweenPunctuation(aState, aForward, false, false,
+ aIsKeyboardSelect))
+ return FOUND;
+ } else {
+ // This is not a punctuation boundary.
+ if (aWordSelectEatSpace && aState->mSawBeforeType) return FOUND;
+ }
+ }
+ // Otherwise skip to the other side and note that we encountered
+ // non-whitespace.
+ *aOffset = 1 - startOffset;
+ aState->Update(false, // not punctuation
+ false // not whitespace
+ );
+ if (!aWordSelectEatSpace) aState->SetSawBeforeType();
+ }
+ return CONTINUE;
+}
+
+// static
+bool nsIFrame::BreakWordBetweenPunctuation(const PeekWordState* aState,
+ bool aForward, bool aPunctAfter,
+ bool aWhitespaceAfter,
+ bool aIsKeyboardSelect) {
+ NS_ASSERTION(aPunctAfter != aState->mLastCharWasPunctuation,
+ "Call this only at punctuation boundaries");
+ if (aState->mLastCharWasWhitespace) {
+ // We always stop between whitespace and punctuation
+ return true;
+ }
+ if (!StaticPrefs::layout_word_select_stop_at_punctuation()) {
+ // When this pref is false, we never stop at a punctuation boundary unless
+ // it's followed by whitespace (in the relevant direction).
+ return aWhitespaceAfter;
+ }
+ if (!aIsKeyboardSelect) {
+ // mouse caret movement (e.g. word selection) always stops at every
+ // punctuation boundary
+ return true;
+ }
+ bool afterPunct = aForward ? aState->mLastCharWasPunctuation : aPunctAfter;
+ if (!afterPunct) {
+ // keyboard caret movement only stops after punctuation (in content order)
+ return false;
+ }
+ // Stop only if we've seen some non-punctuation since the last whitespace;
+ // don't stop after punctuation that follows whitespace.
+ return aState->mSeenNonPunctuationSinceWhitespace;
+}
+
+std::pair<nsIFrame*, nsIFrame*> nsIFrame::GetContainingBlockForLine(
+ bool aLockScroll) const {
+ const nsIFrame* parentFrame = this;
+ const nsIFrame* frame;
+ while (parentFrame) {
+ frame = parentFrame;
+ if (frame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
+ // if we are searching for a frame that is not in flow we will not find
+ // it. we must instead look for its placeholder
+ if (frame->HasAnyStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER)) {
+ // abspos continuations don't have placeholders, get the fif
+ frame = frame->FirstInFlow();
+ }
+ frame = frame->GetPlaceholderFrame();
+ if (!frame) {
+ return std::pair(nullptr, nullptr);
+ }
+ }
+ parentFrame = frame->GetParent();
+ if (parentFrame) {
+ if (aLockScroll && parentFrame->IsScrollFrame()) {
+ return std::pair(nullptr, nullptr);
+ }
+ if (parentFrame->CanProvideLineIterator()) {
+ return std::pair(const_cast<nsIFrame*>(parentFrame),
+ const_cast<nsIFrame*>(frame));
+ }
+ }
+ }
+ return std::pair(nullptr, nullptr);
+}
+
+Result<bool, nsresult> nsIFrame::IsVisuallyAtLineEdge(
+ nsILineIterator* aLineIterator, int32_t aLine, nsDirection aDirection) {
+ auto line = aLineIterator->GetLine(aLine).unwrap();
+
+ const bool lineIsRTL = aLineIterator->IsLineIteratorFlowRTL();
+
+ nsIFrame *firstFrame = nullptr, *lastFrame = nullptr;
+ bool isReordered = false;
+ MOZ_TRY(aLineIterator->CheckLineOrder(aLine, &isReordered, &firstFrame,
+ &lastFrame));
+ if (!firstFrame || !lastFrame) {
+ return true; // XXX: Why true? We check whether `this` is at the edge...
+ }
+
+ nsIFrame* leftmostFrame = lineIsRTL ? lastFrame : firstFrame;
+ nsIFrame* rightmostFrame = lineIsRTL ? firstFrame : lastFrame;
+ auto FrameIsRTL = [](nsIFrame* aFrame) {
+ return nsBidiPresUtils::FrameDirection(aFrame) ==
+ mozilla::intl::BidiDirection::RTL;
+ };
+ if (!lineIsRTL == (aDirection == eDirPrevious)) {
+ nsIFrame* maybeLeftmostFrame = leftmostFrame;
+ for ([[maybe_unused]] int32_t i : IntegerRange(line.mNumFramesOnLine)) {
+ if (maybeLeftmostFrame == this) {
+ return true;
+ }
+ // If left edge of the line starts with placeholder frames, we can ignore
+ // them and should keep checking the following frames.
+ if (!maybeLeftmostFrame->IsPlaceholderFrame()) {
+ if ((FrameIsRTL(maybeLeftmostFrame) == lineIsRTL) ==
+ (aDirection == eDirPrevious)) {
+ nsIFrame::GetFirstLeaf(&maybeLeftmostFrame);
+ } else {
+ nsIFrame::GetLastLeaf(&maybeLeftmostFrame);
+ }
+ return maybeLeftmostFrame == this;
+ }
+ maybeLeftmostFrame = nsBidiPresUtils::GetFrameToRightOf(
+ maybeLeftmostFrame, line.mFirstFrameOnLine, line.mNumFramesOnLine);
+ if (!maybeLeftmostFrame) {
+ return false;
+ }
+ }
+ return false;
+ }
+
+ nsIFrame* maybeRightmostFrame = rightmostFrame;
+ for ([[maybe_unused]] int32_t i : IntegerRange(line.mNumFramesOnLine)) {
+ if (maybeRightmostFrame == this) {
+ return true;
+ }
+ // If the line ends with placehlder frames, we can ignore them and should
+ // keep checking the preceding frames.
+ if (!maybeRightmostFrame->IsPlaceholderFrame()) {
+ if ((FrameIsRTL(maybeRightmostFrame) == lineIsRTL) ==
+ (aDirection == eDirPrevious)) {
+ nsIFrame::GetFirstLeaf(&maybeRightmostFrame);
+ } else {
+ nsIFrame::GetLastLeaf(&maybeRightmostFrame);
+ }
+ return maybeRightmostFrame == this;
+ }
+ maybeRightmostFrame = nsBidiPresUtils::GetFrameToLeftOf(
+ maybeRightmostFrame, line.mFirstFrameOnLine, line.mNumFramesOnLine);
+ if (!maybeRightmostFrame) {
+ return false;
+ }
+ }
+ return false;
+}
+
+Result<bool, nsresult> nsIFrame::IsLogicallyAtLineEdge(
+ nsILineIterator* aLineIterator, int32_t aLine, nsDirection aDirection) {
+ auto line = aLineIterator->GetLine(aLine).unwrap();
+ if (!line.mNumFramesOnLine) {
+ return false;
+ }
+ MOZ_ASSERT(line.mFirstFrameOnLine);
+
+ if (aDirection == eDirPrevious) {
+ nsIFrame* maybeFirstFrame = line.mFirstFrameOnLine;
+ for ([[maybe_unused]] int32_t i : IntegerRange(line.mNumFramesOnLine)) {
+ if (maybeFirstFrame == this) {
+ return true;
+ }
+ // If the line starts with placeholder frames, we can ignore them and
+ // should keep checking the following frames.
+ if (!maybeFirstFrame->IsPlaceholderFrame()) {
+ nsIFrame::GetFirstLeaf(&maybeFirstFrame);
+ return maybeFirstFrame == this;
+ }
+ maybeFirstFrame = maybeFirstFrame->GetNextSibling();
+ if (!maybeFirstFrame) {
+ return false;
+ }
+ }
+ return false;
+ }
+
+ // eDirNext
+ nsIFrame* maybeLastFrame = line.GetLastFrameOnLine();
+ for ([[maybe_unused]] int32_t i : IntegerRange(line.mNumFramesOnLine)) {
+ if (maybeLastFrame == this) {
+ return true;
+ }
+ // If the line ends with placehlder frames, we can ignore them and should
+ // keep checking the preceding frames.
+ if (!maybeLastFrame->IsPlaceholderFrame()) {
+ nsIFrame::GetLastLeaf(&maybeLastFrame);
+ return maybeLastFrame == this;
+ }
+ maybeLastFrame = maybeLastFrame->GetPrevSibling();
+ }
+ return false;
+}
+
+nsIFrame::SelectablePeekReport nsIFrame::GetFrameFromDirection(
+ nsDirection aDirection, const PeekOffsetOptions& aOptions) {
+ SelectablePeekReport result;
+
+ nsPresContext* presContext = PresContext();
+ const bool needsVisualTraversal =
+ aOptions.contains(PeekOffsetOption::Visual) && presContext->BidiEnabled();
+ const bool followOofs =
+ !aOptions.contains(PeekOffsetOption::StopAtPlaceholder);
+ nsFrameIterator frameIterator(
+ presContext, this, nsFrameIterator::Type::Leaf, needsVisualTraversal,
+ aOptions.contains(PeekOffsetOption::StopAtScroller), followOofs,
+ false // aSkipPopupChecks
+ );
+
+ // Find the prev/next selectable frame
+ bool selectable = false;
+ nsIFrame* traversedFrame = this;
+ AutoAssertNoDomMutations guard;
+ const nsIContent* const nativeAnonymousSubtreeContent =
+ GetClosestNativeAnonymousSubtreeRoot();
+ while (!selectable) {
+ auto [blockFrame, lineFrame] = traversedFrame->GetContainingBlockForLine(
+ aOptions.contains(PeekOffsetOption::StopAtScroller));
+ if (!blockFrame) {
+ return result;
+ }
+
+ nsILineIterator* it = blockFrame->GetLineIterator();
+ int32_t thisLine = it->FindLineContaining(lineFrame);
+ if (thisLine < 0) {
+ return result;
+ }
+
+ bool atLineEdge;
+ MOZ_TRY_VAR(
+ atLineEdge,
+ needsVisualTraversal
+ ? traversedFrame->IsVisuallyAtLineEdge(it, thisLine, aDirection)
+ : traversedFrame->IsLogicallyAtLineEdge(it, thisLine, aDirection));
+ if (atLineEdge) {
+ result.mJumpedLine = true;
+ if (!aOptions.contains(PeekOffsetOption::JumpLines)) {
+ return result; // we are done. cannot jump lines
+ }
+ int32_t lineToCheckWrap =
+ aDirection == eDirPrevious ? thisLine - 1 : thisLine;
+ if (lineToCheckWrap < 0 ||
+ !it->GetLine(lineToCheckWrap).unwrap().mIsWrapped) {
+ result.mJumpedHardBreak = true;
+ }
+ }
+
+ traversedFrame = frameIterator.Traverse(aDirection == eDirNext);
+ if (!traversedFrame) {
+ return result;
+ }
+
+ if (aOptions.contains(PeekOffsetOption::StopAtPlaceholder) &&
+ traversedFrame->IsPlaceholderFrame()) {
+ // XXX If the placeholder frame does not have meaningful content, the user
+ // may want to select as a word around the out-of-flow cotent. However,
+ // non-text frame resets context in nsIFrame::PeekOffsetWord(). Therefore,
+ // next text frame considers the new word starts from its edge. So, it's
+ // not enough to implement such behavior with adding a check here whether
+ // the real frame may change the word with its contents if it were not
+ // out-of-flow.
+ result.mFoundPlaceholder = true;
+ return result;
+ }
+
+ auto IsSelectable =
+ [aOptions, nativeAnonymousSubtreeContent](const nsIFrame* aFrame) {
+ if (!aFrame->IsSelectable(nullptr)) {
+ return false;
+ }
+ // If the new frame is in a native anonymous subtree, we should treat
+ // it as not selectable unless the frame and found frame are in same
+ // subtree.
+ if (aFrame->GetClosestNativeAnonymousSubtreeRoot() !=
+ nativeAnonymousSubtreeContent) {
+ return false;
+ }
+ return !aOptions.contains(PeekOffsetOption::ForceEditableRegion) ||
+ aFrame->GetContent()->IsEditable();
+ };
+
+ // Skip br frames, but only if we can select something before hitting the
+ // end of the line or a non-selectable region.
+ if (atLineEdge && aDirection == eDirPrevious &&
+ traversedFrame->IsBrFrame()) {
+ for (nsIFrame* current = traversedFrame->GetPrevSibling(); current;
+ current = current->GetPrevSibling()) {
+ if (!current->IsBlockOutside() && IsSelectable(current)) {
+ if (!current->IsBrFrame()) {
+ result.mIgnoredBrFrame = true;
+ }
+ break;
+ }
+ }
+ if (result.mIgnoredBrFrame) {
+ continue;
+ }
+ }
+
+ selectable = IsSelectable(traversedFrame);
+ if (!selectable) {
+ if (traversedFrame->IsSelectable(nullptr)) {
+ result.mHasSelectableFrame = true;
+ }
+ result.mMovedOverNonSelectableText = true;
+ }
+ } // while (!selectable)
+
+ result.mOffset = (aDirection == eDirNext) ? 0 : -1;
+
+ if (aOptions.contains(PeekOffsetOption::Visual) &&
+ nsBidiPresUtils::IsReversedDirectionFrame(traversedFrame)) {
+ // The new frame is reverse-direction, go to the other end
+ result.mOffset = -1 - result.mOffset;
+ }
+ result.mFrame = traversedFrame;
+ return result;
+}
+
+nsIFrame::SelectablePeekReport nsIFrame::GetFrameFromDirection(
+ const PeekOffsetStruct& aPos) {
+ return GetFrameFromDirection(aPos.mDirection, aPos.mOptions);
+}
+
+nsView* nsIFrame::GetClosestView(nsPoint* aOffset) const {
+ nsPoint offset(0, 0);
+ for (const nsIFrame* f = this; f; f = f->GetParent()) {
+ if (f->HasView()) {
+ if (aOffset) *aOffset = offset;
+ return f->GetView();
+ }
+ offset += f->GetPosition();
+ }
+
+ MOZ_ASSERT_UNREACHABLE("No view on any parent? How did that happen?");
+ return nullptr;
+}
+
+/* virtual */
+void nsIFrame::ChildIsDirty(nsIFrame* aChild) {
+ MOZ_ASSERT_UNREACHABLE(
+ "should never be called on a frame that doesn't "
+ "inherit from nsContainerFrame");
+}
+
+#ifdef ACCESSIBILITY
+a11y::AccType nsIFrame::AccessibleType() {
+ if (IsTableCaption() && !GetRect().IsEmpty()) {
+ return a11y::eHTMLCaptionType;
+ }
+ return a11y::eNoType;
+}
+#endif
+
+bool nsIFrame::ClearOverflowRects() {
+ if (mOverflow.mType == OverflowStorageType::None) {
+ return false;
+ }
+ if (mOverflow.mType == OverflowStorageType::Large) {
+ RemoveProperty(OverflowAreasProperty());
+ }
+ mOverflow.mType = OverflowStorageType::None;
+ return true;
+}
+
+bool nsIFrame::SetOverflowAreas(const OverflowAreas& aOverflowAreas) {
+ if (mOverflow.mType == OverflowStorageType::Large) {
+ OverflowAreas* overflow = GetOverflowAreasProperty();
+ bool changed = *overflow != aOverflowAreas;
+ *overflow = aOverflowAreas;
+
+ // Don't bother with converting to the deltas form if we already
+ // have a property.
+ return changed;
+ }
+
+ const nsRect& vis = aOverflowAreas.InkOverflow();
+ uint32_t l = -vis.x, // left edge: positive delta is leftwards
+ t = -vis.y, // top: positive is upwards
+ r = vis.XMost() - mRect.width, // right: positive is rightwards
+ b = vis.YMost() - mRect.height; // bottom: positive is downwards
+ if (aOverflowAreas.ScrollableOverflow().IsEqualEdges(
+ nsRect(nsPoint(0, 0), GetSize())) &&
+ l <= InkOverflowDeltas::kMax && t <= InkOverflowDeltas::kMax &&
+ r <= InkOverflowDeltas::kMax && b <= InkOverflowDeltas::kMax &&
+ // we have to check these against zero because we *never* want to
+ // set a frame as having no overflow in this function. This is
+ // because FinishAndStoreOverflow calls this function prior to
+ // SetRect based on whether the overflow areas match aNewSize.
+ // In the case where the overflow areas exactly match mRect but
+ // do not match aNewSize, we need to store overflow in a property
+ // so that our eventual SetRect/SetSize will know that it has to
+ // reset our overflow areas.
+ (l | t | r | b) != 0) {
+ InkOverflowDeltas oldDeltas = mOverflow.mInkOverflowDeltas;
+ // It's a "small" overflow area so we store the deltas for each edge
+ // directly in the frame, rather than allocating a separate rect.
+ // If they're all zero, that's fine; we're setting things to
+ // no-overflow.
+ mOverflow.mInkOverflowDeltas.mLeft = l;
+ mOverflow.mInkOverflowDeltas.mTop = t;
+ mOverflow.mInkOverflowDeltas.mRight = r;
+ mOverflow.mInkOverflowDeltas.mBottom = b;
+ // There was no scrollable overflow before, and there isn't now.
+ return oldDeltas != mOverflow.mInkOverflowDeltas;
+ } else {
+ bool changed =
+ !aOverflowAreas.ScrollableOverflow().IsEqualEdges(
+ nsRect(nsPoint(0, 0), GetSize())) ||
+ !aOverflowAreas.InkOverflow().IsEqualEdges(InkOverflowFromDeltas());
+
+ // it's a large overflow area that we need to store as a property
+ mOverflow.mType = OverflowStorageType::Large;
+ AddProperty(OverflowAreasProperty(), new OverflowAreas(aOverflowAreas));
+ return changed;
+ }
+}
+
+enum class ApplyTransform : bool { No, Yes };
+
+/**
+ * Compute the outline inner rect (so without outline-width and outline-offset)
+ * of aFrame, maybe iterating over its descendants, in aFrame's coordinate space
+ * or its post-transform coordinate space (depending on aApplyTransform).
+ */
+static nsRect ComputeOutlineInnerRect(
+ nsIFrame* aFrame, ApplyTransform aApplyTransform, bool& aOutValid,
+ const nsSize* aSizeOverride = nullptr,
+ const OverflowAreas* aOverflowOverride = nullptr) {
+ const nsRect bounds(nsPoint(0, 0),
+ aSizeOverride ? *aSizeOverride : aFrame->GetSize());
+
+ // The SVG container frames besides SVGTextFrame do not maintain
+ // an accurate mRect. It will make the outline be larger than
+ // we expect, we need to make them narrow to their children's outline.
+ // aOutValid is set to false if the returned nsRect is not valid
+ // and should not be included in the outline rectangle.
+ aOutValid = !aFrame->HasAnyStateBits(NS_FRAME_SVG_LAYOUT) ||
+ !aFrame->IsSVGContainerFrame() || aFrame->IsSVGTextFrame();
+
+ nsRect u;
+
+ if (!aFrame->FrameMaintainsOverflow()) {
+ return u;
+ }
+
+ // Start from our border-box, transformed. See comment below about
+ // transform of children.
+ bool doTransform =
+ aApplyTransform == ApplyTransform::Yes && aFrame->IsTransformed();
+ TransformReferenceBox boundsRefBox(nullptr, bounds);
+ if (doTransform) {
+ u = nsDisplayTransform::TransformRect(bounds, aFrame, boundsRefBox);
+ } else {
+ u = bounds;
+ }
+
+ if (aOutValid && !StaticPrefs::layout_outline_include_overflow()) {
+ return u;
+ }
+
+ // Only iterate through the children if the overflow areas suggest
+ // that we might need to, and if the frame doesn't clip its overflow
+ // anyway.
+ if (aOverflowOverride) {
+ if (!doTransform && bounds.IsEqualEdges(aOverflowOverride->InkOverflow()) &&
+ bounds.IsEqualEdges(aOverflowOverride->ScrollableOverflow())) {
+ return u;
+ }
+ } else {
+ if (!doTransform && bounds.IsEqualEdges(aFrame->InkOverflowRect()) &&
+ bounds.IsEqualEdges(aFrame->ScrollableOverflowRect())) {
+ return u;
+ }
+ }
+ const nsStyleDisplay* disp = aFrame->StyleDisplay();
+ LayoutFrameType fType = aFrame->Type();
+ if (fType == LayoutFrameType::Scroll ||
+ fType == LayoutFrameType::ListControl ||
+ fType == LayoutFrameType::SVGOuterSVG) {
+ return u;
+ }
+
+ auto overflowClipAxes = aFrame->ShouldApplyOverflowClipping(disp);
+ auto overflowClipMargin = aFrame->OverflowClipMargin(overflowClipAxes);
+ if (overflowClipAxes == nsIFrame::PhysicalAxes::Both &&
+ overflowClipMargin == nsSize()) {
+ return u;
+ }
+
+ const nsStyleEffects* effects = aFrame->StyleEffects();
+ Maybe<nsRect> clipPropClipRect =
+ aFrame->GetClipPropClipRect(disp, effects, bounds.Size());
+
+ // Iterate over all children except pop-up, absolutely-positioned,
+ // float, and overflow ones.
+ const FrameChildListIDs skip = {
+ FrameChildListID::Popup, FrameChildListID::Absolute,
+ FrameChildListID::Fixed, FrameChildListID::Float,
+ FrameChildListID::Overflow};
+ for (const auto& [list, listID] : aFrame->ChildLists()) {
+ if (skip.contains(listID)) {
+ continue;
+ }
+
+ for (nsIFrame* child : list) {
+ if (child->IsPlaceholderFrame()) {
+ continue;
+ }
+
+ // Note that passing ApplyTransform::Yes when
+ // child->Combines3DTransformWithAncestors() returns true is incorrect if
+ // our aApplyTransform is No... but the opposite would be as well.
+ // This is because elements within a preserve-3d scene are always
+ // transformed up to the top of the scene. This means we don't have a
+ // mechanism for getting a transform up to an intermediate point within
+ // the scene. We choose to over-transform rather than under-transform
+ // because this is consistent with other overflow areas.
+ bool validRect = true;
+ nsRect childRect =
+ ComputeOutlineInnerRect(child, ApplyTransform::Yes, validRect) +
+ child->GetPosition();
+
+ if (!validRect) {
+ continue;
+ }
+
+ if (clipPropClipRect) {
+ // Intersect with the clip before transforming.
+ childRect.IntersectRect(childRect, *clipPropClipRect);
+ }
+
+ // Note that we transform each child separately according to
+ // aFrame's transform, and then union, which gives a different
+ // (smaller) result from unioning and then transforming the
+ // union. This doesn't match the way we handle overflow areas
+ // with 2-D transforms, though it does match the way we handle
+ // overflow areas in preserve-3d 3-D scenes.
+ if (doTransform && !child->Combines3DTransformWithAncestors()) {
+ childRect =
+ nsDisplayTransform::TransformRect(childRect, aFrame, boundsRefBox);
+ }
+
+ // If a SVGContainer has a non-SVGContainer child, we assign
+ // its child's outline to this SVGContainer directly.
+ if (!aOutValid && validRect) {
+ u = childRect;
+ aOutValid = true;
+ } else {
+ u = u.UnionEdges(childRect);
+ }
+ }
+ }
+
+ if (overflowClipAxes != nsIFrame::PhysicalAxes::None) {
+ OverflowAreas::ApplyOverflowClippingOnRect(u, bounds, overflowClipAxes,
+ overflowClipMargin);
+ }
+ return u;
+}
+
+static void ComputeAndIncludeOutlineArea(nsIFrame* aFrame,
+ OverflowAreas& aOverflowAreas,
+ const nsSize& aNewSize) {
+ const nsStyleOutline* outline = aFrame->StyleOutline();
+ if (!outline->ShouldPaintOutline()) {
+ return;
+ }
+
+ // When the outline property is set on a :-moz-block-inside-inline-wrapper
+ // pseudo-element, it inherited that outline from the inline that was broken
+ // because it contained a block. In that case, we don't want a really wide
+ // outline if the block inside the inline is narrow, so union the actual
+ // contents of the anonymous blocks.
+ nsIFrame* frameForArea = aFrame;
+ do {
+ PseudoStyleType pseudoType = frameForArea->Style()->GetPseudoType();
+ if (pseudoType != PseudoStyleType::mozBlockInsideInlineWrapper) break;
+ // If we're done, we really want it and all its later siblings.
+ frameForArea = frameForArea->PrincipalChildList().FirstChild();
+ NS_ASSERTION(frameForArea, "anonymous block with no children?");
+ } while (frameForArea);
+
+ // Find the union of the border boxes of all descendants, or in
+ // the block-in-inline case, all descendants we care about.
+ //
+ // Note that the interesting perspective-related cases are taken
+ // care of by the code that handles those issues for overflow
+ // calling FinishAndStoreOverflow again, which in turn calls this
+ // function again. We still need to deal with preserve-3d a bit.
+ nsRect innerRect;
+ bool validRect = false;
+ if (frameForArea == aFrame) {
+ innerRect = ComputeOutlineInnerRect(aFrame, ApplyTransform::No, validRect,
+ &aNewSize, &aOverflowAreas);
+ } else {
+ for (; frameForArea; frameForArea = frameForArea->GetNextSibling()) {
+ nsRect r =
+ ComputeOutlineInnerRect(frameForArea, ApplyTransform::Yes, validRect);
+
+ // Adjust for offsets transforms up to aFrame's pre-transform
+ // (i.e., normal) coordinate space; see comments in
+ // UnionBorderBoxes for some of the subtlety here.
+ for (nsIFrame *f = frameForArea, *parent = f->GetParent();
+ /* see middle of loop */; f = parent, parent = f->GetParent()) {
+ r += f->GetPosition();
+ if (parent == aFrame) {
+ break;
+ }
+ if (parent->IsTransformed() && !f->Combines3DTransformWithAncestors()) {
+ TransformReferenceBox refBox(parent);
+ r = nsDisplayTransform::TransformRect(r, parent, refBox);
+ }
+ }
+
+ innerRect.UnionRect(innerRect, r);
+ }
+ }
+
+ // Keep this code in sync with nsDisplayOutline::GetInnerRect.
+ if (innerRect == aFrame->GetRectRelativeToSelf()) {
+ aFrame->RemoveProperty(nsIFrame::OutlineInnerRectProperty());
+ } else {
+ SetOrUpdateRectValuedProperty(aFrame, nsIFrame::OutlineInnerRectProperty(),
+ innerRect);
+ }
+
+ nsRect outerRect(innerRect);
+ outerRect.Inflate(outline->EffectiveOffsetFor(outerRect));
+
+ if (outline->mOutlineStyle.IsAuto()) {
+ nsPresContext* pc = aFrame->PresContext();
+
+ pc->Theme()->GetWidgetOverflow(pc->DeviceContext(), aFrame,
+ StyleAppearance::FocusOutline, &outerRect);
+ } else {
+ const nscoord width = outline->GetOutlineWidth();
+ outerRect.Inflate(width);
+ }
+
+ nsRect& vo = aOverflowAreas.InkOverflow();
+ vo = vo.UnionEdges(innerRect.Union(outerRect));
+}
+
+bool nsIFrame::FinishAndStoreOverflow(OverflowAreas& aOverflowAreas,
+ nsSize aNewSize, nsSize* aOldSize,
+ const nsStyleDisplay* aStyleDisplay) {
+ MOZ_ASSERT(FrameMaintainsOverflow(),
+ "Don't call - overflow rects not maintained on these SVG frames");
+
+ const nsStyleDisplay* disp = StyleDisplayWithOptionalParam(aStyleDisplay);
+ bool hasTransform = IsTransformed();
+
+ nsRect bounds(nsPoint(0, 0), aNewSize);
+ // Store the passed in overflow area if we are a preserve-3d frame or we have
+ // a transform, and it's not just the frame bounds.
+ if (hasTransform || Combines3DTransformWithAncestors()) {
+ if (!aOverflowAreas.InkOverflow().IsEqualEdges(bounds) ||
+ !aOverflowAreas.ScrollableOverflow().IsEqualEdges(bounds)) {
+ OverflowAreas* initial = GetProperty(nsIFrame::InitialOverflowProperty());
+ if (!initial) {
+ AddProperty(nsIFrame::InitialOverflowProperty(),
+ new OverflowAreas(aOverflowAreas));
+ } else if (initial != &aOverflowAreas) {
+ *initial = aOverflowAreas;
+ }
+ } else {
+ RemoveProperty(nsIFrame::InitialOverflowProperty());
+ }
+#ifdef DEBUG
+ SetProperty(nsIFrame::DebugInitialOverflowPropertyApplied(), true);
+#endif
+ } else {
+#ifdef DEBUG
+ RemoveProperty(nsIFrame::DebugInitialOverflowPropertyApplied());
+#endif
+ }
+
+ nsSize oldSize = mRect.Size();
+ bool sizeChanged = ((aOldSize ? *aOldSize : oldSize) != aNewSize);
+
+ // Our frame size may not have been computed and set yet, but code under
+ // functions such as ComputeEffectsRect (which we're about to call) use the
+ // values that are stored in our frame rect to compute their results. We
+ // need the results from those functions to be based on the frame size that
+ // we *will* have, so we temporarily set our frame size here before calling
+ // those functions.
+ //
+ // XXX Someone should document here why we revert the frame size before we
+ // return rather than just leaving it set.
+ //
+ // We pass false here to avoid invalidating display items for this temporary
+ // change. We sometimes reflow frames multiple times, with the final size
+ // being the same as the initial. The single call to SetSize after reflow is
+ // done will take care of invalidating display items if the size has actually
+ // changed.
+ SetSize(aNewSize, false);
+
+ const auto overflowClipAxes = ShouldApplyOverflowClipping(disp);
+
+ if (ChildrenHavePerspective(disp) && sizeChanged) {
+ RecomputePerspectiveChildrenOverflow(this);
+
+ if (overflowClipAxes != PhysicalAxes::Both) {
+ aOverflowAreas.SetAllTo(bounds);
+ DebugOnly<bool> ok = ComputeCustomOverflow(aOverflowAreas);
+
+ // ComputeCustomOverflow() should not return false, when
+ // FrameMaintainsOverflow() returns true.
+ MOZ_ASSERT(ok, "FrameMaintainsOverflow() != ComputeCustomOverflow()");
+
+ UnionChildOverflow(aOverflowAreas);
+ }
+ }
+
+ // This is now called FinishAndStoreOverflow() instead of
+ // StoreOverflow() because frame-generic ways of adding overflow
+ // can happen here, e.g. CSS2 outline and native theme.
+ // If the overflow area width or height is nscoord_MAX, then a saturating
+ // union may have encountered an overflow, so the overflow may not contain the
+ // frame border-box. Don't warn in that case.
+ // Don't warn for SVG either, since SVG doesn't need the overflow area
+ // to contain the frame bounds.
+#ifdef DEBUG
+ for (const auto otype : AllOverflowTypes()) {
+ const nsRect& r = aOverflowAreas.Overflow(otype);
+ NS_ASSERTION(aNewSize.width == 0 || aNewSize.height == 0 ||
+ r.width == nscoord_MAX || r.height == nscoord_MAX ||
+ HasAnyStateBits(NS_FRAME_SVG_LAYOUT) ||
+ r.Contains(nsRect(nsPoint(), aNewSize)),
+ "Computed overflow area must contain frame bounds");
+ }
+#endif
+
+ // Overflow area must always include the frame's top-left and bottom-right,
+ // even if the frame rect is empty (so we can scroll to those positions).
+ const bool shouldIncludeBounds = [&] {
+ if (aNewSize.width == 0 && IsInlineFrame()) {
+ // Pending a real fix for bug 426879, don't do this for inline frames with
+ // zero width.
+ return false;
+ }
+ if (HasAnyStateBits(NS_FRAME_SVG_LAYOUT)) {
+ // Do not do this for SVG either, since it will usually massively increase
+ // the area unnecessarily (except for SVG that applies clipping, since
+ // that's the pre-existing behavior, and breaks pre-rendering otherwise).
+ // FIXME(bug 1770704): This check most likely wants to be removed or check
+ // for specific frame types at least.
+ return overflowClipAxes != PhysicalAxes::None;
+ }
+ return true;
+ }();
+
+ if (shouldIncludeBounds) {
+ for (const auto otype : AllOverflowTypes()) {
+ nsRect& o = aOverflowAreas.Overflow(otype);
+ o = o.UnionEdges(bounds);
+ }
+ }
+
+ // If we clip our children, clear accumulated overflow area in the affected
+ // dimension(s). The children are actually clipped to the padding-box, but
+ // since the overflow area should include the entire border-box, just set it
+ // to the border-box size here.
+ if (overflowClipAxes != PhysicalAxes::None) {
+ aOverflowAreas.ApplyClipping(bounds, overflowClipAxes,
+ OverflowClipMargin(overflowClipAxes));
+ }
+
+ ComputeAndIncludeOutlineArea(this, aOverflowAreas, aNewSize);
+
+ // Nothing in here should affect scrollable overflow.
+ aOverflowAreas.InkOverflow() =
+ ComputeEffectsRect(this, aOverflowAreas.InkOverflow(), aNewSize);
+
+ // Absolute position clipping
+ const nsStyleEffects* effects = StyleEffects();
+ Maybe<nsRect> clipPropClipRect = GetClipPropClipRect(disp, effects, aNewSize);
+ if (clipPropClipRect) {
+ for (const auto otype : AllOverflowTypes()) {
+ nsRect& o = aOverflowAreas.Overflow(otype);
+ o.IntersectRect(o, *clipPropClipRect);
+ }
+ }
+
+ /* If we're transformed, transform the overflow rect by the current
+ * transformation. */
+ if (hasTransform) {
+ SetProperty(nsIFrame::PreTransformOverflowAreasProperty(),
+ new OverflowAreas(aOverflowAreas));
+
+ if (Combines3DTransformWithAncestors()) {
+ /* If we're a preserve-3d leaf frame, then our pre-transform overflow
+ * should be correct. Our post-transform overflow is empty though, because
+ * we only contribute to the overflow area of the preserve-3d root frame.
+ * If we're an intermediate frame then the pre-transform overflow should
+ * contain all our non-preserve-3d children, which is what we want. Again
+ * we have no post-transform overflow.
+ */
+ aOverflowAreas.SetAllTo(nsRect());
+ } else {
+ TransformReferenceBox refBox(this);
+ for (const auto otype : AllOverflowTypes()) {
+ nsRect& o = aOverflowAreas.Overflow(otype);
+ o = nsDisplayTransform::TransformRect(o, this, refBox);
+ }
+
+ /* If we're the root of the 3d context, then we want to include the
+ * overflow areas of all the participants. This won't have happened yet as
+ * the code above set their overflow area to empty. Manually collect these
+ * overflow areas now.
+ */
+ if (Extend3DContext(disp, effects)) {
+ ComputePreserve3DChildrenOverflow(aOverflowAreas);
+ }
+ }
+ } else {
+ RemoveProperty(nsIFrame::PreTransformOverflowAreasProperty());
+ }
+
+ /* Revert the size change in case some caller is depending on this. */
+ SetSize(oldSize, false);
+
+ bool anyOverflowChanged;
+ if (aOverflowAreas != OverflowAreas(bounds, bounds)) {
+ anyOverflowChanged = SetOverflowAreas(aOverflowAreas);
+ } else {
+ anyOverflowChanged = ClearOverflowRects();
+ }
+
+ if (anyOverflowChanged) {
+ SVGObserverUtils::InvalidateDirectRenderingObservers(this);
+ if (nsBlockFrame* block = do_QueryFrame(this)) {
+ // NOTE(emilio): we need to use BeforeReflow::Yes, because we want to
+ // invalidate in cases where we _used_ to have an overflow marker and no
+ // longer do.
+ if (TextOverflow::CanHaveOverflowMarkers(
+ block, TextOverflow::BeforeReflow::Yes)) {
+ DiscardDisplayItems(this, [](nsDisplayItem* aItem) {
+ return aItem->GetType() == DisplayItemType::TYPE_TEXT_OVERFLOW;
+ });
+ SchedulePaint(PAINT_DEFAULT);
+ }
+ }
+ }
+ return anyOverflowChanged;
+}
+
+void nsIFrame::RecomputePerspectiveChildrenOverflow(
+ const nsIFrame* aStartFrame) {
+ for (const auto& childList : ChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ if (!child->FrameMaintainsOverflow()) {
+ continue; // frame does not maintain overflow rects
+ }
+ if (child->HasPerspective()) {
+ OverflowAreas* overflow =
+ child->GetProperty(nsIFrame::InitialOverflowProperty());
+ nsRect bounds(nsPoint(0, 0), child->GetSize());
+ if (overflow) {
+ OverflowAreas overflowCopy = *overflow;
+ child->FinishAndStoreOverflow(overflowCopy, bounds.Size());
+ } else {
+ OverflowAreas boundsOverflow;
+ boundsOverflow.SetAllTo(bounds);
+ child->FinishAndStoreOverflow(boundsOverflow, bounds.Size());
+ }
+ } else if (child->GetContent() == aStartFrame->GetContent() ||
+ child->GetClosestFlattenedTreeAncestorPrimaryFrame() ==
+ aStartFrame) {
+ // If a frame is using perspective, then the size used to compute
+ // perspective-origin is the size of the frame belonging to its parent
+ // style. We must find any descendant frames using our size
+ // (by recursing into frames that have the same containing block)
+ // to update their overflow rects too.
+ child->RecomputePerspectiveChildrenOverflow(aStartFrame);
+ }
+ }
+ }
+}
+
+void nsIFrame::ComputePreserve3DChildrenOverflow(
+ OverflowAreas& aOverflowAreas) {
+ // Find all descendants that participate in the 3d context, and include their
+ // overflow. These descendants have an empty overflow, so won't have been
+ // included in the normal overflow calculation. Any children that don't
+ // participate have normal overflow, so will have been included already.
+
+ nsRect childVisual;
+ nsRect childScrollable;
+ for (const auto& childList : ChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ // If this child participates in the 3d context, then take the
+ // pre-transform region (which contains all descendants that aren't
+ // participating in the 3d context) and transform it into the 3d context
+ // root coordinate space.
+ if (child->Combines3DTransformWithAncestors()) {
+ OverflowAreas childOverflow = child->GetOverflowAreasRelativeToSelf();
+ TransformReferenceBox refBox(child);
+ for (const auto otype : AllOverflowTypes()) {
+ nsRect& o = childOverflow.Overflow(otype);
+ o = nsDisplayTransform::TransformRect(o, child, refBox);
+ }
+
+ aOverflowAreas.UnionWith(childOverflow);
+
+ // If this child also extends the 3d context, then recurse into it
+ // looking for more participants.
+ if (child->Extend3DContext()) {
+ child->ComputePreserve3DChildrenOverflow(aOverflowAreas);
+ }
+ }
+ }
+ }
+}
+
+bool nsIFrame::ZIndexApplies() const {
+ return StyleDisplay()->IsPositionedStyle() || IsFlexOrGridItem() ||
+ IsMenuPopupFrame();
+}
+
+Maybe<int32_t> nsIFrame::ZIndex() const {
+ if (!ZIndexApplies()) {
+ return Nothing();
+ }
+ const auto& zIndex = StylePosition()->mZIndex;
+ if (zIndex.IsAuto()) {
+ return Nothing();
+ }
+ return Some(zIndex.AsInteger());
+}
+
+bool nsIFrame::IsScrollAnchor(ScrollAnchorContainer** aOutContainer) {
+ if (!mInScrollAnchorChain) {
+ return false;
+ }
+
+ nsIFrame* f = this;
+
+ // FIXME(emilio, bug 1629280): We should find a non-null anchor if we have the
+ // flag set, but bug 1629280 makes it so that we cannot really assert it /
+ // make this just a `while (true)`, and uncomment the below assertion.
+ while (auto* container = ScrollAnchorContainer::FindFor(f)) {
+ // MOZ_ASSERT(f->IsInScrollAnchorChain());
+ if (nsIFrame* anchor = container->AnchorNode()) {
+ if (anchor != this) {
+ return false;
+ }
+ if (aOutContainer) {
+ *aOutContainer = container;
+ }
+ return true;
+ }
+
+ f = container->Frame();
+ }
+
+ return false;
+}
+
+bool nsIFrame::IsInScrollAnchorChain() const { return mInScrollAnchorChain; }
+
+void nsIFrame::SetInScrollAnchorChain(bool aInChain) {
+ mInScrollAnchorChain = aInChain;
+}
+
+uint32_t nsIFrame::GetDepthInFrameTree() const {
+ uint32_t result = 0;
+ for (nsContainerFrame* ancestor = GetParent(); ancestor;
+ ancestor = ancestor->GetParent()) {
+ result++;
+ }
+ return result;
+}
+
+/**
+ * This function takes a frame that is part of a block-in-inline split,
+ * and _if_ that frame is an anonymous block created by an ib split it
+ * returns the block's preceding inline. This is needed because the
+ * split inline's style is the parent of the anonymous block's style.
+ *
+ * If aFrame is not an anonymous block, null is returned.
+ */
+static nsIFrame* GetIBSplitSiblingForAnonymousBlock(const nsIFrame* aFrame) {
+ MOZ_ASSERT(aFrame, "Must have a non-null frame!");
+ NS_ASSERTION(aFrame->HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT),
+ "GetIBSplitSibling should only be called on ib-split frames");
+
+ if (aFrame->Style()->GetPseudoType() !=
+ PseudoStyleType::mozBlockInsideInlineWrapper) {
+ // it's not an anonymous block
+ return nullptr;
+ }
+
+ // Find the first continuation of the frame. (Ugh. This ends up
+ // being O(N^2) when it is called O(N) times.)
+ aFrame = aFrame->FirstContinuation();
+
+ /*
+ * Now look up the nsGkAtoms::IBSplitPrevSibling
+ * property.
+ */
+ nsIFrame* ibSplitSibling =
+ aFrame->GetProperty(nsIFrame::IBSplitPrevSibling());
+ NS_ASSERTION(ibSplitSibling, "Broken frame tree?");
+ return ibSplitSibling;
+}
+
+/**
+ * Get the parent, corrected for the mangled frame tree resulting from
+ * having a block within an inline. The result only differs from the
+ * result of |GetParent| when |GetParent| returns an anonymous block
+ * that was created for an element that was 'display: inline' because
+ * that element contained a block.
+ *
+ * Also skip anonymous scrolled-content parents; inherit directly from the
+ * outer scroll frame.
+ *
+ * Also skip NAC parents if the child frame is NAC.
+ */
+static nsIFrame* GetCorrectedParent(const nsIFrame* aFrame) {
+ nsIFrame* parent = aFrame->GetParent();
+ if (!parent) {
+ return nullptr;
+ }
+
+ // For a table caption we want the _inner_ table frame (unless it's anonymous)
+ // as the style parent.
+ if (aFrame->IsTableCaption()) {
+ nsIFrame* innerTable = parent->PrincipalChildList().FirstChild();
+ if (!innerTable->Style()->IsAnonBox()) {
+ return innerTable;
+ }
+ }
+
+ // Table wrappers are always anon boxes; if we're in here for an outer
+ // table, that actually means its the _inner_ table that wants to
+ // know its parent. So get the pseudo of the inner in that case.
+ auto pseudo = aFrame->Style()->GetPseudoType();
+ if (pseudo == PseudoStyleType::tableWrapper) {
+ pseudo =
+ aFrame->PrincipalChildList().FirstChild()->Style()->GetPseudoType();
+ }
+
+ // Prevent a NAC pseudo-element from inheriting from its NAC parent, and
+ // inherit from the NAC generator element instead.
+ if (pseudo != PseudoStyleType::NotPseudo) {
+ MOZ_ASSERT(aFrame->GetContent());
+ Element* element = Element::FromNode(aFrame->GetContent());
+ // Make sure to avoid doing the fixup for non-element-backed pseudos like
+ // ::first-line and such.
+ if (element && !element->IsRootOfNativeAnonymousSubtree() &&
+ element->GetPseudoElementType() == aFrame->Style()->GetPseudoType()) {
+ while (parent->GetContent() &&
+ !parent->GetContent()->IsRootOfNativeAnonymousSubtree()) {
+ parent = parent->GetInFlowParent();
+ }
+ parent = parent->GetInFlowParent();
+ }
+ }
+
+ return nsIFrame::CorrectStyleParentFrame(parent, pseudo);
+}
+
+/* static */
+nsIFrame* nsIFrame::CorrectStyleParentFrame(nsIFrame* aProspectiveParent,
+ PseudoStyleType aChildPseudo) {
+ MOZ_ASSERT(aProspectiveParent, "Must have a prospective parent");
+
+ if (aChildPseudo != PseudoStyleType::NotPseudo) {
+ // Non-inheriting anon boxes have no style parent frame at all.
+ if (PseudoStyle::IsNonInheritingAnonBox(aChildPseudo)) {
+ return nullptr;
+ }
+
+ // Other anon boxes are parented to their actual parent already, except
+ // for non-elements. Those should not be treated as an anon box.
+ if (PseudoStyle::IsAnonBox(aChildPseudo) &&
+ !nsCSSAnonBoxes::IsNonElement(aChildPseudo)) {
+ NS_ASSERTION(aChildPseudo != PseudoStyleType::mozBlockInsideInlineWrapper,
+ "Should have dealt with kids that have "
+ "NS_FRAME_PART_OF_IBSPLIT elsewhere");
+ return aProspectiveParent;
+ }
+ }
+
+ // Otherwise, walk up out of all anon boxes. For placeholder frames, walk out
+ // of all pseudo-elements as well. Otherwise ReparentComputedStyle could
+ // cause style data to be out of sync with the frame tree.
+ nsIFrame* parent = aProspectiveParent;
+ do {
+ if (parent->HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT)) {
+ nsIFrame* sibling = GetIBSplitSiblingForAnonymousBlock(parent);
+
+ if (sibling) {
+ // |parent| was a block in an {ib} split; use the inline as
+ // |the style parent.
+ parent = sibling;
+ }
+ }
+
+ if (!parent->Style()->IsPseudoOrAnonBox()) {
+ return parent;
+ }
+
+ if (!parent->Style()->IsAnonBox() && aChildPseudo != PseudoStyleType::MAX) {
+ // nsPlaceholderFrame passes in PseudoStyleType::MAX for
+ // aChildPseudo (even though that's not a valid pseudo-type) just to
+ // trigger this behavior of walking up to the nearest non-pseudo
+ // ancestor.
+ return parent;
+ }
+
+ parent = parent->GetInFlowParent();
+ } while (parent);
+
+ if (aProspectiveParent->Style()->GetPseudoType() ==
+ PseudoStyleType::viewportScroll) {
+ // aProspectiveParent is the scrollframe for a viewport
+ // and the kids are the anonymous scrollbars
+ return aProspectiveParent;
+ }
+
+ // We can get here if the root element is absolutely positioned.
+ // We can't test for this very accurately, but it can only happen
+ // when the prospective parent is a canvas frame.
+ NS_ASSERTION(aProspectiveParent->IsCanvasFrame(),
+ "Should have found a parent before this");
+ return nullptr;
+}
+
+ComputedStyle* nsIFrame::DoGetParentComputedStyle(
+ nsIFrame** aProviderFrame) const {
+ *aProviderFrame = nullptr;
+
+ // Handle display:contents and the root frame, when there's no parent frame
+ // to inherit from.
+ if (MOZ_LIKELY(mContent)) {
+ Element* parentElement = mContent->GetFlattenedTreeParentElement();
+ if (MOZ_LIKELY(parentElement)) {
+ auto pseudo = Style()->GetPseudoType();
+ if (pseudo == PseudoStyleType::NotPseudo || !mContent->IsElement() ||
+ (!PseudoStyle::IsAnonBox(pseudo) &&
+ // Ensure that we don't return the display:contents style
+ // of the parent content for pseudos that have the same content
+ // as their primary frame (like -moz-list-bullets do):
+ IsPrimaryFrame()) ||
+ /* if next is true then it's really a request for the table frame's
+ parent context, see nsTable[Outer]Frame::GetParentComputedStyle. */
+ pseudo == PseudoStyleType::tableWrapper) {
+ // In some edge cases involving display: contents, we may end up here
+ // for something that's pending to be reframed. In this case we return
+ // the wrong style from here (because we've already lost track of it!),
+ // but it's not a big deal as we're going to be reframed anyway.
+ if (MOZ_LIKELY(parentElement->HasServoData()) &&
+ Servo_Element_IsDisplayContents(parentElement)) {
+ RefPtr<ComputedStyle> style =
+ ServoStyleSet::ResolveServoStyle(*parentElement);
+ // NOTE(emilio): we return a weak reference because the element also
+ // holds the style context alive. This is a bit silly (we could've
+ // returned a weak ref directly), but it's probably not worth
+ // optimizing, given this function has just one caller which is rare,
+ // and this path is rare itself.
+ return style;
+ }
+ }
+ } else {
+ if (Style()->GetPseudoType() == PseudoStyleType::NotPseudo) {
+ // We're a frame for the root. We have no style parent.
+ return nullptr;
+ }
+ }
+ }
+
+ if (!HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
+ /*
+ * If this frame is an anonymous block created when an inline with a block
+ * inside it got split, then the parent style is on its preceding inline. We
+ * can get to it using GetIBSplitSiblingForAnonymousBlock.
+ */
+ if (HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT)) {
+ nsIFrame* ibSplitSibling = GetIBSplitSiblingForAnonymousBlock(this);
+ if (ibSplitSibling) {
+ return (*aProviderFrame = ibSplitSibling)->Style();
+ }
+ }
+
+ // If this frame is one of the blocks that split an inline, we must
+ // return the "special" inline parent, i.e., the parent that this
+ // frame would have if we didn't mangle the frame structure.
+ *aProviderFrame = GetCorrectedParent(this);
+ return *aProviderFrame ? (*aProviderFrame)->Style() : nullptr;
+ }
+
+ // We're an out-of-flow frame. For out-of-flow frames, we must
+ // resolve underneath the placeholder's parent. The placeholder is
+ // reached from the first-in-flow.
+ nsPlaceholderFrame* placeholder = FirstInFlow()->GetPlaceholderFrame();
+ if (!placeholder) {
+ MOZ_ASSERT_UNREACHABLE("no placeholder frame for out-of-flow frame");
+ *aProviderFrame = GetCorrectedParent(this);
+ return *aProviderFrame ? (*aProviderFrame)->Style() : nullptr;
+ }
+ return placeholder->GetParentComputedStyleForOutOfFlow(aProviderFrame);
+}
+
+void nsIFrame::GetLastLeaf(nsIFrame** aFrame) {
+ if (!aFrame || !*aFrame) return;
+ nsIFrame* child = *aFrame;
+ // if we are a block frame then go for the last line of 'this'
+ while (1) {
+ child = child->PrincipalChildList().FirstChild();
+ if (!child) return; // nothing to do
+ nsIFrame* siblingFrame;
+ nsIContent* content;
+ // ignore anonymous elements, e.g. mozTableAdd* mozTableRemove*
+ // see bug 278197 comment #12 #13 for details
+ while ((siblingFrame = child->GetNextSibling()) &&
+ (content = siblingFrame->GetContent()) &&
+ !content->IsRootOfNativeAnonymousSubtree())
+ child = siblingFrame;
+ *aFrame = child;
+ }
+}
+
+void nsIFrame::GetFirstLeaf(nsIFrame** aFrame) {
+ if (!aFrame || !*aFrame) return;
+ nsIFrame* child = *aFrame;
+ while (1) {
+ child = child->PrincipalChildList().FirstChild();
+ if (!child) return; // nothing to do
+ *aFrame = child;
+ }
+}
+
+bool nsIFrame::IsFocusableDueToScrollFrame() {
+ if (!IsScrollFrame()) {
+ if (nsFieldSetFrame* fieldset = do_QueryFrame(this)) {
+ // TODO: Do we have similar special-cases like this where we can have
+ // anonymous scrollable boxes hanging off a primary frame?
+ if (nsIFrame* inner = fieldset->GetInner()) {
+ return inner->IsFocusableDueToScrollFrame();
+ }
+ }
+ return false;
+ }
+ if (!mContent->IsHTMLElement()) {
+ return false;
+ }
+ if (mContent->IsRootOfNativeAnonymousSubtree()) {
+ return false;
+ }
+ if (!mContent->GetParent()) {
+ return false;
+ }
+ if (mContent->AsElement()->HasAttr(nsGkAtoms::tabindex)) {
+ return false;
+ }
+ // Elements with scrollable view are focusable with script & tabbable
+ // Otherwise you couldn't scroll them with keyboard, which is an accessibility
+ // issue (e.g. Section 508 rules) However, we don't make them to be focusable
+ // with the mouse, because the extra focus outlines are considered
+ // unnecessarily ugly. When clicked on, the selection position within the
+ // element will be enough to make them keyboard scrollable.
+ nsIScrollableFrame* scrollFrame = do_QueryFrame(this);
+ if (!scrollFrame) {
+ return false;
+ }
+ if (scrollFrame->IsForTextControlWithNoScrollbars()) {
+ return false;
+ }
+ if (scrollFrame->GetScrollStyles().IsHiddenInBothDirections()) {
+ return false;
+ }
+ if (scrollFrame->GetScrollRange().IsEqualEdges(nsRect(0, 0, 0, 0))) {
+ return false;
+ }
+ return true;
+}
+
+Focusable nsIFrame::IsFocusable(bool aWithMouse, bool aCheckVisibility) {
+ // cannot focus content in print preview mode. Only the root can be focused,
+ // but that's handled elsewhere.
+ if (PresContext()->Type() == nsPresContext::eContext_PrintPreview) {
+ return {};
+ }
+
+ if (!mContent || !mContent->IsElement()) {
+ return {};
+ }
+
+ if (aCheckVisibility && !IsVisibleConsideringAncestors()) {
+ return {};
+ }
+
+ const StyleUserFocus uf = StyleUI()->UserFocus();
+ if (uf == StyleUserFocus::None) {
+ return {};
+ }
+ MOZ_ASSERT(!StyleUI()->IsInert(), "inert implies -moz-user-focus: none");
+
+ const PseudoStyleType pseudo = Style()->GetPseudoType();
+ if (pseudo == PseudoStyleType::anonymousItem) {
+ return {};
+ }
+
+ Focusable focusable;
+ if (auto* xul = nsXULElement::FromNode(mContent)) {
+ // As a legacy special-case, -moz-user-focus controls focusability and
+ // tabability of XUL elements in some circumstances (which default to
+ // -moz-user-focus: ignore).
+ auto focusability = xul->GetXULFocusability(aWithMouse);
+ focusable.mFocusable =
+ focusability.mForcedFocusable.valueOr(uf == StyleUserFocus::Normal);
+ if (focusable) {
+ focusable.mTabIndex = focusability.mForcedTabIndexIfFocusable.valueOr(0);
+ }
+ } else {
+ focusable = mContent->IsFocusableWithoutStyle(aWithMouse);
+ }
+
+ if (focusable) {
+ return focusable;
+ }
+
+ // If we're focusing with the mouse we never focus scroll areas.
+ if (!aWithMouse && IsFocusableDueToScrollFrame()) {
+ return {true, 0};
+ }
+
+ // FIXME(emilio): some callers rely on somewhat broken return values
+ // (focusable = false, but non-negative tab-index) from
+ // IsFocusableWithoutStyle (for image maps in particular).
+ return focusable;
+}
+
+/**
+ * @return true if this text frame ends with a newline character which is
+ * treated as preformatted. It should return false if this is not a text frame.
+ */
+bool nsIFrame::HasSignificantTerminalNewline() const { return false; }
+
+static StyleVerticalAlignKeyword ConvertSVGDominantBaselineToVerticalAlign(
+ StyleDominantBaseline aDominantBaseline) {
+ // Most of these are approximate mappings.
+ switch (aDominantBaseline) {
+ case StyleDominantBaseline::Hanging:
+ case StyleDominantBaseline::TextBeforeEdge:
+ return StyleVerticalAlignKeyword::TextTop;
+ case StyleDominantBaseline::TextAfterEdge:
+ case StyleDominantBaseline::Ideographic:
+ return StyleVerticalAlignKeyword::TextBottom;
+ case StyleDominantBaseline::Central:
+ case StyleDominantBaseline::Middle:
+ case StyleDominantBaseline::Mathematical:
+ return StyleVerticalAlignKeyword::Middle;
+ case StyleDominantBaseline::Auto:
+ case StyleDominantBaseline::Alphabetic:
+ return StyleVerticalAlignKeyword::Baseline;
+ default:
+ MOZ_ASSERT_UNREACHABLE("unexpected aDominantBaseline value");
+ return StyleVerticalAlignKeyword::Baseline;
+ }
+}
+
+Maybe<StyleVerticalAlignKeyword> nsIFrame::VerticalAlignEnum() const {
+ if (IsInSVGTextSubtree()) {
+ StyleDominantBaseline dominantBaseline = StyleSVG()->mDominantBaseline;
+ return Some(ConvertSVGDominantBaselineToVerticalAlign(dominantBaseline));
+ }
+
+ const auto& verticalAlign = StyleDisplay()->mVerticalAlign;
+ if (verticalAlign.IsKeyword()) {
+ return Some(verticalAlign.AsKeyword());
+ }
+
+ return Nothing();
+}
+
+void nsIFrame::UpdateStyleOfChildAnonBox(nsIFrame* aChildFrame,
+ ServoRestyleState& aRestyleState) {
+#ifdef DEBUG
+ nsIFrame* parent = aChildFrame->GetInFlowParent();
+ if (aChildFrame->IsTableFrame()) {
+ parent = parent->GetParent();
+ }
+ if (parent->IsLineFrame()) {
+ parent = parent->GetParent();
+ }
+ MOZ_ASSERT(nsLayoutUtils::FirstContinuationOrIBSplitSibling(parent) == this,
+ "This should only be used for children!");
+#endif // DEBUG
+ MOZ_ASSERT(!GetContent() || !aChildFrame->GetContent() ||
+ aChildFrame->GetContent() == GetContent(),
+ "What content node is it a frame for?");
+ MOZ_ASSERT(!aChildFrame->GetPrevContinuation(),
+ "Only first continuations should end up here");
+
+ // We could force the caller to pass in the pseudo, since some callers know it
+ // statically... But this API is a bit nicer.
+ auto pseudo = aChildFrame->Style()->GetPseudoType();
+ MOZ_ASSERT(PseudoStyle::IsAnonBox(pseudo), "Child is not an anon box?");
+ MOZ_ASSERT(!PseudoStyle::IsNonInheritingAnonBox(pseudo),
+ "Why did the caller bother calling us?");
+
+ // Anon boxes inherit from their parent; that's us.
+ RefPtr<ComputedStyle> newContext =
+ aRestyleState.StyleSet().ResolveInheritingAnonymousBoxStyle(pseudo,
+ Style());
+
+ nsChangeHint childHint =
+ UpdateStyleOfOwnedChildFrame(aChildFrame, newContext, aRestyleState);
+
+ // Now that we've updated the style on aChildFrame, check whether it itself
+ // has anon boxes to deal with.
+ ServoRestyleState childrenState(*aChildFrame, aRestyleState, childHint,
+ ServoRestyleState::Type::InFlow);
+ aChildFrame->UpdateStyleOfOwnedAnonBoxes(childrenState);
+
+ // Assuming anon boxes don't have ::backdrop associated with them... if that
+ // ever changes, we'd need to handle that here, like we do in
+ // RestyleManager::ProcessPostTraversal
+
+ // We do need to handle block pseudo-elements here, though. Especially list
+ // bullets.
+ if (nsBlockFrame* block = do_QueryFrame(aChildFrame)) {
+ block->UpdatePseudoElementStyles(childrenState);
+ }
+}
+
+/* static */
+nsChangeHint nsIFrame::UpdateStyleOfOwnedChildFrame(
+ nsIFrame* aChildFrame, ComputedStyle* aNewComputedStyle,
+ ServoRestyleState& aRestyleState,
+ const Maybe<ComputedStyle*>& aContinuationComputedStyle) {
+ MOZ_ASSERT(!aChildFrame->GetAdditionalComputedStyle(0),
+ "We don't handle additional styles here");
+
+ // Figure out whether we have an actual change. It's important that we do
+ // this, for several reasons:
+ //
+ // 1) Even if all the child's changes are due to properties it inherits from
+ // us, it's possible that no one ever asked us for those style structs and
+ // hence changes to them aren't reflected in the changes handled at all.
+ //
+ // 2) Content can change stylesheets that change the styles of pseudos, and
+ // extensions can add/remove stylesheets that change the styles of
+ // anonymous boxes directly.
+ uint32_t equalStructs; // Not used, actually.
+ nsChangeHint childHint = aChildFrame->Style()->CalcStyleDifference(
+ *aNewComputedStyle, &equalStructs);
+
+ // If aChildFrame is out of flow, then aRestyleState's "changes handled by the
+ // parent" doesn't apply to it, because it may have some other parent in the
+ // frame tree.
+ if (!aChildFrame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
+ childHint = NS_RemoveSubsumedHints(
+ childHint, aRestyleState.ChangesHandledFor(aChildFrame));
+ }
+ if (childHint) {
+ if (childHint & nsChangeHint_ReconstructFrame) {
+ // If we generate a reconstruct here, remove any non-reconstruct hints we
+ // may have already generated for this content.
+ aRestyleState.ChangeList().PopChangesForContent(
+ aChildFrame->GetContent());
+ }
+ aRestyleState.ChangeList().AppendChange(
+ aChildFrame, aChildFrame->GetContent(), childHint);
+ }
+
+ aChildFrame->SetComputedStyle(aNewComputedStyle);
+ ComputedStyle* continuationStyle = aContinuationComputedStyle
+ ? *aContinuationComputedStyle
+ : aNewComputedStyle;
+ for (nsIFrame* kid = aChildFrame->GetNextContinuation(); kid;
+ kid = kid->GetNextContinuation()) {
+ MOZ_ASSERT(!kid->GetAdditionalComputedStyle(0));
+ kid->SetComputedStyle(continuationStyle);
+ }
+
+ return childHint;
+}
+
+/* static */
+void nsIFrame::AddInPopupStateBitToDescendants(nsIFrame* aFrame) {
+ if (!aFrame->HasAnyStateBits(NS_FRAME_IN_POPUP) &&
+ aFrame->TrackingVisibility()) {
+ // Assume all frames in popups are visible.
+ aFrame->IncApproximateVisibleCount();
+ }
+
+ aFrame->AddStateBits(NS_FRAME_IN_POPUP);
+
+ for (const auto& childList : aFrame->CrossDocChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ AddInPopupStateBitToDescendants(child);
+ }
+ }
+}
+
+/* static */
+void nsIFrame::RemoveInPopupStateBitFromDescendants(nsIFrame* aFrame) {
+ if (!aFrame->HasAnyStateBits(NS_FRAME_IN_POPUP) ||
+ nsLayoutUtils::IsPopup(aFrame)) {
+ return;
+ }
+
+ aFrame->RemoveStateBits(NS_FRAME_IN_POPUP);
+
+ if (aFrame->TrackingVisibility()) {
+ // We assume all frames in popups are visible, so this decrement balances
+ // out the increment in AddInPopupStateBitToDescendants above.
+ aFrame->DecApproximateVisibleCount();
+ }
+ for (const auto& childList : aFrame->CrossDocChildLists()) {
+ for (nsIFrame* child : childList.mList) {
+ RemoveInPopupStateBitFromDescendants(child);
+ }
+ }
+}
+
+void nsIFrame::SetParent(nsContainerFrame* aParent) {
+ // If our parent is a wrapper anon box, our new parent should be too. We
+ // _can_ change parent if our parent is a wrapper anon box, because some
+ // wrapper anon boxes can have continuations.
+ MOZ_ASSERT_IF(ParentIsWrapperAnonBox(),
+ aParent->Style()->IsInheritingAnonBox());
+
+ // Note that the current mParent may already be destroyed at this point.
+ mParent = aParent;
+ MOZ_DIAGNOSTIC_ASSERT(!mParent || PresShell() == mParent->PresShell());
+
+ if (HasAnyStateBits(NS_FRAME_HAS_VIEW | NS_FRAME_HAS_CHILD_WITH_VIEW)) {
+ for (nsIFrame* f = aParent;
+ f && !f->HasAnyStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW);
+ f = f->GetParent()) {
+ f->AddStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW);
+ }
+ }
+
+ if (HasAnyStateBits(NS_FRAME_CONTAINS_RELATIVE_BSIZE)) {
+ for (nsIFrame* f = aParent; f; f = f->GetParent()) {
+ if (f->HasAnyStateBits(NS_FRAME_CONTAINS_RELATIVE_BSIZE)) {
+ break;
+ }
+ f->AddStateBits(NS_FRAME_CONTAINS_RELATIVE_BSIZE);
+ }
+ }
+
+ if (HasAnyStateBits(NS_FRAME_DESCENDANT_INTRINSIC_ISIZE_DEPENDS_ON_BSIZE)) {
+ for (nsIFrame* f = aParent; f; f = f->GetParent()) {
+ if (f->HasAnyStateBits(
+ NS_FRAME_DESCENDANT_INTRINSIC_ISIZE_DEPENDS_ON_BSIZE)) {
+ break;
+ }
+ f->AddStateBits(NS_FRAME_DESCENDANT_INTRINSIC_ISIZE_DEPENDS_ON_BSIZE);
+ }
+ }
+
+ if (HasInvalidFrameInSubtree()) {
+ for (nsIFrame* f = aParent;
+ f && !f->HasAnyStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT |
+ NS_FRAME_IS_NONDISPLAY);
+ f = nsLayoutUtils::GetCrossDocParentFrameInProcess(f)) {
+ f->AddStateBits(NS_FRAME_DESCENDANT_NEEDS_PAINT);
+ }
+ }
+
+ if (aParent->HasAnyStateBits(NS_FRAME_IN_POPUP)) {
+ AddInPopupStateBitToDescendants(this);
+ } else {
+ RemoveInPopupStateBitFromDescendants(this);
+ }
+
+ // If our new parent only has invalid children, then we just invalidate
+ // ourselves too. This is probably faster than clearing the flag all
+ // the way up the frame tree.
+ if (aParent->HasAnyStateBits(NS_FRAME_ALL_DESCENDANTS_NEED_PAINT)) {
+ InvalidateFrame();
+ } else {
+ SchedulePaint();
+ }
+}
+
+bool nsIFrame::IsStackingContext(const nsStyleDisplay* aStyleDisplay,
+ const nsStyleEffects* aStyleEffects) {
+ // Properties that influence the output of this function should be handled in
+ // change_bits_for_longhand as well.
+ if (HasOpacity(aStyleDisplay, aStyleEffects, nullptr)) {
+ return true;
+ }
+ if (IsTransformed()) {
+ return true;
+ }
+ auto willChange = aStyleDisplay->mWillChange.bits;
+ if (aStyleDisplay->IsContainPaint() || aStyleDisplay->IsContainLayout() ||
+ willChange & StyleWillChangeBits::CONTAIN) {
+ if (SupportsContainLayoutAndPaint()) {
+ return true;
+ }
+ }
+ // strictly speaking, 'perspective' doesn't require visual atomicity,
+ // but the spec says it acts like the rest of these
+ if (aStyleDisplay->HasPerspectiveStyle() ||
+ willChange & StyleWillChangeBits::PERSPECTIVE) {
+ if (SupportsCSSTransforms()) {
+ return true;
+ }
+ }
+ if (!StylePosition()->mZIndex.IsAuto() ||
+ willChange & StyleWillChangeBits::Z_INDEX) {
+ if (ZIndexApplies()) {
+ return true;
+ }
+ }
+ return aStyleEffects->mMixBlendMode != StyleBlend::Normal ||
+ SVGIntegrationUtils::UsingEffectsForFrame(this) ||
+ aStyleDisplay->IsPositionForcingStackingContext() ||
+ aStyleDisplay->mIsolation != StyleIsolation::Auto ||
+ willChange & StyleWillChangeBits::STACKING_CONTEXT_UNCONDITIONAL;
+}
+
+bool nsIFrame::IsStackingContext() {
+ return IsStackingContext(StyleDisplay(), StyleEffects());
+}
+
+static bool IsFrameScrolledOutOfView(const nsIFrame* aTarget,
+ const nsRect& aTargetRect,
+ const nsIFrame* aParent) {
+ // The ancestor frame we are checking if it clips out aTargetRect relative to
+ // aTarget.
+ nsIFrame* clipParent = nullptr;
+
+ // find the first scrollable frame or root frame if we are in a fixed pos
+ // subtree
+ for (nsIFrame* f = const_cast<nsIFrame*>(aParent); f;
+ f = nsLayoutUtils::GetCrossDocParentFrameInProcess(f)) {
+ nsIScrollableFrame* scrollableFrame = do_QueryFrame(f);
+ if (scrollableFrame) {
+ clipParent = f;
+ break;
+ }
+ if (f->StyleDisplay()->mPosition == StylePositionProperty::Fixed &&
+ nsLayoutUtils::IsReallyFixedPos(f)) {
+ clipParent = f->GetParent();
+ break;
+ }
+ }
+
+ if (!clipParent) {
+ // Even if we couldn't find the nearest scrollable frame, it might mean we
+ // are in an out-of-process iframe, try to see if |aTarget| frame is
+ // scrolled out of view in an scrollable frame in a cross-process ancestor
+ // document.
+ return nsLayoutUtils::FrameIsScrolledOutOfViewInCrossProcess(aTarget);
+ }
+
+ nsRect clipRect = clipParent->InkOverflowRectRelativeToSelf();
+ // We consider that the target is scrolled out if the scrollable (or root)
+ // frame is empty.
+ if (clipRect.IsEmpty()) {
+ return true;
+ }
+
+ nsRect transformedRect = nsLayoutUtils::TransformFrameRectToAncestor(
+ aTarget, aTargetRect, clipParent);
+
+ if (transformedRect.IsEmpty()) {
+ // If the transformed rect is empty it represents a line or a point that we
+ // should check is outside the the scrollable rect.
+ if (transformedRect.x > clipRect.XMost() ||
+ transformedRect.y > clipRect.YMost() ||
+ clipRect.x > transformedRect.XMost() ||
+ clipRect.y > transformedRect.YMost()) {
+ return true;
+ }
+ } else if (!transformedRect.Intersects(clipRect)) {
+ return true;
+ }
+
+ nsIFrame* parent = clipParent->GetParent();
+ if (!parent) {
+ return false;
+ }
+
+ return IsFrameScrolledOutOfView(aTarget, aTargetRect, parent);
+}
+
+bool nsIFrame::IsScrolledOutOfView() const {
+ nsRect rect = InkOverflowRectRelativeToSelf();
+ return IsFrameScrolledOutOfView(this, rect, this);
+}
+
+gfx::Matrix nsIFrame::ComputeWidgetTransform() const {
+ const nsStyleUIReset* uiReset = StyleUIReset();
+ if (uiReset->mMozWindowTransform.IsNone()) {
+ return gfx::Matrix();
+ }
+
+ TransformReferenceBox refBox(nullptr, nsRect(nsPoint(), GetSize()));
+
+ nsPresContext* presContext = PresContext();
+ int32_t appUnitsPerDevPixel = presContext->AppUnitsPerDevPixel();
+ gfx::Matrix4x4 matrix = nsStyleTransformMatrix::ReadTransforms(
+ uiReset->mMozWindowTransform, refBox, float(appUnitsPerDevPixel));
+
+ // Apply the -moz-window-transform-origin translation to the matrix.
+ const StyleTransformOrigin& origin = uiReset->mWindowTransformOrigin;
+ Point transformOrigin = nsStyleTransformMatrix::Convert2DPosition(
+ origin.horizontal, origin.vertical, refBox, appUnitsPerDevPixel);
+ matrix.ChangeBasis(Point3D(transformOrigin.x, transformOrigin.y, 0));
+
+ gfx::Matrix result2d;
+ if (!matrix.CanDraw2D(&result2d)) {
+ // FIXME: It would be preferable to reject non-2D transforms at parse time.
+ NS_WARNING(
+ "-moz-window-transform does not describe a 2D transform, "
+ "but only 2d transforms are supported");
+ return gfx::Matrix();
+ }
+
+ return result2d;
+}
+
+void nsIFrame::DoUpdateStyleOfOwnedAnonBoxes(ServoRestyleState& aRestyleState) {
+ // As a special case, we check for {ib}-split block frames here, rather
+ // than have an nsInlineFrame::AppendDirectlyOwnedAnonBoxes implementation
+ // that returns them.
+ //
+ // (If we did handle them in AppendDirectlyOwnedAnonBoxes, we would have to
+ // return *all* of the in-flow {ib}-split block frames, not just the first
+ // one. For restyling, we really just need the first in flow, and the other
+ // user of the AppendOwnedAnonBoxes API, AllChildIterator, doesn't need to
+ // know about them at all, since these block frames never create NAC. So we
+ // avoid any unncessary hashtable lookups for the {ib}-split frames by calling
+ // UpdateStyleOfOwnedAnonBoxesForIBSplit directly here.)
+ if (IsInlineFrame()) {
+ if (HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT)) {
+ static_cast<nsInlineFrame*>(this)->UpdateStyleOfOwnedAnonBoxesForIBSplit(
+ aRestyleState);
+ }
+ return;
+ }
+
+ AutoTArray<OwnedAnonBox, 4> frames;
+ AppendDirectlyOwnedAnonBoxes(frames);
+ for (OwnedAnonBox& box : frames) {
+ if (box.mUpdateStyleFn) {
+ box.mUpdateStyleFn(this, box.mAnonBoxFrame, aRestyleState);
+ } else {
+ UpdateStyleOfChildAnonBox(box.mAnonBoxFrame, aRestyleState);
+ }
+ }
+}
+
+/* virtual */
+void nsIFrame::AppendDirectlyOwnedAnonBoxes(nsTArray<OwnedAnonBox>& aResult) {
+ MOZ_ASSERT(!HasAnyStateBits(NS_FRAME_OWNS_ANON_BOXES));
+ MOZ_ASSERT(false, "Why did this get called?");
+}
+
+void nsIFrame::DoAppendOwnedAnonBoxes(nsTArray<OwnedAnonBox>& aResult) {
+ size_t i = aResult.Length();
+ AppendDirectlyOwnedAnonBoxes(aResult);
+
+ // After appending the directly owned anonymous boxes of this frame to
+ // aResult above, we need to check each of them to see if they own
+ // any anonymous boxes themselves. Note that we keep progressing
+ // through aResult, looking for additional entries in aResult from these
+ // subsequent AppendDirectlyOwnedAnonBoxes calls. (Thus we can't
+ // use a ranged for loop here.)
+
+ while (i < aResult.Length()) {
+ nsIFrame* f = aResult[i].mAnonBoxFrame;
+ if (f->HasAnyStateBits(NS_FRAME_OWNS_ANON_BOXES)) {
+ f->AppendDirectlyOwnedAnonBoxes(aResult);
+ }
+ ++i;
+ }
+}
+
+nsIFrame::CaretPosition::CaretPosition() : mContentOffset(0) {}
+
+nsIFrame::CaretPosition::~CaretPosition() = default;
+
+bool nsIFrame::HasCSSAnimations() {
+ auto* collection = AnimationCollection<CSSAnimation>::Get(this);
+ return collection && !collection->mAnimations.IsEmpty();
+}
+
+bool nsIFrame::HasCSSTransitions() {
+ auto* collection = AnimationCollection<CSSTransition>::Get(this);
+ return collection && !collection->mAnimations.IsEmpty();
+}
+
+void nsIFrame::AddSizeOfExcludingThisForTree(nsWindowSizes& aSizes) const {
+ aSizes.mLayoutFramePropertiesSize +=
+ mProperties.SizeOfExcludingThis(aSizes.mState.mMallocSizeOf);
+
+ // We don't do this for Gecko because this stuff is stored in the nsPresArena
+ // and so measured elsewhere.
+ if (!aSizes.mState.HaveSeenPtr(mComputedStyle)) {
+ mComputedStyle->AddSizeOfIncludingThis(aSizes,
+ &aSizes.mLayoutComputedValuesNonDom);
+ }
+
+ // And our additional styles.
+ int32_t index = 0;
+ while (auto* extra = GetAdditionalComputedStyle(index++)) {
+ if (!aSizes.mState.HaveSeenPtr(extra)) {
+ extra->AddSizeOfIncludingThis(aSizes,
+ &aSizes.mLayoutComputedValuesNonDom);
+ }
+ }
+
+ for (const auto& childList : ChildLists()) {
+ for (const nsIFrame* f : childList.mList) {
+ f->AddSizeOfExcludingThisForTree(aSizes);
+ }
+ }
+}
+
+nsRect nsIFrame::GetCompositorHitTestArea(nsDisplayListBuilder* aBuilder) {
+ nsRect area;
+
+ nsIScrollableFrame* scrollFrame = nsLayoutUtils::GetScrollableFrameFor(this);
+ if (scrollFrame) {
+ // If the frame is content of a scrollframe, then we need to pick up the
+ // area corresponding to the overflow rect as well. Otherwise the parts of
+ // the overflow that are not occupied by descendants get skipped and the
+ // APZ code sends touch events to the content underneath instead.
+ // See https://bugzilla.mozilla.org/show_bug.cgi?id=1127773#c15.
+ area = ScrollableOverflowRect();
+ } else {
+ area = GetRectRelativeToSelf();
+ }
+
+ if (!area.IsEmpty()) {
+ return area + aBuilder->ToReferenceFrame(this);
+ }
+
+ return area;
+}
+
+CompositorHitTestInfo nsIFrame::GetCompositorHitTestInfo(
+ nsDisplayListBuilder* aBuilder) {
+ CompositorHitTestInfo result = CompositorHitTestInvisibleToHit;
+
+ if (aBuilder->IsInsidePointerEventsNoneDoc()) {
+ // Somewhere up the parent document chain is a subdocument with pointer-
+ // events:none set on it.
+ return result;
+ }
+ if (!GetParent()) {
+ MOZ_ASSERT(IsViewportFrame());
+ // Viewport frames are never event targets, other frames, like canvas
+ // frames, are the event targets for any regions viewport frames may cover.
+ return result;
+ }
+ if (Style()->PointerEvents() == StylePointerEvents::None) {
+ return result;
+ }
+ if (!StyleVisibility()->IsVisible()) {
+ return result;
+ }
+
+ // Anything that didn't match the above conditions is visible to hit-testing.
+ result = CompositorHitTestFlags::eVisibleToHitTest;
+ SVGUtils::MaskUsage maskUsage = SVGUtils::DetermineMaskUsage(this, false);
+ if (maskUsage.UsingMaskOrClipPath()) {
+ // If WebRender is enabled, simple clip-paths can be converted into WR
+ // clips that WR knows how to hit-test against, so we don't need to mark
+ // it as an irregular area.
+ if (!maskUsage.IsSimpleClipShape()) {
+ result += CompositorHitTestFlags::eIrregularArea;
+ }
+ }
+
+ if (aBuilder->IsBuildingNonLayerizedScrollbar()) {
+ // Scrollbars may be painted into a layer below the actual layer they will
+ // scroll, and therefore wheel events may be dispatched to the outer frame
+ // instead of the intended scrollframe. To address this, we force a d-t-c
+ // region on scrollbar frames that won't be placed in their own layer. See
+ // bug 1213324 for details.
+ result += CompositorHitTestFlags::eInactiveScrollframe;
+ } else if (aBuilder->GetAncestorHasApzAwareEventHandler()) {
+ result += CompositorHitTestFlags::eApzAwareListeners;
+ } else if (IsRangeFrame()) {
+ // Range frames handle touch events directly without having a touch listener
+ // so we need to let APZ know that this area cares about events.
+ result += CompositorHitTestFlags::eApzAwareListeners;
+ }
+
+ if (aBuilder->IsTouchEventPrefEnabledDoc()) {
+ // Inherit the touch-action flags from the parent, if there is one. We do
+ // this because of how the touch-action on a frame combines the touch-action
+ // from ancestor DOM elements. Refer to the documentation in
+ // TouchActionHelper.cpp for details; this code is meant to be equivalent to
+ // that code, but woven into the top-down recursive display list building
+ // process.
+ CompositorHitTestInfo inheritedTouchAction =
+ aBuilder->GetCompositorHitTestInfo() & CompositorHitTestTouchActionMask;
+
+ nsIFrame* touchActionFrame = this;
+ if (nsIScrollableFrame* scrollFrame =
+ nsLayoutUtils::GetScrollableFrameFor(this)) {
+ ScrollStyles ss = scrollFrame->GetScrollStyles();
+ if (ss.mVertical != StyleOverflow::Hidden ||
+ ss.mHorizontal != StyleOverflow::Hidden) {
+ touchActionFrame = do_QueryFrame(scrollFrame);
+ // On scrollframes, stop inheriting the pan-x and pan-y flags; instead,
+ // reset them back to zero to allow panning on the scrollframe unless we
+ // encounter an element that disables it that's inside the scrollframe.
+ // This is equivalent to the |considerPanning| variable in
+ // TouchActionHelper.cpp, but for a top-down traversal.
+ CompositorHitTestInfo panMask(
+ CompositorHitTestFlags::eTouchActionPanXDisabled,
+ CompositorHitTestFlags::eTouchActionPanYDisabled);
+ inheritedTouchAction -= panMask;
+ }
+ }
+
+ result += inheritedTouchAction;
+
+ const StyleTouchAction touchAction = touchActionFrame->UsedTouchAction();
+ // The CSS allows the syntax auto | none | [pan-x || pan-y] | manipulation
+ // so we can eliminate some combinations of things.
+ if (touchAction == StyleTouchAction::AUTO) {
+ // nothing to do
+ } else if (touchAction & StyleTouchAction::MANIPULATION) {
+ result += CompositorHitTestFlags::eTouchActionAnimatingZoomDisabled;
+ } else {
+ // This path handles the cases none | [pan-x || pan-y || pinch-zoom] so
+ // double-tap is disabled in here.
+ if (!(touchAction & StyleTouchAction::PINCH_ZOOM)) {
+ result += CompositorHitTestFlags::eTouchActionPinchZoomDisabled;
+ }
+
+ result += CompositorHitTestFlags::eTouchActionAnimatingZoomDisabled;
+
+ if (!(touchAction & StyleTouchAction::PAN_X)) {
+ result += CompositorHitTestFlags::eTouchActionPanXDisabled;
+ }
+ if (!(touchAction & StyleTouchAction::PAN_Y)) {
+ result += CompositorHitTestFlags::eTouchActionPanYDisabled;
+ }
+ if (touchAction & StyleTouchAction::NONE) {
+ // all the touch-action disabling flags will already have been set above
+ MOZ_ASSERT(result.contains(CompositorHitTestTouchActionMask));
+ }
+ }
+ }
+
+ const Maybe<ScrollDirection> scrollDirection =
+ aBuilder->GetCurrentScrollbarDirection();
+ if (scrollDirection.isSome()) {
+ if (GetContent()->IsXULElement(nsGkAtoms::thumb)) {
+ const bool thumbGetsLayer = aBuilder->GetCurrentScrollbarTarget() !=
+ layers::ScrollableLayerGuid::NULL_SCROLL_ID;
+ if (thumbGetsLayer) {
+ result += CompositorHitTestFlags::eScrollbarThumb;
+ } else {
+ result += CompositorHitTestFlags::eInactiveScrollframe;
+ }
+ }
+
+ if (*scrollDirection == ScrollDirection::eVertical) {
+ result += CompositorHitTestFlags::eScrollbarVertical;
+ }
+
+ // includes the ScrollbarFrame, SliderFrame, anything else that
+ // might be inside the xul:scrollbar
+ result += CompositorHitTestFlags::eScrollbar;
+ }
+
+ return result;
+}
+
+// Returns true if we can guarantee there is no visible descendants.
+static bool HasNoVisibleDescendants(const nsIFrame* aFrame) {
+ for (const auto& childList : aFrame->ChildLists()) {
+ for (nsIFrame* f : childList.mList) {
+ if (nsPlaceholderFrame::GetRealFrameFor(f)
+ ->IsVisibleOrMayHaveVisibleDescendants()) {
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+nsIScrollableFrame* nsIFrame::GetAsScrollContainer() const {
+ return do_QueryFrame(this);
+}
+
+void nsIFrame::UpdateVisibleDescendantsState() {
+ if (StyleVisibility()->IsVisible()) {
+ // Notify invisible ancestors that a visible descendant exists now.
+ nsIFrame* ancestor;
+ for (ancestor = GetInFlowParent();
+ ancestor && !ancestor->StyleVisibility()->IsVisible();
+ ancestor = ancestor->GetInFlowParent()) {
+ ancestor->mAllDescendantsAreInvisible = false;
+ }
+ } else {
+ mAllDescendantsAreInvisible = HasNoVisibleDescendants(this);
+ }
+}
+
+nsIFrame::PhysicalAxes nsIFrame::ShouldApplyOverflowClipping(
+ const nsStyleDisplay* aDisp) const {
+ MOZ_ASSERT(aDisp == StyleDisplay(), "Wrong display struct");
+
+ // 'contain:paint', which we handle as 'overflow:clip' here. Except for
+ // scrollframes we don't need contain:paint to add any clipping, because
+ // the scrollable frame will already clip overflowing content, and because
+ // 'contain:paint' should prevent all means of escaping that clipping
+ // (e.g. because it forms a fixed-pos containing block).
+ if (aDisp->IsContainPaint() && !IsScrollFrame() &&
+ SupportsContainLayoutAndPaint()) {
+ return PhysicalAxes::Both;
+ }
+
+ // and overflow:hidden that we should interpret as clip
+ if (aDisp->mOverflowX == StyleOverflow::Hidden &&
+ aDisp->mOverflowY == StyleOverflow::Hidden) {
+ // REVIEW: these are the frame types that set up clipping.
+ LayoutFrameType type = Type();
+ switch (type) {
+ case LayoutFrameType::Table:
+ case LayoutFrameType::TableCell:
+ case LayoutFrameType::SVGOuterSVG:
+ case LayoutFrameType::SVGInnerSVG:
+ case LayoutFrameType::SVGSymbol:
+ case LayoutFrameType::SVGForeignObject:
+ return PhysicalAxes::Both;
+ default:
+ if (IsReplacedWithBlock()) {
+ if (type == mozilla::LayoutFrameType::TextInput) {
+ // It has an anonymous scroll frame that handles any overflow.
+ return PhysicalAxes::None;
+ }
+ return PhysicalAxes::Both;
+ }
+ }
+ }
+
+ // clip overflow:clip, except for nsListControlFrame which is
+ // an nsHTMLScrollFrame sub-class.
+ if (MOZ_UNLIKELY((aDisp->mOverflowX == mozilla::StyleOverflow::Clip ||
+ aDisp->mOverflowY == mozilla::StyleOverflow::Clip) &&
+ !IsListControlFrame())) {
+ // FIXME: we could use GetViewportScrollStylesOverrideElement() here instead
+ // if that worked correctly in a print context. (see bug 1654667)
+ const auto* element = Element::FromNodeOrNull(GetContent());
+ if (!element ||
+ !PresContext()->ElementWouldPropagateScrollStyles(*element)) {
+ uint8_t axes = uint8_t(PhysicalAxes::None);
+ if (aDisp->mOverflowX == mozilla::StyleOverflow::Clip) {
+ axes |= uint8_t(PhysicalAxes::Horizontal);
+ }
+ if (aDisp->mOverflowY == mozilla::StyleOverflow::Clip) {
+ axes |= uint8_t(PhysicalAxes::Vertical);
+ }
+ return PhysicalAxes(axes);
+ }
+ }
+
+ if (HasAnyStateBits(NS_FRAME_SVG_LAYOUT)) {
+ return PhysicalAxes::None;
+ }
+
+ // If we're paginated and a block, and have NS_BLOCK_CLIP_PAGINATED_OVERFLOW
+ // set, then we want to clip our overflow.
+ bool clip = HasAnyStateBits(NS_BLOCK_CLIP_PAGINATED_OVERFLOW) &&
+ PresContext()->IsPaginated() && IsBlockFrame();
+ return clip ? PhysicalAxes::Both : PhysicalAxes::None;
+}
+
+#ifdef DEBUG
+static void GetTagName(nsIFrame* aFrame, nsIContent* aContent, int aResultSize,
+ char* aResult) {
+ if (aContent) {
+ snprintf(aResult, aResultSize, "%s@%p",
+ nsAtomCString(aContent->NodeInfo()->NameAtom()).get(), aFrame);
+ } else {
+ snprintf(aResult, aResultSize, "@%p", aFrame);
+ }
+}
+
+void nsIFrame::Trace(const char* aMethod, bool aEnter) {
+ if (NS_FRAME_LOG_TEST(sFrameLogModule, NS_FRAME_TRACE_CALLS)) {
+ char tagbuf[40];
+ GetTagName(this, mContent, sizeof(tagbuf), tagbuf);
+ printf_stderr("%s: %s %s", tagbuf, aEnter ? "enter" : "exit", aMethod);
+ }
+}
+
+void nsIFrame::Trace(const char* aMethod, bool aEnter,
+ const nsReflowStatus& aStatus) {
+ if (NS_FRAME_LOG_TEST(sFrameLogModule, NS_FRAME_TRACE_CALLS)) {
+ char tagbuf[40];
+ GetTagName(this, mContent, sizeof(tagbuf), tagbuf);
+ printf_stderr("%s: %s %s, status=%scomplete%s", tagbuf,
+ aEnter ? "enter" : "exit", aMethod,
+ aStatus.IsIncomplete() ? "not" : "",
+ (aStatus.NextInFlowNeedsReflow()) ? "+reflow" : "");
+ }
+}
+
+void nsIFrame::TraceMsg(const char* aFormatString, ...) {
+ if (NS_FRAME_LOG_TEST(sFrameLogModule, NS_FRAME_TRACE_CALLS)) {
+ // Format arguments into a buffer
+ char argbuf[200];
+ va_list ap;
+ va_start(ap, aFormatString);
+ VsprintfLiteral(argbuf, aFormatString, ap);
+ va_end(ap);
+
+ char tagbuf[40];
+ GetTagName(this, mContent, sizeof(tagbuf), tagbuf);
+ printf_stderr("%s: %s", tagbuf, argbuf);
+ }
+}
+
+void nsIFrame::VerifyDirtyBitSet(const nsFrameList& aFrameList) {
+ for (nsIFrame* f : aFrameList) {
+ NS_ASSERTION(f->HasAnyStateBits(NS_FRAME_IS_DIRTY), "dirty bit not set");
+ }
+}
+
+// Start Display Reflow
+DR_cookie::DR_cookie(nsPresContext* aPresContext, nsIFrame* aFrame,
+ const ReflowInput& aReflowInput, ReflowOutput& aMetrics,
+ nsReflowStatus& aStatus)
+ : mPresContext(aPresContext),
+ mFrame(aFrame),
+ mReflowInput(aReflowInput),
+ mMetrics(aMetrics),
+ mStatus(aStatus) {
+ MOZ_COUNT_CTOR(DR_cookie);
+ mValue = nsIFrame::DisplayReflowEnter(aPresContext, mFrame, mReflowInput);
+}
+
+DR_cookie::~DR_cookie() {
+ MOZ_COUNT_DTOR(DR_cookie);
+ nsIFrame::DisplayReflowExit(mPresContext, mFrame, mMetrics, mStatus, mValue);
+}
+
+DR_layout_cookie::DR_layout_cookie(nsIFrame* aFrame) : mFrame(aFrame) {
+ MOZ_COUNT_CTOR(DR_layout_cookie);
+ mValue = nsIFrame::DisplayLayoutEnter(mFrame);
+}
+
+DR_layout_cookie::~DR_layout_cookie() {
+ MOZ_COUNT_DTOR(DR_layout_cookie);
+ nsIFrame::DisplayLayoutExit(mFrame, mValue);
+}
+
+DR_intrinsic_inline_size_cookie::DR_intrinsic_inline_size_cookie(
+ nsIFrame* aFrame, const char* aType, nscoord& aResult)
+ : mFrame(aFrame), mType(aType), mResult(aResult) {
+ MOZ_COUNT_CTOR(DR_intrinsic_inline_size_cookie);
+ mValue = nsIFrame::DisplayIntrinsicISizeEnter(mFrame, mType);
+}
+
+DR_intrinsic_inline_size_cookie::~DR_intrinsic_inline_size_cookie() {
+ MOZ_COUNT_DTOR(DR_intrinsic_inline_size_cookie);
+ nsIFrame::DisplayIntrinsicISizeExit(mFrame, mType, mResult, mValue);
+}
+
+DR_intrinsic_size_cookie::DR_intrinsic_size_cookie(nsIFrame* aFrame,
+ const char* aType,
+ nsSize& aResult)
+ : mFrame(aFrame), mType(aType), mResult(aResult) {
+ MOZ_COUNT_CTOR(DR_intrinsic_size_cookie);
+ mValue = nsIFrame::DisplayIntrinsicSizeEnter(mFrame, mType);
+}
+
+DR_intrinsic_size_cookie::~DR_intrinsic_size_cookie() {
+ MOZ_COUNT_DTOR(DR_intrinsic_size_cookie);
+ nsIFrame::DisplayIntrinsicSizeExit(mFrame, mType, mResult, mValue);
+}
+
+DR_init_constraints_cookie::DR_init_constraints_cookie(
+ nsIFrame* aFrame, ReflowInput* aState, nscoord aCBWidth, nscoord aCBHeight,
+ const mozilla::Maybe<mozilla::LogicalMargin> aBorder,
+ const mozilla::Maybe<mozilla::LogicalMargin> aPadding)
+ : mFrame(aFrame), mState(aState) {
+ MOZ_COUNT_CTOR(DR_init_constraints_cookie);
+ nsMargin border;
+ if (aBorder) {
+ border = aBorder->GetPhysicalMargin(aFrame->GetWritingMode());
+ }
+ nsMargin padding;
+ if (aPadding) {
+ padding = aPadding->GetPhysicalMargin(aFrame->GetWritingMode());
+ }
+ mValue = ReflowInput::DisplayInitConstraintsEnter(
+ mFrame, mState, aCBWidth, aCBHeight, aBorder ? &border : nullptr,
+ aPadding ? &padding : nullptr);
+}
+
+DR_init_constraints_cookie::~DR_init_constraints_cookie() {
+ MOZ_COUNT_DTOR(DR_init_constraints_cookie);
+ ReflowInput::DisplayInitConstraintsExit(mFrame, mState, mValue);
+}
+
+DR_init_offsets_cookie::DR_init_offsets_cookie(
+ nsIFrame* aFrame, SizeComputationInput* aState, nscoord aPercentBasis,
+ WritingMode aCBWritingMode,
+ const mozilla::Maybe<mozilla::LogicalMargin> aBorder,
+ const mozilla::Maybe<mozilla::LogicalMargin> aPadding)
+ : mFrame(aFrame), mState(aState) {
+ MOZ_COUNT_CTOR(DR_init_offsets_cookie);
+ nsMargin border;
+ if (aBorder) {
+ border = aBorder->GetPhysicalMargin(aFrame->GetWritingMode());
+ }
+ nsMargin padding;
+ if (aPadding) {
+ padding = aPadding->GetPhysicalMargin(aFrame->GetWritingMode());
+ }
+ mValue = SizeComputationInput::DisplayInitOffsetsEnter(
+ mFrame, mState, aPercentBasis, aCBWritingMode,
+ aBorder ? &border : nullptr, aPadding ? &padding : nullptr);
+}
+
+DR_init_offsets_cookie::~DR_init_offsets_cookie() {
+ MOZ_COUNT_DTOR(DR_init_offsets_cookie);
+ SizeComputationInput::DisplayInitOffsetsExit(mFrame, mState, mValue);
+}
+
+struct DR_Rule;
+
+struct DR_FrameTypeInfo {
+ DR_FrameTypeInfo(LayoutFrameType aFrameType, const char* aFrameNameAbbrev,
+ const char* aFrameName);
+ ~DR_FrameTypeInfo();
+
+ LayoutFrameType mType;
+ char mNameAbbrev[16];
+ char mName[32];
+ nsTArray<DR_Rule*> mRules;
+
+ private:
+ DR_FrameTypeInfo& operator=(const DR_FrameTypeInfo&) = delete;
+};
+
+struct DR_FrameTreeNode;
+struct DR_Rule;
+
+struct DR_State {
+ DR_State();
+ ~DR_State();
+ void Init();
+ void AddFrameTypeInfo(LayoutFrameType aFrameType,
+ const char* aFrameNameAbbrev, const char* aFrameName);
+ DR_FrameTypeInfo* GetFrameTypeInfo(LayoutFrameType aFrameType);
+ DR_FrameTypeInfo* GetFrameTypeInfo(char* aFrameName);
+ void InitFrameTypeTable();
+ DR_FrameTreeNode* CreateTreeNode(nsIFrame* aFrame,
+ const ReflowInput* aReflowInput);
+ void FindMatchingRule(DR_FrameTreeNode& aNode);
+ bool RuleMatches(DR_Rule& aRule, DR_FrameTreeNode& aNode);
+ bool GetToken(FILE* aFile, char* aBuf, size_t aBufSize);
+ DR_Rule* ParseRule(FILE* aFile);
+ void ParseRulesFile();
+ void AddRule(nsTArray<DR_Rule*>& aRules, DR_Rule& aRule);
+ bool IsWhiteSpace(int c);
+ bool GetNumber(char* aBuf, int32_t& aNumber);
+ void PrettyUC(nscoord aSize, char* aBuf, int aBufSize);
+ void PrintMargin(const char* tag, const nsMargin* aMargin);
+ void DisplayFrameTypeInfo(nsIFrame* aFrame, int32_t aIndent);
+ void DeleteTreeNode(DR_FrameTreeNode& aNode);
+
+ bool mInited;
+ bool mActive;
+ int32_t mCount;
+ int32_t mAssert;
+ int32_t mIndent;
+ bool mIndentUndisplayedFrames;
+ bool mDisplayPixelErrors;
+ nsTArray<DR_Rule*> mWildRules;
+ nsTArray<DR_FrameTypeInfo> mFrameTypeTable;
+ // reflow specific state
+ nsTArray<DR_FrameTreeNode*> mFrameTreeLeaves;
+};
+
+static DR_State* DR_state; // the one and only DR_State
+
+struct DR_RulePart {
+ explicit DR_RulePart(LayoutFrameType aFrameType)
+ : mFrameType(aFrameType), mNext(0) {}
+
+ void Destroy();
+
+ LayoutFrameType mFrameType;
+ DR_RulePart* mNext;
+};
+
+void DR_RulePart::Destroy() {
+ if (mNext) {
+ mNext->Destroy();
+ }
+ delete this;
+}
+
+struct DR_Rule {
+ DR_Rule() : mLength(0), mTarget(nullptr), mDisplay(false) {
+ MOZ_COUNT_CTOR(DR_Rule);
+ }
+ ~DR_Rule() {
+ if (mTarget) mTarget->Destroy();
+ MOZ_COUNT_DTOR(DR_Rule);
+ }
+ void AddPart(LayoutFrameType aFrameType);
+
+ uint32_t mLength;
+ DR_RulePart* mTarget;
+ bool mDisplay;
+};
+
+void DR_Rule::AddPart(LayoutFrameType aFrameType) {
+ DR_RulePart* newPart = new DR_RulePart(aFrameType);
+ newPart->mNext = mTarget;
+ mTarget = newPart;
+ mLength++;
+}
+
+DR_FrameTypeInfo::~DR_FrameTypeInfo() {
+ int32_t numElements;
+ numElements = mRules.Length();
+ for (int32_t i = numElements - 1; i >= 0; i--) {
+ delete mRules.ElementAt(i);
+ }
+}
+
+DR_FrameTypeInfo::DR_FrameTypeInfo(LayoutFrameType aFrameType,
+ const char* aFrameNameAbbrev,
+ const char* aFrameName) {
+ mType = aFrameType;
+ PL_strncpyz(mNameAbbrev, aFrameNameAbbrev, sizeof(mNameAbbrev));
+ PL_strncpyz(mName, aFrameName, sizeof(mName));
+}
+
+struct DR_FrameTreeNode {
+ DR_FrameTreeNode(nsIFrame* aFrame, DR_FrameTreeNode* aParent)
+ : mFrame(aFrame), mParent(aParent), mDisplay(0), mIndent(0) {
+ MOZ_COUNT_CTOR(DR_FrameTreeNode);
+ }
+
+ MOZ_COUNTED_DTOR(DR_FrameTreeNode)
+
+ nsIFrame* mFrame;
+ DR_FrameTreeNode* mParent;
+ bool mDisplay;
+ uint32_t mIndent;
+};
+
+// DR_State implementation
+
+DR_State::DR_State()
+ : mInited(false),
+ mActive(false),
+ mCount(0),
+ mAssert(-1),
+ mIndent(0),
+ mIndentUndisplayedFrames(false),
+ mDisplayPixelErrors(false) {
+ MOZ_COUNT_CTOR(DR_State);
+}
+
+void DR_State::Init() {
+ char* env = PR_GetEnv("GECKO_DISPLAY_REFLOW_ASSERT");
+ int32_t num;
+ if (env) {
+ if (GetNumber(env, num))
+ mAssert = num;
+ else
+ printf("GECKO_DISPLAY_REFLOW_ASSERT - invalid value = %s", env);
+ }
+
+ env = PR_GetEnv("GECKO_DISPLAY_REFLOW_INDENT_START");
+ if (env) {
+ if (GetNumber(env, num))
+ mIndent = num;
+ else
+ printf("GECKO_DISPLAY_REFLOW_INDENT_START - invalid value = %s", env);
+ }
+
+ env = PR_GetEnv("GECKO_DISPLAY_REFLOW_INDENT_UNDISPLAYED_FRAMES");
+ if (env) {
+ if (GetNumber(env, num))
+ mIndentUndisplayedFrames = num;
+ else
+ printf(
+ "GECKO_DISPLAY_REFLOW_INDENT_UNDISPLAYED_FRAMES - invalid value = %s",
+ env);
+ }
+
+ env = PR_GetEnv("GECKO_DISPLAY_REFLOW_FLAG_PIXEL_ERRORS");
+ if (env) {
+ if (GetNumber(env, num))
+ mDisplayPixelErrors = num;
+ else
+ printf("GECKO_DISPLAY_REFLOW_FLAG_PIXEL_ERRORS - invalid value = %s",
+ env);
+ }
+
+ InitFrameTypeTable();
+ ParseRulesFile();
+ mInited = true;
+}
+
+DR_State::~DR_State() {
+ MOZ_COUNT_DTOR(DR_State);
+ int32_t numElements, i;
+ numElements = mWildRules.Length();
+ for (i = numElements - 1; i >= 0; i--) {
+ delete mWildRules.ElementAt(i);
+ }
+ numElements = mFrameTreeLeaves.Length();
+ for (i = numElements - 1; i >= 0; i--) {
+ delete mFrameTreeLeaves.ElementAt(i);
+ }
+}
+
+bool DR_State::GetNumber(char* aBuf, int32_t& aNumber) {
+ if (sscanf(aBuf, "%d", &aNumber) > 0)
+ return true;
+ else
+ return false;
+}
+
+bool DR_State::IsWhiteSpace(int c) {
+ return (c == ' ') || (c == '\t') || (c == '\n') || (c == '\r');
+}
+
+bool DR_State::GetToken(FILE* aFile, char* aBuf, size_t aBufSize) {
+ bool haveToken = false;
+ aBuf[0] = 0;
+ // get the 1st non whitespace char
+ int c = -1;
+ for (c = getc(aFile); (c > 0) && IsWhiteSpace(c); c = getc(aFile)) {
+ }
+
+ if (c > 0) {
+ haveToken = true;
+ aBuf[0] = c;
+ // get everything up to the next whitespace char
+ size_t cX;
+ for (cX = 1; cX + 1 < aBufSize; cX++) {
+ c = getc(aFile);
+ if (c < 0) { // EOF
+ ungetc(' ', aFile);
+ break;
+ } else {
+ if (IsWhiteSpace(c)) {
+ break;
+ } else {
+ aBuf[cX] = c;
+ }
+ }
+ }
+ aBuf[cX] = 0;
+ }
+ return haveToken;
+}
+
+DR_Rule* DR_State::ParseRule(FILE* aFile) {
+ char buf[128];
+ int32_t doDisplay;
+ DR_Rule* rule = nullptr;
+ while (GetToken(aFile, buf, sizeof(buf))) {
+ if (GetNumber(buf, doDisplay)) {
+ if (rule) {
+ rule->mDisplay = !!doDisplay;
+ break;
+ } else {
+ printf("unexpected token - %s \n", buf);
+ }
+ } else {
+ if (!rule) {
+ rule = new DR_Rule;
+ }
+ if (strcmp(buf, "*") == 0) {
+ rule->AddPart(LayoutFrameType::None);
+ } else {
+ DR_FrameTypeInfo* info = GetFrameTypeInfo(buf);
+ if (info) {
+ rule->AddPart(info->mType);
+ } else {
+ printf("invalid frame type - %s \n", buf);
+ }
+ }
+ }
+ }
+ return rule;
+}
+
+void DR_State::AddRule(nsTArray<DR_Rule*>& aRules, DR_Rule& aRule) {
+ int32_t numRules = aRules.Length();
+ for (int32_t ruleX = 0; ruleX < numRules; ruleX++) {
+ DR_Rule* rule = aRules.ElementAt(ruleX);
+ NS_ASSERTION(rule, "program error");
+ if (aRule.mLength > rule->mLength) {
+ aRules.InsertElementAt(ruleX, &aRule);
+ return;
+ }
+ }
+ aRules.AppendElement(&aRule);
+}
+
+static Maybe<bool> ShouldLogReflow(const char* processes) {
+ switch (processes[0]) {
+ case 'A':
+ case 'a':
+ return Some(true);
+ case 'P':
+ case 'p':
+ return Some(XRE_IsParentProcess());
+ case 'C':
+ case 'c':
+ return Some(XRE_IsContentProcess());
+ default:
+ return Nothing{};
+ }
+}
+
+void DR_State::ParseRulesFile() {
+ char* processes = PR_GetEnv("GECKO_DISPLAY_REFLOW_PROCESSES");
+ if (processes) {
+ Maybe<bool> enableLog = ShouldLogReflow(processes);
+ if (enableLog.isNothing()) {
+ MOZ_CRASH("GECKO_DISPLAY_REFLOW_PROCESSES: [a]ll [p]arent [c]ontent");
+ } else if (enableLog.value()) {
+ DR_Rule* rule = new DR_Rule;
+ rule->AddPart(LayoutFrameType::None);
+ rule->mDisplay = true;
+ AddRule(mWildRules, *rule);
+ mActive = true;
+ }
+ return;
+ }
+
+ char* path = PR_GetEnv("GECKO_DISPLAY_REFLOW_RULES_FILE");
+ if (path) {
+ FILE* inFile = fopen(path, "r");
+ if (!inFile) {
+ MOZ_CRASH(
+ "Failed to open the specified rules file; Try `--setpref "
+ "security.sandbox.content.level=2` if the sandbox is at cause");
+ }
+ for (DR_Rule* rule = ParseRule(inFile); rule; rule = ParseRule(inFile)) {
+ if (rule->mTarget) {
+ LayoutFrameType fType = rule->mTarget->mFrameType;
+ if (fType != LayoutFrameType::None) {
+ DR_FrameTypeInfo* info = GetFrameTypeInfo(fType);
+ AddRule(info->mRules, *rule);
+ } else {
+ AddRule(mWildRules, *rule);
+ }
+ mActive = true;
+ }
+ }
+
+ fclose(inFile);
+ }
+}
+
+void DR_State::AddFrameTypeInfo(LayoutFrameType aFrameType,
+ const char* aFrameNameAbbrev,
+ const char* aFrameName) {
+ mFrameTypeTable.EmplaceBack(aFrameType, aFrameNameAbbrev, aFrameName);
+}
+
+DR_FrameTypeInfo* DR_State::GetFrameTypeInfo(LayoutFrameType aFrameType) {
+ int32_t numEntries = mFrameTypeTable.Length();
+ NS_ASSERTION(numEntries != 0, "empty FrameTypeTable");
+ for (int32_t i = 0; i < numEntries; i++) {
+ DR_FrameTypeInfo& info = mFrameTypeTable.ElementAt(i);
+ if (info.mType == aFrameType) {
+ return &info;
+ }
+ }
+ return &mFrameTypeTable.ElementAt(numEntries -
+ 1); // return unknown frame type
+}
+
+DR_FrameTypeInfo* DR_State::GetFrameTypeInfo(char* aFrameName) {
+ int32_t numEntries = mFrameTypeTable.Length();
+ NS_ASSERTION(numEntries != 0, "empty FrameTypeTable");
+ for (int32_t i = 0; i < numEntries; i++) {
+ DR_FrameTypeInfo& info = mFrameTypeTable.ElementAt(i);
+ if ((strcmp(aFrameName, info.mName) == 0) ||
+ (strcmp(aFrameName, info.mNameAbbrev) == 0)) {
+ return &info;
+ }
+ }
+ return &mFrameTypeTable.ElementAt(numEntries -
+ 1); // return unknown frame type
+}
+
+void DR_State::InitFrameTypeTable() {
+ AddFrameTypeInfo(LayoutFrameType::Block, "block", "block");
+ AddFrameTypeInfo(LayoutFrameType::Br, "br", "br");
+ AddFrameTypeInfo(LayoutFrameType::ColorControl, "color", "colorControl");
+ AddFrameTypeInfo(LayoutFrameType::GfxButtonControl, "button",
+ "gfxButtonControl");
+ AddFrameTypeInfo(LayoutFrameType::HTMLButtonControl, "HTMLbutton",
+ "HTMLButtonControl");
+ AddFrameTypeInfo(LayoutFrameType::HTMLCanvas, "HTMLCanvas", "HTMLCanvas");
+ AddFrameTypeInfo(LayoutFrameType::SubDocument, "subdoc", "subDocument");
+ AddFrameTypeInfo(LayoutFrameType::Image, "img", "image");
+ AddFrameTypeInfo(LayoutFrameType::Inline, "inline", "inline");
+ AddFrameTypeInfo(LayoutFrameType::Letter, "letter", "letter");
+ AddFrameTypeInfo(LayoutFrameType::Line, "line", "line");
+ AddFrameTypeInfo(LayoutFrameType::ListControl, "select", "select");
+ AddFrameTypeInfo(LayoutFrameType::Page, "page", "page");
+ AddFrameTypeInfo(LayoutFrameType::Placeholder, "place", "placeholder");
+ AddFrameTypeInfo(LayoutFrameType::Canvas, "canvas", "canvas");
+ AddFrameTypeInfo(LayoutFrameType::Scroll, "scroll", "scroll");
+ AddFrameTypeInfo(LayoutFrameType::TableCell, "cell", "tableCell");
+ AddFrameTypeInfo(LayoutFrameType::TableCol, "col", "tableCol");
+ AddFrameTypeInfo(LayoutFrameType::TableColGroup, "colG", "tableColGroup");
+ AddFrameTypeInfo(LayoutFrameType::Table, "tbl", "table");
+ AddFrameTypeInfo(LayoutFrameType::TableWrapper, "tblW", "tableWrapper");
+ AddFrameTypeInfo(LayoutFrameType::TableRowGroup, "rowG", "tableRowGroup");
+ AddFrameTypeInfo(LayoutFrameType::TableRow, "row", "tableRow");
+ AddFrameTypeInfo(LayoutFrameType::TextInput, "textCtl", "textInput");
+ AddFrameTypeInfo(LayoutFrameType::Text, "text", "text");
+ AddFrameTypeInfo(LayoutFrameType::Viewport, "VP", "viewport");
+ AddFrameTypeInfo(LayoutFrameType::Slider, "Slider", "Slider");
+ AddFrameTypeInfo(LayoutFrameType::None, "unknown", "unknown");
+}
+
+void DR_State::DisplayFrameTypeInfo(nsIFrame* aFrame, int32_t aIndent) {
+ DR_FrameTypeInfo* frameTypeInfo = GetFrameTypeInfo(aFrame->Type());
+ if (frameTypeInfo) {
+ for (int32_t i = 0; i < aIndent; i++) {
+ printf(" ");
+ }
+ if (!strcmp(frameTypeInfo->mNameAbbrev, "unknown")) {
+ if (aFrame) {
+ nsAutoString name;
+ aFrame->GetFrameName(name);
+ printf("%s %p ", NS_LossyConvertUTF16toASCII(name).get(),
+ (void*)aFrame);
+ } else {
+ printf("%s %p ", frameTypeInfo->mNameAbbrev, (void*)aFrame);
+ }
+ } else {
+ printf("%s %p ", frameTypeInfo->mNameAbbrev, (void*)aFrame);
+ }
+ }
+}
+
+bool DR_State::RuleMatches(DR_Rule& aRule, DR_FrameTreeNode& aNode) {
+ NS_ASSERTION(aRule.mTarget, "program error");
+
+ DR_RulePart* rulePart;
+ DR_FrameTreeNode* parentNode;
+ for (rulePart = aRule.mTarget->mNext, parentNode = aNode.mParent;
+ rulePart && parentNode;
+ rulePart = rulePart->mNext, parentNode = parentNode->mParent) {
+ if (rulePart->mFrameType != LayoutFrameType::None) {
+ if (parentNode->mFrame) {
+ if (rulePart->mFrameType != parentNode->mFrame->Type()) {
+ return false;
+ }
+ } else
+ NS_ASSERTION(false, "program error");
+ }
+ // else wild card match
+ }
+ return true;
+}
+
+void DR_State::FindMatchingRule(DR_FrameTreeNode& aNode) {
+ if (!aNode.mFrame) {
+ NS_ASSERTION(false, "invalid DR_FrameTreeNode \n");
+ return;
+ }
+
+ bool matchingRule = false;
+
+ DR_FrameTypeInfo* info = GetFrameTypeInfo(aNode.mFrame->Type());
+ NS_ASSERTION(info, "program error");
+ int32_t numRules = info->mRules.Length();
+ for (int32_t ruleX = 0; ruleX < numRules; ruleX++) {
+ DR_Rule* rule = info->mRules.ElementAt(ruleX);
+ if (rule && RuleMatches(*rule, aNode)) {
+ aNode.mDisplay = rule->mDisplay;
+ matchingRule = true;
+ break;
+ }
+ }
+ if (!matchingRule) {
+ int32_t numWildRules = mWildRules.Length();
+ for (int32_t ruleX = 0; ruleX < numWildRules; ruleX++) {
+ DR_Rule* rule = mWildRules.ElementAt(ruleX);
+ if (rule && RuleMatches(*rule, aNode)) {
+ aNode.mDisplay = rule->mDisplay;
+ break;
+ }
+ }
+ }
+}
+
+DR_FrameTreeNode* DR_State::CreateTreeNode(nsIFrame* aFrame,
+ const ReflowInput* aReflowInput) {
+ // find the frame of the parent reflow input (usually just the parent of
+ // aFrame)
+ nsIFrame* parentFrame;
+ if (aReflowInput) {
+ const ReflowInput* parentRI = aReflowInput->mParentReflowInput;
+ parentFrame = (parentRI) ? parentRI->mFrame : nullptr;
+ } else {
+ parentFrame = aFrame->GetParent();
+ }
+
+ // find the parent tree node leaf
+ DR_FrameTreeNode* parentNode = nullptr;
+
+ DR_FrameTreeNode* lastLeaf = nullptr;
+ if (mFrameTreeLeaves.Length())
+ lastLeaf = mFrameTreeLeaves.ElementAt(mFrameTreeLeaves.Length() - 1);
+ if (lastLeaf) {
+ for (parentNode = lastLeaf;
+ parentNode && (parentNode->mFrame != parentFrame);
+ parentNode = parentNode->mParent) {
+ }
+ }
+ DR_FrameTreeNode* newNode = new DR_FrameTreeNode(aFrame, parentNode);
+ FindMatchingRule(*newNode);
+
+ newNode->mIndent = mIndent;
+ if (newNode->mDisplay || mIndentUndisplayedFrames) {
+ ++mIndent;
+ }
+
+ if (lastLeaf && (lastLeaf == parentNode)) {
+ mFrameTreeLeaves.RemoveLastElement();
+ }
+ mFrameTreeLeaves.AppendElement(newNode);
+ mCount++;
+
+ return newNode;
+}
+
+void DR_State::PrettyUC(nscoord aSize, char* aBuf, int aBufSize) {
+ if (NS_UNCONSTRAINEDSIZE == aSize) {
+ strcpy(aBuf, "UC");
+ } else {
+ if ((nscoord)0xdeadbeefU == aSize) {
+ strcpy(aBuf, "deadbeef");
+ } else {
+ snprintf(aBuf, aBufSize, "%d", aSize);
+ }
+ }
+}
+
+void DR_State::PrintMargin(const char* tag, const nsMargin* aMargin) {
+ if (aMargin) {
+ char t[16], r[16], b[16], l[16];
+ PrettyUC(aMargin->top, t, 16);
+ PrettyUC(aMargin->right, r, 16);
+ PrettyUC(aMargin->bottom, b, 16);
+ PrettyUC(aMargin->left, l, 16);
+ printf(" %s=%s,%s,%s,%s", tag, t, r, b, l);
+ } else {
+ // use %p here for consistency with other null-pointer printouts
+ printf(" %s=%p", tag, (void*)aMargin);
+ }
+}
+
+void DR_State::DeleteTreeNode(DR_FrameTreeNode& aNode) {
+ mFrameTreeLeaves.RemoveElement(&aNode);
+ int32_t numLeaves = mFrameTreeLeaves.Length();
+ if ((0 == numLeaves) ||
+ (aNode.mParent != mFrameTreeLeaves.ElementAt(numLeaves - 1))) {
+ mFrameTreeLeaves.AppendElement(aNode.mParent);
+ }
+
+ if (aNode.mDisplay || mIndentUndisplayedFrames) {
+ --mIndent;
+ }
+ // delete the tree node
+ delete &aNode;
+}
+
+static void CheckPixelError(nscoord aSize, int32_t aPixelToTwips) {
+ if (NS_UNCONSTRAINEDSIZE != aSize) {
+ if ((aSize % aPixelToTwips) > 0) {
+ printf("VALUE %d is not a whole pixel \n", aSize);
+ }
+ }
+}
+
+static void DisplayReflowEnterPrint(nsPresContext* aPresContext,
+ nsIFrame* aFrame,
+ const ReflowInput& aReflowInput,
+ DR_FrameTreeNode& aTreeNode,
+ bool aChanged) {
+ if (aTreeNode.mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, aTreeNode.mIndent);
+
+ char width[16];
+ char height[16];
+
+ DR_state->PrettyUC(aReflowInput.AvailableWidth(), width, 16);
+ DR_state->PrettyUC(aReflowInput.AvailableHeight(), height, 16);
+ printf("Reflow a=%s,%s ", width, height);
+
+ DR_state->PrettyUC(aReflowInput.ComputedWidth(), width, 16);
+ DR_state->PrettyUC(aReflowInput.ComputedHeight(), height, 16);
+ printf("c=%s,%s ", width, height);
+
+ if (aFrame->HasAnyStateBits(NS_FRAME_IS_DIRTY)) printf("dirty ");
+
+ if (aFrame->HasAnyStateBits(NS_FRAME_HAS_DIRTY_CHILDREN))
+ printf("dirty-children ");
+
+ if (aReflowInput.mFlags.mSpecialBSizeReflow) printf("special-bsize ");
+
+ if (aReflowInput.IsHResize()) printf("h-resize ");
+
+ if (aReflowInput.IsVResize()) printf("v-resize ");
+
+ nsIFrame* inFlow = aFrame->GetPrevInFlow();
+ if (inFlow) {
+ printf("pif=%p ", (void*)inFlow);
+ }
+ inFlow = aFrame->GetNextInFlow();
+ if (inFlow) {
+ printf("nif=%p ", (void*)inFlow);
+ }
+ if (aChanged)
+ printf("CHANGED \n");
+ else
+ printf("cnt=%d \n", DR_state->mCount);
+ if (DR_state->mDisplayPixelErrors) {
+ int32_t d2a = aPresContext->AppUnitsPerDevPixel();
+ CheckPixelError(aReflowInput.AvailableWidth(), d2a);
+ CheckPixelError(aReflowInput.AvailableHeight(), d2a);
+ CheckPixelError(aReflowInput.ComputedWidth(), d2a);
+ CheckPixelError(aReflowInput.ComputedHeight(), d2a);
+ }
+ }
+}
+
+void* nsIFrame::DisplayReflowEnter(nsPresContext* aPresContext,
+ nsIFrame* aFrame,
+ const ReflowInput& aReflowInput) {
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ NS_ASSERTION(aFrame, "invalid call");
+
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, &aReflowInput);
+ if (treeNode) {
+ DisplayReflowEnterPrint(aPresContext, aFrame, aReflowInput, *treeNode,
+ false);
+ }
+ return treeNode;
+}
+
+void* nsIFrame::DisplayLayoutEnter(nsIFrame* aFrame) {
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ NS_ASSERTION(aFrame, "invalid call");
+
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, nullptr);
+ if (treeNode && treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ printf("XULLayout\n");
+ }
+ return treeNode;
+}
+
+void* nsIFrame::DisplayIntrinsicISizeEnter(nsIFrame* aFrame,
+ const char* aType) {
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ NS_ASSERTION(aFrame, "invalid call");
+
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, nullptr);
+ if (treeNode && treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ printf("Get%sISize\n", aType);
+ }
+ return treeNode;
+}
+
+void* nsIFrame::DisplayIntrinsicSizeEnter(nsIFrame* aFrame, const char* aType) {
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ NS_ASSERTION(aFrame, "invalid call");
+
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, nullptr);
+ if (treeNode && treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ printf("Get%sSize\n", aType);
+ }
+ return treeNode;
+}
+
+void nsIFrame::DisplayReflowExit(nsPresContext* aPresContext, nsIFrame* aFrame,
+ ReflowOutput& aMetrics,
+ const nsReflowStatus& aStatus,
+ void* aFrameTreeNode) {
+ if (!DR_state->mActive) return;
+
+ NS_ASSERTION(aFrame, "DisplayReflowExit - invalid call");
+ if (!aFrameTreeNode) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aFrameTreeNode;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+
+ char width[16];
+ char height[16];
+ char x[16];
+ char y[16];
+ DR_state->PrettyUC(aMetrics.Width(), width, 16);
+ DR_state->PrettyUC(aMetrics.Height(), height, 16);
+ printf("Reflow d=%s,%s", width, height);
+
+ if (!aStatus.IsEmpty()) {
+ printf(" status=%s", ToString(aStatus).c_str());
+ }
+ if (aFrame->HasOverflowAreas()) {
+ DR_state->PrettyUC(aMetrics.InkOverflow().x, x, 16);
+ DR_state->PrettyUC(aMetrics.InkOverflow().y, y, 16);
+ DR_state->PrettyUC(aMetrics.InkOverflow().width, width, 16);
+ DR_state->PrettyUC(aMetrics.InkOverflow().height, height, 16);
+ printf(" vis-o=(%s,%s) %s x %s", x, y, width, height);
+
+ nsRect storedOverflow = aFrame->InkOverflowRect();
+ DR_state->PrettyUC(storedOverflow.x, x, 16);
+ DR_state->PrettyUC(storedOverflow.y, y, 16);
+ DR_state->PrettyUC(storedOverflow.width, width, 16);
+ DR_state->PrettyUC(storedOverflow.height, height, 16);
+ printf(" vis-sto=(%s,%s) %s x %s", x, y, width, height);
+
+ DR_state->PrettyUC(aMetrics.ScrollableOverflow().x, x, 16);
+ DR_state->PrettyUC(aMetrics.ScrollableOverflow().y, y, 16);
+ DR_state->PrettyUC(aMetrics.ScrollableOverflow().width, width, 16);
+ DR_state->PrettyUC(aMetrics.ScrollableOverflow().height, height, 16);
+ printf(" scr-o=(%s,%s) %s x %s", x, y, width, height);
+
+ storedOverflow = aFrame->ScrollableOverflowRect();
+ DR_state->PrettyUC(storedOverflow.x, x, 16);
+ DR_state->PrettyUC(storedOverflow.y, y, 16);
+ DR_state->PrettyUC(storedOverflow.width, width, 16);
+ DR_state->PrettyUC(storedOverflow.height, height, 16);
+ printf(" scr-sto=(%s,%s) %s x %s", x, y, width, height);
+ }
+ printf("\n");
+ if (DR_state->mDisplayPixelErrors) {
+ int32_t d2a = aPresContext->AppUnitsPerDevPixel();
+ CheckPixelError(aMetrics.Width(), d2a);
+ CheckPixelError(aMetrics.Height(), d2a);
+ }
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+void nsIFrame::DisplayLayoutExit(nsIFrame* aFrame, void* aFrameTreeNode) {
+ if (!DR_state->mActive) return;
+
+ NS_ASSERTION(aFrame, "non-null frame required");
+ if (!aFrameTreeNode) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aFrameTreeNode;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ nsRect rect = aFrame->GetRect();
+ printf("XULLayout=%d,%d,%d,%d\n", rect.x, rect.y, rect.width, rect.height);
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+void nsIFrame::DisplayIntrinsicISizeExit(nsIFrame* aFrame, const char* aType,
+ nscoord aResult,
+ void* aFrameTreeNode) {
+ if (!DR_state->mActive) return;
+
+ NS_ASSERTION(aFrame, "non-null frame required");
+ if (!aFrameTreeNode) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aFrameTreeNode;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ char iSize[16];
+ DR_state->PrettyUC(aResult, iSize, 16);
+ printf("Get%sISize=%s\n", aType, iSize);
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+void nsIFrame::DisplayIntrinsicSizeExit(nsIFrame* aFrame, const char* aType,
+ nsSize aResult, void* aFrameTreeNode) {
+ if (!DR_state->mActive) return;
+
+ NS_ASSERTION(aFrame, "non-null frame required");
+ if (!aFrameTreeNode) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aFrameTreeNode;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+
+ char width[16];
+ char height[16];
+ DR_state->PrettyUC(aResult.width, width, 16);
+ DR_state->PrettyUC(aResult.height, height, 16);
+ printf("Get%sSize=%s,%s\n", aType, width, height);
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+/* static */
+void nsIFrame::DisplayReflowStartup() { DR_state = new DR_State(); }
+
+/* static */
+void nsIFrame::DisplayReflowShutdown() {
+ delete DR_state;
+ DR_state = nullptr;
+}
+
+void DR_cookie::Change() const {
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)mValue;
+ if (treeNode && treeNode->mDisplay) {
+ DisplayReflowEnterPrint(mPresContext, mFrame, mReflowInput, *treeNode,
+ true);
+ }
+}
+
+/* static */
+void* ReflowInput::DisplayInitConstraintsEnter(nsIFrame* aFrame,
+ ReflowInput* aState,
+ nscoord aContainingBlockWidth,
+ nscoord aContainingBlockHeight,
+ const nsMargin* aBorder,
+ const nsMargin* aPadding) {
+ MOZ_ASSERT(aFrame, "non-null frame required");
+ MOZ_ASSERT(aState, "non-null state required");
+
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, aState);
+ if (treeNode && treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+
+ printf("InitConstraints parent=%p", (void*)aState->mParentReflowInput);
+
+ char width[16];
+ char height[16];
+
+ DR_state->PrettyUC(aContainingBlockWidth, width, 16);
+ DR_state->PrettyUC(aContainingBlockHeight, height, 16);
+ printf(" cb=%s,%s", width, height);
+
+ DR_state->PrettyUC(aState->AvailableWidth(), width, 16);
+ DR_state->PrettyUC(aState->AvailableHeight(), height, 16);
+ printf(" as=%s,%s", width, height);
+
+ DR_state->PrintMargin("b", aBorder);
+ DR_state->PrintMargin("p", aPadding);
+ putchar('\n');
+ }
+ return treeNode;
+}
+
+/* static */
+void ReflowInput::DisplayInitConstraintsExit(nsIFrame* aFrame,
+ ReflowInput* aState,
+ void* aValue) {
+ MOZ_ASSERT(aFrame, "non-null frame required");
+ MOZ_ASSERT(aState, "non-null state required");
+
+ if (!DR_state->mActive) return;
+ if (!aValue) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aValue;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ char cmiw[16], cw[16], cmxw[16], cmih[16], ch[16], cmxh[16];
+ DR_state->PrettyUC(aState->ComputedMinWidth(), cmiw, 16);
+ DR_state->PrettyUC(aState->ComputedWidth(), cw, 16);
+ DR_state->PrettyUC(aState->ComputedMaxWidth(), cmxw, 16);
+ DR_state->PrettyUC(aState->ComputedMinHeight(), cmih, 16);
+ DR_state->PrettyUC(aState->ComputedHeight(), ch, 16);
+ DR_state->PrettyUC(aState->ComputedMaxHeight(), cmxh, 16);
+ printf("InitConstraints= cw=(%s <= %s <= %s) ch=(%s <= %s <= %s)", cmiw, cw,
+ cmxw, cmih, ch, cmxh);
+ const nsMargin m = aState->ComputedPhysicalOffsets();
+ DR_state->PrintMargin("co", &m);
+ putchar('\n');
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+/* static */
+void* SizeComputationInput::DisplayInitOffsetsEnter(
+ nsIFrame* aFrame, SizeComputationInput* aState, nscoord aPercentBasis,
+ WritingMode aCBWritingMode, const nsMargin* aBorder,
+ const nsMargin* aPadding) {
+ MOZ_ASSERT(aFrame, "non-null frame required");
+ MOZ_ASSERT(aState, "non-null state required");
+
+ if (!DR_state->mInited) DR_state->Init();
+ if (!DR_state->mActive) return nullptr;
+
+ // aState is not necessarily a ReflowInput
+ DR_FrameTreeNode* treeNode = DR_state->CreateTreeNode(aFrame, nullptr);
+ if (treeNode && treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+
+ char pctBasisStr[16];
+ DR_state->PrettyUC(aPercentBasis, pctBasisStr, 16);
+ printf("InitOffsets pct_basis=%s", pctBasisStr);
+
+ DR_state->PrintMargin("b", aBorder);
+ DR_state->PrintMargin("p", aPadding);
+ putchar('\n');
+ }
+ return treeNode;
+}
+
+/* static */
+void SizeComputationInput::DisplayInitOffsetsExit(nsIFrame* aFrame,
+ SizeComputationInput* aState,
+ void* aValue) {
+ MOZ_ASSERT(aFrame, "non-null frame required");
+ MOZ_ASSERT(aState, "non-null state required");
+
+ if (!DR_state->mActive) return;
+ if (!aValue) return;
+
+ DR_FrameTreeNode* treeNode = (DR_FrameTreeNode*)aValue;
+ if (treeNode->mDisplay) {
+ DR_state->DisplayFrameTypeInfo(aFrame, treeNode->mIndent);
+ printf("InitOffsets=");
+ const auto m = aState->ComputedPhysicalMargin();
+ DR_state->PrintMargin("m", &m);
+ const auto p = aState->ComputedPhysicalPadding();
+ DR_state->PrintMargin("p", &p);
+ const auto bp = aState->ComputedPhysicalBorderPadding();
+ DR_state->PrintMargin("b+p", &bp);
+ putchar('\n');
+ }
+ DR_state->DeleteTreeNode(*treeNode);
+}
+
+// End Display Reflow
+
+// Validation of SideIsVertical.
+# define CASE(side, result) \
+ static_assert(SideIsVertical(side) == result, "SideIsVertical is wrong")
+CASE(eSideTop, false);
+CASE(eSideRight, true);
+CASE(eSideBottom, false);
+CASE(eSideLeft, true);
+# undef CASE
+
+// Validation of HalfCornerIsX.
+# define CASE(corner, result) \
+ static_assert(HalfCornerIsX(corner) == result, "HalfCornerIsX is wrong")
+CASE(eCornerTopLeftX, true);
+CASE(eCornerTopLeftY, false);
+CASE(eCornerTopRightX, true);
+CASE(eCornerTopRightY, false);
+CASE(eCornerBottomRightX, true);
+CASE(eCornerBottomRightY, false);
+CASE(eCornerBottomLeftX, true);
+CASE(eCornerBottomLeftY, false);
+# undef CASE
+
+// Validation of HalfToFullCorner.
+# define CASE(corner, result) \
+ static_assert(HalfToFullCorner(corner) == result, \
+ "HalfToFullCorner is " \
+ "wrong")
+CASE(eCornerTopLeftX, eCornerTopLeft);
+CASE(eCornerTopLeftY, eCornerTopLeft);
+CASE(eCornerTopRightX, eCornerTopRight);
+CASE(eCornerTopRightY, eCornerTopRight);
+CASE(eCornerBottomRightX, eCornerBottomRight);
+CASE(eCornerBottomRightY, eCornerBottomRight);
+CASE(eCornerBottomLeftX, eCornerBottomLeft);
+CASE(eCornerBottomLeftY, eCornerBottomLeft);
+# undef CASE
+
+// Validation of FullToHalfCorner.
+# define CASE(corner, vert, result) \
+ static_assert(FullToHalfCorner(corner, vert) == result, \
+ "FullToHalfCorner is wrong")
+CASE(eCornerTopLeft, false, eCornerTopLeftX);
+CASE(eCornerTopLeft, true, eCornerTopLeftY);
+CASE(eCornerTopRight, false, eCornerTopRightX);
+CASE(eCornerTopRight, true, eCornerTopRightY);
+CASE(eCornerBottomRight, false, eCornerBottomRightX);
+CASE(eCornerBottomRight, true, eCornerBottomRightY);
+CASE(eCornerBottomLeft, false, eCornerBottomLeftX);
+CASE(eCornerBottomLeft, true, eCornerBottomLeftY);
+# undef CASE
+
+// Validation of SideToFullCorner.
+# define CASE(side, second, result) \
+ static_assert(SideToFullCorner(side, second) == result, \
+ "SideToFullCorner is wrong")
+CASE(eSideTop, false, eCornerTopLeft);
+CASE(eSideTop, true, eCornerTopRight);
+
+CASE(eSideRight, false, eCornerTopRight);
+CASE(eSideRight, true, eCornerBottomRight);
+
+CASE(eSideBottom, false, eCornerBottomRight);
+CASE(eSideBottom, true, eCornerBottomLeft);
+
+CASE(eSideLeft, false, eCornerBottomLeft);
+CASE(eSideLeft, true, eCornerTopLeft);
+# undef CASE
+
+// Validation of SideToHalfCorner.
+# define CASE(side, second, parallel, result) \
+ static_assert(SideToHalfCorner(side, second, parallel) == result, \
+ "SideToHalfCorner is wrong")
+CASE(eSideTop, false, true, eCornerTopLeftX);
+CASE(eSideTop, false, false, eCornerTopLeftY);
+CASE(eSideTop, true, true, eCornerTopRightX);
+CASE(eSideTop, true, false, eCornerTopRightY);
+
+CASE(eSideRight, false, false, eCornerTopRightX);
+CASE(eSideRight, false, true, eCornerTopRightY);
+CASE(eSideRight, true, false, eCornerBottomRightX);
+CASE(eSideRight, true, true, eCornerBottomRightY);
+
+CASE(eSideBottom, false, true, eCornerBottomRightX);
+CASE(eSideBottom, false, false, eCornerBottomRightY);
+CASE(eSideBottom, true, true, eCornerBottomLeftX);
+CASE(eSideBottom, true, false, eCornerBottomLeftY);
+
+CASE(eSideLeft, false, false, eCornerBottomLeftX);
+CASE(eSideLeft, false, true, eCornerBottomLeftY);
+CASE(eSideLeft, true, false, eCornerTopLeftX);
+CASE(eSideLeft, true, true, eCornerTopLeftY);
+# undef CASE
+
+#endif