mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-23 04:41:11 +00:00
e263983c59
Differential Revision: https://phabricator.services.mozilla.com/D229245
11988 lines
477 KiB
C++
11988 lines
477 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/*
|
|
* construction of a frame tree that is nearly isomorphic to the content
|
|
* tree and updating of that tree in response to dynamic changes
|
|
*/
|
|
|
|
#include "nsCSSFrameConstructor.h"
|
|
|
|
#include "ActiveLayerTracker.h"
|
|
#include "ChildIterator.h"
|
|
#include "mozilla/AutoRestore.h"
|
|
#include "mozilla/ComputedStyleInlines.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "mozilla/dom/BindContext.h"
|
|
#include "mozilla/dom/BrowsingContext.h"
|
|
#include "mozilla/dom/CharacterData.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/dom/DocumentInlines.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/ElementInlines.h"
|
|
#include "mozilla/dom/GeneratedImageContent.h"
|
|
#include "mozilla/dom/HTMLInputElement.h"
|
|
#include "mozilla/dom/HTMLSelectElement.h"
|
|
#include "mozilla/dom/HTMLSharedListElement.h"
|
|
#include "mozilla/dom/HTMLSummaryElement.h"
|
|
#include "mozilla/ErrorResult.h"
|
|
#include "mozilla/Likely.h"
|
|
#include "mozilla/LinkedList.h"
|
|
#include "mozilla/ManualNAC.h"
|
|
#include "mozilla/MemoryReporting.h"
|
|
#include "mozilla/PresShell.h"
|
|
#include "mozilla/PresShellInlines.h"
|
|
#include "mozilla/PrintedSheetFrame.h"
|
|
#include "mozilla/ProfilerLabels.h"
|
|
#include "mozilla/ProfilerMarkers.h"
|
|
#include "mozilla/RestyleManager.h"
|
|
#include "mozilla/ScopeExit.h"
|
|
#include "mozilla/ScrollContainerFrame.h"
|
|
#include "mozilla/ServoBindings.h"
|
|
#include "mozilla/ServoStyleSetInlines.h"
|
|
#include "mozilla/StaticPrefs_browser.h"
|
|
#include "mozilla/StaticPrefs_layout.h"
|
|
#include "mozilla/StaticPrefs_mathml.h"
|
|
#include "mozilla/SVGGradientFrame.h"
|
|
#include "mozilla/Unused.h"
|
|
#include "nsAbsoluteContainingBlock.h"
|
|
#include "nsAtom.h"
|
|
#include "nsAutoLayoutPhase.h"
|
|
#include "nsBackdropFrame.h"
|
|
#include "nsBlockFrame.h"
|
|
#include "nsCanvasFrame.h"
|
|
#include "nsCheckboxRadioFrame.h"
|
|
#include "nsComboboxControlFrame.h"
|
|
#include "nsContainerFrame.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsCRT.h"
|
|
#include "nsCSSAnonBoxes.h"
|
|
#include "nsCSSPseudoElements.h"
|
|
#include "nsError.h"
|
|
#include "nsFieldSetFrame.h"
|
|
#include "nsFirstLetterFrame.h"
|
|
#include "nsFlexContainerFrame.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsGridContainerFrame.h"
|
|
#include "nsHTMLParts.h"
|
|
#include "nsIAnonymousContentCreator.h"
|
|
#include "nsIFormControl.h"
|
|
#include "nsIFrameInlines.h"
|
|
#include "nsImageFrame.h"
|
|
#include "nsInlineFrame.h"
|
|
#include "nsIObjectLoadingContent.h"
|
|
#include "nsIPopupContainer.h"
|
|
#include "nsIScriptError.h"
|
|
#include "nsLayoutUtils.h"
|
|
#include "nsListControlFrame.h"
|
|
#include "nsMathMLParts.h"
|
|
#include "nsNameSpaceManager.h"
|
|
#include "nsPageContentFrame.h"
|
|
#include "nsPageFrame.h"
|
|
#include "nsPageSequenceFrame.h"
|
|
#include "nsPlaceholderFrame.h"
|
|
#include "nsPresContext.h"
|
|
#include "nsRefreshDriver.h"
|
|
#include "nsRubyBaseContainerFrame.h"
|
|
#include "nsRubyBaseFrame.h"
|
|
#include "nsRubyFrame.h"
|
|
#include "nsRubyTextContainerFrame.h"
|
|
#include "nsRubyTextFrame.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsStyleStructInlines.h"
|
|
#include "nsTableCellFrame.h"
|
|
#include "nsTableColFrame.h"
|
|
#include "nsTableFrame.h"
|
|
#include "nsTableRowFrame.h"
|
|
#include "nsTableRowGroupFrame.h"
|
|
#include "nsTableWrapperFrame.h"
|
|
#include "nsTArray.h"
|
|
#include "nsTextFragment.h"
|
|
#include "nsTextNode.h"
|
|
#include "nsTransitionManager.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "nsViewManager.h"
|
|
#include "nsXULElement.h"
|
|
#include "RetainedDisplayListBuilder.h"
|
|
#include "RubyUtils.h"
|
|
#include "StickyScrollContainer.h"
|
|
|
|
#ifdef XP_MACOSX
|
|
# include "nsIDocShell.h"
|
|
#endif
|
|
|
|
#ifdef ACCESSIBILITY
|
|
# include "nsAccessibilityService.h"
|
|
#endif
|
|
|
|
#undef NOISY_FIRST_LETTER
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
nsIFrame* NS_NewHTMLCanvasFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewHTMLVideoFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewHTMLAudioFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsContainerFrame* NS_NewSVGOuterSVGFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsContainerFrame* NS_NewSVGOuterSVGAnonChildFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGInnerSVGFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGGeometryFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGGFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsContainerFrame* NS_NewSVGForeignObjectFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGAFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGSwitchFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGSymbolFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGTextFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGContainerFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGUseFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGViewFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
extern nsIFrame* NS_NewSVGLinearGradientFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
extern nsIFrame* NS_NewSVGRadialGradientFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
extern nsIFrame* NS_NewSVGStopFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsContainerFrame* NS_NewSVGMarkerFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsContainerFrame* NS_NewSVGMarkerAnonChildFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
extern nsIFrame* NS_NewSVGImageFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGClipPathFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGFilterFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGPatternFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGMaskFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGFEContainerFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGFELeafFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGFEImageFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewSVGFEUnstyledLeafFrame(PresShell* aPresShell,
|
|
ComputedStyle* aStyle);
|
|
nsIFrame* NS_NewFileControlLabelFrame(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewComboboxLabelFrame(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewMiddleCroppingLabelFrame(PresShell*, ComputedStyle*);
|
|
|
|
#include "mozilla/dom/NodeInfo.h"
|
|
#include "prenv.h"
|
|
#include "nsNodeInfoManager.h"
|
|
#include "nsContentCreatorFunctions.h"
|
|
|
|
#ifdef DEBUG
|
|
// Set the environment variable GECKO_FRAMECTOR_DEBUG_FLAGS to one or
|
|
// more of the following flags (comma separated) for handy debug
|
|
// output.
|
|
static bool gNoisyContentUpdates = false;
|
|
static bool gReallyNoisyContentUpdates = false;
|
|
static bool gNoisyInlineConstruction = false;
|
|
|
|
struct FrameCtorDebugFlags {
|
|
const char* name;
|
|
bool* on;
|
|
};
|
|
|
|
static FrameCtorDebugFlags gFlags[] = {
|
|
{"content-updates", &gNoisyContentUpdates},
|
|
{"really-noisy-content-updates", &gReallyNoisyContentUpdates},
|
|
{"noisy-inline", &gNoisyInlineConstruction}};
|
|
|
|
# define NUM_DEBUG_FLAGS (sizeof(gFlags) / sizeof(gFlags[0]))
|
|
#endif
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
nsIFrame* NS_NewLeafBoxFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewRangeFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewTextBoxFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewSplitterFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewMenuPopupFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewTreeBodyFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewSliderFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewScrollbarFrame(PresShell* aPresShell, ComputedStyle* aStyle);
|
|
|
|
nsIFrame* NS_NewScrollbarButtonFrame(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewSimpleXULLeafFrame(PresShell*, ComputedStyle*);
|
|
|
|
nsIFrame* NS_NewXULImageFrame(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewImageFrameForContentProperty(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewImageFrameForGeneratedContentIndex(PresShell*, ComputedStyle*);
|
|
nsIFrame* NS_NewImageFrameForListStyleImage(PresShell*, ComputedStyle*);
|
|
|
|
// Returns true if aFrame is an anonymous flex/grid item.
|
|
static inline bool IsAnonymousItem(const nsIFrame* aFrame) {
|
|
return aFrame->Style()->GetPseudoType() == PseudoStyleType::anonymousItem;
|
|
}
|
|
|
|
// Returns true IFF the given nsIFrame is a nsFlexContainerFrame and represents
|
|
// a -webkit-{inline-}box container.
|
|
static inline bool IsFlexContainerForLegacyWebKitBox(const nsIFrame* aFrame) {
|
|
return aFrame->IsFlexContainerFrame() &&
|
|
aFrame->HasAnyStateBits(NS_STATE_FLEX_IS_EMULATING_LEGACY_WEBKIT_BOX);
|
|
}
|
|
|
|
#if DEBUG
|
|
static void AssertAnonymousFlexOrGridItemParent(const nsIFrame* aChild,
|
|
const nsIFrame* aParent) {
|
|
MOZ_ASSERT(IsAnonymousItem(aChild), "expected an anonymous item child frame");
|
|
MOZ_ASSERT(aParent, "expected a parent frame");
|
|
MOZ_ASSERT(aParent->IsFlexOrGridContainer(),
|
|
"anonymous items should only exist as children of flex/grid "
|
|
"container frames");
|
|
}
|
|
#else
|
|
# define AssertAnonymousFlexOrGridItemParent(x, y) PR_BEGIN_MACRO PR_END_MACRO
|
|
#endif
|
|
|
|
#define ToCreationFunc(_func) \
|
|
[](PresShell* aPs, ComputedStyle* aStyle) -> nsIFrame* { \
|
|
return _func(aPs, aStyle); \
|
|
}
|
|
|
|
/**
|
|
* True if aFrame is an actual inline frame in the sense of non-replaced
|
|
* display:inline CSS boxes. In other words, it can be affected by {ib}
|
|
* splitting and can contain first-letter frames. Basically, this is either an
|
|
* inline frame (positioned or otherwise) or an line frame (this last because
|
|
* it can contain first-letter and because inserting blocks in the middle of it
|
|
* needs to terminate it).
|
|
*/
|
|
static bool IsInlineFrame(const nsIFrame* aFrame) {
|
|
return aFrame->IsLineParticipant();
|
|
}
|
|
|
|
/**
|
|
* True for display: contents elements.
|
|
*/
|
|
static inline bool IsDisplayContents(const Element* aElement) {
|
|
return aElement->IsDisplayContents();
|
|
}
|
|
|
|
static inline bool IsDisplayContents(const nsIContent* aContent) {
|
|
return aContent->IsElement() && IsDisplayContents(aContent->AsElement());
|
|
}
|
|
|
|
/**
|
|
* True if aFrame is an instance of an SVG frame class or is an inline/block
|
|
* frame being used for SVG text.
|
|
*/
|
|
static bool IsFrameForSVG(const nsIFrame* aFrame) {
|
|
return aFrame->IsSVGFrame() || aFrame->IsInSVGTextSubtree();
|
|
}
|
|
|
|
static bool IsLastContinuationForColumnContent(const nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame);
|
|
return aFrame->Style()->GetPseudoType() == PseudoStyleType::columnContent &&
|
|
!aFrame->GetNextContinuation();
|
|
}
|
|
|
|
/**
|
|
* Returns true iff aFrame explicitly prevents its descendants from floating
|
|
* (at least, down to the level of descendants which themselves are
|
|
* float-containing blocks -- those will manage the floating status of any
|
|
* lower-level descendents inside them, of course).
|
|
*/
|
|
static bool ShouldSuppressFloatingOfDescendants(nsIFrame* aFrame) {
|
|
return aFrame->IsFlexOrGridContainer() || aFrame->IsMathMLFrame();
|
|
}
|
|
|
|
// Return true if column-span descendants should be suppressed under aFrame's
|
|
// subtree (until a multi-column container re-establishing a block formatting
|
|
// context). Basically, this is testing whether aFrame establishes a new block
|
|
// formatting context or not.
|
|
static bool ShouldSuppressColumnSpanDescendants(nsIFrame* aFrame) {
|
|
if (aFrame->Style()->GetPseudoType() == PseudoStyleType::columnContent) {
|
|
// Never suppress column-span under ::-moz-column-content frames.
|
|
return false;
|
|
}
|
|
|
|
if (aFrame->IsInlineFrame()) {
|
|
// Allow inline frames to have column-span block children.
|
|
return false;
|
|
}
|
|
|
|
if (!aFrame->IsBlockFrameOrSubclass() ||
|
|
aFrame->HasAnyStateBits(NS_BLOCK_BFC | NS_FRAME_OUT_OF_FLOW) ||
|
|
aFrame->IsFixedPosContainingBlock()) {
|
|
// Need to suppress column-span if we:
|
|
// - Are a different block formatting context,
|
|
// - Are an out-of-flow frame, OR
|
|
// - Establish a containing block for fixed-position descendants
|
|
//
|
|
// For example, the children of a column-span never need to be further
|
|
// processed even if there is a nested column-span child. Because a
|
|
// column-span always creates its own block formatting context, a nested
|
|
// column-span child won't be in the same block formatting context with the
|
|
// nearest multi-column ancestor. This is the same case as if the
|
|
// column-span is outside of a multi-column hierarchy.
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Reparent a frame into a wrapper frame that is a child of its old parent.
|
|
static void ReparentFrame(RestyleManager* aRestyleManager,
|
|
nsContainerFrame* aNewParentFrame, nsIFrame* aFrame,
|
|
bool aForceStyleReparent) {
|
|
aFrame->SetParent(aNewParentFrame);
|
|
// We reparent frames for two reasons: to put them inside ::first-line, and to
|
|
// put them inside some wrapper anonymous boxes.
|
|
if (aForceStyleReparent) {
|
|
aRestyleManager->ReparentComputedStyleForFirstLine(aFrame);
|
|
}
|
|
}
|
|
|
|
static void ReparentFrames(nsCSSFrameConstructor* aFrameConstructor,
|
|
nsContainerFrame* aNewParentFrame,
|
|
const nsFrameList& aFrameList,
|
|
bool aForceStyleReparent) {
|
|
RestyleManager* restyleManager = aFrameConstructor->RestyleManager();
|
|
for (nsIFrame* f : aFrameList) {
|
|
ReparentFrame(restyleManager, aNewParentFrame, f, aForceStyleReparent);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
//
|
|
// When inline frames get weird and have block frames in them, we
|
|
// annotate them to help us respond to incremental content changes
|
|
// more easily.
|
|
|
|
static inline bool IsFramePartOfIBSplit(nsIFrame* aFrame) {
|
|
bool result = aFrame->HasAnyStateBits(NS_FRAME_PART_OF_IBSPLIT);
|
|
MOZ_ASSERT(!result || static_cast<nsBlockFrame*>(do_QueryFrame(aFrame)) ||
|
|
static_cast<nsInlineFrame*>(do_QueryFrame(aFrame)),
|
|
"only block/inline frames can have NS_FRAME_PART_OF_IBSPLIT");
|
|
return result;
|
|
}
|
|
|
|
static nsContainerFrame* GetIBSplitSibling(nsIFrame* aFrame) {
|
|
MOZ_ASSERT(IsFramePartOfIBSplit(aFrame), "Shouldn't call this");
|
|
|
|
// We only store the "ib-split sibling" annotation with the first
|
|
// frame in the continuation chain. Walk back to find that frame now.
|
|
return aFrame->FirstContinuation()->GetProperty(nsIFrame::IBSplitSibling());
|
|
}
|
|
|
|
static nsContainerFrame* GetIBSplitPrevSibling(nsIFrame* aFrame) {
|
|
MOZ_ASSERT(IsFramePartOfIBSplit(aFrame), "Shouldn't call this");
|
|
|
|
// We only store the ib-split sibling annotation with the first
|
|
// frame in the continuation chain. Walk back to find that frame now.
|
|
return aFrame->FirstContinuation()->GetProperty(
|
|
nsIFrame::IBSplitPrevSibling());
|
|
}
|
|
|
|
static nsContainerFrame* GetLastIBSplitSibling(nsIFrame* aFrame) {
|
|
for (nsIFrame *frame = aFrame, *next;; frame = next) {
|
|
next = GetIBSplitSibling(frame);
|
|
if (!next) {
|
|
return static_cast<nsContainerFrame*>(frame);
|
|
}
|
|
}
|
|
MOZ_ASSERT_UNREACHABLE("unreachable code");
|
|
return nullptr;
|
|
}
|
|
|
|
static void SetFrameIsIBSplit(nsContainerFrame* aFrame,
|
|
nsContainerFrame* aIBSplitSibling) {
|
|
MOZ_ASSERT(aFrame, "bad args!");
|
|
|
|
// We should be the only continuation
|
|
NS_ASSERTION(!aFrame->GetPrevContinuation(),
|
|
"assigning ib-split sibling to other than first continuation!");
|
|
NS_ASSERTION(!aFrame->GetNextContinuation() ||
|
|
IsFramePartOfIBSplit(aFrame->GetNextContinuation()),
|
|
"should have no non-ib-split continuations here");
|
|
|
|
// Mark the frame as ib-split.
|
|
aFrame->AddStateBits(NS_FRAME_PART_OF_IBSPLIT);
|
|
|
|
if (aIBSplitSibling) {
|
|
NS_ASSERTION(!aIBSplitSibling->GetPrevContinuation(),
|
|
"assigning something other than the first continuation as the "
|
|
"ib-split sibling");
|
|
|
|
// Store the ib-split sibling (if we were given one) with the
|
|
// first frame in the flow.
|
|
aFrame->SetProperty(nsIFrame::IBSplitSibling(), aIBSplitSibling);
|
|
aIBSplitSibling->SetProperty(nsIFrame::IBSplitPrevSibling(), aFrame);
|
|
}
|
|
}
|
|
|
|
static nsIFrame* GetIBContainingBlockFor(nsIFrame* aFrame) {
|
|
MOZ_ASSERT(
|
|
IsFramePartOfIBSplit(aFrame),
|
|
"GetIBContainingBlockFor() should only be called on known IB frames");
|
|
|
|
// Get the first "normal" ancestor of the target frame.
|
|
nsIFrame* parentFrame;
|
|
do {
|
|
parentFrame = aFrame->GetParent();
|
|
|
|
if (!parentFrame) {
|
|
NS_ERROR("no unsplit block frame in IB hierarchy");
|
|
return aFrame;
|
|
}
|
|
|
|
// Note that we ignore non-ib-split frames which have a pseudo on their
|
|
// ComputedStyle -- they're not the frames we're looking for! In
|
|
// particular, they may be hiding a real parent that _is_ in an ib-split.
|
|
if (!IsFramePartOfIBSplit(parentFrame) &&
|
|
!parentFrame->Style()->IsPseudoOrAnonBox()) {
|
|
break;
|
|
}
|
|
|
|
aFrame = parentFrame;
|
|
} while (true);
|
|
|
|
// post-conditions
|
|
NS_ASSERTION(parentFrame,
|
|
"no normal ancestor found for ib-split frame "
|
|
"in GetIBContainingBlockFor");
|
|
NS_ASSERTION(parentFrame != aFrame,
|
|
"parentFrame is actually the child frame - bogus reslt");
|
|
|
|
return parentFrame;
|
|
}
|
|
|
|
// Find the multicol containing block suitable for reframing.
|
|
//
|
|
// Note: this function may not return a ColumnSetWrapperFrame. For example, if
|
|
// the multicol containing block has "overflow:scroll" style,
|
|
// ScrollContainerFrame is returned because ColumnSetWrapperFrame is the
|
|
// scrolled frame which has the -moz-scrolled-content pseudo style. We may walk
|
|
// up "too far", but in terms of correctness of reframing, it's OK.
|
|
static nsContainerFrame* GetMultiColumnContainingBlockFor(nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR),
|
|
"Should only be called if the frame has a multi-column ancestor!");
|
|
|
|
nsContainerFrame* current = aFrame->GetParent();
|
|
while (current &&
|
|
(current->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR) ||
|
|
current->Style()->IsPseudoOrAnonBox())) {
|
|
current = current->GetParent();
|
|
}
|
|
|
|
MOZ_ASSERT(current,
|
|
"No multicol containing block in a valid column hierarchy?");
|
|
|
|
return current;
|
|
}
|
|
|
|
static bool InsertSeparatorBeforeAccessKey() {
|
|
static bool sInitialized = false;
|
|
static bool sValue = false;
|
|
if (!sInitialized) {
|
|
sInitialized = true;
|
|
|
|
const char* prefName = "intl.menuitems.insertseparatorbeforeaccesskeys";
|
|
nsAutoString val;
|
|
Preferences::GetLocalizedString(prefName, val);
|
|
sValue = val.EqualsLiteral("true");
|
|
}
|
|
return sValue;
|
|
}
|
|
|
|
static bool AlwaysAppendAccessKey() {
|
|
static bool sInitialized = false;
|
|
static bool sValue = false;
|
|
if (!sInitialized) {
|
|
sInitialized = true;
|
|
const char* prefName = "intl.menuitems.alwaysappendaccesskeys";
|
|
nsAutoString val;
|
|
Preferences::GetLocalizedString(prefName, val);
|
|
sValue = val.EqualsLiteral("true");
|
|
}
|
|
return sValue;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Block/inline frame construction logic. We maintain a few invariants here:
|
|
//
|
|
// 1. Block frames contain block and inline frames.
|
|
//
|
|
// 2. Inline frames only contain inline frames. If an inline parent has a block
|
|
// child then the block child is migrated upward until it lands in a block
|
|
// parent (the inline frames containing block is where it will end up).
|
|
|
|
inline void SetInitialSingleChild(nsContainerFrame* aParent, nsIFrame* aFrame) {
|
|
MOZ_ASSERT(!aFrame->GetNextSibling(), "Should be using a frame list");
|
|
aParent->SetInitialChildList(FrameChildListID::Principal,
|
|
nsFrameList(aFrame, aFrame));
|
|
}
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
// Structure used when constructing formatting object trees. Contains
|
|
// state information needed for absolutely positioned elements
|
|
namespace mozilla {
|
|
struct AbsoluteFrameList final : public nsFrameList {
|
|
// Containing block for absolutely positioned elements.
|
|
nsContainerFrame* mContainingBlock;
|
|
|
|
explicit AbsoluteFrameList(nsContainerFrame* aContainingBlock = nullptr)
|
|
: mContainingBlock(aContainingBlock) {}
|
|
|
|
// Transfer frames in aOther to this list. aOther becomes empty after this
|
|
// operation.
|
|
AbsoluteFrameList(AbsoluteFrameList&& aOther) = default;
|
|
AbsoluteFrameList& operator=(AbsoluteFrameList&& aOther) = default;
|
|
|
|
#ifdef DEBUG
|
|
// XXXbz Does this need a debug-only assignment operator that nulls out the
|
|
// childList in the AbsoluteFrameList we're copying? Introducing a difference
|
|
// between debug and non-debug behavior seems bad, so I guess not...
|
|
~AbsoluteFrameList() {
|
|
NS_ASSERTION(!FirstChild(),
|
|
"Dangling child list. Someone forgot to insert it?");
|
|
}
|
|
#endif
|
|
};
|
|
} // namespace mozilla
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
// Structure for saving the existing state when pushing/poping containing
|
|
// blocks. The destructor restores the state to its previous state
|
|
class MOZ_STACK_CLASS nsFrameConstructorSaveState {
|
|
public:
|
|
~nsFrameConstructorSaveState();
|
|
|
|
private:
|
|
// Pointer to struct whose data we save/restore.
|
|
AbsoluteFrameList* mList = nullptr;
|
|
|
|
// The saved pointer to the fixed list.
|
|
AbsoluteFrameList* mSavedFixedList = nullptr;
|
|
|
|
// Copy of original frame list. This can be the original absolute list or a
|
|
// float list.
|
|
AbsoluteFrameList mSavedList;
|
|
|
|
// The name of the child list in which our frames would belong.
|
|
mozilla::FrameChildListID mChildListID = FrameChildListID::Principal;
|
|
nsFrameConstructorState* mState = nullptr;
|
|
|
|
friend class nsFrameConstructorState;
|
|
};
|
|
|
|
// Structure used for maintaining state information during the
|
|
// frame construction process
|
|
class MOZ_STACK_CLASS nsFrameConstructorState {
|
|
public:
|
|
nsPresContext* mPresContext;
|
|
PresShell* mPresShell;
|
|
nsCSSFrameConstructor* mFrameConstructor;
|
|
|
|
// Containing block information for out-of-flow frames.
|
|
//
|
|
// Floats are easy. Whatever is our float CB.
|
|
//
|
|
// Regular abspos elements are easy too. Its containing block can be the
|
|
// nearest abspos element, or the ICB (the canvas frame).
|
|
//
|
|
// Top layer abspos elements are always children of the ICB, but we can get
|
|
// away with having two different lists (mAbsoluteList and
|
|
// mTopLayerAbsoluteList), because because top layer frames cause
|
|
// non-top-layer frames to be contained inside (so any descendants of a top
|
|
// layer abspos can never share containing block with it, unless they're also
|
|
// in the top layer).
|
|
//
|
|
// Regular fixed elements however are trickier. Fixed elements can be
|
|
// contained in one of three lists:
|
|
//
|
|
// * mAbsoluteList, if our abspos cb is also a fixpos cb (e.g., is
|
|
// transformed or has a filter).
|
|
//
|
|
// * mAncestorFixedList, if the fixpos cb is an ancestor element other than
|
|
// the viewport frame, (so, a transformed / filtered
|
|
// ancestor).
|
|
//
|
|
// * mRealFixedList, which is also the fixed list used for the top layer
|
|
// fixed items, which is the fixed list of the viewport
|
|
// frame.
|
|
//
|
|
// It is important that mRealFixedList is shared between regular and top layer
|
|
// fixpos elements, since no-top-layer descendants of top layer fixed elements
|
|
// could share ICB and vice versa, so without that there would be no guarantee
|
|
// of layout ordering between them.
|
|
AbsoluteFrameList mFloatedList;
|
|
AbsoluteFrameList mAbsoluteList;
|
|
AbsoluteFrameList mTopLayerAbsoluteList;
|
|
AbsoluteFrameList mAncestorFixedList;
|
|
AbsoluteFrameList mRealFixedList;
|
|
|
|
// Never null, always pointing to one of the lists documented above.
|
|
AbsoluteFrameList* mFixedList;
|
|
|
|
// What `page: auto` resolves to. This is the used page-name of the parent
|
|
// frame. Updated by AutoFrameConstructionPageName.
|
|
const nsAtom* mAutoPageNameValue = nullptr;
|
|
|
|
nsCOMPtr<nsILayoutHistoryState> mFrameState;
|
|
// These bits will be added to the state bits of any frame we construct
|
|
// using this state.
|
|
nsFrameState mAdditionalStateBits{0};
|
|
|
|
// If false (which is the default) then call SetPrimaryFrame() as needed
|
|
// during frame construction. If true, don't make any SetPrimaryFrame()
|
|
// calls, except for generated content which doesn't have a primary frame
|
|
// yet. The mCreatingExtraFrames == true mode is meant to be used for
|
|
// construction of random "extra" frames for elements via normal frame
|
|
// construction APIs (e.g. replication of things across pages in paginated
|
|
// mode).
|
|
bool mCreatingExtraFrames;
|
|
|
|
// This keeps track of whether we have found a "rendered legend" for
|
|
// the current FieldSetFrame.
|
|
bool mHasRenderedLegend;
|
|
|
|
nsTArray<RefPtr<nsIContent>> mGeneratedContentWithInitializer;
|
|
|
|
#ifdef DEBUG
|
|
// Record the float containing block candidate passed into
|
|
// MaybePushFloatContainingBlock() to keep track that we've call the method to
|
|
// handle the float CB scope before processing the CB's children. It is reset
|
|
// in ConstructFramesFromItemList().
|
|
nsContainerFrame* mFloatCBCandidate = nullptr;
|
|
#endif
|
|
|
|
// Constructor
|
|
// Use the passed-in history state.
|
|
nsFrameConstructorState(
|
|
PresShell* aPresShell, nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock,
|
|
already_AddRefed<nsILayoutHistoryState> aHistoryState);
|
|
// Get the history state from the pres context's pres shell.
|
|
nsFrameConstructorState(PresShell* aPresShell,
|
|
nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock);
|
|
|
|
~nsFrameConstructorState();
|
|
|
|
// Process the frame insertions for all the out-of-flow nsAbsoluteItems.
|
|
void ProcessFrameInsertionsForAllLists();
|
|
|
|
// Function to push the existing absolute containing block state and
|
|
// create a new scope. Code that uses this function should get matching
|
|
// logic in GetAbsoluteContainingBlock.
|
|
// Also makes aNewAbsoluteContainingBlock the containing block for
|
|
// fixed-pos elements if necessary.
|
|
// aPositionedFrame is the frame whose style actually makes
|
|
// aNewAbsoluteContainingBlock a containing block. E.g. for a scrollable
|
|
// element aPositionedFrame is the element's primary frame and
|
|
// aNewAbsoluteContainingBlock is the scrolled frame.
|
|
void PushAbsoluteContainingBlock(
|
|
nsContainerFrame* aNewAbsoluteContainingBlock, nsIFrame* aPositionedFrame,
|
|
nsFrameConstructorSaveState& aSaveState);
|
|
|
|
// Function to forbid floats descendants under aFloatCBCandidate, or open a
|
|
// new float containing block scope for aFloatCBCandidate. The current
|
|
// state is saved in aSaveState if a new scope is pushed.
|
|
void MaybePushFloatContainingBlock(nsContainerFrame* aFloatCBCandidate,
|
|
nsFrameConstructorSaveState& aSaveState);
|
|
|
|
// Helper function for MaybePushFloatContainingBlock().
|
|
void PushFloatContainingBlock(nsContainerFrame* aNewFloatContainingBlock,
|
|
nsFrameConstructorSaveState& aSaveState);
|
|
|
|
// Function to return the proper geometric parent for a frame with display
|
|
// struct given by aStyleDisplay and parent's frame given by
|
|
// aContentParentFrame.
|
|
nsContainerFrame* GetGeometricParent(
|
|
const nsStyleDisplay& aStyleDisplay,
|
|
nsContainerFrame* aContentParentFrame) const;
|
|
|
|
// Collect absolute frames in mAbsoluteList which are proper descendants
|
|
// of aNewParent, and reparent them to aNewParent.
|
|
//
|
|
// Note: This function does something unusual that moves absolute items
|
|
// after their frames are constructed under a column hierarchy which has
|
|
// column-span elements. Do not use this if you're not dealing with
|
|
// columns.
|
|
void ReparentAbsoluteItems(nsContainerFrame* aNewParent);
|
|
|
|
// Collect floats in mFloatedList which are proper descendants of aNewParent,
|
|
// and reparent them to aNewParent.
|
|
//
|
|
// Note: This function does something unusual that moves floats after their
|
|
// frames are constructed under a column hierarchy which has column-span
|
|
// elements. Do not use this if you're not dealing with columns.
|
|
void ReparentFloats(nsContainerFrame* aNewParent);
|
|
|
|
/**
|
|
* Function to add a new frame to the right frame list. This MUST be called
|
|
* on frames before their children have been processed if the frames might
|
|
* conceivably be out-of-flow; otherwise cleanup in error cases won't work
|
|
* right. Also, this MUST be called on frames after they have been
|
|
* initialized.
|
|
* @param aNewFrame the frame to add
|
|
* @param aFrameList the list to add in-flow frames to
|
|
* @param aContent the content pointer for aNewFrame
|
|
* @param aParentFrame the parent frame for the content if it were in-flow
|
|
* @param aCanBePositioned pass false if the frame isn't allowed to be
|
|
* positioned
|
|
* @param aCanBeFloated pass false if the frame isn't allowed to be
|
|
* floated
|
|
*/
|
|
void AddChild(nsIFrame* aNewFrame, nsFrameList& aFrameList,
|
|
nsIContent* aContent, nsContainerFrame* aParentFrame,
|
|
bool aCanBePositioned = true, bool aCanBeFloated = true,
|
|
bool aInsertAfter = false,
|
|
nsIFrame* aInsertAfterFrame = nullptr);
|
|
|
|
/**
|
|
* Function to return the fixed-pos element list. Normally this will just
|
|
* hand back the fixed-pos element list, but in case we're dealing with a
|
|
* transformed element that's acting as an abs-pos and fixed-pos container,
|
|
* we'll hand back the abs-pos list. Callers should use this function if they
|
|
* want to get the list acting as the fixed-pos item parent.
|
|
*/
|
|
AbsoluteFrameList& GetFixedList() { return *mFixedList; }
|
|
const AbsoluteFrameList& GetFixedList() const { return *mFixedList; }
|
|
|
|
protected:
|
|
friend class nsFrameConstructorSaveState;
|
|
|
|
/**
|
|
* ProcessFrameInsertions takes the frames in aFrameList and adds them as
|
|
* kids to the aChildListID child list of |aFrameList.containingBlock|.
|
|
*/
|
|
void ProcessFrameInsertions(AbsoluteFrameList& aFrameList,
|
|
mozilla::FrameChildListID aChildListID);
|
|
|
|
/**
|
|
* GetOutOfFlowFrameList selects the out-of-flow frame list the new
|
|
* frame should be added to. If the frame shouldn't be added to any
|
|
* out-of-flow list, it returns nullptr. The corresponding type of
|
|
* placeholder is also returned via the aPlaceholderType parameter
|
|
* if this method doesn't return nullptr. The caller should check
|
|
* whether the returned list really has a containing block.
|
|
*/
|
|
AbsoluteFrameList* GetOutOfFlowFrameList(nsIFrame* aNewFrame,
|
|
bool aCanBePositioned,
|
|
bool aCanBeFloated,
|
|
nsFrameState* aPlaceholderType);
|
|
|
|
void ConstructBackdropFrameFor(nsIContent* aContent, nsIFrame* aFrame);
|
|
};
|
|
|
|
nsFrameConstructorState::nsFrameConstructorState(
|
|
PresShell* aPresShell, nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock,
|
|
already_AddRefed<nsILayoutHistoryState> aHistoryState)
|
|
: mPresContext(aPresShell->GetPresContext()),
|
|
mPresShell(aPresShell),
|
|
mFrameConstructor(aPresShell->FrameConstructor()),
|
|
mFloatedList(aFloatContainingBlock),
|
|
mAbsoluteList(aAbsoluteContainingBlock),
|
|
mTopLayerAbsoluteList(mFrameConstructor->GetCanvasFrame()),
|
|
mAncestorFixedList(aFixedContainingBlock),
|
|
mRealFixedList(
|
|
static_cast<nsContainerFrame*>(mFrameConstructor->GetRootFrame())),
|
|
// See PushAbsoluteContaningBlock below
|
|
mFrameState(aHistoryState),
|
|
mCreatingExtraFrames(false),
|
|
mHasRenderedLegend(false) {
|
|
MOZ_COUNT_CTOR(nsFrameConstructorState);
|
|
mFixedList = [&] {
|
|
if (aFixedContainingBlock == aAbsoluteContainingBlock) {
|
|
return &mAbsoluteList;
|
|
}
|
|
if (aAbsoluteContainingBlock == mRealFixedList.mContainingBlock) {
|
|
return &mRealFixedList;
|
|
}
|
|
return &mAncestorFixedList;
|
|
}();
|
|
}
|
|
|
|
nsFrameConstructorState::nsFrameConstructorState(
|
|
PresShell* aPresShell, nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock)
|
|
: nsFrameConstructorState(
|
|
aPresShell, aFixedContainingBlock, aAbsoluteContainingBlock,
|
|
aFloatContainingBlock,
|
|
aPresShell->GetDocument()->GetLayoutHistoryState()) {}
|
|
|
|
nsFrameConstructorState::~nsFrameConstructorState() {
|
|
MOZ_COUNT_DTOR(nsFrameConstructorState);
|
|
ProcessFrameInsertionsForAllLists();
|
|
for (auto& content : Reversed(mGeneratedContentWithInitializer)) {
|
|
content->RemoveProperty(nsGkAtoms::genConInitializerProperty);
|
|
}
|
|
}
|
|
|
|
void nsFrameConstructorState::ProcessFrameInsertionsForAllLists() {
|
|
ProcessFrameInsertions(mFloatedList, FrameChildListID::Float);
|
|
ProcessFrameInsertions(mAbsoluteList, FrameChildListID::Absolute);
|
|
ProcessFrameInsertions(mTopLayerAbsoluteList, FrameChildListID::Absolute);
|
|
ProcessFrameInsertions(*mFixedList, FrameChildListID::Fixed);
|
|
ProcessFrameInsertions(mRealFixedList, FrameChildListID::Fixed);
|
|
}
|
|
|
|
void nsFrameConstructorState::PushAbsoluteContainingBlock(
|
|
nsContainerFrame* aNewAbsoluteContainingBlock, nsIFrame* aPositionedFrame,
|
|
nsFrameConstructorSaveState& aSaveState) {
|
|
MOZ_ASSERT(!!aNewAbsoluteContainingBlock == !!aPositionedFrame,
|
|
"We should have both or none");
|
|
aSaveState.mList = &mAbsoluteList;
|
|
aSaveState.mChildListID = FrameChildListID::Absolute;
|
|
aSaveState.mState = this;
|
|
aSaveState.mSavedList = std::move(mAbsoluteList);
|
|
aSaveState.mSavedFixedList = mFixedList;
|
|
mAbsoluteList = AbsoluteFrameList(aNewAbsoluteContainingBlock);
|
|
mFixedList = [&] {
|
|
if (!aPositionedFrame || aPositionedFrame->IsFixedPosContainingBlock()) {
|
|
// See if we need to treat abspos and fixedpos the same. This happens if
|
|
// we're a transformed/filtered/etc element, or if we force a null abspos
|
|
// containing block (for mathml for example).
|
|
return &mAbsoluteList;
|
|
}
|
|
if (aPositionedFrame->StyleDisplay()->mTopLayer == StyleTopLayer::Auto) {
|
|
// If our new CB is in the top layer, and isn't a fixed CB itself, we also
|
|
// escape the usual containment.
|
|
return &mRealFixedList;
|
|
}
|
|
if (mFixedList == &mAbsoluteList) {
|
|
// If we were pointing to our old absolute list, keep pointing to it.
|
|
return &aSaveState.mSavedList;
|
|
}
|
|
// Otherwise keep pointing to the current thing (another ancestor's
|
|
// absolute list, or the real fixed list, doesn't matter).
|
|
return mFixedList;
|
|
}();
|
|
|
|
if (aNewAbsoluteContainingBlock) {
|
|
aNewAbsoluteContainingBlock->MarkAsAbsoluteContainingBlock();
|
|
}
|
|
}
|
|
|
|
void nsFrameConstructorState::MaybePushFloatContainingBlock(
|
|
nsContainerFrame* aFloatCBCandidate,
|
|
nsFrameConstructorSaveState& aSaveState) {
|
|
// The logic here needs to match the logic in GetFloatContainingBlock().
|
|
if (ShouldSuppressFloatingOfDescendants(aFloatCBCandidate)) {
|
|
// Pushing a null float containing block forbids any frames from being
|
|
// floated until a new float containing block is pushed. See implementation
|
|
// of nsFrameConstructorState::AddChild().
|
|
//
|
|
// XXX we should get rid of null float containing blocks and teach the
|
|
// various frame classes to deal with floats instead.
|
|
PushFloatContainingBlock(nullptr, aSaveState);
|
|
} else if (aFloatCBCandidate->IsFloatContainingBlock()) {
|
|
PushFloatContainingBlock(aFloatCBCandidate, aSaveState);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
mFloatCBCandidate = aFloatCBCandidate;
|
|
#endif
|
|
}
|
|
|
|
void nsFrameConstructorState::PushFloatContainingBlock(
|
|
nsContainerFrame* aNewFloatContainingBlock,
|
|
nsFrameConstructorSaveState& aSaveState) {
|
|
MOZ_ASSERT(!aNewFloatContainingBlock ||
|
|
aNewFloatContainingBlock->IsFloatContainingBlock(),
|
|
"Please push a real float containing block!");
|
|
NS_ASSERTION(
|
|
!aNewFloatContainingBlock ||
|
|
!ShouldSuppressFloatingOfDescendants(aNewFloatContainingBlock),
|
|
"We should not push a frame that is supposed to _suppress_ "
|
|
"floats as a float containing block!");
|
|
aSaveState.mList = &mFloatedList;
|
|
aSaveState.mSavedList = std::move(mFloatedList);
|
|
aSaveState.mChildListID = FrameChildListID::Float;
|
|
aSaveState.mState = this;
|
|
mFloatedList = AbsoluteFrameList(aNewFloatContainingBlock);
|
|
}
|
|
|
|
nsContainerFrame* nsFrameConstructorState::GetGeometricParent(
|
|
const nsStyleDisplay& aStyleDisplay,
|
|
nsContainerFrame* aContentParentFrame) const {
|
|
// If there is no container for a fixed, absolute, or floating root
|
|
// frame, we will ignore the positioning. This hack is originally
|
|
// brought to you by the letter T: tables, since other roots don't
|
|
// even call into this code. See bug 178855.
|
|
//
|
|
// XXX Disabling positioning in this case is a hack. If one was so inclined,
|
|
// one could support this either by (1) inserting a dummy block between the
|
|
// table and the canvas or (2) teaching the canvas how to reflow positioned
|
|
// elements. (1) has the usual problems when multiple frames share the same
|
|
// content (notice all the special cases in this file dealing with inner
|
|
// tables and table wrappers which share the same content). (2) requires some
|
|
// work and possible factoring.
|
|
//
|
|
// XXXbz couldn't we just force position to "static" on roots and
|
|
// float to "none"? That's OK per CSS 2.1, as far as I can tell.
|
|
|
|
if (aContentParentFrame && aContentParentFrame->IsInSVGTextSubtree()) {
|
|
return aContentParentFrame;
|
|
}
|
|
|
|
if (aStyleDisplay.IsFloatingStyle() && mFloatedList.mContainingBlock) {
|
|
NS_ASSERTION(!aStyleDisplay.IsAbsolutelyPositionedStyle(),
|
|
"Absolutely positioned _and_ floating?");
|
|
return mFloatedList.mContainingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay.mTopLayer != StyleTopLayer::None) {
|
|
MOZ_ASSERT(aStyleDisplay.mTopLayer == StyleTopLayer::Auto,
|
|
"-moz-top-layer should be either none or auto");
|
|
MOZ_ASSERT(aStyleDisplay.IsAbsolutelyPositionedStyle(),
|
|
"Top layer items should always be absolutely positioned");
|
|
if (aStyleDisplay.mPosition == StylePositionProperty::Fixed) {
|
|
MOZ_ASSERT(mRealFixedList.mContainingBlock, "No root frame?");
|
|
return mRealFixedList.mContainingBlock;
|
|
}
|
|
MOZ_ASSERT(aStyleDisplay.mPosition == StylePositionProperty::Absolute);
|
|
MOZ_ASSERT(mTopLayerAbsoluteList.mContainingBlock);
|
|
return mTopLayerAbsoluteList.mContainingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay.mPosition == StylePositionProperty::Absolute &&
|
|
mAbsoluteList.mContainingBlock) {
|
|
return mAbsoluteList.mContainingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay.mPosition == StylePositionProperty::Fixed &&
|
|
mFixedList->mContainingBlock) {
|
|
return mFixedList->mContainingBlock;
|
|
}
|
|
|
|
return aContentParentFrame;
|
|
}
|
|
|
|
void nsFrameConstructorState::ReparentAbsoluteItems(
|
|
nsContainerFrame* aNewParent) {
|
|
// Bug 1491727: This function might not conform to the spec. See
|
|
// https://github.com/w3c/csswg-drafts/issues/1894.
|
|
|
|
MOZ_ASSERT(aNewParent->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR),
|
|
"Restrict the usage under column hierarchy.");
|
|
|
|
AbsoluteFrameList newAbsoluteItems(aNewParent);
|
|
|
|
nsIFrame* current = mAbsoluteList.FirstChild();
|
|
while (current) {
|
|
nsIFrame* placeholder = current->GetPlaceholderFrame();
|
|
|
|
if (nsLayoutUtils::IsProperAncestorFrame(aNewParent, placeholder)) {
|
|
nsIFrame* next = current->GetNextSibling();
|
|
mAbsoluteList.RemoveFrame(current);
|
|
newAbsoluteItems.AppendFrame(aNewParent, current);
|
|
current = next;
|
|
} else {
|
|
current = current->GetNextSibling();
|
|
}
|
|
}
|
|
|
|
if (newAbsoluteItems.NotEmpty()) {
|
|
// ~nsFrameConstructorSaveState() will move newAbsoluteItems to
|
|
// aNewParent's absolute child list.
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
// It doesn't matter whether aNewParent has position style or not. Caller
|
|
// won't call us if we can't have absolute children.
|
|
PushAbsoluteContainingBlock(aNewParent, aNewParent, absoluteSaveState);
|
|
mAbsoluteList = std::move(newAbsoluteItems);
|
|
}
|
|
}
|
|
|
|
void nsFrameConstructorState::ReparentFloats(nsContainerFrame* aNewParent) {
|
|
MOZ_ASSERT(aNewParent->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR),
|
|
"Restrict the usage under column hierarchy.");
|
|
MOZ_ASSERT(
|
|
aNewParent->IsFloatContainingBlock(),
|
|
"Why calling this method if aNewParent is not a float containing block?");
|
|
|
|
// Gather floats that should reparent under aNewParent.
|
|
AbsoluteFrameList floats(aNewParent);
|
|
nsIFrame* current = mFloatedList.FirstChild();
|
|
while (current) {
|
|
nsIFrame* placeholder = current->GetPlaceholderFrame();
|
|
nsIFrame* next = current->GetNextSibling();
|
|
if (nsLayoutUtils::IsProperAncestorFrame(aNewParent, placeholder)) {
|
|
mFloatedList.RemoveFrame(current);
|
|
floats.AppendFrame(aNewParent, current);
|
|
}
|
|
current = next;
|
|
}
|
|
|
|
if (floats.NotEmpty()) {
|
|
// Make floats move into aNewParent's float child list in
|
|
// ~nsFrameConstructorSaveState() when destructing floatSaveState.
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
PushFloatContainingBlock(aNewParent, floatSaveState);
|
|
mFloatedList = std::move(floats);
|
|
}
|
|
}
|
|
|
|
AbsoluteFrameList* nsFrameConstructorState::GetOutOfFlowFrameList(
|
|
nsIFrame* aNewFrame, bool aCanBePositioned, bool aCanBeFloated,
|
|
nsFrameState* aPlaceholderType) {
|
|
const nsStyleDisplay* disp = aNewFrame->StyleDisplay();
|
|
if (aCanBeFloated && disp->IsFloatingStyle()) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_FLOAT;
|
|
return &mFloatedList;
|
|
}
|
|
|
|
if (aCanBePositioned) {
|
|
if (disp->mTopLayer != StyleTopLayer::None) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_TOPLAYER;
|
|
if (disp->mPosition == StylePositionProperty::Fixed) {
|
|
*aPlaceholderType |= PLACEHOLDER_FOR_FIXEDPOS;
|
|
return &mRealFixedList;
|
|
}
|
|
*aPlaceholderType |= PLACEHOLDER_FOR_ABSPOS;
|
|
return &mTopLayerAbsoluteList;
|
|
}
|
|
if (disp->mPosition == StylePositionProperty::Absolute) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_ABSPOS;
|
|
return &mAbsoluteList;
|
|
}
|
|
if (disp->mPosition == StylePositionProperty::Fixed) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_FIXEDPOS;
|
|
return mFixedList;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void nsFrameConstructorState::ConstructBackdropFrameFor(nsIContent* aContent,
|
|
nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame->StyleDisplay()->mTopLayer == StyleTopLayer::Auto);
|
|
nsContainerFrame* frame = do_QueryFrame(aFrame);
|
|
if (!frame) {
|
|
NS_WARNING("Cannot create backdrop frame for non-container frame");
|
|
return;
|
|
}
|
|
|
|
ComputedStyle* parentStyle = nsLayoutUtils::GetStyleFrame(aFrame)->Style();
|
|
RefPtr<ComputedStyle> style =
|
|
mPresShell->StyleSet()->ResolvePseudoElementStyle(
|
|
*aContent->AsElement(), PseudoStyleType::backdrop, nullptr,
|
|
parentStyle);
|
|
MOZ_ASSERT(style->StyleDisplay()->mTopLayer == StyleTopLayer::Auto);
|
|
nsContainerFrame* parentFrame =
|
|
GetGeometricParent(*style->StyleDisplay(), nullptr);
|
|
|
|
nsBackdropFrame* backdropFrame =
|
|
new (mPresShell) nsBackdropFrame(style, mPresShell->GetPresContext());
|
|
backdropFrame->Init(aContent, parentFrame, nullptr);
|
|
|
|
nsFrameState placeholderType;
|
|
AbsoluteFrameList* frameList =
|
|
GetOutOfFlowFrameList(backdropFrame, true, true, &placeholderType);
|
|
MOZ_ASSERT(placeholderType & PLACEHOLDER_FOR_TOPLAYER);
|
|
|
|
nsIFrame* placeholder = nsCSSFrameConstructor::CreatePlaceholderFrameFor(
|
|
mPresShell, aContent, backdropFrame, frame, nullptr, placeholderType);
|
|
frame->SetInitialChildList(FrameChildListID::Backdrop,
|
|
nsFrameList(placeholder, placeholder));
|
|
|
|
frameList->AppendFrame(nullptr, backdropFrame);
|
|
}
|
|
|
|
void nsFrameConstructorState::AddChild(
|
|
nsIFrame* aNewFrame, nsFrameList& aFrameList, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, bool aCanBePositioned, bool aCanBeFloated,
|
|
bool aInsertAfter, nsIFrame* aInsertAfterFrame) {
|
|
MOZ_ASSERT(!aNewFrame->GetNextSibling(), "Shouldn't happen");
|
|
|
|
nsFrameState placeholderType;
|
|
AbsoluteFrameList* outOfFlowFrameList = GetOutOfFlowFrameList(
|
|
aNewFrame, aCanBePositioned, aCanBeFloated, &placeholderType);
|
|
|
|
// The comments in GetGeometricParent regarding root table frames
|
|
// all apply here, unfortunately. Thus, we need to check whether
|
|
// the returned frame items really has containing block.
|
|
nsFrameList* frameList;
|
|
if (outOfFlowFrameList && outOfFlowFrameList->mContainingBlock) {
|
|
MOZ_ASSERT(aNewFrame->GetParent() == outOfFlowFrameList->mContainingBlock,
|
|
"Parent of the frame is not the containing block?");
|
|
frameList = outOfFlowFrameList;
|
|
} else {
|
|
frameList = &aFrameList;
|
|
placeholderType = nsFrameState(0);
|
|
}
|
|
|
|
if (placeholderType) {
|
|
NS_ASSERTION(frameList != &aFrameList,
|
|
"Putting frame in-flow _and_ want a placeholder?");
|
|
nsIFrame* placeholderFrame =
|
|
nsCSSFrameConstructor::CreatePlaceholderFrameFor(
|
|
mPresShell, aContent, aNewFrame, aParentFrame, nullptr,
|
|
placeholderType);
|
|
|
|
placeholderFrame->AddStateBits(mAdditionalStateBits);
|
|
// Add the placeholder frame to the flow
|
|
aFrameList.AppendFrame(nullptr, placeholderFrame);
|
|
|
|
if (placeholderType & PLACEHOLDER_FOR_TOPLAYER) {
|
|
ConstructBackdropFrameFor(aContent, aNewFrame);
|
|
}
|
|
}
|
|
#ifdef DEBUG
|
|
else {
|
|
NS_ASSERTION(aNewFrame->GetParent() == aParentFrame,
|
|
"In-flow frame has wrong parent");
|
|
}
|
|
#endif
|
|
|
|
if (aInsertAfter) {
|
|
frameList->InsertFrame(nullptr, aInsertAfterFrame, aNewFrame);
|
|
} else {
|
|
frameList->AppendFrame(nullptr, aNewFrame);
|
|
}
|
|
}
|
|
|
|
// Some of this function's callers recurse 1000 levels deep in crashtests. On
|
|
// platforms where stack limits are low, we can't afford to incorporate this
|
|
// function's `AutoTArray`s into its callers' stack frames, so disable inlining.
|
|
MOZ_NEVER_INLINE void nsFrameConstructorState::ProcessFrameInsertions(
|
|
AbsoluteFrameList& aFrameList, FrameChildListID aChildListID) {
|
|
MOZ_ASSERT(&aFrameList == &mFloatedList || &aFrameList == &mAbsoluteList ||
|
|
&aFrameList == &mTopLayerAbsoluteList ||
|
|
&aFrameList == &mAncestorFixedList || &aFrameList == mFixedList ||
|
|
&aFrameList == &mRealFixedList);
|
|
MOZ_ASSERT_IF(&aFrameList == &mFloatedList,
|
|
aChildListID == FrameChildListID::Float);
|
|
MOZ_ASSERT_IF(&aFrameList == &mAbsoluteList || &aFrameList == mFixedList,
|
|
aChildListID == FrameChildListID::Absolute ||
|
|
aChildListID == FrameChildListID::Fixed);
|
|
MOZ_ASSERT_IF(&aFrameList == &mTopLayerAbsoluteList,
|
|
aChildListID == FrameChildListID::Absolute);
|
|
MOZ_ASSERT_IF(&aFrameList == mFixedList && &aFrameList != &mAbsoluteList,
|
|
aChildListID == FrameChildListID::Fixed);
|
|
MOZ_ASSERT_IF(&aFrameList == &mAncestorFixedList,
|
|
aChildListID == FrameChildListID::Fixed);
|
|
MOZ_ASSERT_IF(&aFrameList == &mRealFixedList,
|
|
aChildListID == FrameChildListID::Fixed);
|
|
|
|
if (aFrameList.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
nsContainerFrame* containingBlock = aFrameList.mContainingBlock;
|
|
|
|
NS_ASSERTION(containingBlock, "Child list without containing block?");
|
|
|
|
if (aChildListID == FrameChildListID::Fixed) {
|
|
// Put this frame on the transformed-frame's abs-pos list instead, if
|
|
// it has abs-pos children instead of fixed-pos children.
|
|
aChildListID = containingBlock->GetAbsoluteListID();
|
|
}
|
|
|
|
// Insert the frames hanging out in aItems. We can use SetInitialChildList()
|
|
// if the containing block hasn't been reflowed yet (so NS_FRAME_FIRST_REFLOW
|
|
// is set) and doesn't have any frames in the aChildListID child list yet.
|
|
const nsFrameList& childList = containingBlock->GetChildList(aChildListID);
|
|
if (childList.IsEmpty() &&
|
|
containingBlock->HasAnyStateBits(NS_FRAME_FIRST_REFLOW)) {
|
|
// If we're injecting absolutely positioned frames, inject them on the
|
|
// absolute containing block
|
|
if (aChildListID == containingBlock->GetAbsoluteListID()) {
|
|
containingBlock->GetAbsoluteContainingBlock()->SetInitialChildList(
|
|
containingBlock, aChildListID, std::move(aFrameList));
|
|
} else {
|
|
containingBlock->SetInitialChildList(aChildListID, std::move(aFrameList));
|
|
}
|
|
} else if (aChildListID == FrameChildListID::Fixed ||
|
|
aChildListID == FrameChildListID::Absolute) {
|
|
// The order is not important for abs-pos/fixed-pos frame list, just
|
|
// append the frame items to the list directly.
|
|
mFrameConstructor->AppendFrames(containingBlock, aChildListID,
|
|
std::move(aFrameList));
|
|
} else {
|
|
// Note that whether the frame construction context is doing an append or
|
|
// not is not helpful here, since it could be appending to some frame in
|
|
// the middle of the document, which means we're not necessarily
|
|
// appending to the children of the containing block.
|
|
//
|
|
// We need to make sure the 'append to the end of document' case is fast.
|
|
// So first test the last child of the containing block
|
|
nsIFrame* lastChild = childList.LastChild();
|
|
|
|
// CompareTreePosition uses placeholder hierarchy for out of flow frames,
|
|
// so this will make out-of-flows respect the ordering of placeholders,
|
|
// which is great because it takes care of anonymous content.
|
|
nsIFrame* firstNewFrame = aFrameList.FirstChild();
|
|
|
|
// Cache the ancestor chain so that we can reuse it if needed.
|
|
AutoTArray<nsIFrame*, 20> firstNewFrameAncestors;
|
|
nsIFrame* notCommonAncestor = nullptr;
|
|
if (lastChild) {
|
|
notCommonAncestor = nsLayoutUtils::FillAncestors(
|
|
firstNewFrame, containingBlock, &firstNewFrameAncestors);
|
|
}
|
|
|
|
if (!lastChild || nsLayoutUtils::CompareTreePosition(
|
|
lastChild, firstNewFrame, firstNewFrameAncestors,
|
|
notCommonAncestor ? containingBlock : nullptr) < 0) {
|
|
// no lastChild, or lastChild comes before the new children, so just
|
|
// append
|
|
mFrameConstructor->AppendFrames(containingBlock, aChildListID,
|
|
std::move(aFrameList));
|
|
} else {
|
|
// Try the other children. First collect them to an array so that a
|
|
// reasonable fast binary search can be used to find the insertion point.
|
|
AutoTArray<nsIFrame*, 128> children;
|
|
for (nsIFrame* f = childList.FirstChild(); f != lastChild;
|
|
f = f->GetNextSibling()) {
|
|
children.AppendElement(f);
|
|
}
|
|
|
|
nsIFrame* insertionPoint = nullptr;
|
|
int32_t imin = 0;
|
|
int32_t max = children.Length();
|
|
while (max > imin) {
|
|
int32_t imid = imin + ((max - imin) / 2);
|
|
nsIFrame* f = children[imid];
|
|
int32_t compare = nsLayoutUtils::CompareTreePosition(
|
|
f, firstNewFrame, firstNewFrameAncestors,
|
|
notCommonAncestor ? containingBlock : nullptr);
|
|
if (compare > 0) {
|
|
// f is after the new frame.
|
|
max = imid;
|
|
insertionPoint = imid > 0 ? children[imid - 1] : nullptr;
|
|
} else if (compare < 0) {
|
|
// f is before the new frame.
|
|
imin = imid + 1;
|
|
insertionPoint = f;
|
|
} else {
|
|
// This is for the old behavior. Should be removed once it is
|
|
// guaranteed that CompareTreePosition can't return 0!
|
|
// See bug 928645.
|
|
NS_WARNING("Something odd happening???");
|
|
insertionPoint = nullptr;
|
|
for (uint32_t i = 0; i < children.Length(); ++i) {
|
|
nsIFrame* f = children[i];
|
|
if (nsLayoutUtils::CompareTreePosition(
|
|
f, firstNewFrame, firstNewFrameAncestors,
|
|
notCommonAncestor ? containingBlock : nullptr) > 0) {
|
|
break;
|
|
}
|
|
insertionPoint = f;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
mFrameConstructor->InsertFrames(containingBlock, aChildListID,
|
|
insertionPoint, std::move(aFrameList));
|
|
}
|
|
}
|
|
|
|
MOZ_ASSERT(aFrameList.IsEmpty(), "How did that happen?");
|
|
}
|
|
|
|
nsFrameConstructorSaveState::~nsFrameConstructorSaveState() {
|
|
// Restore the state
|
|
if (mList) {
|
|
MOZ_ASSERT(mState, "Can't have mList set without having a state!");
|
|
mState->ProcessFrameInsertions(*mList, mChildListID);
|
|
|
|
if (mList == &mState->mAbsoluteList) {
|
|
mState->mAbsoluteList = std::move(mSavedList);
|
|
mState->mFixedList = mSavedFixedList;
|
|
} else {
|
|
mState->mFloatedList = std::move(mSavedList);
|
|
}
|
|
|
|
MOZ_ASSERT(mSavedList.IsEmpty(),
|
|
"Frames in mSavedList should've moved back into mState!");
|
|
MOZ_ASSERT(!mList->LastChild() || !mList->LastChild()->GetNextSibling(),
|
|
"Something corrupted our list!");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Moves aFrameList from aOldParent to aNewParent. This updates the parent
|
|
* pointer of the frames in the list, and reparents their views as needed.
|
|
* nsIFrame::SetParent sets the NS_FRAME_HAS_VIEW bit on aNewParent and its
|
|
* ancestors as needed. Then it sets the list as the initial child list
|
|
* on aNewParent, unless aNewParent either already has kids or has been
|
|
* reflowed; in that case it appends the new frames. Note that this
|
|
* method differs from ReparentFrames in that it doesn't change the kids'
|
|
* style.
|
|
*/
|
|
// XXXbz Since this is only used for {ib} splits, could we just copy the view
|
|
// bits from aOldParent to aNewParent and then use the
|
|
// nsFrameList::ApplySetParent? That would still leave us doing two passes
|
|
// over the list, of course; if we really wanted to we could factor out the
|
|
// relevant part of ReparentFrameViewList, I suppose... Or just get rid of
|
|
// views, which would make most of this function go away.
|
|
static void MoveChildrenTo(nsIFrame* aOldParent, nsContainerFrame* aNewParent,
|
|
nsFrameList& aFrameList) {
|
|
#ifdef DEBUG
|
|
bool sameGrandParent = aOldParent->GetParent() == aNewParent->GetParent();
|
|
|
|
if (aNewParent->HasView() || aOldParent->HasView() || !sameGrandParent) {
|
|
// Move the frames into the new view
|
|
nsContainerFrame::ReparentFrameViewList(aFrameList, aOldParent, aNewParent);
|
|
}
|
|
#endif
|
|
|
|
aFrameList.ApplySetParent(aNewParent);
|
|
|
|
if (aNewParent->PrincipalChildList().IsEmpty() &&
|
|
aNewParent->HasAnyStateBits(NS_FRAME_FIRST_REFLOW)) {
|
|
aNewParent->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(aFrameList));
|
|
} else {
|
|
aNewParent->AppendFrames(FrameChildListID::Principal,
|
|
std::move(aFrameList));
|
|
}
|
|
}
|
|
|
|
static void EnsureAutoPageName(nsFrameConstructorState& aState,
|
|
const nsContainerFrame* const aFrame) {
|
|
// Check if we need to figure out our used page name.
|
|
// When building the entire document, this should only happen for the
|
|
// root, which will mean the loop will immediately end. Either way, this will
|
|
// only happen once for each time the frame constructor is run.
|
|
if (aState.mAutoPageNameValue) {
|
|
return;
|
|
}
|
|
|
|
for (const nsContainerFrame* frame = aFrame; frame;
|
|
frame = frame->GetParent()) {
|
|
if (const nsAtom* maybePageName = frame->GetStylePageName()) {
|
|
aState.mAutoPageNameValue = maybePageName;
|
|
return;
|
|
}
|
|
}
|
|
// Ensure that a root with `page: auto` gets an empty page name
|
|
// https://drafts.csswg.org/css-page-3/#using-named-pages
|
|
aState.mAutoPageNameValue = nsGkAtoms::_empty;
|
|
}
|
|
|
|
nsCSSFrameConstructor::AutoFrameConstructionPageName::
|
|
AutoFrameConstructionPageName(nsFrameConstructorState& aState,
|
|
nsIFrame* const aFrame)
|
|
: mState(aState), mNameToRestore(nullptr) {
|
|
if (!aState.mPresContext->IsPaginated()) {
|
|
MOZ_ASSERT(!aState.mAutoPageNameValue,
|
|
"Page name should not have been set");
|
|
return;
|
|
}
|
|
#ifdef DEBUG
|
|
MOZ_ASSERT(!aFrame->mWasVisitedByAutoFrameConstructionPageName,
|
|
"Frame should only have been visited once");
|
|
aFrame->mWasVisitedByAutoFrameConstructionPageName = true;
|
|
#endif
|
|
|
|
EnsureAutoPageName(aState, aFrame->GetParent());
|
|
mNameToRestore = aState.mAutoPageNameValue;
|
|
|
|
MOZ_ASSERT(mNameToRestore,
|
|
"Page name should have been found by EnsureAutoPageName");
|
|
if (const nsAtom* maybePageName = aFrame->GetStylePageName()) {
|
|
aState.mAutoPageNameValue = maybePageName;
|
|
}
|
|
aFrame->SetAutoPageValue(aState.mAutoPageNameValue);
|
|
}
|
|
|
|
nsCSSFrameConstructor::AutoFrameConstructionPageName::
|
|
~AutoFrameConstructionPageName() {
|
|
// This isn't actually useful when not in paginated layout, but it's very
|
|
// likely cheaper to unconditionally write this pointer than to test for
|
|
// paginated layout and then branch on the result.
|
|
mState.mAutoPageNameValue = mNameToRestore;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
nsCSSFrameConstructor::nsCSSFrameConstructor(Document* aDocument,
|
|
PresShell* aPresShell)
|
|
: nsFrameManager(aPresShell),
|
|
mDocument(aDocument),
|
|
mFirstFreeFCItem(nullptr),
|
|
mFCItemsInUse(0),
|
|
mCurrentDepth(0),
|
|
mQuotesDirty(false),
|
|
mCountersDirty(false),
|
|
mAlwaysCreateFramesForIgnorableWhitespace(false) {
|
|
#ifdef DEBUG
|
|
static bool gFirstTime = true;
|
|
if (gFirstTime) {
|
|
gFirstTime = false;
|
|
char* flags = PR_GetEnv("GECKO_FRAMECTOR_DEBUG_FLAGS");
|
|
if (flags) {
|
|
bool error = false;
|
|
for (;;) {
|
|
char* comma = strchr(flags, ',');
|
|
if (comma) *comma = '\0';
|
|
|
|
bool found = false;
|
|
FrameCtorDebugFlags* flag = gFlags;
|
|
FrameCtorDebugFlags* limit = gFlags + NUM_DEBUG_FLAGS;
|
|
while (flag < limit) {
|
|
if (nsCRT::strcasecmp(flag->name, flags) == 0) {
|
|
*(flag->on) = true;
|
|
printf("nsCSSFrameConstructor: setting %s debug flag on\n",
|
|
flag->name);
|
|
found = true;
|
|
break;
|
|
}
|
|
++flag;
|
|
}
|
|
|
|
if (!found) error = true;
|
|
|
|
if (!comma) break;
|
|
|
|
*comma = ',';
|
|
flags = comma + 1;
|
|
}
|
|
|
|
if (error) {
|
|
printf("Here are the available GECKO_FRAMECTOR_DEBUG_FLAGS:\n");
|
|
FrameCtorDebugFlags* flag = gFlags;
|
|
FrameCtorDebugFlags* limit = gFlags + NUM_DEBUG_FLAGS;
|
|
while (flag < limit) {
|
|
printf(" %s\n", flag->name);
|
|
++flag;
|
|
}
|
|
printf(
|
|
"Note: GECKO_FRAMECTOR_DEBUG_FLAGS is a comma separated list of "
|
|
"flag\n");
|
|
printf("names (no whitespace)\n");
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void nsCSSFrameConstructor::NotifyDestroyingFrame(nsIFrame* aFrame) {
|
|
if (aFrame->StyleDisplay()->IsContainStyle()) {
|
|
mContainStyleScopeManager.DestroyScopesFor(aFrame);
|
|
}
|
|
|
|
if (aFrame->HasAnyStateBits(NS_FRAME_GENERATED_CONTENT) &&
|
|
mContainStyleScopeManager.DestroyQuoteNodesFor(aFrame)) {
|
|
QuotesDirty();
|
|
}
|
|
|
|
if (aFrame->HasAnyStateBits(NS_FRAME_HAS_CSS_COUNTER_STYLE) &&
|
|
mContainStyleScopeManager.DestroyCounterNodesFor(aFrame)) {
|
|
// Technically we don't need to update anything if we destroyed only
|
|
// USE nodes. However, this is unlikely to happen in the real world
|
|
// since USE nodes generally go along with INCREMENT nodes.
|
|
CountersDirty();
|
|
}
|
|
|
|
RestyleManager()->NotifyDestroyingFrame(aFrame);
|
|
}
|
|
|
|
struct nsGenConInitializer {
|
|
UniquePtr<nsGenConNode> mNode;
|
|
nsGenConList* mList;
|
|
void (nsCSSFrameConstructor::*mDirtyAll)();
|
|
|
|
nsGenConInitializer(UniquePtr<nsGenConNode> aNode, nsGenConList* aList,
|
|
void (nsCSSFrameConstructor::*aDirtyAll)())
|
|
: mNode(std::move(aNode)), mList(aList), mDirtyAll(aDirtyAll) {}
|
|
};
|
|
|
|
already_AddRefed<nsIContent> nsCSSFrameConstructor::CreateGenConTextNode(
|
|
nsFrameConstructorState& aState, const nsAString& aString,
|
|
UniquePtr<nsGenConInitializer> aInitializer) {
|
|
RefPtr<nsTextNode> content = new (mDocument->NodeInfoManager())
|
|
nsTextNode(mDocument->NodeInfoManager());
|
|
content->SetText(aString, false);
|
|
if (aInitializer) {
|
|
aInitializer->mNode->mText = content;
|
|
content->SetProperty(nsGkAtoms::genConInitializerProperty,
|
|
aInitializer.release(),
|
|
nsINode::DeleteProperty<nsGenConInitializer>);
|
|
aState.mGeneratedContentWithInitializer.AppendElement(content);
|
|
}
|
|
return content.forget();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CreateGeneratedContent(
|
|
nsFrameConstructorState& aState, Element& aOriginatingElement,
|
|
ComputedStyle& aPseudoStyle, const StyleContentItem& aItem,
|
|
size_t aContentIndex, const FunctionRef<void(nsIContent*)> aAddChild) {
|
|
using Type = StyleContentItem::Tag;
|
|
// Get the content value
|
|
const Type type = aItem.tag;
|
|
|
|
switch (type) {
|
|
case Type::Image: {
|
|
RefPtr c = GeneratedImageContent::Create(*mDocument, aContentIndex);
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
|
|
case Type::String: {
|
|
const auto string = aItem.AsString().AsString();
|
|
if (string.IsEmpty()) {
|
|
return;
|
|
}
|
|
RefPtr text =
|
|
CreateGenConTextNode(aState, NS_ConvertUTF8toUTF16(string), nullptr);
|
|
aAddChild(text);
|
|
return;
|
|
}
|
|
|
|
case Type::Attr: {
|
|
const auto& attr = aItem.AsAttr();
|
|
RefPtr<nsAtom> attrName = attr.attribute.AsAtom();
|
|
int32_t attrNameSpace = kNameSpaceID_None;
|
|
RefPtr<nsAtom> ns = attr.namespace_url.AsAtom();
|
|
if (!ns->IsEmpty()) {
|
|
nsresult rv = nsNameSpaceManager::GetInstance()->RegisterNameSpace(
|
|
ns.forget(), attrNameSpace);
|
|
NS_ENSURE_SUCCESS_VOID(rv);
|
|
}
|
|
|
|
if (mDocument->IsHTMLDocument() && aOriginatingElement.IsHTMLElement()) {
|
|
ToLowerCaseASCII(attrName);
|
|
}
|
|
|
|
RefPtr<nsAtom> fallback = attr.fallback.AsAtom();
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(), attrNameSpace,
|
|
attrName, fallback, getter_AddRefs(content));
|
|
aAddChild(content);
|
|
return;
|
|
}
|
|
|
|
case Type::Counter:
|
|
case Type::Counters: {
|
|
RefPtr<nsAtom> name;
|
|
const StyleCounterStyle* style;
|
|
nsString separator;
|
|
if (type == Type::Counter) {
|
|
const auto& counter = aItem.AsCounter();
|
|
name = counter._0.AsAtom();
|
|
style = &counter._1;
|
|
} else {
|
|
const auto& counters = aItem.AsCounters();
|
|
name = counters._0.AsAtom();
|
|
CopyUTF8toUTF16(counters._1.AsString(), separator);
|
|
style = &counters._2;
|
|
}
|
|
|
|
auto* counterList = mContainStyleScopeManager.GetOrCreateCounterList(
|
|
aOriginatingElement, name);
|
|
auto node = MakeUnique<nsCounterUseNode>(
|
|
*style, std::move(separator), aContentIndex,
|
|
/* aAllCounters = */ type == Type::Counters);
|
|
|
|
auto initializer = MakeUnique<nsGenConInitializer>(
|
|
std::move(node), counterList, &nsCSSFrameConstructor::CountersDirty);
|
|
RefPtr c = CreateGenConTextNode(aState, u""_ns, std::move(initializer));
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
case Type::OpenQuote:
|
|
case Type::CloseQuote:
|
|
case Type::NoOpenQuote:
|
|
case Type::NoCloseQuote: {
|
|
auto node = MakeUnique<nsQuoteNode>(type, aContentIndex);
|
|
auto* quoteList =
|
|
mContainStyleScopeManager.QuoteListFor(aOriginatingElement);
|
|
auto initializer = MakeUnique<nsGenConInitializer>(
|
|
std::move(node), quoteList, &nsCSSFrameConstructor::QuotesDirty);
|
|
RefPtr c = CreateGenConTextNode(aState, u""_ns, std::move(initializer));
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
|
|
case Type::MozLabelContent: {
|
|
nsAutoString accesskey;
|
|
if (!aOriginatingElement.GetAttr(nsGkAtoms::accesskey, accesskey) ||
|
|
accesskey.IsEmpty() || !LookAndFeel::GetMenuAccessKey()) {
|
|
// Easy path: just return a regular value attribute content.
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(), kNameSpaceID_None,
|
|
nsGkAtoms::value, nsGkAtoms::_empty,
|
|
getter_AddRefs(content));
|
|
aAddChild(content);
|
|
return;
|
|
}
|
|
|
|
nsAutoString value;
|
|
aOriginatingElement.GetAttr(nsGkAtoms::value, value);
|
|
|
|
auto AppendAccessKeyLabel = [&] {
|
|
// Always append accesskey text in uppercase, see bug 1806167.
|
|
ToUpperCase(accesskey);
|
|
nsAutoString accessKeyLabel = u"("_ns + accesskey + u")"_ns;
|
|
if (!StringEndsWith(value, accessKeyLabel)) {
|
|
if (InsertSeparatorBeforeAccessKey() && !value.IsEmpty() &&
|
|
!NS_IS_SPACE(value.Last())) {
|
|
value.Append(' ');
|
|
}
|
|
value.Append(accessKeyLabel);
|
|
}
|
|
};
|
|
if (AlwaysAppendAccessKey()) {
|
|
AppendAccessKeyLabel();
|
|
RefPtr c = CreateGenConTextNode(aState, value, nullptr);
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
|
|
const auto accessKeyStart = [&]() -> Maybe<size_t> {
|
|
nsAString::const_iterator start, end;
|
|
value.BeginReading(start);
|
|
value.EndReading(end);
|
|
|
|
const auto originalStart = start;
|
|
// not appending access key - do case-sensitive search
|
|
// first
|
|
bool found = true;
|
|
if (!FindInReadable(accesskey, start, end)) {
|
|
start = originalStart;
|
|
// didn't find it - perform a case-insensitive search
|
|
found = FindInReadable(accesskey, start, end,
|
|
nsCaseInsensitiveStringComparator);
|
|
}
|
|
if (!found) {
|
|
return Nothing();
|
|
}
|
|
return Some(Distance(originalStart, start));
|
|
}();
|
|
|
|
if (accessKeyStart.isNothing()) {
|
|
AppendAccessKeyLabel();
|
|
RefPtr c = CreateGenConTextNode(aState, value, nullptr);
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
|
|
if (*accessKeyStart != 0) {
|
|
RefPtr beginning = CreateGenConTextNode(
|
|
aState, Substring(value, 0, *accessKeyStart), nullptr);
|
|
aAddChild(beginning);
|
|
}
|
|
|
|
{
|
|
RefPtr accessKeyText = CreateGenConTextNode(
|
|
aState, Substring(value, *accessKeyStart, accesskey.Length()),
|
|
nullptr);
|
|
RefPtr<nsIContent> underline =
|
|
mDocument->CreateHTMLElement(nsGkAtoms::u);
|
|
underline->AppendChildTo(accessKeyText, /* aNotify = */ false,
|
|
IgnoreErrors());
|
|
aAddChild(underline);
|
|
}
|
|
|
|
size_t accessKeyEnd = *accessKeyStart + accesskey.Length();
|
|
if (accessKeyEnd != value.Length()) {
|
|
RefPtr valueEnd = CreateGenConTextNode(
|
|
aState, Substring(value, *accessKeyStart + accesskey.Length()),
|
|
nullptr);
|
|
aAddChild(valueEnd);
|
|
}
|
|
break;
|
|
}
|
|
case Type::MozAltContent: {
|
|
// Use the "alt" attribute; if that fails and the node is an HTML
|
|
// <input>, try the value attribute and then fall back to some default
|
|
// localized text we have.
|
|
// XXX what if the 'alt' attribute is added later, how will we
|
|
// detect that and do the right thing here?
|
|
if (aOriginatingElement.HasAttr(nsGkAtoms::alt)) {
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(), kNameSpaceID_None,
|
|
nsGkAtoms::alt, nsGkAtoms::_empty,
|
|
getter_AddRefs(content));
|
|
aAddChild(content);
|
|
return;
|
|
}
|
|
|
|
if (aOriginatingElement.IsHTMLElement(nsGkAtoms::input)) {
|
|
if (aOriginatingElement.HasAttr(nsGkAtoms::value)) {
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(),
|
|
kNameSpaceID_None, nsGkAtoms::value,
|
|
nsGkAtoms::_empty, getter_AddRefs(content));
|
|
aAddChild(content);
|
|
return;
|
|
}
|
|
|
|
nsAutoString temp;
|
|
nsContentUtils::GetMaybeLocalizedString(
|
|
nsContentUtils::eFORMS_PROPERTIES, "Submit", mDocument, temp);
|
|
RefPtr c = CreateGenConTextNode(aState, temp, nullptr);
|
|
aAddChild(c);
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CreateGeneratedContentFromListStyle(
|
|
nsFrameConstructorState& aState, Element& aOriginatingElement,
|
|
const ComputedStyle& aPseudoStyle,
|
|
const FunctionRef<void(nsIContent*)> aAddChild) {
|
|
const nsStyleList* styleList = aPseudoStyle.StyleList();
|
|
if (!styleList->mListStyleImage.IsNone()) {
|
|
RefPtr<nsIContent> child =
|
|
GeneratedImageContent::CreateForListStyleImage(*mDocument);
|
|
aAddChild(child);
|
|
child = CreateGenConTextNode(aState, u" "_ns, nullptr);
|
|
aAddChild(child);
|
|
return;
|
|
}
|
|
CreateGeneratedContentFromListStyleType(aState, aOriginatingElement,
|
|
aPseudoStyle, aAddChild);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CreateGeneratedContentFromListStyleType(
|
|
nsFrameConstructorState& aState, Element& aOriginatingElement,
|
|
const ComputedStyle& aPseudoStyle,
|
|
const FunctionRef<void(nsIContent*)> aAddChild) {
|
|
using Tag = StyleCounterStyle::Tag;
|
|
const auto& styleType = aPseudoStyle.StyleList()->mListStyleType;
|
|
switch (styleType.tag) {
|
|
case Tag::None:
|
|
return;
|
|
case Tag::String: {
|
|
nsDependentAtomString string(styleType.AsString().AsAtom());
|
|
RefPtr<nsIContent> child = CreateGenConTextNode(aState, string, nullptr);
|
|
aAddChild(child);
|
|
return;
|
|
}
|
|
case Tag::Name:
|
|
case Tag::Symbols:
|
|
break;
|
|
}
|
|
|
|
auto node = MakeUnique<nsCounterUseNode>(nsCounterUseNode::ForLegacyBullet,
|
|
styleType);
|
|
if (styleType.IsName()) {
|
|
nsAtom* name = styleType.AsName().AsAtom();
|
|
if (name == nsGkAtoms::disc || name == nsGkAtoms::circle ||
|
|
name == nsGkAtoms::square || name == nsGkAtoms::disclosure_closed ||
|
|
name == nsGkAtoms::disclosure_open) {
|
|
// We don't need a use node inserted for these.
|
|
CounterStyle* counterStyle = mPresShell->GetPresContext()
|
|
->CounterStyleManager()
|
|
->ResolveCounterStyle(name);
|
|
nsAutoString text;
|
|
node->GetText(WritingMode(&aPseudoStyle), counterStyle, text);
|
|
// Note that we're done with 'node' in this case. It's not inserted into
|
|
// any list so it's deleted when we return.
|
|
RefPtr<nsIContent> child = CreateGenConTextNode(aState, text, nullptr);
|
|
aAddChild(child);
|
|
return;
|
|
}
|
|
}
|
|
|
|
auto* counterList = mContainStyleScopeManager.GetOrCreateCounterList(
|
|
aOriginatingElement, nsGkAtoms::list_item);
|
|
auto initializer = MakeUnique<nsGenConInitializer>(
|
|
std::move(node), counterList, &nsCSSFrameConstructor::CountersDirty);
|
|
RefPtr<nsIContent> child =
|
|
CreateGenConTextNode(aState, EmptyString(), std::move(initializer));
|
|
aAddChild(child);
|
|
}
|
|
|
|
// Frames for these may not be leaves in the proper sense, but we still don't
|
|
// want to expose generated content on them. For the purposes of the page they
|
|
// should be leaves.
|
|
static bool HasUAWidget(const Element& aOriginatingElement) {
|
|
const ShadowRoot* sr = aOriginatingElement.GetShadowRoot();
|
|
return sr && sr->IsUAWidget();
|
|
}
|
|
|
|
/*
|
|
* aParentFrame - the frame that should be the parent of the generated
|
|
* content. This is the frame for the corresponding content node,
|
|
* which must not be a leaf frame.
|
|
*
|
|
* Any items created are added to aItems.
|
|
*
|
|
* We create an XML element (tag _moz_generated_content_before/after/marker)
|
|
* representing the pseudoelement. We create a DOM node for each 'content'
|
|
* item and make those nodes the children of the XML element. Then we create
|
|
* a frame subtree for the XML element as if it were a regular child of
|
|
* aParentFrame/aParentContent, giving the XML element the ::before, ::after
|
|
* or ::marker style.
|
|
*/
|
|
void nsCSSFrameConstructor::CreateGeneratedContentItem(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aParentFrame,
|
|
Element& aOriginatingElement, ComputedStyle& aStyle,
|
|
PseudoStyleType aPseudoElement, FrameConstructionItemList& aItems,
|
|
ItemFlags aExtraFlags) {
|
|
MOZ_ASSERT(aPseudoElement == PseudoStyleType::before ||
|
|
aPseudoElement == PseudoStyleType::after ||
|
|
aPseudoElement == PseudoStyleType::marker,
|
|
"unexpected aPseudoElement");
|
|
|
|
if (HasUAWidget(aOriginatingElement) &&
|
|
!aOriginatingElement.IsHTMLElement(nsGkAtoms::details)) {
|
|
return;
|
|
}
|
|
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
|
|
// Probe for the existence of the pseudo-element.
|
|
// |ProbePseudoElementStyle| checks the relevant properties for the pseudo.
|
|
// It only returns a non-null value if the pseudo should exist.
|
|
RefPtr<ComputedStyle> pseudoStyle = styleSet->ProbePseudoElementStyle(
|
|
aOriginatingElement, aPseudoElement, nullptr, &aStyle);
|
|
if (!pseudoStyle) {
|
|
return;
|
|
}
|
|
|
|
nsAtom* elemName = nullptr;
|
|
nsAtom* property = nullptr;
|
|
switch (aPseudoElement) {
|
|
case PseudoStyleType::before:
|
|
elemName = nsGkAtoms::mozgeneratedcontentbefore;
|
|
property = nsGkAtoms::beforePseudoProperty;
|
|
break;
|
|
case PseudoStyleType::after:
|
|
elemName = nsGkAtoms::mozgeneratedcontentafter;
|
|
property = nsGkAtoms::afterPseudoProperty;
|
|
break;
|
|
case PseudoStyleType::marker:
|
|
// We want to get a marker style even if we match no rules, but we still
|
|
// want to check the result of GeneratedContentPseudoExists.
|
|
elemName = nsGkAtoms::mozgeneratedcontentmarker;
|
|
property = nsGkAtoms::markerPseudoProperty;
|
|
break;
|
|
default:
|
|
MOZ_ASSERT_UNREACHABLE("unexpected aPseudoElement");
|
|
}
|
|
|
|
RefPtr<NodeInfo> nodeInfo = mDocument->NodeInfoManager()->GetNodeInfo(
|
|
elemName, nullptr, kNameSpaceID_None, nsINode::ELEMENT_NODE);
|
|
RefPtr<Element> container;
|
|
nsresult rv = NS_NewXMLElement(getter_AddRefs(container), nodeInfo.forget());
|
|
if (NS_FAILED(rv)) {
|
|
return;
|
|
}
|
|
|
|
// Cleared when the pseudo is unbound from the tree, so no need to store a
|
|
// strong reference, nor a destructor.
|
|
aOriginatingElement.SetProperty(property, container.get());
|
|
|
|
container->SetIsNativeAnonymousRoot();
|
|
container->SetPseudoElementType(aPseudoElement);
|
|
|
|
BindContext context(aOriginatingElement, BindContext::ForNativeAnonymous);
|
|
rv = container->BindToTree(context, aOriginatingElement);
|
|
if (NS_FAILED(rv)) {
|
|
container->UnbindFromTree();
|
|
return;
|
|
}
|
|
|
|
if (mDocument->DevToolsAnonymousAndShadowEventsEnabled()) {
|
|
container->QueueDevtoolsAnonymousEvent(/* aIsRemove = */ false);
|
|
}
|
|
|
|
// Servo has already eagerly computed the style for the container, so we can
|
|
// just stick the style on the element and avoid an additional traversal.
|
|
//
|
|
// We don't do this for pseudos that may trigger animations or transitions,
|
|
// since those need to be kicked off by the traversal machinery.
|
|
//
|
|
// Note that when a pseudo-element animates, we flag the originating element,
|
|
// so we check that flag, but we could also a more expensive (but exhaustive)
|
|
// check using EffectSet::GetEffectSet, for example.
|
|
if (!Servo_ComputedValues_SpecifiesAnimationsOrTransitions(pseudoStyle) &&
|
|
!aOriginatingElement.MayHaveAnimations()) {
|
|
Servo_SetExplicitStyle(container, pseudoStyle);
|
|
} else {
|
|
// If animations are involved, we avoid the SetExplicitStyle optimization
|
|
// above. We need to grab style with animations from the pseudo element and
|
|
// replace old one.
|
|
mPresShell->StyleSet()->StyleNewSubtree(container);
|
|
pseudoStyle = ServoStyleSet::ResolveServoStyle(*container);
|
|
}
|
|
|
|
auto AppendChild = [&container, this](nsIContent* aChild) {
|
|
// We don't strictly have to set NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE
|
|
// here; it would get set under AppendChildTo. But AppendChildTo might
|
|
// think that we're going from not being anonymous to being anonymous and
|
|
// do some extra work; setting the flag here avoids that.
|
|
aChild->SetFlags(NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE);
|
|
container->AppendChildTo(aChild, false, IgnoreErrors());
|
|
if (auto* childElement = Element::FromNode(aChild)) {
|
|
// If we created any children elements, Servo needs to traverse them, but
|
|
// the root is already set up.
|
|
mPresShell->StyleSet()->StyleNewSubtree(childElement);
|
|
}
|
|
};
|
|
auto items = pseudoStyle->StyleContent()->NonAltContentItems();
|
|
size_t index = 0;
|
|
for (const auto& item : items) {
|
|
CreateGeneratedContent(aState, aOriginatingElement, *pseudoStyle, item,
|
|
index++, AppendChild);
|
|
}
|
|
// If a ::marker has no 'content' then generate it from its 'list-style-*'.
|
|
if (index == 0 && aPseudoElement == PseudoStyleType::marker) {
|
|
CreateGeneratedContentFromListStyle(aState, aOriginatingElement,
|
|
*pseudoStyle, AppendChild);
|
|
}
|
|
auto flags = ItemFlags{ItemFlag::IsGeneratedContent} + aExtraFlags;
|
|
AddFrameConstructionItemsInternal(aState, container, aParentFrame, true,
|
|
pseudoStyle, flags, aItems);
|
|
}
|
|
|
|
/****************************************************
|
|
** BEGIN TABLE SECTION
|
|
****************************************************/
|
|
|
|
// The term pseudo frame is being used instead of anonymous frame, since
|
|
// anonymous frame has been used elsewhere to refer to frames that have
|
|
// generated content
|
|
|
|
// Return whether the given frame is a table pseudo-frame. Note that
|
|
// cell-content and table-outer frames have pseudo-types, but are always
|
|
// created, even for non-anonymous cells and tables respectively. So for those
|
|
// we have to examine the cell or table frame to see whether it's a pseudo
|
|
// frame. In particular, a lone table caption will have a table wrapper as its
|
|
// parent, but will also trigger construction of an empty inner table, which
|
|
// will be the one we can examine to see whether the wrapper was a pseudo-frame.
|
|
static bool IsTablePseudo(nsIFrame* aFrame) {
|
|
auto pseudoType = aFrame->Style()->GetPseudoType();
|
|
if (pseudoType == PseudoStyleType::NotPseudo) {
|
|
return false;
|
|
}
|
|
return pseudoType == PseudoStyleType::table ||
|
|
pseudoType == PseudoStyleType::inlineTable ||
|
|
pseudoType == PseudoStyleType::tableColGroup ||
|
|
pseudoType == PseudoStyleType::tableRowGroup ||
|
|
pseudoType == PseudoStyleType::tableRow ||
|
|
pseudoType == PseudoStyleType::tableCell ||
|
|
(pseudoType == PseudoStyleType::cellContent &&
|
|
aFrame->GetParent()->Style()->GetPseudoType() ==
|
|
PseudoStyleType::tableCell) ||
|
|
(pseudoType == PseudoStyleType::tableWrapper &&
|
|
aFrame->PrincipalChildList()
|
|
.FirstChild()
|
|
->Style()
|
|
->IsPseudoOrAnonBox());
|
|
}
|
|
|
|
static bool IsRubyPseudo(nsIFrame* aFrame) {
|
|
return RubyUtils::IsRubyPseudo(aFrame->Style()->GetPseudoType());
|
|
}
|
|
|
|
// Note that this is (subtly) different to ParentIsWrapperAnonBox, since
|
|
// ParentIsWrapperAnonBox is really just about restyles, but there are wrapper
|
|
// anon boxes that don't need to return true for that...
|
|
// FIXME(emilio): This should be less complicated, ParentIsWrapperAnonBox should
|
|
// probably be renamed to something else, and this should just use
|
|
// IsWrapperAnonBox or similar...
|
|
static bool IsWrapperPseudo(nsIFrame* aFrame) {
|
|
auto pseudoType = aFrame->Style()->GetPseudoType();
|
|
if (!PseudoStyle::IsAnonBox(pseudoType)) {
|
|
return false;
|
|
}
|
|
return PseudoStyle::IsWrapperAnonBox(pseudoType) || IsTablePseudo(aFrame);
|
|
}
|
|
|
|
/* static */
|
|
nsCSSFrameConstructor::ParentType nsCSSFrameConstructor::GetParentType(
|
|
LayoutFrameType aFrameType) {
|
|
if (aFrameType == LayoutFrameType::Table) {
|
|
return eTypeTable;
|
|
}
|
|
if (aFrameType == LayoutFrameType::TableRowGroup) {
|
|
return eTypeRowGroup;
|
|
}
|
|
if (aFrameType == LayoutFrameType::TableRow) {
|
|
return eTypeRow;
|
|
}
|
|
if (aFrameType == LayoutFrameType::TableColGroup) {
|
|
return eTypeColGroup;
|
|
}
|
|
if (aFrameType == LayoutFrameType::RubyBaseContainer) {
|
|
return eTypeRubyBaseContainer;
|
|
}
|
|
if (aFrameType == LayoutFrameType::RubyTextContainer) {
|
|
return eTypeRubyTextContainer;
|
|
}
|
|
if (aFrameType == LayoutFrameType::Ruby) {
|
|
return eTypeRuby;
|
|
}
|
|
|
|
return eTypeBlock;
|
|
}
|
|
|
|
// Pull all the captions present in aItems out into aCaptions.
|
|
static void PullOutCaptionFrames(nsFrameList& aList, nsFrameList& aCaptions) {
|
|
nsIFrame* child = aList.FirstChild();
|
|
while (child) {
|
|
nsIFrame* nextSibling = child->GetNextSibling();
|
|
if (child->StyleDisplay()->mDisplay == StyleDisplay::TableCaption) {
|
|
aList.RemoveFrame(child);
|
|
aCaptions.AppendFrame(nullptr, child);
|
|
}
|
|
child = nextSibling;
|
|
}
|
|
}
|
|
|
|
// Construct the outer, inner table frames and the children frames for the
|
|
// table.
|
|
// XXX Page break frames for pseudo table frames are not constructed to avoid
|
|
// the risk associated with revising the pseudo frame mechanism. The long term
|
|
// solution of having frames handle page-break-before/after will solve the
|
|
// problem.
|
|
nsIFrame* nsCSSFrameConstructor::ConstructTable(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
MOZ_ASSERT(aDisplay->mDisplay == StyleDisplay::Table ||
|
|
aDisplay->mDisplay == StyleDisplay::InlineTable,
|
|
"Unexpected call");
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
const bool isMathMLContent = content->IsMathMLElement();
|
|
|
|
// create the pseudo SC for the table wrapper as a child of the inner SC
|
|
RefPtr<ComputedStyle> outerComputedStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::tableWrapper, computedStyle);
|
|
|
|
// Create the table wrapper frame which holds the caption and inner table
|
|
// frame
|
|
nsContainerFrame* newFrame;
|
|
if (isMathMLContent) {
|
|
newFrame = NS_NewMathMLmtableOuterFrame(mPresShell, outerComputedStyle);
|
|
} else {
|
|
newFrame = NS_NewTableWrapperFrame(mPresShell, outerComputedStyle);
|
|
}
|
|
|
|
nsContainerFrame* geometricParent = aState.GetGeometricParent(
|
|
*outerComputedStyle->StyleDisplay(), aParentFrame);
|
|
|
|
// Init the table wrapper frame
|
|
InitAndRestoreFrame(aState, content, geometricParent, newFrame);
|
|
|
|
// Create the inner table frame
|
|
nsContainerFrame* innerFrame;
|
|
if (isMathMLContent) {
|
|
innerFrame = NS_NewMathMLmtableFrame(mPresShell, computedStyle);
|
|
} else {
|
|
innerFrame = NS_NewTableFrame(mPresShell, computedStyle);
|
|
}
|
|
|
|
InitAndRestoreFrame(aState, content, newFrame, innerFrame);
|
|
innerFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Put the newly created frames into the right child list
|
|
SetInitialSingleChild(newFrame, innerFrame);
|
|
|
|
aState.AddChild(newFrame, aFrameList, content, aParentFrame);
|
|
|
|
if (!mRootElementFrame) {
|
|
mRootElementFrame = newFrame;
|
|
}
|
|
|
|
nsFrameList childList;
|
|
|
|
// Process children
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
// Mark the table frame as an absolute container if needed
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (newFrame->IsAbsPosContainingBlock()) {
|
|
aState.PushAbsoluteContainingBlock(newFrame, newFrame, absoluteSaveState);
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(innerFrame, floatSaveState);
|
|
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(
|
|
aState, aItem.mChildItems, innerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX, childList);
|
|
} else {
|
|
ProcessChildren(aState, content, computedStyle, innerFrame, true, childList,
|
|
false);
|
|
}
|
|
|
|
nsFrameList captionList;
|
|
PullOutCaptionFrames(childList, captionList);
|
|
|
|
// Set the inner table frame's initial primary list
|
|
innerFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
|
|
// Set the table wrapper frame's secondary childlist lists
|
|
if (captionList.NotEmpty()) {
|
|
captionList.ApplySetParent(newFrame);
|
|
newFrame->SetInitialChildList(FrameChildListID::Caption,
|
|
std::move(captionList));
|
|
}
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
static void MakeTablePartAbsoluteContainingBlock(
|
|
nsFrameConstructorState& aState, nsFrameConstructorSaveState& aAbsSaveState,
|
|
nsContainerFrame* aFrame) {
|
|
// If we're positioned, then we need to become an absolute containing block
|
|
// for any absolutely positioned children.
|
|
aFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (aFrame->IsAbsPosContainingBlock()) {
|
|
aState.PushAbsoluteContainingBlock(aFrame, aFrame, aAbsSaveState);
|
|
}
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructTableRowOrRowGroup(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
MOZ_ASSERT(aDisplay->mDisplay == StyleDisplay::TableRow ||
|
|
aDisplay->mDisplay == StyleDisplay::TableRowGroup ||
|
|
aDisplay->mDisplay == StyleDisplay::TableFooterGroup ||
|
|
aDisplay->mDisplay == StyleDisplay::TableHeaderGroup,
|
|
"Not a row or row group");
|
|
MOZ_ASSERT(aItem.mComputedStyle->StyleDisplay() == aDisplay,
|
|
"Display style doesn't match style");
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsContainerFrame* newFrame;
|
|
if (aDisplay->mDisplay == StyleDisplay::TableRow) {
|
|
if (content->IsMathMLElement()) {
|
|
newFrame = NS_NewMathMLmtrFrame(mPresShell, computedStyle);
|
|
} else {
|
|
newFrame = NS_NewTableRowFrame(mPresShell, computedStyle);
|
|
}
|
|
} else {
|
|
newFrame = NS_NewTableRowGroupFrame(mPresShell, computedStyle);
|
|
}
|
|
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newFrame);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlock(aState, absoluteSaveState, newFrame);
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(newFrame, floatSaveState);
|
|
|
|
nsFrameList childList;
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(
|
|
aState, aItem.mChildItems, newFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX, childList);
|
|
} else {
|
|
ProcessChildren(aState, content, computedStyle, newFrame, true, childList,
|
|
false);
|
|
}
|
|
|
|
newFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
aFrameList.AppendFrame(nullptr, newFrame);
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructTableCol(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameList& aFrameList) {
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsTableColFrame* colFrame = NS_NewTableColFrame(mPresShell, computedStyle);
|
|
InitAndRestoreFrame(aState, content, aParentFrame, colFrame);
|
|
|
|
NS_ASSERTION(colFrame->Style() == computedStyle, "Unexpected style");
|
|
|
|
aFrameList.AppendFrame(nullptr, colFrame);
|
|
|
|
// construct additional col frames if the col frame has a span > 1
|
|
int32_t span = colFrame->GetSpan();
|
|
for (int32_t spanX = 1; spanX < span; spanX++) {
|
|
nsTableColFrame* newCol = NS_NewTableColFrame(mPresShell, computedStyle);
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newCol, false);
|
|
aFrameList.LastChild()->SetNextContinuation(newCol);
|
|
newCol->SetPrevContinuation(aFrameList.LastChild());
|
|
aFrameList.AppendFrame(nullptr, newCol);
|
|
newCol->SetColType(eColAnonymousCol);
|
|
}
|
|
|
|
return colFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructTableCell(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
MOZ_ASSERT(aDisplay->mDisplay == StyleDisplay::TableCell, "Unexpected call");
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
const bool isMathMLContent = content->IsMathMLElement();
|
|
|
|
nsTableFrame* tableFrame =
|
|
static_cast<nsTableRowFrame*>(aParentFrame)->GetTableFrame();
|
|
nsContainerFrame* cellFrame;
|
|
// <mtable> is border separate in mathml.css and the MathML code doesn't
|
|
// implement border collapse. For those users who style <mtable> with border
|
|
// collapse, give them the default non-MathML table frames that understand
|
|
// border collapse. This won't break us because MathML table frames are all
|
|
// subclasses of the default table code, and so we can freely mix <mtable>
|
|
// with <mtr> or <tr>, <mtd> or <td>. What will happen is just that non-MathML
|
|
// frames won't understand MathML attributes and will therefore miss the
|
|
// special handling that the MathML code does.
|
|
if (isMathMLContent && !tableFrame->IsBorderCollapse()) {
|
|
cellFrame = NS_NewMathMLmtdFrame(mPresShell, computedStyle, tableFrame);
|
|
} else {
|
|
// Warning: If you change this and add a wrapper frame around table cell
|
|
// frames, make sure Bug 368554 doesn't regress!
|
|
// See IsInAutoWidthTableCellForQuirk() in nsImageFrame.cpp.
|
|
cellFrame = NS_NewTableCellFrame(mPresShell, computedStyle, tableFrame);
|
|
}
|
|
|
|
// Initialize the table cell frame
|
|
InitAndRestoreFrame(aState, content, aParentFrame, cellFrame);
|
|
cellFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Resolve pseudo style and initialize the body cell frame
|
|
RefPtr<ComputedStyle> innerPseudoStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::cellContent, computedStyle);
|
|
|
|
nsContainerFrame* cellInnerFrame;
|
|
nsContainerFrame* scrollFrame = nullptr;
|
|
bool isScrollable = false;
|
|
// Create a block frame that will format the cell's content
|
|
if (isMathMLContent) {
|
|
cellInnerFrame = NS_NewMathMLmtdInnerFrame(mPresShell, innerPseudoStyle);
|
|
} else {
|
|
isScrollable = innerPseudoStyle->StyleDisplay()->IsScrollableOverflow() &&
|
|
!aState.mPresContext->IsPaginated() &&
|
|
StaticPrefs::layout_tables_scrollable_cells();
|
|
if (isScrollable) {
|
|
innerPseudoStyle = BeginBuildingScrollContainerFrame(
|
|
aState, content, innerPseudoStyle, cellFrame,
|
|
PseudoStyleType::scrolledContent, false, scrollFrame);
|
|
}
|
|
cellInnerFrame = NS_NewBlockFrame(mPresShell, innerPseudoStyle);
|
|
}
|
|
auto* parent = scrollFrame ? scrollFrame : cellFrame;
|
|
InitAndRestoreFrame(aState, content, parent, cellInnerFrame);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlock(aState, absoluteSaveState, cellFrame);
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(cellInnerFrame, floatSaveState);
|
|
|
|
nsFrameList childList;
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
AutoFrameConstructionPageName pageNameTracker(aState, cellInnerFrame);
|
|
ConstructFramesFromItemList(
|
|
aState, aItem.mChildItems, cellInnerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX, childList);
|
|
} else {
|
|
// Process the child content
|
|
ProcessChildren(aState, content, computedStyle, cellInnerFrame, true,
|
|
childList, !isMathMLContent);
|
|
}
|
|
|
|
cellInnerFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
|
|
if (isScrollable) {
|
|
FinishBuildingScrollContainerFrame(scrollFrame, cellInnerFrame);
|
|
}
|
|
SetInitialSingleChild(cellFrame, scrollFrame ? scrollFrame : cellInnerFrame);
|
|
aFrameList.AppendFrame(nullptr, cellFrame);
|
|
return cellFrame;
|
|
}
|
|
|
|
static inline bool NeedFrameFor(const nsFrameConstructorState& aState,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIContent* aChildContent) {
|
|
// XXX the GetContent() != aChildContent check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
MOZ_ASSERT(
|
|
!aChildContent->GetPrimaryFrame() || aState.mCreatingExtraFrames ||
|
|
aChildContent->GetPrimaryFrame()->GetContent() != aChildContent,
|
|
"Why did we get called?");
|
|
|
|
// don't create a whitespace frame if aParentFrame doesn't want it.
|
|
// always create frames for children in generated content. counter(),
|
|
// quotes, and attr() content can easily change dynamically and we don't
|
|
// want to be reconstructing frames. It's not even clear that these
|
|
// should be considered ignorable just because they evaluate to
|
|
// whitespace.
|
|
|
|
// We could handle all this in CreateNeededPseudoContainers or some other
|
|
// place after we build our frame construction items, but that would involve
|
|
// creating frame construction items for whitespace kids that ignores
|
|
// white-space, where we know we'll be dropping them all anyway, and involve
|
|
// an extra walk down the frame construction item list.
|
|
auto excludesIgnorableWhitespace = [](nsIFrame* aParentFrame) {
|
|
return aParentFrame->IsMathMLFrame();
|
|
};
|
|
if (!aParentFrame || !excludesIgnorableWhitespace(aParentFrame) ||
|
|
aParentFrame->IsGeneratedContentFrame() || !aChildContent->IsText()) {
|
|
return true;
|
|
}
|
|
|
|
aChildContent->SetFlags(NS_CREATE_FRAME_IF_NON_WHITESPACE |
|
|
NS_REFRAME_IF_WHITESPACE);
|
|
return !aChildContent->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
/***********************************************
|
|
* END TABLE SECTION
|
|
***********************************************/
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructDocElementFrame(
|
|
Element* aDocElement) {
|
|
MOZ_ASSERT(GetRootFrame(),
|
|
"No viewport? Someone forgot to call ConstructRootFrame!");
|
|
MOZ_ASSERT(!mDocElementContainingBlock,
|
|
"Shouldn't have a doc element containing block here");
|
|
|
|
// Resolve a new style for the viewport since it may be affected by a new root
|
|
// element style (e.g. a propagated 'direction').
|
|
//
|
|
// @see ComputedStyle::ApplyStyleFixups
|
|
{
|
|
RefPtr<ComputedStyle> sc =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::viewport, nullptr);
|
|
GetRootFrame()->SetComputedStyleWithoutNotification(sc);
|
|
}
|
|
|
|
// Ensure the document element is styled at this point.
|
|
if (!aDocElement->HasServoData()) {
|
|
mPresShell->StyleSet()->StyleNewSubtree(aDocElement);
|
|
}
|
|
aDocElement->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME);
|
|
|
|
// Make sure to call UpdateViewportScrollStylesOverride before
|
|
// SetUpDocElementContainingBlock, since it sets up our scrollbar state
|
|
// properly.
|
|
DebugOnly<nsIContent*> propagatedScrollFrom;
|
|
if (nsPresContext* presContext = mPresShell->GetPresContext()) {
|
|
propagatedScrollFrom = presContext->UpdateViewportScrollStylesOverride();
|
|
}
|
|
|
|
SetUpDocElementContainingBlock(aDocElement);
|
|
|
|
// This has the side-effect of getting `mFrameTreeState` from our docshell.
|
|
//
|
|
// FIXME(emilio): There may be a more sensible time to do this.
|
|
if (!mFrameTreeState) {
|
|
mPresShell->CaptureHistoryState(getter_AddRefs(mFrameTreeState));
|
|
}
|
|
|
|
NS_ASSERTION(mDocElementContainingBlock, "Should have parent by now");
|
|
nsFrameConstructorState state(
|
|
mPresShell,
|
|
GetAbsoluteContainingBlock(mDocElementContainingBlock, FIXED_POS),
|
|
nullptr, nullptr, do_AddRef(mFrameTreeState));
|
|
|
|
RefPtr<ComputedStyle> computedStyle =
|
|
ServoStyleSet::ResolveServoStyle(*aDocElement);
|
|
|
|
const nsStyleDisplay* display = computedStyle->StyleDisplay();
|
|
|
|
// --------- IF SCROLLABLE WRAP IN SCROLLFRAME --------
|
|
|
|
NS_ASSERTION(!display->IsScrollableOverflow() ||
|
|
state.mPresContext->IsPaginated() ||
|
|
propagatedScrollFrom == aDocElement,
|
|
"Scrollbars should have been propagated to the viewport");
|
|
|
|
if (MOZ_UNLIKELY(display->mDisplay == StyleDisplay::None)) {
|
|
return nullptr;
|
|
}
|
|
|
|
// This implements "The Principal Writing Mode".
|
|
// https://drafts.csswg.org/css-writing-modes-3/#principal-flow
|
|
//
|
|
// If there's a <body> element in an HTML document, its writing-mode,
|
|
// direction, and text-orientation override the root element's used value.
|
|
//
|
|
// We need to copy <body>'s WritingMode to mDocElementContainingBlock before
|
|
// construct mRootElementFrame so that anonymous internal frames such as
|
|
// <html> with table style can copy their parent frame's mWritingMode in
|
|
// nsIFrame::Init().
|
|
MOZ_ASSERT(!mRootElementFrame,
|
|
"We need to copy <body>'s principal writing-mode before "
|
|
"constructing mRootElementFrame.");
|
|
|
|
const WritingMode propagatedWM = [&] {
|
|
const WritingMode rootWM(computedStyle);
|
|
if (computedStyle->StyleDisplay()->IsContainAny()) {
|
|
return rootWM;
|
|
}
|
|
Element* body = mDocument->GetBodyElement();
|
|
if (!body) {
|
|
return rootWM;
|
|
}
|
|
RefPtr<ComputedStyle> bodyStyle = ResolveComputedStyle(body);
|
|
if (bodyStyle->StyleDisplay()->IsContainAny()) {
|
|
return rootWM;
|
|
}
|
|
const WritingMode bodyWM(bodyStyle);
|
|
if (bodyWM != rootWM) {
|
|
nsContentUtils::ReportToConsole(nsIScriptError::warningFlag, "Layout"_ns,
|
|
mDocument,
|
|
nsContentUtils::eLAYOUT_PROPERTIES,
|
|
"PrincipalWritingModePropagationWarning");
|
|
}
|
|
return bodyWM;
|
|
}();
|
|
|
|
mDocElementContainingBlock->PropagateWritingModeToSelfAndAncestors(
|
|
propagatedWM);
|
|
|
|
// Push the absolute containing block now so we can absolutely position the
|
|
// root element
|
|
nsFrameConstructorSaveState canvasCbSaveState;
|
|
mCanvasFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
|
|
state.PushAbsoluteContainingBlock(mCanvasFrame, mCanvasFrame,
|
|
canvasCbSaveState);
|
|
|
|
nsFrameConstructorSaveState docElementCbSaveState;
|
|
if (mCanvasFrame != mDocElementContainingBlock) {
|
|
mDocElementContainingBlock->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
state.PushAbsoluteContainingBlock(mDocElementContainingBlock,
|
|
mDocElementContainingBlock,
|
|
docElementCbSaveState);
|
|
}
|
|
|
|
// The rules from CSS 2.1, section 9.2.4, have already been applied
|
|
// by the style system, so we can assume that display->mDisplay is
|
|
// either NONE, BLOCK, or TABLE.
|
|
|
|
// contentFrame is the primary frame for the root element. frameList contains
|
|
// the children of the initial containing block.
|
|
//
|
|
// The first of those frames is usually `contentFrame`, but it can be
|
|
// different, in particular if the root frame is positioned, in which case
|
|
// contentFrame is the out-of-flow frame and frameList.FirstChild() is the
|
|
// placeholder.
|
|
//
|
|
// The rest of the frames in frameList are the anonymous content of the canvas
|
|
// frame.
|
|
nsContainerFrame* contentFrame;
|
|
nsFrameList frameList;
|
|
bool processChildren = false;
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
if (aDocElement->IsSVGElement()) {
|
|
if (!aDocElement->IsSVGElement(nsGkAtoms::svg)) {
|
|
return nullptr;
|
|
}
|
|
// We're going to call the right function ourselves, so no need to give a
|
|
// function to this FrameConstructionData.
|
|
|
|
// XXXbz on the other hand, if we converted this whole function to
|
|
// FrameConstructionData/Item, then we'd need the right function
|
|
// here... but would probably be able to get away with less code in this
|
|
// function in general.
|
|
static constexpr FrameConstructionData rootSVGData;
|
|
AutoFrameConstructionItem item(this, &rootSVGData, aDocElement,
|
|
do_AddRef(computedStyle), true);
|
|
|
|
contentFrame = static_cast<nsContainerFrame*>(ConstructOuterSVG(
|
|
state, item, mDocElementContainingBlock, display, frameList));
|
|
} else if (display->mDisplay == StyleDisplay::Flex ||
|
|
display->mDisplay == StyleDisplay::WebkitBox ||
|
|
display->mDisplay == StyleDisplay::Grid) {
|
|
auto func = [&] {
|
|
if (display->mDisplay == StyleDisplay::Grid) {
|
|
return NS_NewGridContainerFrame;
|
|
}
|
|
return NS_NewFlexContainerFrame;
|
|
}();
|
|
contentFrame = func(mPresShell, computedStyle);
|
|
InitAndRestoreFrame(
|
|
state, aDocElement,
|
|
state.GetGeometricParent(*display, mDocElementContainingBlock),
|
|
contentFrame);
|
|
state.AddChild(contentFrame, frameList, aDocElement,
|
|
mDocElementContainingBlock);
|
|
processChildren = true;
|
|
|
|
contentFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (contentFrame->IsAbsPosContainingBlock()) {
|
|
state.PushAbsoluteContainingBlock(contentFrame, contentFrame,
|
|
absoluteSaveState);
|
|
}
|
|
} else if (display->mDisplay == StyleDisplay::Table) {
|
|
// We're going to call the right function ourselves, so no need to give a
|
|
// function to this FrameConstructionData.
|
|
|
|
// XXXbz on the other hand, if we converted this whole function to
|
|
// FrameConstructionData/Item, then we'd need the right function
|
|
// here... but would probably be able to get away with less code in this
|
|
// function in general.
|
|
static constexpr FrameConstructionData rootTableData;
|
|
AutoFrameConstructionItem item(this, &rootTableData, aDocElement,
|
|
do_AddRef(computedStyle), true);
|
|
|
|
// if the document is a table then just populate it.
|
|
contentFrame = static_cast<nsContainerFrame*>(ConstructTable(
|
|
state, item, mDocElementContainingBlock, display, frameList));
|
|
} else if (display->DisplayInside() == StyleDisplayInside::Ruby) {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructBlockRubyFrame);
|
|
AutoFrameConstructionItem item(this, &data, aDocElement,
|
|
do_AddRef(computedStyle), true);
|
|
contentFrame = static_cast<nsContainerFrame*>(ConstructBlockRubyFrame(
|
|
state, item,
|
|
state.GetGeometricParent(*display, mDocElementContainingBlock), display,
|
|
frameList));
|
|
} else {
|
|
MOZ_ASSERT(display->mDisplay == StyleDisplay::Block ||
|
|
display->mDisplay == StyleDisplay::FlowRoot,
|
|
"Unhandled display type for root element");
|
|
contentFrame = NS_NewBlockFrame(mPresShell, computedStyle);
|
|
ConstructBlock(
|
|
state, aDocElement,
|
|
state.GetGeometricParent(*display, mDocElementContainingBlock),
|
|
mDocElementContainingBlock, computedStyle, &contentFrame, frameList,
|
|
contentFrame->IsAbsPosContainingBlock() ? contentFrame : nullptr);
|
|
}
|
|
|
|
MOZ_ASSERT(frameList.FirstChild());
|
|
MOZ_ASSERT(frameList.FirstChild()->GetContent() == aDocElement);
|
|
MOZ_ASSERT(contentFrame);
|
|
|
|
MOZ_ASSERT(
|
|
processChildren ? !mRootElementFrame : mRootElementFrame == contentFrame,
|
|
"unexpected mRootElementFrame");
|
|
if (processChildren) {
|
|
mRootElementFrame = contentFrame;
|
|
}
|
|
|
|
// Figure out which frame has the main style for the document element,
|
|
// assigning it to mRootElementStyleFrame.
|
|
// Backgrounds should be propagated from that frame to the viewport.
|
|
contentFrame->GetParentComputedStyle(&mRootElementStyleFrame);
|
|
bool isChild = mRootElementStyleFrame &&
|
|
mRootElementStyleFrame->GetParent() == contentFrame;
|
|
if (!isChild) {
|
|
mRootElementStyleFrame = mRootElementFrame;
|
|
}
|
|
|
|
if (processChildren) {
|
|
// Still need to process the child content
|
|
nsFrameList childList;
|
|
|
|
NS_ASSERTION(
|
|
!contentFrame->IsBlockFrameOrSubclass() && !contentFrame->IsSVGFrame(),
|
|
"Only XUL frames should reach here");
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
state.MaybePushFloatContainingBlock(contentFrame, floatSaveState);
|
|
|
|
ProcessChildren(state, aDocElement, computedStyle, contentFrame, true,
|
|
childList, false);
|
|
|
|
// Set the initial child lists
|
|
contentFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
}
|
|
|
|
nsIFrame* newFrame = frameList.FirstChild();
|
|
// set the primary frame
|
|
aDocElement->SetPrimaryFrame(contentFrame);
|
|
mDocElementContainingBlock->AppendFrames(FrameChildListID::Principal,
|
|
std::move(frameList));
|
|
|
|
// NOTE(emilio): This is in the reverse order compared to normal anonymous
|
|
// children. We usually generate anonymous kids first, then non-anonymous,
|
|
// but we generate the doc element frame the other way around. This is fine
|
|
// either way, but generating anonymous children in a different order requires
|
|
// changing nsCanvasFrame (and a whole lot of other potentially unknown code)
|
|
// to look at the last child to find the root frame rather than the first
|
|
// child.
|
|
ConstructAnonymousContentForCanvas(
|
|
state, mCanvasFrame, mRootElementFrame->GetContent(), frameList);
|
|
mCanvasFrame->AppendFrames(FrameChildListID::Principal, std::move(frameList));
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
RestyleManager* nsCSSFrameConstructor::RestyleManager() const {
|
|
return mPresShell->GetPresContext()->RestyleManager();
|
|
}
|
|
|
|
ViewportFrame* nsCSSFrameConstructor::ConstructRootFrame() {
|
|
AUTO_PROFILER_LABEL_HOT("nsCSSFrameConstructor::ConstructRootFrame",
|
|
LAYOUT_FrameConstruction);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
|
|
// --------- BUILD VIEWPORT -----------
|
|
RefPtr<ComputedStyle> viewportPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(PseudoStyleType::viewport,
|
|
nullptr);
|
|
ViewportFrame* viewportFrame =
|
|
NS_NewViewportFrame(mPresShell, viewportPseudoStyle);
|
|
|
|
// XXXbz do we _have_ to pass a null content pointer to that frame?
|
|
// Would it really kill us to pass in the root element or something?
|
|
// What would that break?
|
|
viewportFrame->Init(nullptr, nullptr, nullptr);
|
|
|
|
viewportFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Bind the viewport frame to the root view
|
|
if (nsView* rootView = mPresShell->GetViewManager()->GetRootView()) {
|
|
viewportFrame->SetView(rootView);
|
|
viewportFrame->SyncFrameViewProperties(rootView);
|
|
rootView->SetNeedsWindowPropertiesSync();
|
|
}
|
|
|
|
// Make it an absolute container for fixed-pos elements
|
|
viewportFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
viewportFrame->MarkAsAbsoluteContainingBlock();
|
|
|
|
return viewportFrame;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::SetUpDocElementContainingBlock(
|
|
nsIContent* aDocElement) {
|
|
MOZ_ASSERT(aDocElement, "No element?");
|
|
MOZ_ASSERT(!aDocElement->GetParent(), "Not root content?");
|
|
MOZ_ASSERT(aDocElement->GetUncomposedDoc(), "Not in a document?");
|
|
MOZ_ASSERT(aDocElement->GetUncomposedDoc()->GetRootElement() == aDocElement,
|
|
"Not the root of the document?");
|
|
|
|
/*
|
|
how the root frame hierarchy should look
|
|
|
|
Galley presentation, with scrolling:
|
|
|
|
ViewportFrame [fixed-cb]
|
|
ScrollContainerFrame (if needed)
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, SVGOuterSVGFrame,
|
|
nsTableWrapperFrame, nsPlaceholderFrame,
|
|
nsFlexContainerFrame, nsGridContainerFrame)
|
|
|
|
Print presentation, non-XUL
|
|
|
|
ViewportFrame
|
|
nsCanvasFrame
|
|
nsPageSequenceFrame
|
|
PrintedSheetFrame
|
|
nsPageFrame
|
|
nsPageContentFrame [fixed-cb]
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, SVGOuterSVGFrame,
|
|
nsTableWrapperFrame, nsPlaceholderFrame,
|
|
nsFlexContainerFrame,
|
|
nsGridContainerFrame)
|
|
|
|
Print-preview presentation, non-XUL
|
|
|
|
ViewportFrame
|
|
ScrollContainerFrame
|
|
nsCanvasFrame
|
|
nsPageSequenceFrame
|
|
PrintedSheetFrame
|
|
nsPageFrame
|
|
nsPageContentFrame [fixed-cb]
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, SVGOuterSVGFrame,
|
|
nsTableWrapperFrame,
|
|
nsPlaceholderFrame,
|
|
nsFlexContainerFrame,
|
|
nsGridContainerFrame)
|
|
|
|
Print/print preview of XUL is not supported.
|
|
[fixed-cb]: the default containing block for fixed-pos content
|
|
[abs-cb]: the default containing block for abs-pos content
|
|
|
|
Meaning of nsCSSFrameConstructor fields:
|
|
mRootElementFrame is "root element frame". This is the primary frame for
|
|
the root element.
|
|
mDocElementContainingBlock is the parent of mRootElementFrame
|
|
(i.e. nsCanvasFrame)
|
|
mPageSequenceFrame is the nsPageSequenceFrame, or null if there isn't
|
|
one
|
|
*/
|
|
|
|
// --------- CREATE ROOT FRAME -------
|
|
|
|
// Create the root frame. The document element's frame is a child of the
|
|
// root frame.
|
|
//
|
|
// The root frame serves two purposes:
|
|
// - reserves space for any margins needed for the document element's frame
|
|
// - renders the document element's background. This ensures the background
|
|
// covers the entire canvas as specified by the CSS2 spec
|
|
|
|
nsPresContext* presContext = mPresShell->GetPresContext();
|
|
const bool isPaginated = presContext->IsRootPaginatedDocument();
|
|
|
|
const bool isHTML = aDocElement->IsHTMLElement();
|
|
const bool isXUL = !isHTML && aDocElement->IsXULElement();
|
|
|
|
const bool isScrollable = [&] {
|
|
if (isPaginated) {
|
|
return presContext->HasPaginatedScrolling();
|
|
}
|
|
// Never create scrollbars for XUL documents or top level XHTML documents
|
|
// that disable scrolling.
|
|
if (isXUL) {
|
|
return false;
|
|
}
|
|
if (aDocElement->OwnerDoc()->ChromeRulesEnabled() &&
|
|
aDocElement->AsElement()->AttrValueIs(
|
|
kNameSpaceID_None, nsGkAtoms::scrolling, nsGkAtoms::_false,
|
|
eCaseMatters)) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}();
|
|
|
|
nsContainerFrame* viewportFrame =
|
|
static_cast<nsContainerFrame*>(GetRootFrame());
|
|
ComputedStyle* viewportPseudoStyle = viewportFrame->Style();
|
|
|
|
nsCanvasFrame* rootCanvasFrame =
|
|
NS_NewCanvasFrame(mPresShell, viewportPseudoStyle);
|
|
PseudoStyleType rootPseudo = PseudoStyleType::canvas;
|
|
mCanvasFrame = rootCanvasFrame;
|
|
mDocElementContainingBlock = rootCanvasFrame;
|
|
|
|
// --------- IF SCROLLABLE WRAP IN SCROLLFRAME --------
|
|
|
|
// If the device supports scrolling (e.g., in galley mode on the screen and
|
|
// for print-preview, but not when printing), then create a scroll frame that
|
|
// will act as the scrolling mechanism for the viewport.
|
|
// XXX Do we even need a viewport when printing to a printer?
|
|
|
|
// We no longer need to do overflow propagation here. It's taken care of
|
|
// when we construct frames for the element whose overflow might be
|
|
// propagated
|
|
NS_ASSERTION(!isScrollable || !isXUL,
|
|
"XUL documents should never be scrollable - see above");
|
|
|
|
nsContainerFrame* newFrame = rootCanvasFrame;
|
|
RefPtr<ComputedStyle> rootPseudoStyle;
|
|
// we must create a state because if the scrollbars are GFX it needs the
|
|
// state to build the scrollbar frames.
|
|
nsFrameConstructorState state(mPresShell, nullptr, nullptr, nullptr);
|
|
|
|
// Start off with the viewport as parent; we'll adjust it as needed.
|
|
nsContainerFrame* parentFrame = viewportFrame;
|
|
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
// If paginated, make sure we don't put scrollbars in
|
|
if (!isScrollable) {
|
|
rootPseudoStyle = styleSet->ResolveInheritingAnonymousBoxStyle(
|
|
rootPseudo, viewportPseudoStyle);
|
|
} else {
|
|
rootPseudo = PseudoStyleType::scrolledCanvas;
|
|
|
|
// Build the frame. We give it the content we are wrapping which is the
|
|
// document element, the root frame, the parent view port frame, and we
|
|
// should get back the new frame and the scrollable view if one was
|
|
// created.
|
|
|
|
// resolve a context for the scrollframe
|
|
RefPtr<ComputedStyle> computedStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::viewportScroll, viewportPseudoStyle);
|
|
|
|
// Note that the viewport scrollframe is always built with
|
|
// overflow:auto style. This forces the scroll frame to create
|
|
// anonymous content for both scrollbars. This is necessary even
|
|
// if the HTML or BODY elements are overriding the viewport
|
|
// scroll style to 'hidden' --- dynamic style changes might put
|
|
// scrollbars back on the viewport and we don't want to have to
|
|
// reframe the viewport to create the scrollbar content.
|
|
newFrame = nullptr;
|
|
rootPseudoStyle = BeginBuildingScrollContainerFrame(
|
|
state, aDocElement, computedStyle, viewportFrame, rootPseudo, true,
|
|
newFrame);
|
|
parentFrame = newFrame;
|
|
}
|
|
|
|
rootCanvasFrame->SetComputedStyleWithoutNotification(rootPseudoStyle);
|
|
rootCanvasFrame->Init(aDocElement, parentFrame, nullptr);
|
|
|
|
if (isScrollable) {
|
|
FinishBuildingScrollContainerFrame(parentFrame, rootCanvasFrame);
|
|
}
|
|
|
|
if (isPaginated) {
|
|
// Create a page sequence frame
|
|
{
|
|
RefPtr<ComputedStyle> pageSequenceStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::pageSequence, viewportPseudoStyle);
|
|
mPageSequenceFrame =
|
|
NS_NewPageSequenceFrame(mPresShell, pageSequenceStyle);
|
|
mPageSequenceFrame->Init(aDocElement, rootCanvasFrame, nullptr);
|
|
SetInitialSingleChild(rootCanvasFrame, mPageSequenceFrame);
|
|
}
|
|
|
|
// Create the first printed sheet frame, as the sole child (for now) of our
|
|
// page sequence frame (mPageSequenceFrame).
|
|
auto* printedSheetFrame =
|
|
ConstructPrintedSheetFrame(mPresShell, mPageSequenceFrame, nullptr);
|
|
SetInitialSingleChild(mPageSequenceFrame, printedSheetFrame);
|
|
|
|
MOZ_ASSERT(!mNextPageContentFramePageName,
|
|
"Next page name should not have been set.");
|
|
|
|
// Create the first page, as the sole child (for now) of the printed sheet
|
|
// frame that we just created.
|
|
nsCanvasFrame* canvasFrame;
|
|
nsContainerFrame* pageFrame =
|
|
ConstructPageFrame(mPresShell, printedSheetFrame, nullptr, canvasFrame);
|
|
pageFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
SetInitialSingleChild(printedSheetFrame, pageFrame);
|
|
|
|
// The eventual parent of the document element frame.
|
|
// XXX should this be set for every new page (in ConstructPageFrame)?
|
|
mDocElementContainingBlock = canvasFrame;
|
|
}
|
|
|
|
if (viewportFrame->HasAnyStateBits(NS_FRAME_FIRST_REFLOW)) {
|
|
SetInitialSingleChild(viewportFrame, newFrame);
|
|
} else {
|
|
viewportFrame->AppendFrames(FrameChildListID::Principal,
|
|
nsFrameList(newFrame, newFrame));
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ConstructAnonymousContentForCanvas(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aFrame,
|
|
nsIContent* aDocElement, nsFrameList& aFrameList) {
|
|
NS_ASSERTION(aFrame->IsCanvasFrame(), "aFrame should be canvas frame!");
|
|
MOZ_ASSERT(mRootElementFrame->GetContent() == aDocElement);
|
|
|
|
AutoTArray<nsIAnonymousContentCreator::ContentInfo, 4> anonymousItems;
|
|
GetAnonymousContent(aDocElement, aFrame, anonymousItems);
|
|
if (anonymousItems.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
AutoFrameConstructionItemList itemsToConstruct(this);
|
|
AutoFrameConstructionPageName pageNameTracker(aState, aFrame);
|
|
AddFCItemsForAnonymousContent(aState, aFrame, anonymousItems,
|
|
itemsToConstruct, pageNameTracker);
|
|
ConstructFramesFromItemList(aState, itemsToConstruct, aFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
aFrameList);
|
|
}
|
|
|
|
PrintedSheetFrame* nsCSSFrameConstructor::ConstructPrintedSheetFrame(
|
|
PresShell* aPresShell, nsContainerFrame* aParentFrame,
|
|
nsIFrame* aPrevSheetFrame) {
|
|
RefPtr<ComputedStyle> printedSheetPseudoStyle =
|
|
aPresShell->StyleSet()->ResolveNonInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::printedSheet);
|
|
|
|
auto* printedSheetFrame =
|
|
NS_NewPrintedSheetFrame(aPresShell, printedSheetPseudoStyle);
|
|
|
|
printedSheetFrame->Init(nullptr, aParentFrame, aPrevSheetFrame);
|
|
|
|
return printedSheetFrame;
|
|
}
|
|
|
|
nsContainerFrame* nsCSSFrameConstructor::ConstructPageFrame(
|
|
PresShell* aPresShell, nsContainerFrame* aParentFrame,
|
|
nsIFrame* aPrevPageFrame, nsCanvasFrame*& aCanvasFrame) {
|
|
ServoStyleSet* styleSet = aPresShell->StyleSet();
|
|
|
|
RefPtr<ComputedStyle> pagePseudoStyle =
|
|
styleSet->ResolveNonInheritingAnonymousBoxStyle(PseudoStyleType::page);
|
|
|
|
nsContainerFrame* pageFrame = NS_NewPageFrame(aPresShell, pagePseudoStyle);
|
|
|
|
// Initialize the page frame and force it to have a view. This makes printing
|
|
// of the pages easier and faster.
|
|
pageFrame->Init(nullptr, aParentFrame, aPrevPageFrame);
|
|
|
|
RefPtr<const nsAtom> pageName;
|
|
if (mNextPageContentFramePageName) {
|
|
pageName = mNextPageContentFramePageName.forget();
|
|
} else if (aPrevPageFrame) {
|
|
pageName = aPrevPageFrame->ComputePageValue();
|
|
MOZ_ASSERT(pageName,
|
|
"Page name from prev-in-flow should not have been null");
|
|
}
|
|
RefPtr<ComputedStyle> pageContentPseudoStyle =
|
|
styleSet->ResolvePageContentStyle(pageName,
|
|
StylePagePseudoClassFlags::NONE);
|
|
|
|
nsContainerFrame* pageContentFrame = NS_NewPageContentFrame(
|
|
aPresShell, pageContentPseudoStyle, pageName.forget());
|
|
|
|
nsPageContentFrame* prevPageContentFrame = nullptr;
|
|
if (aPrevPageFrame) {
|
|
MOZ_ASSERT(aPrevPageFrame->IsPageFrame());
|
|
prevPageContentFrame =
|
|
static_cast<nsPageFrame*>(aPrevPageFrame)->PageContentFrame();
|
|
}
|
|
pageContentFrame->Init(nullptr, pageFrame, prevPageContentFrame);
|
|
if (!prevPageContentFrame) {
|
|
// The canvas is an inheriting anon box, so needs to be "owned" by the page
|
|
// content.
|
|
pageContentFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
}
|
|
SetInitialSingleChild(pageFrame, pageContentFrame);
|
|
// Make it an absolute container for fixed-pos elements
|
|
pageContentFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
pageContentFrame->MarkAsAbsoluteContainingBlock();
|
|
|
|
RefPtr<ComputedStyle> canvasPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(PseudoStyleType::canvas,
|
|
pageContentPseudoStyle);
|
|
|
|
aCanvasFrame = NS_NewCanvasFrame(aPresShell, canvasPseudoStyle);
|
|
|
|
nsIFrame* prevCanvasFrame = nullptr;
|
|
if (prevPageContentFrame) {
|
|
prevCanvasFrame = prevPageContentFrame->PrincipalChildList().FirstChild();
|
|
NS_ASSERTION(prevCanvasFrame, "missing canvas frame");
|
|
}
|
|
aCanvasFrame->Init(nullptr, pageContentFrame, prevCanvasFrame);
|
|
SetInitialSingleChild(pageContentFrame, aCanvasFrame);
|
|
return pageFrame;
|
|
}
|
|
|
|
/* static */
|
|
nsIFrame* nsCSSFrameConstructor::CreatePlaceholderFrameFor(
|
|
PresShell* aPresShell, nsIContent* aContent, nsIFrame* aFrame,
|
|
nsContainerFrame* aParentFrame, nsIFrame* aPrevInFlow,
|
|
nsFrameState aTypeBit) {
|
|
RefPtr<ComputedStyle> placeholderStyle =
|
|
aPresShell->StyleSet()->ResolveStyleForPlaceholder();
|
|
|
|
// The placeholder frame gets a pseudo style.
|
|
nsPlaceholderFrame* placeholderFrame =
|
|
NS_NewPlaceholderFrame(aPresShell, placeholderStyle, aTypeBit);
|
|
|
|
placeholderFrame->Init(aContent, aParentFrame, aPrevInFlow);
|
|
|
|
// Associate the placeholder/out-of-flow with each other.
|
|
placeholderFrame->SetOutOfFlowFrame(aFrame);
|
|
aFrame->SetProperty(nsIFrame::PlaceholderFrameProperty(), placeholderFrame);
|
|
|
|
aFrame->AddStateBits(NS_FRAME_OUT_OF_FLOW);
|
|
|
|
return placeholderFrame;
|
|
}
|
|
|
|
// Clears any lazy bits set in the range [aStartContent, aEndContent). If
|
|
// aEndContent is null, that means to clear bits in all siblings starting with
|
|
// aStartContent. aStartContent must not be null unless aEndContent is also
|
|
// null. We do this so that when new children are inserted under elements whose
|
|
// frame is a leaf the new children don't cause us to try to construct frames
|
|
// for the existing children again.
|
|
static inline void ClearLazyBits(nsIContent* aStartContent,
|
|
nsIContent* aEndContent) {
|
|
MOZ_ASSERT(aStartContent || !aEndContent,
|
|
"Must have start child if we have an end child");
|
|
|
|
for (nsIContent* cur = aStartContent; cur != aEndContent;
|
|
cur = cur->GetNextSibling()) {
|
|
cur->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindSelectData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
// Construct a frame-based listbox or combobox
|
|
const auto* sel = dom::HTMLSelectElement::FromNode(aElement);
|
|
MOZ_ASSERT(sel);
|
|
if (sel->IsCombobox()) {
|
|
static constexpr FrameConstructionData sComboboxData{
|
|
ToCreationFunc(NS_NewComboboxControlFrame), 0,
|
|
PseudoStyleType::buttonContent};
|
|
return &sComboboxData;
|
|
}
|
|
// FIXME: Can we simplify this to avoid needing ConstructListboxSelectFrame,
|
|
// and reuse ConstructScrollableBlock or so?
|
|
static constexpr FrameConstructionData sListBoxData{
|
|
&nsCSSFrameConstructor::ConstructListBoxSelectFrame};
|
|
return &sListBoxData;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructListBoxSelectFrame(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameList& aFrameList) {
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
// Listbox, not combobox
|
|
nsContainerFrame* listFrame =
|
|
NS_NewListControlFrame(mPresShell, computedStyle);
|
|
|
|
nsContainerFrame* scrolledFrame =
|
|
NS_NewSelectsAreaFrame(mPresShell, computedStyle);
|
|
|
|
// ******* this code stolen from Initialze ScrollFrame ********
|
|
// please adjust this code to use BuildScrollFrame.
|
|
|
|
InitializeListboxSelect(aState, listFrame, scrolledFrame, content,
|
|
aParentFrame, computedStyle, aFrameList);
|
|
|
|
return listFrame;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::InitializeListboxSelect(
|
|
nsFrameConstructorState& aState, nsContainerFrame* scrollFrame,
|
|
nsContainerFrame* scrolledFrame, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, ComputedStyle* aComputedStyle,
|
|
nsFrameList& aFrameList) {
|
|
// Initialize it
|
|
nsContainerFrame* geometricParent =
|
|
aState.GetGeometricParent(*aComputedStyle->StyleDisplay(), aParentFrame);
|
|
|
|
// We don't call InitAndRestoreFrame for scrollFrame because we can only
|
|
// restore the frame state after its parts have been created (in particular,
|
|
// the scrollable view). So we have to split Init and Restore.
|
|
|
|
scrollFrame->Init(aContent, geometricParent, nullptr);
|
|
aState.AddChild(scrollFrame, aFrameList, aContent, aParentFrame);
|
|
BuildScrollContainerFrame(aState, aContent, aComputedStyle, scrolledFrame,
|
|
geometricParent, scrollFrame);
|
|
if (aState.mFrameState) {
|
|
// Restore frame state for the scroll frame
|
|
RestoreFrameStateFor(scrollFrame, aState.mFrameState);
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(scrolledFrame, floatSaveState);
|
|
|
|
// Process children
|
|
nsFrameList childList;
|
|
|
|
ProcessChildren(aState, aContent, aComputedStyle, scrolledFrame, false,
|
|
childList, false);
|
|
|
|
// Set the scrolled frame's initial child lists
|
|
scrolledFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructFieldSetFrame(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameList& aFrameList) {
|
|
AutoRestore<bool> savedHasRenderedLegend(aState.mHasRenderedLegend);
|
|
aState.mHasRenderedLegend = false;
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsContainerFrame* fieldsetFrame =
|
|
NS_NewFieldSetFrame(mPresShell, computedStyle);
|
|
|
|
// Initialize it
|
|
InitAndRestoreFrame(aState, content,
|
|
aState.GetGeometricParent(*aStyleDisplay, aParentFrame),
|
|
fieldsetFrame);
|
|
|
|
fieldsetFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Resolve style and initialize the frame
|
|
RefPtr<ComputedStyle> fieldsetContentStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::fieldsetContent, computedStyle);
|
|
|
|
const nsStyleDisplay* fieldsetContentDisplay =
|
|
fieldsetContentStyle->StyleDisplay();
|
|
const bool isScrollable = fieldsetContentDisplay->IsScrollableOverflow();
|
|
nsContainerFrame* scrollFrame = nullptr;
|
|
if (isScrollable) {
|
|
fieldsetContentStyle = BeginBuildingScrollContainerFrame(
|
|
aState, content, fieldsetContentStyle, fieldsetFrame,
|
|
PseudoStyleType::scrolledContent, false, scrollFrame);
|
|
}
|
|
|
|
// Create the inner ::-moz-fieldset-content frame.
|
|
nsContainerFrame* contentFrameTop;
|
|
nsContainerFrame* contentFrame;
|
|
auto* parent = scrollFrame ? scrollFrame : fieldsetFrame;
|
|
MOZ_ASSERT(fieldsetContentDisplay->DisplayOutside() ==
|
|
StyleDisplayOutside::Block);
|
|
switch (fieldsetContentDisplay->DisplayInside()) {
|
|
case StyleDisplayInside::Flex:
|
|
contentFrame = NS_NewFlexContainerFrame(mPresShell, fieldsetContentStyle);
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
break;
|
|
case StyleDisplayInside::Grid:
|
|
contentFrame = NS_NewGridContainerFrame(mPresShell, fieldsetContentStyle);
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
break;
|
|
default: {
|
|
MOZ_ASSERT(fieldsetContentDisplay->mDisplay == StyleDisplay::Block,
|
|
"bug in StyleAdjuster::adjust_for_fieldset_content?");
|
|
|
|
contentFrame = NS_NewBlockFrame(mPresShell, fieldsetContentStyle);
|
|
if (fieldsetContentStyle->StyleColumn()->IsColumnContainerStyle()) {
|
|
contentFrameTop = BeginBuildingColumns(
|
|
aState, content, parent, contentFrame, fieldsetContentStyle);
|
|
} else {
|
|
// No need to create column container. Initialize content frame.
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
aState.AddChild(fieldsetFrame, aFrameList, content, aParentFrame);
|
|
|
|
// Process children
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
nsFrameList childList;
|
|
|
|
contentFrameTop->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (fieldsetFrame->IsAbsPosContainingBlock()) {
|
|
aState.PushAbsoluteContainingBlock(contentFrameTop, fieldsetFrame,
|
|
absoluteSaveState);
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(contentFrame, floatSaveState);
|
|
|
|
ProcessChildren(aState, content, computedStyle, contentFrame, true, childList,
|
|
true);
|
|
nsFrameList fieldsetKids;
|
|
fieldsetKids.AppendFrame(nullptr,
|
|
scrollFrame ? scrollFrame : contentFrameTop);
|
|
|
|
if (!MayNeedToCreateColumnSpanSiblings(contentFrame, childList)) {
|
|
// Set the inner frame's initial child lists.
|
|
contentFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
} else {
|
|
// Extract any initial non-column-span kids, and put them in inner frame's
|
|
// child list.
|
|
nsFrameList initialNonColumnSpanKids =
|
|
childList.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
contentFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(initialNonColumnSpanKids));
|
|
|
|
if (childList.NotEmpty()) {
|
|
nsFrameList columnSpanSiblings = CreateColumnSpanSiblings(
|
|
aState, contentFrame, childList,
|
|
// Column content should never be a absolute/fixed positioned
|
|
// containing block. Pass nullptr as aPositionedFrame.
|
|
nullptr);
|
|
FinishBuildingColumns(aState, contentFrameTop, contentFrame,
|
|
columnSpanSiblings);
|
|
}
|
|
}
|
|
|
|
if (isScrollable) {
|
|
FinishBuildingScrollContainerFrame(scrollFrame, contentFrameTop);
|
|
}
|
|
|
|
// We use AppendFrames here because the rendered legend will already
|
|
// be present in the principal child list if it exists.
|
|
fieldsetFrame->AppendFrames(FrameChildListID::NoReflowPrincipal,
|
|
std::move(fieldsetKids));
|
|
|
|
return fieldsetFrame;
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDetailsData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
if (!StaticPrefs::layout_details_force_block_layout()) {
|
|
return nullptr;
|
|
}
|
|
static constexpr FrameConstructionData sBlockData[2] = {
|
|
{&nsCSSFrameConstructor::ConstructNonScrollableBlock},
|
|
{&nsCSSFrameConstructor::ConstructScrollableBlock},
|
|
};
|
|
return &sBlockData[aStyle.StyleDisplay()->IsScrollableOverflow()];
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructBlockRubyFrame(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameList& aFrameList) {
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsBlockFrame* blockFrame = NS_NewBlockFrame(mPresShell, computedStyle);
|
|
nsContainerFrame* newFrame = blockFrame;
|
|
nsContainerFrame* geometricParent =
|
|
aState.GetGeometricParent(*aStyleDisplay, aParentFrame);
|
|
AutoFrameConstructionPageName pageNameTracker(aState, blockFrame);
|
|
if ((aItem.mFCData->mBits & FCDATA_MAY_NEED_SCROLLFRAME) &&
|
|
aStyleDisplay->IsScrollableOverflow()) {
|
|
nsContainerFrame* scrollframe = nullptr;
|
|
BuildScrollContainerFrame(aState, content, computedStyle, blockFrame,
|
|
geometricParent, scrollframe);
|
|
newFrame = scrollframe;
|
|
} else {
|
|
InitAndRestoreFrame(aState, content, geometricParent, blockFrame);
|
|
}
|
|
|
|
RefPtr<ComputedStyle> rubyStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::blockRubyContent, computedStyle);
|
|
nsContainerFrame* rubyFrame = NS_NewRubyFrame(mPresShell, rubyStyle);
|
|
InitAndRestoreFrame(aState, content, blockFrame, rubyFrame);
|
|
SetInitialSingleChild(blockFrame, rubyFrame);
|
|
blockFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
aState.AddChild(newFrame, aFrameList, content, aParentFrame);
|
|
|
|
if (!mRootElementFrame) {
|
|
mRootElementFrame = newFrame;
|
|
}
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
blockFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (newFrame->IsAbsPosContainingBlock()) {
|
|
aState.PushAbsoluteContainingBlock(blockFrame, blockFrame,
|
|
absoluteSaveState);
|
|
}
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(blockFrame, floatSaveState);
|
|
|
|
nsFrameList childList;
|
|
ProcessChildren(aState, content, rubyStyle, rubyFrame, true, childList, false,
|
|
nullptr);
|
|
rubyFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
static nsIFrame* FindAncestorWithGeneratedContentPseudo(nsIFrame* aFrame) {
|
|
for (nsIFrame* f = aFrame->GetParent(); f; f = f->GetParent()) {
|
|
NS_ASSERTION(f->IsGeneratedContentFrame(),
|
|
"should not have exited generated content");
|
|
auto pseudo = f->Style()->GetPseudoType();
|
|
if (pseudo == PseudoStyleType::before || pseudo == PseudoStyleType::after ||
|
|
pseudo == PseudoStyleType::marker) {
|
|
return f;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindTextData(const Text& aTextContent,
|
|
nsIFrame* aParentFrame) {
|
|
if (aParentFrame && IsFrameForSVG(aParentFrame)) {
|
|
if (!aParentFrame->IsInSVGTextSubtree()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// FIXME(bug 1588477) Don't render stuff in display: contents / Shadow DOM
|
|
// subtrees, because TextCorrespondenceRecorder in the SVG text code doesn't
|
|
// really know how to deal with it. This kinda sucks. :(
|
|
if (aParentFrame->GetContent() != aTextContent.GetParent()) {
|
|
return nullptr;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sSVGTextData(
|
|
NS_NewTextFrame, FCDATA_IS_LINE_PARTICIPANT | FCDATA_IS_SVG_TEXT);
|
|
return &sSVGTextData;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sTextData(NS_NewTextFrame,
|
|
FCDATA_IS_LINE_PARTICIPANT);
|
|
return &sTextData;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ConstructTextFrame(
|
|
const FrameConstructionData* aData, nsFrameConstructorState& aState,
|
|
nsIContent* aContent, nsContainerFrame* aParentFrame,
|
|
ComputedStyle* aComputedStyle, nsFrameList& aFrameList) {
|
|
MOZ_ASSERT(aData, "Must have frame construction data");
|
|
|
|
nsIFrame* newFrame =
|
|
(*aData->mFunc.mCreationFunc)(mPresShell, aComputedStyle);
|
|
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, newFrame);
|
|
|
|
// We never need to create a view for a text frame.
|
|
|
|
if (newFrame->IsGeneratedContentFrame()) {
|
|
UniquePtr<nsGenConInitializer> initializer(
|
|
static_cast<nsGenConInitializer*>(
|
|
aContent->TakeProperty(nsGkAtoms::genConInitializerProperty)));
|
|
if (initializer) {
|
|
if (initializer->mNode.release()->InitTextFrame(
|
|
initializer->mList,
|
|
FindAncestorWithGeneratedContentPseudo(newFrame), newFrame)) {
|
|
(this->*(initializer->mDirtyAll))();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add the newly constructed frame to the flow
|
|
aFrameList.AppendFrame(nullptr, newFrame);
|
|
|
|
if (!aState.mCreatingExtraFrames || (aContent->IsInNativeAnonymousSubtree() &&
|
|
!aContent->GetPrimaryFrame())) {
|
|
aContent->SetPrimaryFrame(newFrame);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDataByInt(int32_t aInt, const Element& aElement,
|
|
ComputedStyle& aComputedStyle,
|
|
const FrameConstructionDataByInt* aDataPtr,
|
|
uint32_t aDataLength) {
|
|
for (const FrameConstructionDataByInt *curData = aDataPtr,
|
|
*endData = aDataPtr + aDataLength;
|
|
curData != endData; ++curData) {
|
|
if (curData->mInt == aInt) {
|
|
const FrameConstructionData* data = &curData->mData;
|
|
if (data->mBits & FCDATA_FUNC_IS_DATA_GETTER) {
|
|
return data->mFunc.mDataGetter(aElement, aComputedStyle);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDataByTag(const Element& aElement,
|
|
ComputedStyle& aStyle,
|
|
const FrameConstructionDataByTag* aDataPtr,
|
|
uint32_t aDataLength) {
|
|
const nsAtom* tag = aElement.NodeInfo()->NameAtom();
|
|
for (const FrameConstructionDataByTag *curData = aDataPtr,
|
|
*endData = aDataPtr + aDataLength;
|
|
curData != endData; ++curData) {
|
|
if (curData->mTag == tag) {
|
|
const FrameConstructionData* data = &curData->mData;
|
|
if (data->mBits & FCDATA_FUNC_IS_DATA_GETTER) {
|
|
return data->mFunc.mDataGetter(aElement, aStyle);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
#define SUPPRESS_FCDATA() FrameConstructionData(nullptr, FCDATA_SUPPRESS_FRAME)
|
|
#define SIMPLE_INT_CREATE(_int, _func) \
|
|
{int32_t(_int), FrameConstructionData(_func)}
|
|
#define SIMPLE_INT_CHAIN(_int, _func) \
|
|
{int32_t(_int), FrameConstructionData(_func)}
|
|
#define COMPLEX_INT_CREATE(_int, _func) \
|
|
{int32_t(_int), FrameConstructionData(_func)}
|
|
|
|
#define SIMPLE_TAG_CREATE(_tag, _func) \
|
|
{nsGkAtoms::_tag, FrameConstructionData(_func)}
|
|
#define SIMPLE_TAG_CHAIN(_tag, _func) \
|
|
{nsGkAtoms::_tag, FrameConstructionData(_func)}
|
|
#define COMPLEX_TAG_CREATE(_tag, _func) \
|
|
{nsGkAtoms::_tag, FrameConstructionData(_func)}
|
|
|
|
static nsFieldSetFrame* GetFieldSetFrameFor(nsIFrame* aFrame) {
|
|
auto pseudo = aFrame->Style()->GetPseudoType();
|
|
if (pseudo == PseudoStyleType::fieldsetContent ||
|
|
pseudo == PseudoStyleType::scrolledContent ||
|
|
pseudo == PseudoStyleType::columnSet ||
|
|
pseudo == PseudoStyleType::columnContent) {
|
|
return GetFieldSetFrameFor(aFrame->GetParent());
|
|
}
|
|
return do_QueryFrame(aFrame);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindHTMLData(const Element& aElement,
|
|
nsIFrame* aParentFrame,
|
|
ComputedStyle& aStyle) {
|
|
MOZ_ASSERT(aElement.IsHTMLElement());
|
|
NS_ASSERTION(!aParentFrame ||
|
|
aParentFrame->Style()->GetPseudoType() !=
|
|
PseudoStyleType::fieldsetContent ||
|
|
aParentFrame->GetParent()->IsFieldSetFrame(),
|
|
"Unexpected parent for fieldset content anon box");
|
|
|
|
if (aElement.IsInNativeAnonymousSubtree() &&
|
|
aElement.NodeInfo()->NameAtom() == nsGkAtoms::label && aParentFrame) {
|
|
if (aParentFrame->IsFileControlFrame()) {
|
|
static constexpr FrameConstructionData sFileLabelData(
|
|
NS_NewFileControlLabelFrame);
|
|
return &sFileLabelData;
|
|
}
|
|
if (aParentFrame->GetParent() &&
|
|
aParentFrame->GetParent()->IsComboboxControlFrame()) {
|
|
static constexpr FrameConstructionData sComboboxLabelData(
|
|
NS_NewComboboxLabelFrame);
|
|
return &sComboboxLabelData;
|
|
}
|
|
}
|
|
|
|
static constexpr FrameConstructionDataByTag sHTMLData[] = {
|
|
SIMPLE_TAG_CHAIN(img, nsCSSFrameConstructor::FindImgData),
|
|
SIMPLE_TAG_CHAIN(mozgeneratedcontentimage,
|
|
nsCSSFrameConstructor::FindGeneratedImageData),
|
|
{nsGkAtoms::br,
|
|
{NS_NewBRFrame, FCDATA_IS_LINE_PARTICIPANT | FCDATA_IS_LINE_BREAK}},
|
|
SIMPLE_TAG_CREATE(wbr, NS_NewWBRFrame),
|
|
SIMPLE_TAG_CHAIN(input, nsCSSFrameConstructor::FindInputData),
|
|
SIMPLE_TAG_CREATE(textarea, NS_NewTextControlFrame),
|
|
SIMPLE_TAG_CHAIN(select, nsCSSFrameConstructor::FindSelectData),
|
|
SIMPLE_TAG_CHAIN(object, nsCSSFrameConstructor::FindObjectData),
|
|
SIMPLE_TAG_CHAIN(embed, nsCSSFrameConstructor::FindObjectData),
|
|
COMPLEX_TAG_CREATE(fieldset,
|
|
&nsCSSFrameConstructor::ConstructFieldSetFrame),
|
|
SIMPLE_TAG_CREATE(frameset, NS_NewHTMLFramesetFrame),
|
|
SIMPLE_TAG_CREATE(iframe, NS_NewSubDocumentFrame),
|
|
{nsGkAtoms::button,
|
|
{ToCreationFunc(NS_NewHTMLButtonControlFrame),
|
|
FCDATA_ALLOW_BLOCK_STYLES | FCDATA_ALLOW_GRID_FLEX_COLUMN,
|
|
PseudoStyleType::buttonContent}},
|
|
SIMPLE_TAG_CHAIN(canvas, nsCSSFrameConstructor::FindCanvasData),
|
|
SIMPLE_TAG_CREATE(video, NS_NewHTMLVideoFrame),
|
|
SIMPLE_TAG_CREATE(audio, NS_NewHTMLAudioFrame),
|
|
SIMPLE_TAG_CREATE(progress, NS_NewProgressFrame),
|
|
SIMPLE_TAG_CREATE(meter, NS_NewMeterFrame),
|
|
SIMPLE_TAG_CHAIN(details, nsCSSFrameConstructor::FindDetailsData),
|
|
};
|
|
|
|
return FindDataByTag(aElement, aStyle, sHTMLData, std::size(sHTMLData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindGeneratedImageData(const Element& aElement,
|
|
ComputedStyle&) {
|
|
if (!aElement.IsInNativeAnonymousSubtree()) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto& generatedContent = static_cast<const GeneratedImageContent&>(aElement);
|
|
if (generatedContent.IsForListStyleImageMarker()) {
|
|
static constexpr FrameConstructionData sImgData(
|
|
NS_NewImageFrameForListStyleImage);
|
|
return &sImgData;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sImgData(
|
|
NS_NewImageFrameForGeneratedContentIndex);
|
|
return &sImgData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindImgData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
if (nsImageFrame::ImageFrameTypeFor(aElement, aStyle) !=
|
|
nsImageFrame::ImageFrameType::ForElementRequest) {
|
|
// content: url gets handled by the generic code-path.
|
|
return nullptr;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sImgData(NS_NewImageFrame);
|
|
return &sImgData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindImgControlData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
if (nsImageFrame::ImageFrameTypeFor(aElement, aStyle) !=
|
|
nsImageFrame::ImageFrameType::ForElementRequest) {
|
|
return nullptr;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sImgControlData(
|
|
NS_NewImageControlFrame);
|
|
return &sImgControlData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindSearchControlData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
if (StaticPrefs::layout_forms_input_type_search_enabled()) {
|
|
static constexpr FrameConstructionData sSearchControlData(
|
|
NS_NewSearchControlFrame);
|
|
return &sSearchControlData;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sTextControlData(
|
|
NS_NewTextControlFrame);
|
|
return &sTextControlData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindInputData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
static constexpr FrameConstructionDataByInt sInputData[] = {
|
|
SIMPLE_INT_CREATE(FormControlType::InputCheckbox,
|
|
ToCreationFunc(NS_NewCheckboxRadioFrame)),
|
|
SIMPLE_INT_CREATE(FormControlType::InputRadio,
|
|
ToCreationFunc(NS_NewCheckboxRadioFrame)),
|
|
SIMPLE_INT_CREATE(FormControlType::InputFile, NS_NewFileControlFrame),
|
|
SIMPLE_INT_CHAIN(FormControlType::InputImage,
|
|
nsCSSFrameConstructor::FindImgControlData),
|
|
SIMPLE_INT_CREATE(FormControlType::InputEmail, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputText, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputTel, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputUrl, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputRange, NS_NewRangeFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputPassword, NS_NewTextControlFrame),
|
|
{int32_t(FormControlType::InputColor),
|
|
{NS_NewColorControlFrame, 0, PseudoStyleType::buttonContent}},
|
|
|
|
SIMPLE_INT_CHAIN(FormControlType::InputSearch,
|
|
nsCSSFrameConstructor::FindSearchControlData),
|
|
SIMPLE_INT_CREATE(FormControlType::InputNumber, NS_NewNumberControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputTime, NS_NewDateTimeControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputDate, NS_NewDateTimeControlFrame),
|
|
SIMPLE_INT_CREATE(FormControlType::InputDatetimeLocal,
|
|
NS_NewDateTimeControlFrame),
|
|
// TODO: this is temporary until a frame is written: bug 888320
|
|
SIMPLE_INT_CREATE(FormControlType::InputMonth, NS_NewTextControlFrame),
|
|
// TODO: this is temporary until a frame is written: bug 888320
|
|
SIMPLE_INT_CREATE(FormControlType::InputWeek, NS_NewTextControlFrame),
|
|
{int32_t(FormControlType::InputSubmit),
|
|
{ToCreationFunc(NS_NewGfxButtonControlFrame), 0,
|
|
PseudoStyleType::buttonContent}},
|
|
{int32_t(FormControlType::InputReset),
|
|
{ToCreationFunc(NS_NewGfxButtonControlFrame), 0,
|
|
PseudoStyleType::buttonContent}},
|
|
{int32_t(FormControlType::InputButton),
|
|
{ToCreationFunc(NS_NewGfxButtonControlFrame), 0,
|
|
PseudoStyleType::buttonContent}}
|
|
// Keeping hidden inputs out of here on purpose for so they get frames by
|
|
// display (in practice, none).
|
|
};
|
|
|
|
auto controlType = HTMLInputElement::FromNode(aElement)->ControlType();
|
|
|
|
// radio and checkbox inputs with appearance:none should be constructed
|
|
// by display type. (Note that we're not checking that appearance is
|
|
// not (respectively) StyleAppearance::Radio and StyleAppearance::Checkbox.)
|
|
if ((controlType == FormControlType::InputCheckbox ||
|
|
controlType == FormControlType::InputRadio) &&
|
|
!aStyle.StyleDisplay()->HasAppearance()) {
|
|
return nullptr;
|
|
}
|
|
|
|
return FindDataByInt(int32_t(controlType), aElement, aStyle, sInputData,
|
|
std::size(sInputData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindObjectData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
uint32_t type;
|
|
nsCOMPtr<nsIObjectLoadingContent> objContent =
|
|
do_QueryInterface(const_cast<Element*>(&aElement));
|
|
NS_ASSERTION(objContent,
|
|
"embed and object must implement "
|
|
"nsIObjectLoadingContent!");
|
|
objContent->GetDisplayedType(&type);
|
|
|
|
static constexpr FrameConstructionDataByInt sObjectData[] = {
|
|
// TODO(emilio): Can we remove the NS_NewEmptyFrame case and just use a
|
|
// subdocument frame here?
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_LOADING,
|
|
NS_NewEmptyFrame),
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_DOCUMENT,
|
|
NS_NewSubDocumentFrame),
|
|
// Nothing for TYPE_FALLBACK so we'll construct frames by display there
|
|
};
|
|
|
|
return FindDataByInt((int32_t)type, aElement, aStyle, sObjectData,
|
|
std::size(sObjectData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindCanvasData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
// We want to check whether script is enabled on the document that
|
|
// could be painting to the canvas. That's the owner document of
|
|
// the canvas, except when the owner document is a static document,
|
|
// in which case it's the original document it was cloned from.
|
|
Document* doc = aElement.OwnerDoc();
|
|
if (doc->IsStaticDocument()) {
|
|
doc = doc->GetOriginalDocument();
|
|
}
|
|
if (!doc->IsScriptEnabled()) {
|
|
return nullptr;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sCanvasData(
|
|
NS_NewHTMLCanvasFrame, 0, PseudoStyleType::htmlCanvasContent);
|
|
return &sCanvasData;
|
|
}
|
|
|
|
static MOZ_NEVER_INLINE void DestroyFramesInList(PresShell* aPs,
|
|
nsFrameList& aList) {
|
|
nsIFrame::DestroyContext context(aPs);
|
|
aList.DestroyFrames(context);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ConstructFrameFromItemInternal(
|
|
FrameConstructionItem& aItem, nsFrameConstructorState& aState,
|
|
nsContainerFrame* aParentFrame, nsFrameList& aFrameList) {
|
|
const FrameConstructionData* data = aItem.mFCData;
|
|
NS_ASSERTION(data, "Must have frame construction data");
|
|
|
|
uint32_t bits = data->mBits;
|
|
|
|
NS_ASSERTION(!(bits & FCDATA_FUNC_IS_DATA_GETTER),
|
|
"Should have dealt with this inside the data finder");
|
|
|
|
// Some sets of bits are not compatible with each other
|
|
#define CHECK_ONLY_ONE_BIT(_bit1, _bit2) \
|
|
NS_ASSERTION(!(bits & _bit1) || !(bits & _bit2), \
|
|
"Only one of these bits should be set")
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR,
|
|
FCDATA_FORCE_NULL_ABSPOS_CONTAINER);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR, FCDATA_WRAP_KIDS_IN_BLOCKS);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR, FCDATA_IS_POPUP);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR, FCDATA_SKIP_ABSPOS_PUSH);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR,
|
|
FCDATA_DISALLOW_GENERATED_CONTENT);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR, FCDATA_ALLOW_BLOCK_STYLES);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_FUNC_IS_FULL_CTOR,
|
|
FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS);
|
|
CHECK_ONLY_ONE_BIT(FCDATA_WRAP_KIDS_IN_BLOCKS,
|
|
FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS);
|
|
#undef CHECK_ONLY_ONE_BIT
|
|
MOZ_ASSERT(
|
|
!(bits & FCDATA_IS_WRAPPER_ANON_BOX) || (bits & FCDATA_USE_CHILD_ITEMS),
|
|
"Wrapper anon boxes should always have FCDATA_USE_CHILD_ITEMS");
|
|
MOZ_ASSERT(!(bits & FCDATA_ALLOW_GRID_FLEX_COLUMN) ||
|
|
(bits & FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS),
|
|
"Need the block wrapper bit to create grid/flex/column.");
|
|
|
|
// Don't create a subdocument frame for iframes if we're creating extra frames
|
|
if (aState.mCreatingExtraFrames &&
|
|
aItem.mContent->IsHTMLElement(nsGkAtoms::iframe)) {
|
|
return;
|
|
}
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
nsIFrame* newFrame;
|
|
nsIFrame* primaryFrame;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
const nsStyleDisplay* display = computedStyle->StyleDisplay();
|
|
if (bits & FCDATA_FUNC_IS_FULL_CTOR) {
|
|
newFrame = (this->*(data->mFunc.mFullConstructor))(
|
|
aState, aItem, aParentFrame, display, aFrameList);
|
|
MOZ_ASSERT(newFrame, "Full constructor failed");
|
|
primaryFrame = newFrame;
|
|
} else {
|
|
newFrame = (*data->mFunc.mCreationFunc)(mPresShell, computedStyle);
|
|
|
|
const bool allowOutOfFlow = !(bits & FCDATA_DISALLOW_OUT_OF_FLOW);
|
|
const bool isPopup = aItem.mIsPopup;
|
|
|
|
nsContainerFrame* geometricParent =
|
|
(isPopup || allowOutOfFlow)
|
|
? aState.GetGeometricParent(*display, aParentFrame)
|
|
: aParentFrame;
|
|
|
|
// In the non-scrollframe case, primaryFrame and newFrame are equal; in the
|
|
// scrollframe case, newFrame is the scrolled frame while primaryFrame is
|
|
// the scrollframe.
|
|
if ((bits & FCDATA_MAY_NEED_SCROLLFRAME) &&
|
|
display->IsScrollableOverflow()) {
|
|
nsContainerFrame* scrollframe = nullptr;
|
|
BuildScrollContainerFrame(aState, content, computedStyle, newFrame,
|
|
geometricParent, scrollframe);
|
|
primaryFrame = scrollframe;
|
|
} else {
|
|
InitAndRestoreFrame(aState, content, geometricParent, newFrame);
|
|
primaryFrame = newFrame;
|
|
}
|
|
|
|
// If we need to create a block formatting context to wrap our
|
|
// kids, do it now.
|
|
nsIFrame* maybeAbsoluteContainingBlockStyleFrame = primaryFrame;
|
|
nsIFrame* maybeAbsoluteContainingBlock = newFrame;
|
|
nsIFrame* possiblyLeafFrame = newFrame;
|
|
nsContainerFrame* outerFrame = nullptr;
|
|
if (bits & FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS) {
|
|
RefPtr<ComputedStyle> outerStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
data->mAnonBoxPseudo, computedStyle);
|
|
#ifdef DEBUG
|
|
nsContainerFrame* containerFrame = do_QueryFrame(newFrame);
|
|
MOZ_ASSERT(containerFrame);
|
|
#endif
|
|
nsContainerFrame* container = static_cast<nsContainerFrame*>(newFrame);
|
|
nsContainerFrame* innerFrame;
|
|
if (bits & FCDATA_ALLOW_GRID_FLEX_COLUMN) {
|
|
switch (display->DisplayInside()) {
|
|
case StyleDisplayInside::Flex:
|
|
outerFrame = NS_NewFlexContainerFrame(mPresShell, outerStyle);
|
|
InitAndRestoreFrame(aState, content, container, outerFrame);
|
|
innerFrame = outerFrame;
|
|
break;
|
|
case StyleDisplayInside::Grid:
|
|
outerFrame = NS_NewGridContainerFrame(mPresShell, outerStyle);
|
|
InitAndRestoreFrame(aState, content, container, outerFrame);
|
|
innerFrame = outerFrame;
|
|
break;
|
|
default: {
|
|
innerFrame = NS_NewBlockFrame(mPresShell, outerStyle);
|
|
if (outerStyle->StyleColumn()->IsColumnContainerStyle()) {
|
|
outerFrame = BeginBuildingColumns(aState, content, container,
|
|
innerFrame, outerStyle);
|
|
} else {
|
|
// No need to create column container. Initialize innerFrame.
|
|
InitAndRestoreFrame(aState, content, container, innerFrame);
|
|
outerFrame = innerFrame;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
innerFrame = NS_NewBlockFrame(mPresShell, outerStyle);
|
|
InitAndRestoreFrame(aState, content, container, innerFrame);
|
|
outerFrame = innerFrame;
|
|
}
|
|
|
|
SetInitialSingleChild(container, outerFrame);
|
|
|
|
container->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Now figure out whether newFrame or outerFrame should be the
|
|
// absolute container.
|
|
if (outerFrame->IsAbsPosContainingBlock()) {
|
|
maybeAbsoluteContainingBlock = outerFrame;
|
|
maybeAbsoluteContainingBlockStyleFrame = outerFrame;
|
|
innerFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
}
|
|
|
|
// Our kids should go into the innerFrame.
|
|
newFrame = innerFrame;
|
|
}
|
|
|
|
aState.AddChild(primaryFrame, aFrameList, content, aParentFrame,
|
|
allowOutOfFlow, allowOutOfFlow);
|
|
|
|
nsContainerFrame* newFrameAsContainer = do_QueryFrame(newFrame);
|
|
if (newFrameAsContainer) {
|
|
// Process the child content if requested
|
|
nsFrameList childList;
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
if (bits & FCDATA_FORCE_NULL_ABSPOS_CONTAINER) {
|
|
aState.PushAbsoluteContainingBlock(nullptr, nullptr, absoluteSaveState);
|
|
} else if (!(bits & FCDATA_SKIP_ABSPOS_PUSH)) {
|
|
maybeAbsoluteContainingBlock->AddStateBits(
|
|
NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (maybeAbsoluteContainingBlockStyleFrame->IsAbsPosContainingBlock()) {
|
|
auto* cf =
|
|
static_cast<nsContainerFrame*>(maybeAbsoluteContainingBlock);
|
|
aState.PushAbsoluteContainingBlock(
|
|
cf, maybeAbsoluteContainingBlockStyleFrame, absoluteSaveState);
|
|
}
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(newFrameAsContainer, floatSaveState);
|
|
|
|
if (bits & FCDATA_USE_CHILD_ITEMS) {
|
|
// At this point, we have not set up the auto value for this frame, and
|
|
// no caller will have set it so it is not redundant and therefor will
|
|
// not assert.
|
|
AutoFrameConstructionPageName pageNameTracker(aState,
|
|
newFrameAsContainer);
|
|
ConstructFramesFromItemList(
|
|
aState, aItem.mChildItems, newFrameAsContainer,
|
|
bits & FCDATA_IS_WRAPPER_ANON_BOX, childList);
|
|
} else {
|
|
// Process the child frames.
|
|
ProcessChildren(aState, content, computedStyle, newFrameAsContainer,
|
|
!(bits & FCDATA_DISALLOW_GENERATED_CONTENT), childList,
|
|
(bits & FCDATA_ALLOW_BLOCK_STYLES) != 0,
|
|
possiblyLeafFrame);
|
|
}
|
|
|
|
if (bits & FCDATA_WRAP_KIDS_IN_BLOCKS) {
|
|
nsFrameList newList;
|
|
nsFrameList currentBlockList;
|
|
nsIFrame* f;
|
|
while ((f = childList.FirstChild()) != nullptr) {
|
|
bool wrapFrame = IsInlineFrame(f) || IsFramePartOfIBSplit(f);
|
|
if (!wrapFrame) {
|
|
FlushAccumulatedBlock(aState, content, newFrameAsContainer,
|
|
currentBlockList, newList);
|
|
}
|
|
|
|
childList.RemoveFrame(f);
|
|
if (wrapFrame) {
|
|
currentBlockList.AppendFrame(nullptr, f);
|
|
} else {
|
|
newList.AppendFrame(nullptr, f);
|
|
}
|
|
}
|
|
FlushAccumulatedBlock(aState, content, newFrameAsContainer,
|
|
currentBlockList, newList);
|
|
|
|
if (childList.NotEmpty()) {
|
|
// an error must have occurred, delete unprocessed frames
|
|
DestroyFramesInList(mPresShell, childList);
|
|
}
|
|
|
|
childList = std::move(newList);
|
|
}
|
|
|
|
if (!(bits & FCDATA_ALLOW_GRID_FLEX_COLUMN) ||
|
|
!MayNeedToCreateColumnSpanSiblings(newFrameAsContainer, childList)) {
|
|
// Set the frame's initial child list. Note that MathML depends on this
|
|
// being called even if childList is empty!
|
|
newFrameAsContainer->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
} else {
|
|
// Extract any initial non-column-span kids, and put them in inner
|
|
// frame's child list.
|
|
nsFrameList initialNonColumnSpanKids =
|
|
childList.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
newFrameAsContainer->SetInitialChildList(
|
|
FrameChildListID::Principal, std::move(initialNonColumnSpanKids));
|
|
|
|
if (childList.NotEmpty()) {
|
|
nsFrameList columnSpanSiblings = CreateColumnSpanSiblings(
|
|
aState, newFrameAsContainer, childList,
|
|
// Column content should never be a absolute/fixed positioned
|
|
// containing block. Pass nullptr as aPositionedFrame.
|
|
nullptr);
|
|
|
|
MOZ_ASSERT(outerFrame,
|
|
"outerFrame should be non-null if multi-column container "
|
|
"is created.");
|
|
FinishBuildingColumns(aState, outerFrame, newFrameAsContainer,
|
|
columnSpanSiblings);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
NS_ASSERTION(newFrame->IsLineParticipant() ==
|
|
((bits & FCDATA_IS_LINE_PARTICIPANT) != 0),
|
|
"Incorrectly set FCDATA_IS_LINE_PARTICIPANT bits");
|
|
|
|
// Even if mCreatingExtraFrames is set, we may need to SetPrimaryFrame for
|
|
// generated content that doesn't have one yet. Note that we have to examine
|
|
// the frame bit, because by this point mIsGeneratedContent has been cleared
|
|
// on aItem.
|
|
if ((!aState.mCreatingExtraFrames ||
|
|
(aItem.mContent->IsRootOfNativeAnonymousSubtree() &&
|
|
!aItem.mContent->GetPrimaryFrame())) &&
|
|
!(bits & FCDATA_SKIP_FRAMESET)) {
|
|
aItem.mContent->SetPrimaryFrame(primaryFrame);
|
|
ActiveLayerTracker::TransferActivityToFrame(aItem.mContent, primaryFrame);
|
|
}
|
|
}
|
|
|
|
static void GatherSubtreeElements(Element* aElement,
|
|
nsTArray<Element*>& aElements) {
|
|
aElements.AppendElement(aElement);
|
|
StyleChildrenIterator iter(aElement);
|
|
for (nsIContent* c = iter.GetNextChild(); c; c = iter.GetNextChild()) {
|
|
if (!c->IsElement()) {
|
|
continue;
|
|
}
|
|
GatherSubtreeElements(c->AsElement(), aElements);
|
|
}
|
|
}
|
|
|
|
nsresult nsCSSFrameConstructor::GetAnonymousContent(
|
|
nsIContent* aParent, nsIFrame* aParentFrame,
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>& aContent) {
|
|
nsIAnonymousContentCreator* creator = do_QueryFrame(aParentFrame);
|
|
if (!creator) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult rv = creator->CreateAnonymousContent(aContent);
|
|
if (NS_FAILED(rv)) {
|
|
// CreateAnonymousContent failed, e.g. because the page has a <use> loop.
|
|
return rv;
|
|
}
|
|
|
|
if (aContent.IsEmpty()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
const bool devtoolsEventsEnabled =
|
|
mDocument->DevToolsAnonymousAndShadowEventsEnabled();
|
|
|
|
MOZ_ASSERT(aParent->IsElement());
|
|
for (const auto& info : aContent) {
|
|
// get our child's content and set its parent to our content
|
|
nsIContent* content = info.mContent;
|
|
content->SetIsNativeAnonymousRoot();
|
|
|
|
BindContext context(*aParent->AsElement(), BindContext::ForNativeAnonymous);
|
|
rv = content->BindToTree(context, *aParent);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
content->UnbindFromTree();
|
|
return rv;
|
|
}
|
|
|
|
if (devtoolsEventsEnabled) {
|
|
content->QueueDevtoolsAnonymousEvent(/* aIsRemove = */ false);
|
|
}
|
|
}
|
|
|
|
// Some situations where we don't cache anonymous content styles:
|
|
//
|
|
// * when visibility or pointer-events is anything other than the initial
|
|
// value; we rely on visibility and pointer-events inheriting into anonymous
|
|
// content, but don't bother adding this state to the AnonymousContentKey,
|
|
// since it's not so common. Note that with overlay scrollbars, scrollbars
|
|
// always start off with pointer-events: none so we don't need to check for
|
|
// that in that case.
|
|
//
|
|
// * when the medium is anything other than screen; some UA style sheet rules
|
|
// apply in e.g. print medium, and will give different results from the
|
|
// cached styles
|
|
Maybe<bool> computedAllowStyleCaching;
|
|
auto ComputeAllowStyleCaching = [&] {
|
|
if (!StaticPrefs::layout_css_cached_scrollbar_styles_enabled()) {
|
|
return false;
|
|
}
|
|
if (aParentFrame->StyleVisibility()->mVisible != StyleVisibility::Visible) {
|
|
return false;
|
|
}
|
|
nsPresContext* pc = mPresShell->GetPresContext();
|
|
if (!pc->UseOverlayScrollbars() &&
|
|
aParentFrame->StyleUI()->ComputedPointerEvents() !=
|
|
StylePointerEvents::Auto) {
|
|
return false;
|
|
}
|
|
if (pc->Medium() != nsGkAtoms::screen) {
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
|
|
auto AllowStyleCaching = [&] {
|
|
if (computedAllowStyleCaching.isNothing()) {
|
|
computedAllowStyleCaching.emplace(ComputeAllowStyleCaching());
|
|
}
|
|
return computedAllowStyleCaching.value();
|
|
};
|
|
|
|
// Compute styles for the anonymous content tree.
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
for (auto& info : aContent) {
|
|
Element* e = Element::FromNode(info.mContent);
|
|
if (!e) {
|
|
continue;
|
|
}
|
|
|
|
if (info.mKey == AnonymousContentKey::None || !AllowStyleCaching()) {
|
|
// Most NAC subtrees do not use caching of computed styles. Just go
|
|
// ahead and eagerly style the subtree.
|
|
styleSet->StyleNewSubtree(e);
|
|
continue;
|
|
}
|
|
|
|
// We have a NAC subtree for which we can use cached styles.
|
|
AutoTArray<RefPtr<ComputedStyle>, 2> cachedStyles;
|
|
AutoTArray<Element*, 2> elements;
|
|
|
|
GatherSubtreeElements(e, elements);
|
|
styleSet->GetCachedAnonymousContentStyles(info.mKey, cachedStyles);
|
|
|
|
if (cachedStyles.IsEmpty()) {
|
|
// We haven't stored cached styles for this kind of NAC subtree yet.
|
|
// Eagerly compute those styles, then cache them for later.
|
|
styleSet->StyleNewSubtree(e);
|
|
for (Element* e : elements) {
|
|
if (e->HasServoData()) {
|
|
cachedStyles.AppendElement(ServoStyleSet::ResolveServoStyle(*e));
|
|
} else {
|
|
cachedStyles.AppendElement(nullptr);
|
|
}
|
|
}
|
|
styleSet->PutCachedAnonymousContentStyles(info.mKey,
|
|
std::move(cachedStyles));
|
|
continue;
|
|
}
|
|
|
|
// We previously stored cached styles for this kind of NAC subtree.
|
|
// Iterate over them and set them on the subtree's elements.
|
|
MOZ_ASSERT(cachedStyles.Length() == elements.Length(),
|
|
"should always produce the same size NAC subtree");
|
|
for (size_t i = 0, len = cachedStyles.Length(); i != len; ++i) {
|
|
if (cachedStyles[i]) {
|
|
#ifdef DEBUG
|
|
// Assert that our cached style is the same as one we could compute.
|
|
RefPtr<ComputedStyle> cs = styleSet->ResolveStyleLazily(*elements[i]);
|
|
MOZ_ASSERT(
|
|
cachedStyles[i]->EqualForCachedAnonymousContentStyle(*cs),
|
|
"cached anonymous content styles should be identical to those we "
|
|
"would compute normally");
|
|
// All overlay scrollbars start off as inactive, so we can rely on their
|
|
// pointer-events value being always none.
|
|
MOZ_ASSERT(!mPresShell->GetPresContext()->UseOverlayScrollbars() ||
|
|
cs->StyleUI()->ComputedPointerEvents() ==
|
|
StylePointerEvents::None);
|
|
#endif
|
|
Servo_SetExplicitStyle(elements[i], cachedStyles[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// XUL frames are not allowed to be out of flow.
|
|
#define SIMPLE_XUL_FCDATA(_func) \
|
|
FrameConstructionData(_func, \
|
|
FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_SKIP_ABSPOS_PUSH)
|
|
#define SCROLLABLE_XUL_FCDATA(_func) \
|
|
FrameConstructionData(_func, FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_SKIP_ABSPOS_PUSH | \
|
|
FCDATA_MAY_NEED_SCROLLFRAME)
|
|
// .. but we allow some XUL frames to be _containers_ for out-of-flow content
|
|
// (This is the same as SCROLLABLE_XUL_FCDATA, but w/o FCDATA_SKIP_ABSPOS_PUSH)
|
|
#define SCROLLABLE_ABSPOS_CONTAINER_XUL_FCDATA(_func) \
|
|
FrameConstructionData( \
|
|
_func, FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_MAY_NEED_SCROLLFRAME)
|
|
|
|
#define SIMPLE_XUL_CREATE(_tag, _func) \
|
|
{nsGkAtoms::_tag, SIMPLE_XUL_FCDATA(_func)}
|
|
#define SCROLLABLE_XUL_CREATE(_tag, _func) \
|
|
{nsGkAtoms::_tag, SCROLLABLE_XUL_FCDATA(_func)}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULTagData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
MOZ_ASSERT(aElement.IsXULElement());
|
|
static constexpr FrameConstructionData kPopupData(
|
|
NS_NewMenuPopupFrame, FCDATA_IS_POPUP | FCDATA_SKIP_ABSPOS_PUSH);
|
|
|
|
static constexpr FrameConstructionDataByTag sXULTagData[] = {
|
|
SIMPLE_XUL_CREATE(image, NS_NewXULImageFrame),
|
|
SIMPLE_XUL_CREATE(treechildren, NS_NewTreeBodyFrame),
|
|
SIMPLE_TAG_CHAIN(label,
|
|
nsCSSFrameConstructor::FindXULLabelOrDescriptionData),
|
|
SIMPLE_TAG_CHAIN(description,
|
|
nsCSSFrameConstructor::FindXULLabelOrDescriptionData),
|
|
#ifdef XP_MACOSX
|
|
SIMPLE_TAG_CHAIN(menubar, nsCSSFrameConstructor::FindXULMenubarData),
|
|
#endif /* XP_MACOSX */
|
|
SIMPLE_XUL_CREATE(iframe, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(editor, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(browser, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(splitter, NS_NewSplitterFrame),
|
|
SIMPLE_XUL_CREATE(scrollbar, NS_NewScrollbarFrame),
|
|
SIMPLE_XUL_CREATE(slider, NS_NewSliderFrame),
|
|
SIMPLE_XUL_CREATE(thumb, NS_NewSimpleXULLeafFrame),
|
|
SIMPLE_XUL_CREATE(scrollcorner, NS_NewSimpleXULLeafFrame),
|
|
SIMPLE_XUL_CREATE(resizer, NS_NewSimpleXULLeafFrame),
|
|
SIMPLE_XUL_CREATE(scrollbarbutton, NS_NewScrollbarButtonFrame),
|
|
{nsGkAtoms::panel, kPopupData},
|
|
{nsGkAtoms::menupopup, kPopupData},
|
|
{nsGkAtoms::tooltip, kPopupData},
|
|
};
|
|
|
|
return FindDataByTag(aElement, aStyle, sXULTagData, std::size(sXULTagData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULLabelOrDescriptionData(const Element& aElement,
|
|
ComputedStyle&) {
|
|
// Follow CSS display value if no value attribute
|
|
if (!aElement.HasAttr(nsGkAtoms::value)) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Follow CSS display if there's no crop="center".
|
|
if (!aElement.AttrValueIs(kNameSpaceID_None, nsGkAtoms::crop,
|
|
nsGkAtoms::center, eCaseMatters)) {
|
|
return nullptr;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sMiddleCroppingData =
|
|
SIMPLE_XUL_FCDATA(NS_NewMiddleCroppingLabelFrame);
|
|
return &sMiddleCroppingData;
|
|
}
|
|
|
|
#ifdef XP_MACOSX
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULMenubarData(const Element& aElement,
|
|
ComputedStyle&) {
|
|
if (aElement.OwnerDoc()->IsInChromeDocShell()) {
|
|
BrowsingContext* bc = aElement.OwnerDoc()->GetBrowsingContext();
|
|
bool isRoot = bc && !bc->GetParent();
|
|
if (isRoot) {
|
|
// This is the root. Suppress the menubar, since on Mac
|
|
// window menus are not attached to the window.
|
|
static constexpr FrameConstructionData sSuppressData = SUPPRESS_FCDATA();
|
|
return &sSuppressData;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
#endif /* XP_MACOSX */
|
|
|
|
already_AddRefed<ComputedStyle>
|
|
nsCSSFrameConstructor::BeginBuildingScrollContainerFrame(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
ComputedStyle* aContentStyle, nsContainerFrame* aParentFrame,
|
|
PseudoStyleType aScrolledPseudo, bool aIsRoot,
|
|
nsContainerFrame*& aNewFrame) {
|
|
nsContainerFrame* scrollContainerFrame = aNewFrame;
|
|
|
|
if (!scrollContainerFrame) {
|
|
scrollContainerFrame =
|
|
NS_NewScrollContainerFrame(mPresShell, aContentStyle, aIsRoot);
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, scrollContainerFrame);
|
|
}
|
|
|
|
MOZ_ASSERT(scrollContainerFrame);
|
|
|
|
// if there are any anonymous children for the scroll frame, create
|
|
// frames for them.
|
|
//
|
|
// We can't take the normal ProcessChildren path, because the NAC needs to
|
|
// be parented to the scrollframe, and everything else needs to be parented
|
|
// to the scrolledframe.
|
|
AutoTArray<nsIAnonymousContentCreator::ContentInfo, 4> scrollNAC;
|
|
DebugOnly<nsresult> rv =
|
|
GetAnonymousContent(aContent, scrollContainerFrame, scrollNAC);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
nsFrameList anonymousList;
|
|
if (!scrollNAC.IsEmpty()) {
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(scrollContainerFrame, floatSaveState);
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
AutoFrameConstructionPageName pageNameTracker(aState, scrollContainerFrame);
|
|
AddFCItemsForAnonymousContent(aState, scrollContainerFrame, scrollNAC,
|
|
items, pageNameTracker);
|
|
ConstructFramesFromItemList(aState, items, scrollContainerFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
anonymousList);
|
|
}
|
|
|
|
aNewFrame = scrollContainerFrame;
|
|
scrollContainerFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// we used the style that was passed in. So resolve another one.
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
RefPtr<ComputedStyle> scrolledChildStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(aScrolledPseudo,
|
|
aContentStyle);
|
|
|
|
scrollContainerFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(anonymousList));
|
|
|
|
return scrolledChildStyle.forget();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FinishBuildingScrollContainerFrame(
|
|
nsContainerFrame* aScrollContainerFrame, nsIFrame* aScrolledFrame) {
|
|
aScrollContainerFrame->AppendFrames(
|
|
FrameChildListID::Principal, nsFrameList(aScrolledFrame, aScrolledFrame));
|
|
}
|
|
|
|
void nsCSSFrameConstructor::BuildScrollContainerFrame(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
ComputedStyle* aContentStyle, nsIFrame* aScrolledFrame,
|
|
nsContainerFrame* aParentFrame, nsContainerFrame*& aNewFrame) {
|
|
RefPtr<ComputedStyle> scrolledContentStyle =
|
|
BeginBuildingScrollContainerFrame(
|
|
aState, aContent, aContentStyle, aParentFrame,
|
|
PseudoStyleType::scrolledContent, false, aNewFrame);
|
|
|
|
aScrolledFrame->SetComputedStyleWithoutNotification(scrolledContentStyle);
|
|
InitAndRestoreFrame(aState, aContent, aNewFrame, aScrolledFrame);
|
|
|
|
FinishBuildingScrollContainerFrame(aNewFrame, aScrolledFrame);
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDisplayData(const nsStyleDisplay& aDisplay,
|
|
const Element& aElement) {
|
|
static_assert(eParentTypeCount < (1 << (32 - FCDATA_PARENT_TYPE_OFFSET)),
|
|
"Check eParentTypeCount should not overflow");
|
|
|
|
// The style system ensures that floated and positioned frames are
|
|
// block-level.
|
|
NS_ASSERTION(
|
|
!(aDisplay.IsFloatingStyle() || aDisplay.IsAbsolutelyPositionedStyle()) ||
|
|
aDisplay.IsBlockOutsideStyle(),
|
|
"Style system did not apply CSS2.1 section 9.7 fixups");
|
|
|
|
// If this is "body", try propagating its scroll style to the viewport
|
|
// Note that we need to do this even if the body is NOT scrollable;
|
|
// it might have dynamically changed from scrollable to not scrollable,
|
|
// and that might need to be propagated.
|
|
// XXXbz is this the right place to do this? If this code moves,
|
|
// make this function static.
|
|
bool propagatedScrollToViewport = false;
|
|
if (aElement.IsHTMLElement(nsGkAtoms::body)) {
|
|
if (nsPresContext* presContext = mPresShell->GetPresContext()) {
|
|
propagatedScrollToViewport =
|
|
presContext->UpdateViewportScrollStylesOverride() == &aElement;
|
|
MOZ_ASSERT(!propagatedScrollToViewport ||
|
|
!mPresShell->GetPresContext()->IsPaginated(),
|
|
"Shouldn't propagate scroll in paginated contexts");
|
|
}
|
|
}
|
|
|
|
switch (aDisplay.DisplayInside()) {
|
|
case StyleDisplayInside::Flow:
|
|
case StyleDisplayInside::FlowRoot: {
|
|
if (aDisplay.IsInlineFlow()) {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructInline,
|
|
FCDATA_IS_INLINE | FCDATA_IS_LINE_PARTICIPANT);
|
|
return &data;
|
|
}
|
|
|
|
// If the frame is a block-level frame and is scrollable, then wrap it in
|
|
// a scroll frame. Except we don't want to do that for paginated contexts
|
|
// for frames that are block-outside and aren't frames for native
|
|
// anonymous stuff.
|
|
// XXX Ignore tables for the time being (except caption)
|
|
const uint32_t kCaptionCtorFlags =
|
|
FCDATA_IS_TABLE_PART | FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable);
|
|
const bool caption = aDisplay.mDisplay == StyleDisplay::TableCaption;
|
|
const bool needScrollFrame =
|
|
aDisplay.IsScrollableOverflow() && !propagatedScrollToViewport;
|
|
if (needScrollFrame) {
|
|
const bool suppressScrollFrame =
|
|
mPresShell->GetPresContext()->IsPaginated() &&
|
|
aDisplay.IsBlockOutsideStyle() &&
|
|
!aElement.IsInNativeAnonymousSubtree();
|
|
if (!suppressScrollFrame) {
|
|
static constexpr FrameConstructionData sScrollableBlockData[2] = {
|
|
{&nsCSSFrameConstructor::ConstructScrollableBlock},
|
|
{&nsCSSFrameConstructor::ConstructScrollableBlock,
|
|
kCaptionCtorFlags}};
|
|
return &sScrollableBlockData[caption];
|
|
}
|
|
}
|
|
|
|
// Handle various non-scrollable blocks.
|
|
static constexpr FrameConstructionData sNonScrollableBlockData[2] = {
|
|
{&nsCSSFrameConstructor::ConstructNonScrollableBlock},
|
|
{&nsCSSFrameConstructor::ConstructNonScrollableBlock,
|
|
kCaptionCtorFlags}};
|
|
return &sNonScrollableBlockData[caption];
|
|
}
|
|
case StyleDisplayInside::Table: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTable);
|
|
return &data;
|
|
}
|
|
// NOTE: In the unlikely event that we add another table-part here that
|
|
// has a desired-parent-type (& hence triggers table fixup), we'll need to
|
|
// also update the flexbox chunk in ComputedStyle::ApplyStyleFixups().
|
|
case StyleDisplayInside::TableRowGroup: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableColumn: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableCol,
|
|
FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeColGroup));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableColumnGroup: {
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewTableColGroupFrame),
|
|
FCDATA_IS_TABLE_PART | FCDATA_DISALLOW_OUT_OF_FLOW |
|
|
FCDATA_SKIP_ABSPOS_PUSH |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableHeaderGroup: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableFooterGroup: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableRow: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRowGroup));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::TableCell: {
|
|
static constexpr FrameConstructionData data(
|
|
&nsCSSFrameConstructor::ConstructTableCell,
|
|
FCDATA_IS_TABLE_PART | FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRow));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::Flex:
|
|
case StyleDisplayInside::WebkitBox: {
|
|
static constexpr FrameConstructionData nonScrollableData(
|
|
ToCreationFunc(NS_NewFlexContainerFrame));
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewFlexContainerFrame),
|
|
FCDATA_MAY_NEED_SCROLLFRAME);
|
|
return MOZ_UNLIKELY(propagatedScrollToViewport) ? &nonScrollableData
|
|
: &data;
|
|
}
|
|
case StyleDisplayInside::Grid: {
|
|
static constexpr FrameConstructionData nonScrollableData(
|
|
ToCreationFunc(NS_NewGridContainerFrame));
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewGridContainerFrame),
|
|
FCDATA_MAY_NEED_SCROLLFRAME);
|
|
return MOZ_UNLIKELY(propagatedScrollToViewport) ? &nonScrollableData
|
|
: &data;
|
|
}
|
|
case StyleDisplayInside::Ruby: {
|
|
static constexpr FrameConstructionData data[] = {
|
|
{&nsCSSFrameConstructor::ConstructBlockRubyFrame,
|
|
FCDATA_MAY_NEED_SCROLLFRAME},
|
|
{ToCreationFunc(NS_NewRubyFrame), FCDATA_IS_LINE_PARTICIPANT}};
|
|
bool isInline = aDisplay.DisplayOutside() == StyleDisplayOutside::Inline;
|
|
return &data[isInline];
|
|
}
|
|
case StyleDisplayInside::RubyBase: {
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewRubyBaseFrame),
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyBaseContainer));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::RubyBaseContainer: {
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewRubyBaseContainerFrame),
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::RubyText: {
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewRubyTextFrame),
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyTextContainer));
|
|
return &data;
|
|
}
|
|
case StyleDisplayInside::RubyTextContainer: {
|
|
static constexpr FrameConstructionData data(
|
|
ToCreationFunc(NS_NewRubyTextContainerFrame),
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby));
|
|
return &data;
|
|
}
|
|
default:
|
|
MOZ_ASSERT_UNREACHABLE("unknown 'display' value");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructScrollableBlock(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsContainerFrame* newFrame = nullptr;
|
|
RefPtr<ComputedStyle> scrolledContentStyle =
|
|
BeginBuildingScrollContainerFrame(
|
|
aState, content, computedStyle,
|
|
aState.GetGeometricParent(*aDisplay, aParentFrame),
|
|
PseudoStyleType::scrolledContent, false, newFrame);
|
|
|
|
// Create our block frame
|
|
// pass a temporary stylecontext, the correct one will be set later
|
|
nsContainerFrame* scrolledFrame = NS_NewBlockFrame(mPresShell, computedStyle);
|
|
|
|
// Make sure to AddChild before we call ConstructBlock so that we
|
|
// end up before our descendants in fixed-pos lists as needed.
|
|
aState.AddChild(newFrame, aFrameList, content, aParentFrame);
|
|
|
|
nsFrameList blockList;
|
|
ConstructBlock(aState, content, newFrame, newFrame, scrolledContentStyle,
|
|
&scrolledFrame, blockList,
|
|
newFrame->IsAbsPosContainingBlock() ? newFrame : nullptr);
|
|
|
|
MOZ_ASSERT(blockList.OnlyChild() == scrolledFrame,
|
|
"Scrollframe's frameList should be exactly the scrolled frame!");
|
|
FinishBuildingScrollContainerFrame(newFrame, scrolledFrame);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructNonScrollableBlock(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
nsContainerFrame* newFrame = NS_NewBlockFrame(mPresShell, computedStyle);
|
|
ConstructBlock(aState, aItem.mContent,
|
|
aState.GetGeometricParent(*aDisplay, aParentFrame),
|
|
aParentFrame, computedStyle, &newFrame, aFrameList,
|
|
newFrame->IsAbsPosContainingBlock() ? newFrame : nullptr);
|
|
return newFrame;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::InitAndRestoreFrame(
|
|
const nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, nsIFrame* aNewFrame, bool aAllowCounters) {
|
|
MOZ_ASSERT(aNewFrame, "Null frame cannot be initialized");
|
|
|
|
// Initialize the frame
|
|
aNewFrame->Init(aContent, aParentFrame, nullptr);
|
|
aNewFrame->AddStateBits(aState.mAdditionalStateBits);
|
|
|
|
if (aState.mFrameState) {
|
|
// Restore frame state for just the newly created frame.
|
|
RestoreFrameStateFor(aNewFrame, aState.mFrameState);
|
|
}
|
|
|
|
if (aAllowCounters &&
|
|
mContainStyleScopeManager.AddCounterChanges(aNewFrame)) {
|
|
CountersDirty();
|
|
}
|
|
}
|
|
|
|
already_AddRefed<ComputedStyle> nsCSSFrameConstructor::ResolveComputedStyle(
|
|
nsIContent* aContent) {
|
|
if (auto* element = Element::FromNode(aContent)) {
|
|
return ServoStyleSet::ResolveServoStyle(*element);
|
|
}
|
|
|
|
MOZ_ASSERT(aContent->IsText(),
|
|
"shouldn't waste time creating ComputedStyles for "
|
|
"comments and processing instructions");
|
|
|
|
Element* parent = aContent->GetFlattenedTreeParentElement();
|
|
MOZ_ASSERT(parent, "Text out of the flattened tree?");
|
|
|
|
// FIXME(emilio): The const_cast is unfortunate, but it's not worse than what
|
|
// we did before.
|
|
//
|
|
// We could use ResolveServoStyle, but that would involve extra unnecessary
|
|
// refcount traffic...
|
|
auto* parentStyle =
|
|
const_cast<ComputedStyle*>(Servo_Element_GetMaybeOutOfDateStyle(parent));
|
|
MOZ_ASSERT(parentStyle,
|
|
"How are we inserting text frames in an unstyled element?");
|
|
return mPresShell->StyleSet()->ResolveStyleForText(aContent, parentStyle);
|
|
}
|
|
|
|
// MathML Mod - RBS
|
|
void nsCSSFrameConstructor::FlushAccumulatedBlock(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, nsFrameList& aBlockList,
|
|
nsFrameList& aNewList) {
|
|
if (aBlockList.IsEmpty()) {
|
|
// Nothing to do
|
|
return;
|
|
}
|
|
|
|
auto anonPseudo = PseudoStyleType::mozMathMLAnonymousBlock;
|
|
|
|
ComputedStyle* parentContext =
|
|
nsIFrame::CorrectStyleParentFrame(aParentFrame, anonPseudo)->Style();
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
RefPtr<ComputedStyle> blockContext =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(anonPseudo, parentContext);
|
|
|
|
// then, create a block frame that will wrap the child frames. Make it a
|
|
// MathML frame so that Get(Absolute/Float)ContainingBlockFor know that this
|
|
// is not a suitable block.
|
|
nsContainerFrame* blockFrame =
|
|
NS_NewMathMLmathBlockFrame(mPresShell, blockContext);
|
|
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, blockFrame);
|
|
ReparentFrames(this, blockFrame, aBlockList, false);
|
|
// We have to walk over aBlockList before we hand it over to blockFrame.
|
|
for (nsIFrame* f : aBlockList) {
|
|
f->SetParentIsWrapperAnonBox();
|
|
}
|
|
// abs-pos and floats are disabled in MathML children so we don't have to
|
|
// worry about messing up those.
|
|
blockFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(aBlockList));
|
|
aNewList.AppendFrame(nullptr, blockFrame);
|
|
}
|
|
|
|
// Only <math> elements can be floated or positioned. All other MathML
|
|
// should be in-flow.
|
|
#define SIMPLE_MATHML_CREATE(_tag, _func) \
|
|
{ \
|
|
nsGkAtoms::_tag, { \
|
|
_func, FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_FORCE_NULL_ABSPOS_CONTAINER | \
|
|
FCDATA_WRAP_KIDS_IN_BLOCKS \
|
|
} \
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindMathMLData(const Element& aElement,
|
|
ComputedStyle& aStyle) {
|
|
MOZ_ASSERT(aElement.IsMathMLElement());
|
|
|
|
nsAtom* tag = aElement.NodeInfo()->NameAtom();
|
|
|
|
// Handle <math> specially, because it sometimes produces inlines
|
|
if (tag == nsGkAtoms::math) {
|
|
// The IsBlockOutsideStyle() check must match what
|
|
// specified::Display::equivalent_block_display is checking for
|
|
// already-block-outside things. Though the behavior here for the
|
|
// display:table case is pretty weird...
|
|
if (aStyle.StyleDisplay()->IsBlockOutsideStyle()) {
|
|
static constexpr FrameConstructionData sBlockMathData(
|
|
ToCreationFunc(NS_NewMathMLmathBlockFrame),
|
|
FCDATA_FORCE_NULL_ABSPOS_CONTAINER | FCDATA_WRAP_KIDS_IN_BLOCKS);
|
|
return &sBlockMathData;
|
|
}
|
|
|
|
static constexpr FrameConstructionData sInlineMathData(
|
|
ToCreationFunc(NS_NewMathMLmathInlineFrame),
|
|
FCDATA_FORCE_NULL_ABSPOS_CONTAINER | FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_WRAP_KIDS_IN_BLOCKS);
|
|
return &sInlineMathData;
|
|
}
|
|
|
|
static constexpr FrameConstructionDataByTag sMathMLData[] = {
|
|
SIMPLE_MATHML_CREATE(annotation_, NS_NewMathMLTokenFrame),
|
|
SIMPLE_MATHML_CREATE(annotation_xml_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mi_, NS_NewMathMLTokenFrame),
|
|
SIMPLE_MATHML_CREATE(mn_, NS_NewMathMLTokenFrame),
|
|
SIMPLE_MATHML_CREATE(ms_, NS_NewMathMLTokenFrame),
|
|
SIMPLE_MATHML_CREATE(mtext_, NS_NewMathMLTokenFrame),
|
|
SIMPLE_MATHML_CREATE(mo_, NS_NewMathMLmoFrame),
|
|
SIMPLE_MATHML_CREATE(mfrac_, NS_NewMathMLmfracFrame),
|
|
SIMPLE_MATHML_CREATE(msup_, NS_NewMathMLmmultiscriptsFrame),
|
|
SIMPLE_MATHML_CREATE(msub_, NS_NewMathMLmmultiscriptsFrame),
|
|
SIMPLE_MATHML_CREATE(msubsup_, NS_NewMathMLmmultiscriptsFrame),
|
|
SIMPLE_MATHML_CREATE(munder_, NS_NewMathMLmunderoverFrame),
|
|
SIMPLE_MATHML_CREATE(mover_, NS_NewMathMLmunderoverFrame),
|
|
SIMPLE_MATHML_CREATE(munderover_, NS_NewMathMLmunderoverFrame),
|
|
SIMPLE_MATHML_CREATE(mphantom_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mpadded_, NS_NewMathMLmpaddedFrame),
|
|
SIMPLE_MATHML_CREATE(mspace_, NS_NewMathMLmspaceFrame),
|
|
SIMPLE_MATHML_CREATE(none, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mprescripts_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mfenced_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mmultiscripts_, NS_NewMathMLmmultiscriptsFrame),
|
|
SIMPLE_MATHML_CREATE(mstyle_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(msqrt_, NS_NewMathMLmrootFrame),
|
|
SIMPLE_MATHML_CREATE(mroot_, NS_NewMathMLmrootFrame),
|
|
SIMPLE_MATHML_CREATE(maction_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(mrow_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(merror_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(menclose_, NS_NewMathMLmencloseFrame),
|
|
SIMPLE_MATHML_CREATE(semantics_, NS_NewMathMLmrowFrame)};
|
|
|
|
return FindDataByTag(aElement, aStyle, sMathMLData, std::size(sMathMLData));
|
|
}
|
|
|
|
nsContainerFrame* nsCSSFrameConstructor::ConstructFrameWithAnonymousChild(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, nsFrameList& aFrameList,
|
|
ContainerFrameCreationFunc aConstructor,
|
|
ContainerFrameCreationFunc aInnerConstructor, PseudoStyleType aInnerPseudo,
|
|
bool aCandidateRootFrame) {
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
// Create the outer frame:
|
|
nsContainerFrame* newFrame = aConstructor(mPresShell, computedStyle);
|
|
|
|
InitAndRestoreFrame(aState, content,
|
|
aCandidateRootFrame
|
|
? aState.GetGeometricParent(
|
|
*computedStyle->StyleDisplay(), aParentFrame)
|
|
: aParentFrame,
|
|
newFrame);
|
|
newFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Create the pseudo SC for the anonymous wrapper child as a child of the SC:
|
|
RefPtr<ComputedStyle> scForAnon =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(aInnerPseudo,
|
|
computedStyle);
|
|
|
|
// Create the anonymous inner wrapper frame
|
|
nsContainerFrame* innerFrame = aInnerConstructor(mPresShell, scForAnon);
|
|
|
|
InitAndRestoreFrame(aState, content, newFrame, innerFrame);
|
|
|
|
// Put the newly created frames into the right child list
|
|
SetInitialSingleChild(newFrame, innerFrame);
|
|
|
|
aState.AddChild(newFrame, aFrameList, content, aParentFrame,
|
|
aCandidateRootFrame, aCandidateRootFrame);
|
|
|
|
if (!mRootElementFrame && aCandidateRootFrame) {
|
|
mRootElementFrame = newFrame;
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(innerFrame, floatSaveState);
|
|
|
|
nsFrameList childList;
|
|
|
|
// Process children
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(
|
|
aState, aItem.mChildItems, innerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX, childList);
|
|
} else {
|
|
ProcessChildren(aState, content, computedStyle, innerFrame, true, childList,
|
|
false);
|
|
}
|
|
|
|
// Set the inner wrapper frame's initial primary list
|
|
innerFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructOuterSVG(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
return ConstructFrameWithAnonymousChild(
|
|
aState, aItem, aParentFrame, aFrameList, NS_NewSVGOuterSVGFrame,
|
|
NS_NewSVGOuterSVGAnonChildFrame, PseudoStyleType::mozSVGOuterSVGAnonChild,
|
|
true);
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructMarker(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
return ConstructFrameWithAnonymousChild(
|
|
aState, aItem, aParentFrame, aFrameList, NS_NewSVGMarkerFrame,
|
|
NS_NewSVGMarkerAnonChildFrame, PseudoStyleType::mozSVGMarkerAnonChild,
|
|
false);
|
|
}
|
|
|
|
// Only outer <svg> elements can be floated or positioned. All other SVG
|
|
// should be in-flow.
|
|
#define SIMPLE_SVG_FCDATA(_func) \
|
|
FrameConstructionData(ToCreationFunc(_func), \
|
|
FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_SKIP_ABSPOS_PUSH | \
|
|
FCDATA_DISALLOW_GENERATED_CONTENT)
|
|
#define SIMPLE_SVG_CREATE(_tag, _func) \
|
|
{nsGkAtoms::_tag, SIMPLE_SVG_FCDATA(_func)}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindSVGData(const Element& aElement,
|
|
nsIFrame* aParentFrame,
|
|
bool aIsWithinSVGText,
|
|
bool aAllowsTextPathChild,
|
|
ComputedStyle& aStyle) {
|
|
MOZ_ASSERT(aElement.IsSVGElement());
|
|
|
|
static constexpr FrameConstructionData sSuppressData = SUPPRESS_FCDATA();
|
|
static constexpr FrameConstructionData sContainerData =
|
|
SIMPLE_SVG_FCDATA(NS_NewSVGContainerFrame);
|
|
|
|
bool parentIsSVG = aIsWithinSVGText;
|
|
nsIContent* parentContent =
|
|
aParentFrame ? aParentFrame->GetContent() : nullptr;
|
|
|
|
nsAtom* tag = aElement.NodeInfo()->NameAtom();
|
|
|
|
// XXXbz should this really be based on the tag of the parent frame's content?
|
|
// Should it not be based on the type of the parent frame (e.g. whether it's
|
|
// an SVG frame)?
|
|
if (parentContent) {
|
|
// It's not clear whether the SVG spec intends to allow any SVG
|
|
// content within svg:foreignObject at all (SVG 1.1, section
|
|
// 23.2), but if it does, it better be svg:svg. So given that
|
|
// we're allowing it, treat it as a non-SVG parent.
|
|
parentIsSVG =
|
|
parentContent->IsSVGElement() &&
|
|
parentContent->NodeInfo()->NameAtom() != nsGkAtoms::foreignObject;
|
|
}
|
|
|
|
if ((tag != nsGkAtoms::svg && !parentIsSVG) ||
|
|
(tag == nsGkAtoms::desc || tag == nsGkAtoms::title ||
|
|
tag == nsGkAtoms::metadata)) {
|
|
// Sections 5.1 and G.4 of SVG 1.1 say that SVG elements other than
|
|
// svg:svg not contained within svg:svg are incorrect, although they
|
|
// don't seem to specify error handling. Ignore them, since many of
|
|
// our frame classes can't deal. It *may* be that the document
|
|
// should at that point be considered in error according to F.2, but
|
|
// it's hard to tell.
|
|
//
|
|
// Style mutation can't change this situation, so don't bother
|
|
// adding to the undisplayed content map.
|
|
//
|
|
// We don't currently handle any UI for desc/title/metadata
|
|
return &sSuppressData;
|
|
}
|
|
|
|
// We don't need frames for animation elements
|
|
if (aElement.IsSVGAnimationElement()) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
if (tag == nsGkAtoms::svg && !parentIsSVG) {
|
|
// We need outer <svg> elements to have an SVGOuterSVGFrame regardless
|
|
// of whether they fail conditional processing attributes, since various
|
|
// SVG frames assume that one exists. We handle the non-rendering
|
|
// of failing outer <svg> element contents like <switch> statements,
|
|
// and do the PassesConditionalProcessingTests call in
|
|
// SVGOuterSVGFrame::Init.
|
|
static constexpr FrameConstructionData sOuterSVGData(
|
|
&nsCSSFrameConstructor::ConstructOuterSVG);
|
|
return &sOuterSVGData;
|
|
}
|
|
|
|
if (tag == nsGkAtoms::marker) {
|
|
static constexpr FrameConstructionData sMarkerSVGData(
|
|
&nsCSSFrameConstructor::ConstructMarker);
|
|
return &sMarkerSVGData;
|
|
}
|
|
|
|
if (!aElement.PassesConditionalProcessingTests()) {
|
|
// Elements with failing conditional processing attributes never get
|
|
// rendered. Note that this is not where we select which frame in a
|
|
// <switch> to render! That happens in SVGSwitchFrame::PaintSVG.
|
|
if (aIsWithinSVGText) {
|
|
// SVGTextFrame doesn't handle conditional processing attributes,
|
|
// so don't create frames for descendants of <text> with failing
|
|
// attributes. We need frames not to be created so that text layout
|
|
// is correct.
|
|
return &sSuppressData;
|
|
}
|
|
// If we're not inside <text>, create an SVGContainerFrame (which is a
|
|
// frame that doesn't render) so that paint servers can still be referenced,
|
|
// even if they live inside an element with failing conditional processing
|
|
// attributes.
|
|
return &sContainerData;
|
|
}
|
|
|
|
// Ensure that a stop frame is a child of a gradient and that gradients
|
|
// can only have stop children.
|
|
bool parentIsGradient = aParentFrame && static_cast<SVGGradientFrame*>(
|
|
do_QueryFrame(aParentFrame));
|
|
bool stop = (tag == nsGkAtoms::stop);
|
|
if ((parentIsGradient && !stop) || (!parentIsGradient && stop)) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
// Prevent bad frame types being children of filters or parents of filter
|
|
// primitives. If aParentFrame is null, we know that the frame that will
|
|
// be created will be an nsInlineFrame, so it can never be a filter.
|
|
bool parentIsFilter = aParentFrame && aParentFrame->IsSVGFilterFrame();
|
|
if ((parentIsFilter && !aElement.IsSVGFilterPrimitiveElement()) ||
|
|
(!parentIsFilter && aElement.IsSVGFilterPrimitiveElement())) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
// Prevent bad frame types being children of filter primitives or parents of
|
|
// filter primitive children. If aParentFrame is null, we know that the frame
|
|
// that will be created will be an nsInlineFrame, so it can never be a filter
|
|
// primitive.
|
|
bool parentIsFEContainerFrame =
|
|
aParentFrame && aParentFrame->IsSVGFEContainerFrame();
|
|
if ((parentIsFEContainerFrame &&
|
|
!aElement.IsSVGFilterPrimitiveChildElement()) ||
|
|
(!parentIsFEContainerFrame &&
|
|
aElement.IsSVGFilterPrimitiveChildElement())) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
// Special cases for text/tspan/textPath, because the kind of frame
|
|
// they get depends on the parent frame. We ignore 'a' elements when
|
|
// determining the parent, however.
|
|
if (aIsWithinSVGText) {
|
|
// If aIsWithinSVGText is true, then we know that the "SVG text uses
|
|
// CSS frames" pref was true when this SVG fragment was first constructed.
|
|
//
|
|
// FIXME(bug 1588477) Don't render stuff in display: contents / Shadow DOM
|
|
// subtrees, because TextCorrespondenceRecorder in the SVG text code doesn't
|
|
// really know how to deal with it. This kinda sucks. :(
|
|
if (aParentFrame && aParentFrame->GetContent() != aElement.GetParent()) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
// We don't use ConstructInline because we want different behavior
|
|
// for generated content.
|
|
static constexpr FrameConstructionData sTSpanData(
|
|
ToCreationFunc(NS_NewInlineFrame),
|
|
FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_SKIP_ABSPOS_PUSH |
|
|
FCDATA_DISALLOW_GENERATED_CONTENT | FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_INLINE | FCDATA_USE_CHILD_ITEMS);
|
|
if (tag == nsGkAtoms::textPath) {
|
|
if (aAllowsTextPathChild) {
|
|
return &sTSpanData;
|
|
}
|
|
} else if (tag == nsGkAtoms::tspan || tag == nsGkAtoms::a) {
|
|
return &sTSpanData;
|
|
}
|
|
return &sSuppressData;
|
|
} else if (tag == nsGkAtoms::tspan || tag == nsGkAtoms::textPath) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
static constexpr FrameConstructionDataByTag sSVGData[] = {
|
|
SIMPLE_SVG_CREATE(svg, NS_NewSVGInnerSVGFrame),
|
|
SIMPLE_SVG_CREATE(g, NS_NewSVGGFrame),
|
|
SIMPLE_SVG_CREATE(svgSwitch, NS_NewSVGSwitchFrame),
|
|
SIMPLE_SVG_CREATE(symbol, NS_NewSVGSymbolFrame),
|
|
SIMPLE_SVG_CREATE(polygon, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(polyline, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(circle, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(ellipse, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(line, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(rect, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(path, NS_NewSVGGeometryFrame),
|
|
SIMPLE_SVG_CREATE(defs, NS_NewSVGContainerFrame),
|
|
{nsGkAtoms::text,
|
|
{NS_NewSVGTextFrame,
|
|
FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_ALLOW_BLOCK_STYLES,
|
|
PseudoStyleType::mozSVGText}},
|
|
{nsGkAtoms::foreignObject,
|
|
{ToCreationFunc(NS_NewSVGForeignObjectFrame),
|
|
FCDATA_DISALLOW_OUT_OF_FLOW, PseudoStyleType::mozSVGForeignContent}},
|
|
SIMPLE_SVG_CREATE(a, NS_NewSVGAFrame),
|
|
SIMPLE_SVG_CREATE(linearGradient, NS_NewSVGLinearGradientFrame),
|
|
SIMPLE_SVG_CREATE(radialGradient, NS_NewSVGRadialGradientFrame),
|
|
SIMPLE_SVG_CREATE(stop, NS_NewSVGStopFrame),
|
|
SIMPLE_SVG_CREATE(use, NS_NewSVGUseFrame),
|
|
SIMPLE_SVG_CREATE(view, NS_NewSVGViewFrame),
|
|
SIMPLE_SVG_CREATE(image, NS_NewSVGImageFrame),
|
|
SIMPLE_SVG_CREATE(clipPath, NS_NewSVGClipPathFrame),
|
|
SIMPLE_SVG_CREATE(filter, NS_NewSVGFilterFrame),
|
|
SIMPLE_SVG_CREATE(pattern, NS_NewSVGPatternFrame),
|
|
SIMPLE_SVG_CREATE(mask, NS_NewSVGMaskFrame),
|
|
SIMPLE_SVG_CREATE(feDistantLight, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(fePointLight, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feSpotLight, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feBlend, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feColorMatrix, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feFuncR, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feFuncG, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feFuncB, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feFuncA, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feComposite, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feComponentTransfer, NS_NewSVGFEContainerFrame),
|
|
SIMPLE_SVG_CREATE(feConvolveMatrix, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feDiffuseLighting, NS_NewSVGFEContainerFrame),
|
|
SIMPLE_SVG_CREATE(feDisplacementMap, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feDropShadow, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feFlood, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feGaussianBlur, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feImage, NS_NewSVGFEImageFrame),
|
|
SIMPLE_SVG_CREATE(feMerge, NS_NewSVGFEContainerFrame),
|
|
SIMPLE_SVG_CREATE(feMergeNode, NS_NewSVGFEUnstyledLeafFrame),
|
|
SIMPLE_SVG_CREATE(feMorphology, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feOffset, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feSpecularLighting, NS_NewSVGFEContainerFrame),
|
|
SIMPLE_SVG_CREATE(feTile, NS_NewSVGFELeafFrame),
|
|
SIMPLE_SVG_CREATE(feTurbulence, NS_NewSVGFELeafFrame)};
|
|
|
|
const FrameConstructionData* data =
|
|
FindDataByTag(aElement, aStyle, sSVGData, std::size(sSVGData));
|
|
|
|
if (!data) {
|
|
data = &sContainerData;
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AppendPageBreakItem(
|
|
nsIContent* aContent, FrameConstructionItemList& aItems) {
|
|
RefPtr<ComputedStyle> pseudoStyle =
|
|
mPresShell->StyleSet()->ResolveNonInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::pageBreak);
|
|
|
|
MOZ_ASSERT(pseudoStyle->StyleDisplay()->mDisplay == StyleDisplay::Block,
|
|
"Unexpected display");
|
|
|
|
static constexpr FrameConstructionData sPageBreakData(NS_NewPageBreakFrame,
|
|
FCDATA_SKIP_FRAMESET);
|
|
aItems.AppendItem(this, &sPageBreakData, aContent, pseudoStyle.forget(),
|
|
true);
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::ShouldCreateItemsForChild(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame) {
|
|
aContent->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME);
|
|
// XXX the GetContent() != aContent check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
if (aContent->GetPrimaryFrame() &&
|
|
aContent->GetPrimaryFrame()->GetContent() == aContent &&
|
|
!aState.mCreatingExtraFrames) {
|
|
MOZ_ASSERT(false,
|
|
"asked to create frame construction item for a node that "
|
|
"already has a frame");
|
|
return false;
|
|
}
|
|
|
|
// don't create a whitespace frame if aParent doesn't want it
|
|
if (!NeedFrameFor(aState, aParentFrame, aContent)) {
|
|
return false;
|
|
}
|
|
|
|
// never create frames for comments or PIs
|
|
if (aContent->IsComment() || aContent->IsProcessingInstruction()) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AddFrameConstructionItems(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
bool aSuppressWhiteSpaceOptimizations, const ComputedStyle& aParentStyle,
|
|
const InsertionPoint& aInsertion, FrameConstructionItemList& aItems,
|
|
ItemFlags aFlags) {
|
|
nsContainerFrame* parentFrame = aInsertion.mParentFrame;
|
|
if (!ShouldCreateItemsForChild(aState, aContent, parentFrame)) {
|
|
return;
|
|
}
|
|
if (MOZ_UNLIKELY(aParentStyle.StyleContent()->mContent.IsNone()) &&
|
|
StaticPrefs::layout_css_element_content_none_enabled()) {
|
|
return;
|
|
}
|
|
|
|
RefPtr<ComputedStyle> computedStyle = ResolveComputedStyle(aContent);
|
|
auto flags = aFlags + ItemFlag::AllowPageBreak;
|
|
if (parentFrame) {
|
|
if (parentFrame->IsInSVGTextSubtree()) {
|
|
flags += ItemFlag::IsWithinSVGText;
|
|
}
|
|
if (parentFrame->IsBlockFrame() && parentFrame->GetParent() &&
|
|
parentFrame->GetParent()->IsSVGTextFrame()) {
|
|
flags += ItemFlag::AllowTextPathChild;
|
|
}
|
|
}
|
|
AddFrameConstructionItemsInternal(aState, aContent, parentFrame,
|
|
aSuppressWhiteSpaceOptimizations,
|
|
computedStyle, flags, aItems);
|
|
}
|
|
|
|
// Whether we should suppress frames for a child under a <select> frame.
|
|
//
|
|
// Never create frames for non-option/optgroup kids of <select> and non-option
|
|
// kids of <optgroup> inside a <select>.
|
|
static bool ShouldSuppressFrameInSelect(const nsIContent* aParent,
|
|
const nsIContent& aChild) {
|
|
if (!aParent ||
|
|
!aParent->IsAnyOfHTMLElements(nsGkAtoms::select, nsGkAtoms::optgroup,
|
|
nsGkAtoms::option)) {
|
|
return false;
|
|
}
|
|
|
|
// Allow native anonymous content no matter what.
|
|
if (aChild.IsRootOfNativeAnonymousSubtree()) {
|
|
return false;
|
|
}
|
|
|
|
// Options with labels have their label text added in ::before by forms.css.
|
|
// Suppress frames for their child text.
|
|
if (aParent->IsHTMLElement(nsGkAtoms::option)) {
|
|
return aParent->AsElement()->HasNonEmptyAttr(nsGkAtoms::label);
|
|
}
|
|
|
|
// If we're in any display: contents subtree, just suppress the frame.
|
|
//
|
|
// We can't be regular NAC, since display: contents has no frame to generate
|
|
// them off.
|
|
if (aChild.GetParent() != aParent) {
|
|
return true;
|
|
}
|
|
|
|
// Option is always fine.
|
|
if (aChild.IsHTMLElement(nsGkAtoms::option)) {
|
|
return false;
|
|
}
|
|
|
|
// <optgroup> is OK in <select> but not in <optgroup>.
|
|
if (aChild.IsHTMLElement(nsGkAtoms::optgroup) &&
|
|
aParent->IsHTMLElement(nsGkAtoms::select)) {
|
|
return false;
|
|
}
|
|
|
|
// Anything else is not ok.
|
|
return true;
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDataForContent(nsIContent& aContent,
|
|
ComputedStyle& aStyle,
|
|
nsIFrame* aParentFrame,
|
|
ItemFlags aFlags) {
|
|
MOZ_ASSERT(aStyle.StyleDisplay()->mDisplay != StyleDisplay::None &&
|
|
aStyle.StyleDisplay()->mDisplay != StyleDisplay::Contents,
|
|
"These two special display values should be handled earlier");
|
|
|
|
if (auto* text = Text::FromNode(aContent)) {
|
|
return FindTextData(*text, aParentFrame);
|
|
}
|
|
|
|
return FindElementData(*aContent.AsElement(), aStyle, aParentFrame, aFlags);
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindElementData(const Element& aElement,
|
|
ComputedStyle& aStyle,
|
|
nsIFrame* aParentFrame,
|
|
ItemFlags aFlags) {
|
|
// Don't create frames for non-SVG element children of SVG elements.
|
|
if (!aElement.IsSVGElement()) {
|
|
if (aParentFrame && IsFrameForSVG(aParentFrame) &&
|
|
!aParentFrame->IsSVGForeignObjectFrame()) {
|
|
return nullptr;
|
|
}
|
|
if (aFlags.contains(ItemFlag::IsWithinSVGText)) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
if (auto* data = FindElementTagData(aElement, aStyle, aParentFrame, aFlags)) {
|
|
return data;
|
|
}
|
|
|
|
// Check for 'content: <image-url>' on the element (which makes us ignore
|
|
// 'display' values other than 'none' or 'contents').
|
|
if (nsImageFrame::ShouldCreateImageFrameForContentProperty(aElement,
|
|
aStyle)) {
|
|
static constexpr FrameConstructionData sImgData(
|
|
NS_NewImageFrameForContentProperty);
|
|
return &sImgData;
|
|
}
|
|
|
|
const bool shouldBlockify = aFlags.contains(ItemFlag::IsForRenderedLegend) ||
|
|
aFlags.contains(ItemFlag::IsForOutsideMarker);
|
|
if (shouldBlockify && !aStyle.StyleDisplay()->IsBlockOutsideStyle()) {
|
|
// Make a temp copy of StyleDisplay and blockify its mDisplay value.
|
|
auto display = *aStyle.StyleDisplay();
|
|
bool isRootElement = false;
|
|
uint16_t rawDisplayValue =
|
|
Servo_ComputedValues_BlockifiedDisplay(&aStyle, isRootElement);
|
|
display.mDisplay = StyleDisplay{rawDisplayValue};
|
|
return FindDisplayData(display, aElement);
|
|
}
|
|
|
|
const auto& display = *aStyle.StyleDisplay();
|
|
return FindDisplayData(display, aElement);
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindElementTagData(const Element& aElement,
|
|
ComputedStyle& aStyle,
|
|
nsIFrame* aParentFrame,
|
|
ItemFlags aFlags) {
|
|
switch (aElement.GetNameSpaceID()) {
|
|
case kNameSpaceID_XHTML:
|
|
return FindHTMLData(aElement, aParentFrame, aStyle);
|
|
case kNameSpaceID_MathML:
|
|
return FindMathMLData(aElement, aStyle);
|
|
case kNameSpaceID_SVG:
|
|
return FindSVGData(aElement, aParentFrame,
|
|
aFlags.contains(ItemFlag::IsWithinSVGText),
|
|
aFlags.contains(ItemFlag::AllowTextPathChild), aStyle);
|
|
case kNameSpaceID_XUL:
|
|
return FindXULTagData(aElement, aStyle);
|
|
default:
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AddFrameConstructionItemsInternal(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, bool aSuppressWhiteSpaceOptimizations,
|
|
ComputedStyle* aComputedStyle, ItemFlags aFlags,
|
|
FrameConstructionItemList& aItems) {
|
|
MOZ_ASSERT(aContent->IsText() || aContent->IsElement(),
|
|
"Shouldn't get anything else here!");
|
|
MOZ_ASSERT(aContent->IsInComposedDoc());
|
|
MOZ_ASSERT(!aContent->GetPrimaryFrame() || aState.mCreatingExtraFrames ||
|
|
aContent->NodeInfo()->NameAtom() == nsGkAtoms::area);
|
|
|
|
const bool withinSVGText = aFlags.contains(ItemFlag::IsWithinSVGText);
|
|
const bool isGeneratedContent = aFlags.contains(ItemFlag::IsGeneratedContent);
|
|
MOZ_ASSERT(!isGeneratedContent || aComputedStyle->IsPseudoElement(),
|
|
"Generated content should be a pseudo-element");
|
|
|
|
FrameConstructionItem* item = nullptr;
|
|
auto cleanupGeneratedContent = mozilla::MakeScopeExit([&]() {
|
|
if (isGeneratedContent && !item) {
|
|
MOZ_ASSERT(!IsDisplayContents(aContent),
|
|
"This would need to change if we support display: contents "
|
|
"in generated content");
|
|
aContent->UnbindFromTree();
|
|
}
|
|
});
|
|
|
|
// 'display:none' elements never creates any frames at all.
|
|
const nsStyleDisplay& display = *aComputedStyle->StyleDisplay();
|
|
if (display.mDisplay == StyleDisplay::None) {
|
|
return;
|
|
}
|
|
|
|
if (display.mDisplay == StyleDisplay::Contents) {
|
|
// See the mDisplay fixup code in StyleAdjuster::adjust.
|
|
MOZ_ASSERT(!aContent->AsElement()->IsRootOfNativeAnonymousSubtree(),
|
|
"display:contents on anonymous content is unsupported");
|
|
|
|
// FIXME(bug 1588477): <svg:text>'s TextNodeCorrespondenceRecorder has
|
|
// trouble with everything that looks like display: contents.
|
|
if (withinSVGText) {
|
|
return;
|
|
}
|
|
|
|
CreateGeneratedContentItem(aState, aParentFrame, *aContent->AsElement(),
|
|
*aComputedStyle, PseudoStyleType::before,
|
|
aItems);
|
|
|
|
FlattenedChildIterator iter(aContent);
|
|
InsertionPoint insertion(aParentFrame, aContent);
|
|
for (nsIContent* child = iter.GetNextChild(); child;
|
|
child = iter.GetNextChild()) {
|
|
AddFrameConstructionItems(aState, child, aSuppressWhiteSpaceOptimizations,
|
|
*aComputedStyle, insertion, aItems, aFlags);
|
|
}
|
|
aItems.SetParentHasNoShadowDOM(!iter.ShadowDOMInvolved());
|
|
|
|
CreateGeneratedContentItem(aState, aParentFrame, *aContent->AsElement(),
|
|
*aComputedStyle, PseudoStyleType::after, aItems);
|
|
return;
|
|
}
|
|
|
|
nsIContent* parent = aParentFrame ? aParentFrame->GetContent() : nullptr;
|
|
if (ShouldSuppressFrameInSelect(parent, *aContent)) {
|
|
return;
|
|
}
|
|
|
|
if (aContent->IsHTMLElement(nsGkAtoms::legend) && aParentFrame) {
|
|
const nsFieldSetFrame* const fs = GetFieldSetFrameFor(aParentFrame);
|
|
if (fs && !fs->GetLegend() && !aState.mHasRenderedLegend &&
|
|
!aComputedStyle->StyleDisplay()->IsFloatingStyle() &&
|
|
!aComputedStyle->StyleDisplay()->IsAbsolutelyPositionedStyle()) {
|
|
aState.mHasRenderedLegend = true;
|
|
aFlags += ItemFlag::IsForRenderedLegend;
|
|
}
|
|
}
|
|
|
|
const FrameConstructionData* const data =
|
|
FindDataForContent(*aContent, *aComputedStyle, aParentFrame, aFlags);
|
|
if (!data || data->mBits & FCDATA_SUPPRESS_FRAME) {
|
|
return;
|
|
}
|
|
|
|
const bool isPopup = data->mBits & FCDATA_IS_POPUP;
|
|
|
|
const uint32_t bits = data->mBits;
|
|
|
|
// Inside colgroups, suppress everything except columns.
|
|
if (aParentFrame && aParentFrame->IsTableColGroupFrame() &&
|
|
(!(bits & FCDATA_IS_TABLE_PART) ||
|
|
display.mDisplay != StyleDisplay::TableColumn)) {
|
|
return;
|
|
}
|
|
|
|
const bool canHavePageBreak =
|
|
aFlags.contains(ItemFlag::AllowPageBreak) &&
|
|
aState.mPresContext->IsPaginated() &&
|
|
!display.IsAbsolutelyPositionedStyle() &&
|
|
!(aParentFrame && aParentFrame->IsFlexOrGridContainer()) &&
|
|
!(bits & FCDATA_IS_TABLE_PART) && !(bits & FCDATA_IS_SVG_TEXT);
|
|
if (canHavePageBreak && display.BreakBefore()) {
|
|
AppendPageBreakItem(aContent, aItems);
|
|
}
|
|
|
|
if (!item) {
|
|
item = aItems.AppendItem(this, data, aContent, do_AddRef(aComputedStyle),
|
|
aSuppressWhiteSpaceOptimizations);
|
|
if (aFlags.contains(ItemFlag::IsForRenderedLegend)) {
|
|
item->mIsRenderedLegend = true;
|
|
}
|
|
}
|
|
item->mIsText = !aContent->IsElement();
|
|
item->mIsGeneratedContent = isGeneratedContent;
|
|
if (isGeneratedContent) {
|
|
// We need to keep this alive until the frame takes ownership.
|
|
// This corresponds to the Release in ConstructFramesFromItem.
|
|
item->mContent->AddRef();
|
|
}
|
|
item->mIsPopup = isPopup;
|
|
|
|
if (canHavePageBreak && display.BreakAfter()) {
|
|
AppendPageBreakItem(aContent, aItems);
|
|
}
|
|
|
|
if (bits & FCDATA_IS_INLINE) {
|
|
// To correctly set item->mIsAllInline we need to build up our child items
|
|
// right now.
|
|
BuildInlineChildItems(aState, *item,
|
|
aFlags.contains(ItemFlag::IsWithinSVGText),
|
|
aFlags.contains(ItemFlag::AllowTextPathChild));
|
|
item->mIsBlock = false;
|
|
} else {
|
|
// Compute a boolean isInline which is guaranteed to be false for blocks
|
|
// (but may also be false for some inlines).
|
|
const bool isInline =
|
|
// Table-internal things are inline-outside if and only if they're kids
|
|
// of inlines, since they'll trigger construction of inline-table
|
|
// pseudos.
|
|
((bits & FCDATA_IS_TABLE_PART) &&
|
|
(!aParentFrame || // No aParentFrame means inline
|
|
aParentFrame->StyleDisplay()->IsInlineFlow())) ||
|
|
// Things that are inline-outside but aren't inline frames are inline
|
|
display.IsInlineOutsideStyle() ||
|
|
// Popups that are certainly out of flow.
|
|
isPopup;
|
|
|
|
// Set mIsAllInline conservatively. It just might be that even an inline
|
|
// that has mIsAllInline false doesn't need an {ib} split. So this is just
|
|
// an optimization to keep from doing too much work in cases when we can
|
|
// show that mIsAllInline is true..
|
|
item->mIsAllInline =
|
|
isInline ||
|
|
// Figure out whether we're guaranteed this item will be out of flow.
|
|
// This is not a precise test, since one of our ancestor inlines might
|
|
// add an absolute containing block (if it's relatively positioned) when
|
|
// there wasn't such a containing block before. But it's conservative
|
|
// in the sense that anything that will really end up as an in-flow
|
|
// non-inline will test false here. In other words, if this test is
|
|
// true we're guaranteed to be inline; if it's false we don't know what
|
|
// we'll end up as.
|
|
//
|
|
// If we make this test precise, we can remove some of the code dealing
|
|
// with the imprecision in ConstructInline and adjust the comments on
|
|
// mIsAllInline and mIsBlock in the header.
|
|
(!(bits & FCDATA_DISALLOW_OUT_OF_FLOW) &&
|
|
aState.GetGeometricParent(display, nullptr));
|
|
|
|
// Set mIsBlock conservatively. It's OK to set it false for some real
|
|
// blocks, but not OK to set it true for things that aren't blocks. Since
|
|
// isOutOfFlow might be false even in cases when the frame will end up
|
|
// out-of-flow, we can't use it here. But we _can_ say that the frame will
|
|
// for sure end up in-flow if it's not floated or absolutely positioned.
|
|
item->mIsBlock = !isInline && !display.IsAbsolutelyPositionedStyle() &&
|
|
!display.IsFloatingStyle() && !(bits & FCDATA_IS_SVG_TEXT);
|
|
}
|
|
|
|
if (item->mIsAllInline) {
|
|
aItems.InlineItemAdded();
|
|
} else if (item->mIsBlock) {
|
|
aItems.BlockItemAdded();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return true if the frame construction item pointed to by aIter will
|
|
* create a frame adjacent to a line boundary in the frame tree, and that
|
|
* line boundary is induced by a content node adjacent to the frame's
|
|
* content node in the content tree. The latter condition is necessary so
|
|
* that ContentAppended/ContentInserted/ContentRemoved can easily find any
|
|
* text nodes that were suppressed here.
|
|
*/
|
|
bool nsCSSFrameConstructor::AtLineBoundary(FCItemIterator& aIter) {
|
|
if (aIter.item().mSuppressWhiteSpaceOptimizations) {
|
|
return false;
|
|
}
|
|
|
|
if (aIter.AtStart()) {
|
|
if (aIter.List()->HasLineBoundaryAtStart() &&
|
|
!aIter.item().mContent->GetPreviousSibling()) {
|
|
return true;
|
|
}
|
|
} else {
|
|
FCItemIterator prev = aIter;
|
|
prev.Prev();
|
|
if (prev.item().IsLineBoundary() &&
|
|
!prev.item().mSuppressWhiteSpaceOptimizations &&
|
|
aIter.item().mContent->GetPreviousSibling() == prev.item().mContent) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
FCItemIterator next = aIter;
|
|
next.Next();
|
|
if (next.IsDone()) {
|
|
if (aIter.List()->HasLineBoundaryAtEnd() &&
|
|
!aIter.item().mContent->GetNextSibling()) {
|
|
return true;
|
|
}
|
|
} else {
|
|
if (next.item().IsLineBoundary() &&
|
|
!next.item().mSuppressWhiteSpaceOptimizations &&
|
|
aIter.item().mContent->GetNextSibling() == next.item().mContent) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ConstructFramesFromItem(
|
|
nsFrameConstructorState& aState, FCItemIterator& aIter,
|
|
nsContainerFrame* aParentFrame, nsFrameList& aFrameList) {
|
|
FrameConstructionItem& item = aIter.item();
|
|
ComputedStyle* computedStyle = item.mComputedStyle;
|
|
if (item.mIsText) {
|
|
// If this is collapsible whitespace next to a line boundary,
|
|
// don't create a frame. item.IsWhitespace() also sets the
|
|
// NS_CREATE_FRAME_IF_NON_WHITESPACE flag in the text node. (If we
|
|
// end up creating a frame, nsTextFrame::Init will clear the flag.)
|
|
// We don't do this for generated content, because some generated
|
|
// text content is empty text nodes that are about to be initialized.
|
|
// (We check mAdditionalStateBits because only the generated content
|
|
// container's frame construction item is marked with
|
|
// mIsGeneratedContent, and we might not have an aParentFrame.)
|
|
// We don't do it for content that may have Shadow DOM siblings / insertion
|
|
// points, because they make it difficult to correctly create the frame due
|
|
// to dynamic changes.
|
|
// We don't do it for SVG text, since we might need to position and
|
|
// measure the white space glyphs due to x/y/dx/dy attributes.
|
|
if (AtLineBoundary(aIter) &&
|
|
!computedStyle->StyleText()->WhiteSpaceOrNewlineIsSignificant() &&
|
|
aIter.List()->ParentHasNoShadowDOM() &&
|
|
!(aState.mAdditionalStateBits & NS_FRAME_GENERATED_CONTENT) &&
|
|
(item.mFCData->mBits & FCDATA_IS_LINE_PARTICIPANT) &&
|
|
!(item.mFCData->mBits & FCDATA_IS_SVG_TEXT) &&
|
|
!mAlwaysCreateFramesForIgnorableWhitespace &&
|
|
item.IsWhitespace(aState)) {
|
|
return;
|
|
}
|
|
|
|
ConstructTextFrame(item.mFCData, aState, item.mContent, aParentFrame,
|
|
computedStyle, aFrameList);
|
|
return;
|
|
}
|
|
|
|
AutoRestore<nsFrameState> savedStateBits(aState.mAdditionalStateBits);
|
|
if (item.mIsGeneratedContent) {
|
|
// Ensure that frames created here are all tagged with
|
|
// NS_FRAME_GENERATED_CONTENT.
|
|
aState.mAdditionalStateBits |= NS_FRAME_GENERATED_CONTENT;
|
|
}
|
|
|
|
// XXXbz maybe just inline ConstructFrameFromItemInternal here or something?
|
|
ConstructFrameFromItemInternal(item, aState, aParentFrame, aFrameList);
|
|
|
|
if (item.mIsGeneratedContent) {
|
|
// This corresponds to the AddRef in AddFrameConstructionItemsInternal.
|
|
// The frame owns the generated content now.
|
|
item.mContent->Release();
|
|
|
|
// Now that we've passed ownership of item.mContent to the frame, unset
|
|
// our generated content flag so we don't release or unbind it ourselves.
|
|
item.mIsGeneratedContent = false;
|
|
}
|
|
}
|
|
|
|
nsContainerFrame* nsCSSFrameConstructor::GetAbsoluteContainingBlock(
|
|
nsIFrame* aFrame, ContainingBlockType aType) {
|
|
// Starting with aFrame, look for a frame that is absolutely positioned or
|
|
// relatively positioned (and transformed, if aType is FIXED)
|
|
for (nsIFrame* frame = aFrame; frame; frame = frame->GetParent()) {
|
|
if (frame->IsMathMLFrame()) {
|
|
// If it's mathml, bail out -- no absolute positioning out from inside
|
|
// mathml frames. Note that we don't make this part of the loop
|
|
// condition because of the stuff at the end of this method...
|
|
return nullptr;
|
|
}
|
|
|
|
// Look for the ICB.
|
|
if (aType == FIXED_POS) {
|
|
LayoutFrameType t = frame->Type();
|
|
if (t == LayoutFrameType::Viewport || t == LayoutFrameType::PageContent) {
|
|
return static_cast<nsContainerFrame*>(frame);
|
|
}
|
|
}
|
|
|
|
// If the frame is positioned, we will probably return it as the containing
|
|
// block (see the exceptions below). Otherwise, we'll start looking at the
|
|
// parent frame, unless we're dealing with a scrollframe.
|
|
// Scrollframes are special since they're not positioned, but their
|
|
// scrolledframe might be. So, we need to check this special case to return
|
|
// the correct containing block (the scrolledframe) in that case.
|
|
// If we're looking for a fixed-pos containing block and the frame is
|
|
// not transformed, skip it.
|
|
if (!frame->IsAbsPosContainingBlock()) {
|
|
continue;
|
|
}
|
|
if (aType == FIXED_POS && !frame->IsFixedPosContainingBlock()) {
|
|
continue;
|
|
}
|
|
nsIFrame* absPosCBCandidate = frame;
|
|
LayoutFrameType type = absPosCBCandidate->Type();
|
|
if (type == LayoutFrameType::FieldSet) {
|
|
absPosCBCandidate =
|
|
static_cast<nsFieldSetFrame*>(absPosCBCandidate)->GetInner();
|
|
if (!absPosCBCandidate) {
|
|
continue;
|
|
}
|
|
type = absPosCBCandidate->Type();
|
|
}
|
|
if (type == LayoutFrameType::ScrollContainer) {
|
|
ScrollContainerFrame* scrollContainerFrame =
|
|
do_QueryFrame(absPosCBCandidate);
|
|
absPosCBCandidate = scrollContainerFrame->GetScrolledFrame();
|
|
if (!absPosCBCandidate) {
|
|
continue;
|
|
}
|
|
type = absPosCBCandidate->Type();
|
|
}
|
|
// Only first continuations can be containing blocks.
|
|
absPosCBCandidate = absPosCBCandidate->FirstContinuation();
|
|
// Is the frame really an absolute container?
|
|
if (!absPosCBCandidate->IsAbsoluteContainer()) {
|
|
continue;
|
|
}
|
|
|
|
// For tables, skip the inner frame and consider the table wrapper frame.
|
|
if (type == LayoutFrameType::Table) {
|
|
continue;
|
|
}
|
|
// For table wrapper frames, we can just return absPosCBCandidate.
|
|
MOZ_ASSERT((nsContainerFrame*)do_QueryFrame(absPosCBCandidate),
|
|
"abs.pos. containing block must be nsContainerFrame sub-class");
|
|
return static_cast<nsContainerFrame*>(absPosCBCandidate);
|
|
}
|
|
|
|
MOZ_ASSERT(aType != FIXED_POS, "no ICB in this frame tree?");
|
|
|
|
// It is possible for the search for the containing block to fail, because
|
|
// no absolute container can be found in the parent chain. In those cases,
|
|
// we fall back to the document element's containing block.
|
|
return mDocElementContainingBlock;
|
|
}
|
|
|
|
nsContainerFrame* nsCSSFrameConstructor::GetFloatContainingBlock(
|
|
nsIFrame* aFrame) {
|
|
// Starting with aFrame, look for a frame that is a float containing block.
|
|
// If we hit a frame which prevents its descendants from floating, bail out.
|
|
// The logic here needs to match the logic in MaybePushFloatContainingBlock().
|
|
for (nsIFrame* containingBlock = aFrame;
|
|
containingBlock && !ShouldSuppressFloatingOfDescendants(containingBlock);
|
|
containingBlock = containingBlock->GetParent()) {
|
|
if (containingBlock->IsFloatContainingBlock()) {
|
|
MOZ_ASSERT((nsContainerFrame*)do_QueryFrame(containingBlock),
|
|
"float containing block must be nsContainerFrame sub-class");
|
|
return static_cast<nsContainerFrame*>(containingBlock);
|
|
}
|
|
}
|
|
|
|
// If we didn't find a containing block, then there just isn't
|
|
// one.... return null
|
|
return nullptr;
|
|
}
|
|
|
|
/**
|
|
* This function will get the previous sibling to use for an append operation.
|
|
*
|
|
* It takes a parent frame (must not be null) and the next insertion sibling, if
|
|
* the parent content is display: contents or has ::after content (may be null).
|
|
*/
|
|
static nsIFrame* FindAppendPrevSibling(nsIFrame* aParentFrame,
|
|
nsIFrame* aNextSibling) {
|
|
aParentFrame->DrainSelfOverflowList();
|
|
|
|
if (aNextSibling) {
|
|
MOZ_ASSERT(
|
|
aNextSibling->GetParent()->GetContentInsertionFrame() == aParentFrame,
|
|
"Wrong parent");
|
|
return aNextSibling->GetPrevSibling();
|
|
}
|
|
|
|
return aParentFrame->PrincipalChildList().LastChild();
|
|
}
|
|
|
|
/**
|
|
* Finds the right parent frame to append content to aParentFrame.
|
|
*
|
|
* Cannot return or receive null.
|
|
*/
|
|
static nsContainerFrame* ContinuationToAppendTo(
|
|
nsContainerFrame* aParentFrame) {
|
|
MOZ_ASSERT(aParentFrame);
|
|
|
|
if (IsFramePartOfIBSplit(aParentFrame)) {
|
|
// If the frame we are manipulating is a ib-split frame (that is, one that's
|
|
// been created as a result of a block-in-inline situation) then we need to
|
|
// append to the last ib-split sibling, not to the frame itself.
|
|
//
|
|
// Always make sure to look at the last continuation of the frame for the
|
|
// {ib} case, even if that continuation is empty.
|
|
//
|
|
// We don't do this for the non-ib-split-frame case, since in the other
|
|
// cases appending to the last nonempty continuation is fine and in fact not
|
|
// doing that can confuse code that doesn't know to pull kids from
|
|
// continuations other than its next one.
|
|
return static_cast<nsContainerFrame*>(
|
|
GetLastIBSplitSibling(aParentFrame)->LastContinuation());
|
|
}
|
|
|
|
return nsLayoutUtils::LastContinuationWithChild(aParentFrame);
|
|
}
|
|
|
|
/**
|
|
* This function will get the next sibling for a frame insert operation given
|
|
* the parent and previous sibling. aPrevSibling may be null.
|
|
*/
|
|
static nsIFrame* GetInsertNextSibling(nsIFrame* aParentFrame,
|
|
nsIFrame* aPrevSibling) {
|
|
if (aPrevSibling) {
|
|
return aPrevSibling->GetNextSibling();
|
|
}
|
|
|
|
return aParentFrame->PrincipalChildList().FirstChild();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AppendFramesToParent(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aParentFrame,
|
|
nsFrameList& aFrameList, nsIFrame* aPrevSibling, bool aIsRecursiveCall) {
|
|
MOZ_ASSERT(
|
|
!IsFramePartOfIBSplit(aParentFrame) || !GetIBSplitSibling(aParentFrame) ||
|
|
!GetIBSplitSibling(aParentFrame)->PrincipalChildList().FirstChild(),
|
|
"aParentFrame has a ib-split sibling with kids?");
|
|
MOZ_ASSERT(!aPrevSibling || aPrevSibling->GetParent() == aParentFrame,
|
|
"Parent and prevsibling don't match");
|
|
MOZ_ASSERT(
|
|
!aParentFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR) ||
|
|
!IsFramePartOfIBSplit(aParentFrame),
|
|
"We should have wiped aParentFrame in WipeContainingBlock() "
|
|
"if it's part of an IB split!");
|
|
|
|
nsIFrame* nextSibling = ::GetInsertNextSibling(aParentFrame, aPrevSibling);
|
|
|
|
NS_ASSERTION(nextSibling || !aParentFrame->GetNextContinuation() ||
|
|
!aParentFrame->GetNextContinuation()
|
|
->PrincipalChildList()
|
|
.FirstChild() ||
|
|
aIsRecursiveCall,
|
|
"aParentFrame has later continuations with kids?");
|
|
NS_ASSERTION(
|
|
nextSibling || !IsFramePartOfIBSplit(aParentFrame) ||
|
|
(IsInlineFrame(aParentFrame) && !GetIBSplitSibling(aParentFrame) &&
|
|
!aParentFrame->GetNextContinuation()) ||
|
|
aIsRecursiveCall,
|
|
"aParentFrame is not last?");
|
|
|
|
// If we're inserting a list of frames at the end of the trailing inline
|
|
// of an {ib} split, we may need to create additional {ib} siblings to parent
|
|
// them.
|
|
if (!nextSibling && IsFramePartOfIBSplit(aParentFrame)) {
|
|
// When we get here, our frame list might start with a block. If it does
|
|
// so, and aParentFrame is an inline, and it and all its previous
|
|
// continuations have no siblings, then put the initial blocks from the
|
|
// frame list into the previous block of the {ib} split. Note that we
|
|
// didn't want to stop at the block part of the split when figuring out
|
|
// initial parent, because that could screw up float parenting; it's easier
|
|
// to do this little fixup here instead.
|
|
if (aFrameList.NotEmpty() && aFrameList.FirstChild()->IsBlockOutside()) {
|
|
// See whether our trailing inline is empty
|
|
nsIFrame* firstContinuation = aParentFrame->FirstContinuation();
|
|
if (firstContinuation->PrincipalChildList().IsEmpty()) {
|
|
// Our trailing inline is empty. Collect our starting blocks from
|
|
// aFrameList, get the right parent frame for them, and put them in.
|
|
nsFrameList blockKids =
|
|
aFrameList.Split([](nsIFrame* f) { return !f->IsBlockOutside(); });
|
|
NS_ASSERTION(blockKids.NotEmpty(), "No blocks?");
|
|
|
|
nsContainerFrame* prevBlock = GetIBSplitPrevSibling(firstContinuation);
|
|
prevBlock =
|
|
static_cast<nsContainerFrame*>(prevBlock->LastContinuation());
|
|
NS_ASSERTION(prevBlock, "Should have previous block here");
|
|
|
|
MoveChildrenTo(aParentFrame, prevBlock, blockKids);
|
|
}
|
|
}
|
|
|
|
// We want to put some of the frames into this inline frame.
|
|
nsFrameList inlineKids =
|
|
aFrameList.Split([](nsIFrame* f) { return f->IsBlockOutside(); });
|
|
|
|
if (!inlineKids.IsEmpty()) {
|
|
AppendFrames(aParentFrame, FrameChildListID::Principal,
|
|
std::move(inlineKids));
|
|
}
|
|
|
|
if (!aFrameList.IsEmpty()) {
|
|
nsFrameList ibSiblings;
|
|
CreateIBSiblings(aState, aParentFrame,
|
|
aParentFrame->IsAbsPosContainingBlock(), aFrameList,
|
|
ibSiblings);
|
|
|
|
// Make sure to trigger reflow of the inline that used to be our
|
|
// last one and now isn't anymore, since its GetSkipSides() has
|
|
// changed.
|
|
mPresShell->FrameNeedsReflow(aParentFrame,
|
|
IntrinsicDirty::FrameAndAncestors,
|
|
NS_FRAME_HAS_DIRTY_CHILDREN);
|
|
|
|
// Recurse so we create new ib siblings as needed for aParentFrame's
|
|
// parent
|
|
return AppendFramesToParent(aState, aParentFrame->GetParent(), ibSiblings,
|
|
aParentFrame, true);
|
|
}
|
|
return;
|
|
}
|
|
|
|
// If we're appending a list of frames to the last continuations of a
|
|
// ::-moz-column-content, we may need to create column-span siblings for them.
|
|
if (!nextSibling && IsLastContinuationForColumnContent(aParentFrame)) {
|
|
// Extract any initial non-column-span kids, and append them to
|
|
// ::-moz-column-content's child list.
|
|
nsFrameList initialNonColumnSpanKids =
|
|
aFrameList.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
AppendFrames(aParentFrame, FrameChildListID::Principal,
|
|
std::move(initialNonColumnSpanKids));
|
|
|
|
if (aFrameList.IsEmpty()) {
|
|
// No more kids to process (there weren't any column-span kids).
|
|
return;
|
|
}
|
|
|
|
nsFrameList columnSpanSiblings = CreateColumnSpanSiblings(
|
|
aState, aParentFrame, aFrameList,
|
|
// Column content should never be a absolute/fixed positioned containing
|
|
// block. Pass nullptr as aPositionedFrame.
|
|
nullptr);
|
|
|
|
nsContainerFrame* columnSetWrapper = aParentFrame->GetParent();
|
|
while (!columnSetWrapper->IsColumnSetWrapperFrame()) {
|
|
columnSetWrapper = columnSetWrapper->GetParent();
|
|
}
|
|
MOZ_ASSERT(columnSetWrapper,
|
|
"No ColumnSetWrapperFrame ancestor for -moz-column-content?");
|
|
|
|
FinishBuildingColumns(aState, columnSetWrapper, aParentFrame,
|
|
columnSpanSiblings);
|
|
|
|
MOZ_ASSERT(columnSpanSiblings.IsEmpty(),
|
|
"The column-span siblings should be moved to the proper place!");
|
|
return;
|
|
}
|
|
|
|
// Insert the frames after our aPrevSibling
|
|
InsertFrames(aParentFrame, FrameChildListID::Principal, aPrevSibling,
|
|
std::move(aFrameList));
|
|
}
|
|
|
|
// This gets called to see if the frames corresponding to aSibling and aContent
|
|
// should be siblings in the frame tree. Although (1) rows and cols, (2) row
|
|
// groups and col groups, (3) row groups and captions, (4) legends and content
|
|
// inside fieldsets, (5) popups and other kids of the menu are siblings from a
|
|
// content perspective, they are not considered siblings in the frame tree.
|
|
bool nsCSSFrameConstructor::IsValidSibling(nsIFrame* aSibling,
|
|
nsIContent* aContent,
|
|
Maybe<StyleDisplay>& aDisplay) {
|
|
StyleDisplay siblingDisplay = aSibling->GetDisplay();
|
|
if (StyleDisplay::TableColumnGroup == siblingDisplay ||
|
|
StyleDisplay::TableColumn == siblingDisplay ||
|
|
StyleDisplay::TableCaption == siblingDisplay ||
|
|
StyleDisplay::TableHeaderGroup == siblingDisplay ||
|
|
StyleDisplay::TableRowGroup == siblingDisplay ||
|
|
StyleDisplay::TableFooterGroup == siblingDisplay) {
|
|
// if we haven't already, resolve a style to find the display type of
|
|
// aContent.
|
|
if (aDisplay.isNothing()) {
|
|
if (aContent->IsComment() || aContent->IsProcessingInstruction()) {
|
|
// Comments and processing instructions never have frames, so we should
|
|
// not try to generate styles for them.
|
|
return false;
|
|
}
|
|
// FIXME(emilio): This is buggy some times, see bug 1424656.
|
|
RefPtr<ComputedStyle> computedStyle = ResolveComputedStyle(aContent);
|
|
const nsStyleDisplay* display = computedStyle->StyleDisplay();
|
|
aDisplay.emplace(display->mDisplay);
|
|
}
|
|
|
|
StyleDisplay display = aDisplay.value();
|
|
// To have decent performance we want to return false in cases in which
|
|
// reordering the two siblings has no effect on display. To ensure
|
|
// correctness, we MUST return false in cases where the two siblings have
|
|
// the same desired parent type and live on different display lists.
|
|
// Specificaly, columns and column groups should only consider columns and
|
|
// column groups as valid siblings. Captions should only consider other
|
|
// captions. All other things should consider each other as valid
|
|
// siblings. The restriction in the |if| above on siblingDisplay is ok,
|
|
// because for correctness the only part that really needs to happen is to
|
|
// not consider captions, column groups, and row/header/footer groups
|
|
// siblings of each other. Treating a column or colgroup as a valid
|
|
// sibling of a non-table-related frame will just mean we end up reframing.
|
|
if ((siblingDisplay == StyleDisplay::TableCaption) !=
|
|
(display == StyleDisplay::TableCaption)) {
|
|
// One's a caption and the other is not. Not valid siblings.
|
|
return false;
|
|
}
|
|
|
|
if ((siblingDisplay == StyleDisplay::TableColumnGroup ||
|
|
siblingDisplay == StyleDisplay::TableColumn) !=
|
|
(display == StyleDisplay::TableColumnGroup ||
|
|
display == StyleDisplay::TableColumn)) {
|
|
// One's a column or column group and the other is not. Not valid
|
|
// siblings.
|
|
return false;
|
|
}
|
|
// Fall through; it's possible that the display type was overridden and
|
|
// a different sort of frame was constructed, so we may need to return false
|
|
// below.
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// FIXME(emilio): If we ever kill IsValidSibling() we can simplify this quite a
|
|
// bit (no need to pass aTargetContent or aTargetContentDisplay, and the
|
|
// adjust() calls can be responsibility of the caller).
|
|
template <nsCSSFrameConstructor::SiblingDirection aDirection>
|
|
nsIFrame* nsCSSFrameConstructor::FindSiblingInternal(
|
|
FlattenedChildIterator& aIter, nsIContent* aTargetContent,
|
|
Maybe<StyleDisplay>& aTargetContentDisplay) {
|
|
auto adjust = [&](nsIFrame* aPotentialSiblingFrame) -> nsIFrame* {
|
|
return AdjustSiblingFrame(aPotentialSiblingFrame, aTargetContent,
|
|
aTargetContentDisplay, aDirection);
|
|
};
|
|
|
|
auto nextDomSibling = [](FlattenedChildIterator& aIter) -> nsIContent* {
|
|
return aDirection == SiblingDirection::Forward ? aIter.GetNextChild()
|
|
: aIter.GetPreviousChild();
|
|
};
|
|
|
|
auto getInsideMarkerFrame = [](const nsIContent* aContent) -> nsIFrame* {
|
|
auto* marker = nsLayoutUtils::GetMarkerFrame(aContent);
|
|
const bool isInsideMarker =
|
|
marker && marker->GetInFlowParent()->StyleList()->mListStylePosition ==
|
|
StyleListStylePosition::Inside;
|
|
return isInsideMarker ? marker : nullptr;
|
|
};
|
|
|
|
auto getNearPseudo = [&](const nsIContent* aContent) -> nsIFrame* {
|
|
if (aDirection == SiblingDirection::Forward) {
|
|
if (auto* marker = getInsideMarkerFrame(aContent)) {
|
|
return marker;
|
|
}
|
|
return nsLayoutUtils::GetBeforeFrame(aContent);
|
|
}
|
|
return nsLayoutUtils::GetAfterFrame(aContent);
|
|
};
|
|
|
|
auto getFarPseudo = [&](const nsIContent* aContent) -> nsIFrame* {
|
|
if (aDirection == SiblingDirection::Forward) {
|
|
return nsLayoutUtils::GetAfterFrame(aContent);
|
|
}
|
|
if (auto* before = nsLayoutUtils::GetBeforeFrame(aContent)) {
|
|
return before;
|
|
}
|
|
return getInsideMarkerFrame(aContent);
|
|
};
|
|
|
|
while (nsIContent* sibling = nextDomSibling(aIter)) {
|
|
// NOTE(emilio): It's important to check GetPrimaryFrame() before
|
|
// IsDisplayContents to get the correct insertion point when multiple
|
|
// siblings go from display: non-none to display: contents.
|
|
if (nsIFrame* primaryFrame = sibling->GetPrimaryFrame()) {
|
|
// XXX the GetContent() == sibling check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
if (primaryFrame->GetContent() == sibling) {
|
|
if (nsIFrame* frame = adjust(primaryFrame)) {
|
|
return frame;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (IsDisplayContents(sibling)) {
|
|
if (nsIFrame* frame = adjust(getNearPseudo(sibling))) {
|
|
return frame;
|
|
}
|
|
|
|
const bool startFromBeginning = aDirection == SiblingDirection::Forward;
|
|
FlattenedChildIterator iter(sibling, startFromBeginning);
|
|
nsIFrame* sibling = FindSiblingInternal<aDirection>(
|
|
iter, aTargetContent, aTargetContentDisplay);
|
|
if (sibling) {
|
|
return sibling;
|
|
}
|
|
}
|
|
}
|
|
|
|
return adjust(getFarPseudo(aIter.Parent()));
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::AdjustSiblingFrame(
|
|
nsIFrame* aSibling, nsIContent* aTargetContent,
|
|
Maybe<StyleDisplay>& aTargetContentDisplay, SiblingDirection aDirection) {
|
|
if (!aSibling) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (aSibling->IsRenderedLegend()) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (aSibling->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
|
|
aSibling = aSibling->GetPlaceholderFrame();
|
|
MOZ_ASSERT(aSibling);
|
|
}
|
|
|
|
MOZ_ASSERT(!aSibling->GetPrevContinuation(), "How?");
|
|
if (aDirection == SiblingDirection::Backward) {
|
|
// The frame may be a ib-split frame (a split inline frame that contains a
|
|
// block). Get the last part of that split.
|
|
if (IsFramePartOfIBSplit(aSibling)) {
|
|
aSibling = GetLastIBSplitSibling(aSibling);
|
|
}
|
|
|
|
// The frame may have a continuation. If so, we want the last
|
|
// non-overflow-container continuation as our previous sibling.
|
|
aSibling = aSibling->GetTailContinuation();
|
|
}
|
|
|
|
if (!IsValidSibling(aSibling, aTargetContent, aTargetContentDisplay)) {
|
|
return nullptr;
|
|
}
|
|
|
|
return aSibling;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::FindPreviousSibling(
|
|
const FlattenedChildIterator& aIter,
|
|
Maybe<StyleDisplay>& aTargetContentDisplay) {
|
|
return FindSibling<SiblingDirection::Backward>(aIter, aTargetContentDisplay);
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::FindNextSibling(
|
|
const FlattenedChildIterator& aIter,
|
|
Maybe<StyleDisplay>& aTargetContentDisplay) {
|
|
return FindSibling<SiblingDirection::Forward>(aIter, aTargetContentDisplay);
|
|
}
|
|
|
|
template <nsCSSFrameConstructor::SiblingDirection aDirection>
|
|
nsIFrame* nsCSSFrameConstructor::FindSibling(
|
|
const FlattenedChildIterator& aIter,
|
|
Maybe<StyleDisplay>& aTargetContentDisplay) {
|
|
nsIContent* targetContent = aIter.Get();
|
|
FlattenedChildIterator siblingIter = aIter;
|
|
nsIFrame* sibling = FindSiblingInternal<aDirection>(
|
|
siblingIter, targetContent, aTargetContentDisplay);
|
|
if (sibling) {
|
|
return sibling;
|
|
}
|
|
|
|
// Our siblings (if any) do not have a frame to guide us. The frame for the
|
|
// target content should be inserted whereever a frame for the container would
|
|
// be inserted. This is needed when inserting into display: contents nodes.
|
|
const nsIContent* current = aIter.Parent();
|
|
while (IsDisplayContents(current)) {
|
|
const nsIContent* parent = current->GetFlattenedTreeParent();
|
|
MOZ_ASSERT(parent, "No display: contents on the root");
|
|
|
|
FlattenedChildIterator iter(parent);
|
|
iter.Seek(current);
|
|
sibling = FindSiblingInternal<aDirection>(iter, targetContent,
|
|
aTargetContentDisplay);
|
|
if (sibling) {
|
|
return sibling;
|
|
}
|
|
|
|
current = parent;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
// For fieldsets, returns the area frame, if the child is not a legend.
|
|
static nsContainerFrame* GetAdjustedParentFrame(nsContainerFrame* aParentFrame,
|
|
nsIContent* aChildContent) {
|
|
MOZ_ASSERT(!aParentFrame->IsTableWrapperFrame(), "Shouldn't be happening!");
|
|
|
|
nsContainerFrame* newParent = nullptr;
|
|
if (aParentFrame->IsFieldSetFrame()) {
|
|
// If the parent is a fieldSet, use the fieldSet's area frame as the
|
|
// parent unless the new content is a legend.
|
|
if (!aChildContent->IsHTMLElement(nsGkAtoms::legend)) {
|
|
newParent = static_cast<nsFieldSetFrame*>(aParentFrame)->GetInner();
|
|
if (newParent) {
|
|
newParent = newParent->GetContentInsertionFrame();
|
|
}
|
|
}
|
|
}
|
|
return newParent ? newParent : aParentFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::GetInsertionPrevSibling(
|
|
InsertionPoint* aInsertion, nsIContent* aChild, bool* aIsAppend,
|
|
bool* aIsRangeInsertSafe, nsIContent* aStartSkipChild,
|
|
nsIContent* aEndSkipChild) {
|
|
MOZ_ASSERT(aInsertion->mParentFrame, "Must have parent frame to start with");
|
|
|
|
*aIsAppend = false;
|
|
|
|
// Find the frame that precedes the insertion point.
|
|
FlattenedChildIterator iter(aInsertion->mContainer);
|
|
if (!aChild->IsRootOfNativeAnonymousSubtree()) {
|
|
// The check for IsRootOfNativeAnonymousSubtree() is because editor is
|
|
// severely broken and calls us directly for native anonymous
|
|
// nodes that it creates.
|
|
if (aStartSkipChild) {
|
|
iter.Seek(aStartSkipChild);
|
|
} else {
|
|
iter.Seek(aChild);
|
|
}
|
|
} else {
|
|
// Prime the iterator for the call to FindPreviousSibling.
|
|
iter.GetNextChild();
|
|
MOZ_ASSERT(aChild->GetProperty(nsGkAtoms::restylableAnonymousNode),
|
|
"Someone passed native anonymous content directly into frame "
|
|
"construction. Stop doing that!");
|
|
}
|
|
|
|
// Note that FindPreviousSibling is passed the iterator by value, so that
|
|
// the later usage of the iterator starts from the same place.
|
|
Maybe<StyleDisplay> childDisplay;
|
|
nsIFrame* prevSibling = FindPreviousSibling(iter, childDisplay);
|
|
|
|
// Now, find the geometric parent so that we can handle
|
|
// continuations properly. Use the prev sibling if we have it;
|
|
// otherwise use the next sibling.
|
|
if (prevSibling) {
|
|
aInsertion->mParentFrame =
|
|
prevSibling->GetParent()->GetContentInsertionFrame();
|
|
} else {
|
|
// If there is no previous sibling, then find the frame that follows
|
|
//
|
|
// FIXME(emilio): This is really complex and probably shouldn't be.
|
|
if (aEndSkipChild) {
|
|
iter.Seek(aEndSkipChild);
|
|
iter.GetPreviousChild();
|
|
}
|
|
if (nsIFrame* nextSibling = FindNextSibling(iter, childDisplay)) {
|
|
aInsertion->mParentFrame =
|
|
nextSibling->GetParent()->GetContentInsertionFrame();
|
|
} else {
|
|
// No previous or next sibling, so treat this like an appended frame.
|
|
*aIsAppend = true;
|
|
|
|
// Deal with fieldsets.
|
|
aInsertion->mParentFrame =
|
|
::GetAdjustedParentFrame(aInsertion->mParentFrame, aChild);
|
|
|
|
aInsertion->mParentFrame =
|
|
::ContinuationToAppendTo(aInsertion->mParentFrame);
|
|
|
|
prevSibling = ::FindAppendPrevSibling(aInsertion->mParentFrame, nullptr);
|
|
}
|
|
}
|
|
|
|
*aIsRangeInsertSafe = childDisplay.isNothing();
|
|
return prevSibling;
|
|
}
|
|
|
|
nsContainerFrame* nsCSSFrameConstructor::GetContentInsertionFrameFor(
|
|
nsIContent* aContent) {
|
|
nsIFrame* frame;
|
|
while (!(frame = aContent->GetPrimaryFrame())) {
|
|
if (!IsDisplayContents(aContent)) {
|
|
return nullptr;
|
|
}
|
|
|
|
aContent = aContent->GetFlattenedTreeParent();
|
|
if (!aContent) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
// If the content of the frame is not the desired content then this is not
|
|
// really a frame for the desired content.
|
|
// XXX This check is needed due to bug 135040. Remove it once that's fixed.
|
|
if (frame->GetContent() != aContent) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsContainerFrame* insertionFrame = frame->GetContentInsertionFrame();
|
|
|
|
NS_ASSERTION(!insertionFrame || insertionFrame == frame || !frame->IsLeaf(),
|
|
"The insertion frame is the primary frame or the primary frame "
|
|
"isn't a leaf");
|
|
|
|
return insertionFrame;
|
|
}
|
|
|
|
static bool IsSpecialFramesetChild(nsIContent* aContent) {
|
|
// IMPORTANT: This must match the conditions in nsHTMLFramesetFrame::Init.
|
|
return aContent->IsAnyOfHTMLElements(nsGkAtoms::frameset, nsGkAtoms::frame);
|
|
}
|
|
|
|
static void InvalidateCanvasIfNeeded(PresShell* aPresShell, nsIContent* aNode);
|
|
|
|
void nsCSSFrameConstructor::AddTextItemIfNeeded(
|
|
nsFrameConstructorState& aState, const ComputedStyle& aParentStyle,
|
|
const InsertionPoint& aInsertion, nsIContent* aPossibleTextContent,
|
|
FrameConstructionItemList& aItems) {
|
|
MOZ_ASSERT(aPossibleTextContent, "Must have node");
|
|
if (!aPossibleTextContent->IsText() ||
|
|
!aPossibleTextContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE) ||
|
|
aPossibleTextContent->HasFlag(NODE_NEEDS_FRAME)) {
|
|
// Not text, or not suppressed due to being all-whitespace (if it were being
|
|
// suppressed, it would have the NS_CREATE_FRAME_IF_NON_WHITESPACE flag), or
|
|
// going to be reframed anyway.
|
|
return;
|
|
}
|
|
MOZ_ASSERT(!aPossibleTextContent->GetPrimaryFrame(),
|
|
"Text node has a frame and NS_CREATE_FRAME_IF_NON_WHITESPACE");
|
|
AddFrameConstructionItems(aState, aPossibleTextContent, false, aParentStyle,
|
|
aInsertion, aItems);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ReframeTextIfNeeded(nsIContent* aContent) {
|
|
if (!aContent->IsText() ||
|
|
!aContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE) ||
|
|
aContent->HasFlag(NODE_NEEDS_FRAME)) {
|
|
// Not text, or not suppressed due to being all-whitespace (if it were being
|
|
// suppressed, it would have the NS_CREATE_FRAME_IF_NON_WHITESPACE flag), or
|
|
// going to be reframed anyway.
|
|
return;
|
|
}
|
|
MOZ_ASSERT(!aContent->GetPrimaryFrame(),
|
|
"Text node has a frame and NS_CREATE_FRAME_IF_NON_WHITESPACE");
|
|
ContentInserted(aContent, InsertionKind::Async);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
void nsCSSFrameConstructor::CheckBitsForLazyFrameConstruction(
|
|
nsIContent* aParent) {
|
|
// If we hit a node with no primary frame, or the NODE_NEEDS_FRAME bit set
|
|
// we want to assert, but leaf frames that process their own children and may
|
|
// ignore anonymous children (eg framesets) make this complicated. So we set
|
|
// these two booleans if we encounter these situations and unset them if we
|
|
// hit a node with a leaf frame.
|
|
//
|
|
// It's fine if one of node without primary frame is in a display:none
|
|
// subtree.
|
|
//
|
|
// Also, it's fine if one of the nodes without primary frame is a display:
|
|
// contents node.
|
|
bool noPrimaryFrame = false;
|
|
bool needsFrameBitSet = false;
|
|
nsIContent* content = aParent;
|
|
while (content && !content->HasFlag(NODE_DESCENDANTS_NEED_FRAMES)) {
|
|
if (content->GetPrimaryFrame() && content->GetPrimaryFrame()->IsLeaf()) {
|
|
noPrimaryFrame = needsFrameBitSet = false;
|
|
}
|
|
if (!noPrimaryFrame && !content->GetPrimaryFrame()) {
|
|
noPrimaryFrame = !IsDisplayContents(content);
|
|
}
|
|
if (!needsFrameBitSet && content->HasFlag(NODE_NEEDS_FRAME)) {
|
|
needsFrameBitSet = true;
|
|
}
|
|
|
|
content = content->GetFlattenedTreeParent();
|
|
}
|
|
if (content && content->GetPrimaryFrame() &&
|
|
content->GetPrimaryFrame()->IsLeaf()) {
|
|
noPrimaryFrame = needsFrameBitSet = false;
|
|
}
|
|
MOZ_ASSERT(!noPrimaryFrame,
|
|
"Ancestors of nodes with frames to be "
|
|
"constructed lazily should have frames");
|
|
MOZ_ASSERT(!needsFrameBitSet,
|
|
"Ancestors of nodes with frames to be "
|
|
"constructed lazily should not have NEEDS_FRAME bit set");
|
|
}
|
|
#endif
|
|
|
|
// Returns true if this operation can be lazy, false if not.
|
|
//
|
|
// FIXME(emilio, bug 1410020): This function assumes that the flattened tree
|
|
// parent of all the appended children is the same, which, afaict, is not
|
|
// necessarily true.
|
|
void nsCSSFrameConstructor::ConstructLazily(Operation aOperation,
|
|
nsIContent* aChild) {
|
|
MOZ_ASSERT(aChild->GetParent());
|
|
|
|
// We can construct lazily; just need to set suitable bits in the content
|
|
// tree.
|
|
Element* parent = aChild->GetFlattenedTreeParentElement();
|
|
if (!parent) {
|
|
// Not part of the flat tree, nothing to do.
|
|
return;
|
|
}
|
|
|
|
if (Servo_Element_IsDisplayNone(parent)) {
|
|
// Nothing to do either.
|
|
//
|
|
// FIXME(emilio): This should be an assert, except for weird <frameset>
|
|
// stuff that does its own frame construction. Such an assert would fire in
|
|
// layout/style/crashtests/1411478.html, for example.
|
|
return;
|
|
}
|
|
|
|
// Set NODE_NEEDS_FRAME on the new nodes.
|
|
if (aOperation == CONTENTINSERT) {
|
|
NS_ASSERTION(!aChild->GetPrimaryFrame() ||
|
|
aChild->GetPrimaryFrame()->GetContent() != aChild,
|
|
// XXX the aChild->GetPrimaryFrame()->GetContent() != aChild
|
|
// check is needed due to bug 135040. Remove it once that's
|
|
// fixed.
|
|
"setting NEEDS_FRAME on a node that already has a frame?");
|
|
aChild->SetFlags(NODE_NEEDS_FRAME);
|
|
} else { // CONTENTAPPEND
|
|
for (nsIContent* child = aChild; child; child = child->GetNextSibling()) {
|
|
NS_ASSERTION(!child->GetPrimaryFrame() ||
|
|
child->GetPrimaryFrame()->GetContent() != child,
|
|
// XXX the child->GetPrimaryFrame()->GetContent() != child
|
|
// check is needed due to bug 135040. Remove it once that's
|
|
// fixed.
|
|
"setting NEEDS_FRAME on a node that already has a frame?");
|
|
child->SetFlags(NODE_NEEDS_FRAME);
|
|
}
|
|
}
|
|
|
|
CheckBitsForLazyFrameConstruction(parent);
|
|
parent->NoteDescendantsNeedFramesForServo();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::IssueSingleInsertNofications(
|
|
nsIContent* aStartChild, nsIContent* aEndChild,
|
|
InsertionKind aInsertionKind) {
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
// XXX the GetContent() != child check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
MOZ_ASSERT(!child->GetPrimaryFrame() ||
|
|
child->GetPrimaryFrame()->GetContent() != child);
|
|
|
|
// Call ContentRangeInserted with this node.
|
|
ContentRangeInserted(child, child->GetNextSibling(), aInsertionKind);
|
|
}
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::InsertionPoint::IsMultiple() const {
|
|
// Fieldset frames have multiple normal flow child frame lists so handle it
|
|
// the same as if it had multiple content insertion points.
|
|
return mParentFrame && mParentFrame->IsFieldSetFrame();
|
|
}
|
|
|
|
nsCSSFrameConstructor::InsertionPoint
|
|
nsCSSFrameConstructor::GetRangeInsertionPoint(nsIContent* aStartChild,
|
|
nsIContent* aEndChild,
|
|
InsertionKind aInsertionKind) {
|
|
MOZ_ASSERT(aStartChild);
|
|
|
|
nsIContent* parent = aStartChild->GetParent();
|
|
if (!parent) {
|
|
IssueSingleInsertNofications(aStartChild, aEndChild, aInsertionKind);
|
|
return {};
|
|
}
|
|
|
|
// If the children of the container may be distributed to different insertion
|
|
// points, insert them separately and bail out, letting ContentInserted handle
|
|
// the mess.
|
|
if (parent->GetShadowRoot()) {
|
|
IssueSingleInsertNofications(aStartChild, aEndChild, aInsertionKind);
|
|
return {};
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
{
|
|
nsIContent* expectedParent = aStartChild->GetFlattenedTreeParent();
|
|
for (nsIContent* child = aStartChild->GetNextSibling(); child;
|
|
child = child->GetNextSibling()) {
|
|
MOZ_ASSERT(child->GetFlattenedTreeParent() == expectedParent);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Now the flattened tree parent of all the siblings is the same, just use the
|
|
// same insertion point and take the fast path, unless it's a multiple
|
|
// insertion point.
|
|
InsertionPoint ip = GetInsertionPoint(aStartChild);
|
|
if (ip.IsMultiple()) {
|
|
IssueSingleInsertNofications(aStartChild, aEndChild, aInsertionKind);
|
|
return {};
|
|
}
|
|
|
|
return ip;
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::MaybeRecreateForFrameset(nsIFrame* aParentFrame,
|
|
nsIContent* aStartChild,
|
|
nsIContent* aEndChild) {
|
|
if (aParentFrame->IsFrameSetFrame()) {
|
|
// Check whether we have any kids we care about.
|
|
for (nsIContent* cur = aStartChild; cur != aEndChild;
|
|
cur = cur->GetNextSibling()) {
|
|
if (IsSpecialFramesetChild(cur)) {
|
|
// Just reframe the parent, since framesets are weird like that.
|
|
RecreateFramesForContent(aParentFrame->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::LazilyStyleNewChildRange(nsIContent* aStartChild,
|
|
nsIContent* aEndChild) {
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
if (child->IsElement()) {
|
|
child->AsElement()->NoteDirtyForServo();
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
static bool IsFlattenedTreeChild(nsIContent* aParent, nsIContent* aChild) {
|
|
FlattenedChildIterator iter(aParent);
|
|
for (nsIContent* node = iter.GetNextChild(); node;
|
|
node = iter.GetNextChild()) {
|
|
if (node == aChild) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
void nsCSSFrameConstructor::StyleNewChildRange(nsIContent* aStartChild,
|
|
nsIContent* aEndChild) {
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
if (!child->IsElement()) {
|
|
continue;
|
|
}
|
|
|
|
Element* childElement = child->AsElement();
|
|
|
|
// We only come in here from non-lazy frame construction, so the children
|
|
// should be unstyled.
|
|
MOZ_ASSERT(!childElement->HasServoData());
|
|
|
|
#ifdef DEBUG
|
|
{
|
|
// Furthermore, all of them should have the same flattened tree parent
|
|
// (GetRangeInsertionPoint ensures it). And that parent should be styled,
|
|
// otherwise we would've never found an insertion point at all.
|
|
Element* parent = childElement->GetFlattenedTreeParentElement();
|
|
MOZ_ASSERT(parent);
|
|
MOZ_ASSERT(parent->HasServoData());
|
|
MOZ_ASSERT(
|
|
IsFlattenedTreeChild(parent, child),
|
|
"GetFlattenedTreeParent and ChildIterator don't agree, fix this!");
|
|
}
|
|
#endif
|
|
|
|
styleSet->StyleNewSubtree(childElement);
|
|
}
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::FindNextSiblingForAppend(
|
|
const InsertionPoint& aInsertion) {
|
|
auto SlowPath = [&]() -> nsIFrame* {
|
|
FlattenedChildIterator iter(aInsertion.mContainer,
|
|
/* aStartAtBeginning = */ false);
|
|
iter.GetPreviousChild(); // Prime the iterator.
|
|
Maybe<StyleDisplay> unused;
|
|
return FindNextSibling(iter, unused);
|
|
};
|
|
|
|
if (!IsDisplayContents(aInsertion.mContainer) &&
|
|
!nsLayoutUtils::GetAfterFrame(aInsertion.mContainer)) {
|
|
MOZ_ASSERT(!SlowPath());
|
|
return nullptr;
|
|
}
|
|
|
|
return SlowPath();
|
|
}
|
|
|
|
// This is a bit slow, but sometimes we need it.
|
|
static bool ParentIsWrapperAnonBox(nsIFrame* aParent) {
|
|
nsIFrame* maybeAnonBox = aParent;
|
|
if (maybeAnonBox->Style()->GetPseudoType() == PseudoStyleType::cellContent) {
|
|
// The thing that would maybe be a wrapper anon box is the cell.
|
|
maybeAnonBox = maybeAnonBox->GetParent();
|
|
}
|
|
return maybeAnonBox->Style()->IsWrapperAnonBox();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ContentAppended(nsIContent* aFirstNewContent,
|
|
InsertionKind aInsertionKind) {
|
|
MOZ_ASSERT(aInsertionKind == InsertionKind::Sync ||
|
|
!RestyleManager()->IsInStyleRefresh());
|
|
|
|
AUTO_PROFILER_LABEL_HOT("nsCSSFrameConstructor::ContentAppended",
|
|
LAYOUT_FrameConstruction);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf(
|
|
"nsCSSFrameConstructor::ContentAppended container=%p "
|
|
"first-child=%p lazy=%d\n",
|
|
aFirstNewContent->GetParent(), aFirstNewContent,
|
|
aInsertionKind == InsertionKind::Async);
|
|
if (gReallyNoisyContentUpdates && aFirstNewContent->GetParent()) {
|
|
aFirstNewContent->GetParent()->List(stdout, 0);
|
|
}
|
|
}
|
|
|
|
for (nsIContent* child = aFirstNewContent; child;
|
|
child = child->GetNextSibling()) {
|
|
// XXX the GetContent() != child check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
MOZ_ASSERT(
|
|
!child->GetPrimaryFrame() ||
|
|
child->GetPrimaryFrame()->GetContent() != child,
|
|
"asked to construct a frame for a node that already has a frame");
|
|
}
|
|
#endif
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
InsertionPoint insertion =
|
|
GetRangeInsertionPoint(aFirstNewContent, nullptr, aInsertionKind);
|
|
nsContainerFrame*& parentFrame = insertion.mParentFrame;
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
if (!parentFrame) {
|
|
// We're punting on frame construction because there's no container frame.
|
|
// The Servo-backed style system handles this case like the lazy frame
|
|
// construction case, except when we're already constructing frames, in
|
|
// which case we shouldn't need to do anything else.
|
|
if (aInsertionKind == InsertionKind::Async) {
|
|
LazilyStyleNewChildRange(aFirstNewContent, nullptr);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (aInsertionKind == InsertionKind::Async) {
|
|
ConstructLazily(CONTENTAPPEND, aFirstNewContent);
|
|
LazilyStyleNewChildRange(aFirstNewContent, nullptr);
|
|
return;
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (MaybeRecreateForFrameset(parentFrame, aFirstNewContent, nullptr)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
if (parentFrame->IsLeaf()) {
|
|
// Nothing to do here; we shouldn't be constructing kids of leaves
|
|
// Clear lazy bits so we don't try to construct again.
|
|
ClearLazyBits(aFirstNewContent, nullptr);
|
|
return;
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (WipeInsertionParent(parentFrame)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates && IsFramePartOfIBSplit(parentFrame)) {
|
|
printf("nsCSSFrameConstructor::ContentAppended: parentFrame=");
|
|
parentFrame->ListTag(stdout);
|
|
printf(" is ib-split\n");
|
|
}
|
|
#endif
|
|
|
|
// We should never get here with fieldsets, since they have
|
|
// multiple insertion points.
|
|
MOZ_ASSERT(!parentFrame->IsFieldSetFrame(),
|
|
"Parent frame should not be fieldset!");
|
|
|
|
nsIFrame* nextSibling = FindNextSiblingForAppend(insertion);
|
|
if (nextSibling) {
|
|
parentFrame = nextSibling->GetParent()->GetContentInsertionFrame();
|
|
} else {
|
|
parentFrame = ::ContinuationToAppendTo(parentFrame);
|
|
}
|
|
|
|
nsContainerFrame* containingBlock = GetFloatContainingBlock(parentFrame);
|
|
|
|
// See if the containing block has :first-letter style applied.
|
|
const bool haveFirstLetterStyle =
|
|
containingBlock && HasFirstLetterStyle(containingBlock);
|
|
|
|
const bool haveFirstLineStyle =
|
|
containingBlock && ShouldHaveFirstLineStyle(containingBlock->GetContent(),
|
|
containingBlock->Style());
|
|
|
|
if (haveFirstLetterStyle) {
|
|
AutoWeakFrame wf(nextSibling);
|
|
|
|
// Before we get going, remove the current letter frames
|
|
RemoveLetterFrames(mPresShell, containingBlock);
|
|
|
|
// Reget nextSibling, since we may have killed it.
|
|
//
|
|
// FIXME(emilio): This kinda sucks! :(
|
|
if (nextSibling && !wf) {
|
|
nextSibling = FindNextSiblingForAppend(insertion);
|
|
if (nextSibling) {
|
|
parentFrame = nextSibling->GetParent()->GetContentInsertionFrame();
|
|
containingBlock = GetFloatContainingBlock(parentFrame);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Create some new frames
|
|
nsFrameConstructorState state(
|
|
mPresShell, GetAbsoluteContainingBlock(parentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(parentFrame, ABS_POS), containingBlock);
|
|
|
|
if (mPresShell->GetPresContext()->IsPaginated()) {
|
|
// Because this function can be called outside frame construction, we need
|
|
// to set state.mAutoPageNameValue based on what the parent frame's auto
|
|
// value is.
|
|
// Calling this from outside the frame constructor can violate many of the
|
|
// expectations in AutoFrameConstructionPageName, and unlike during frame
|
|
// construction we already have an auto value from parentFrame, so we do
|
|
// not use AutoFrameConstructionPageName here.
|
|
state.mAutoPageNameValue = parentFrame->GetAutoPageValue();
|
|
#ifdef DEBUG
|
|
parentFrame->mWasVisitedByAutoFrameConstructionPageName = true;
|
|
#endif
|
|
}
|
|
|
|
LayoutFrameType frameType = parentFrame->Type();
|
|
|
|
RefPtr<ComputedStyle> parentStyle =
|
|
ResolveComputedStyle(insertion.mContainer);
|
|
FlattenedChildIterator iter(insertion.mContainer);
|
|
const bool haveNoShadowDOM =
|
|
!iter.ShadowDOMInvolved() || !iter.GetNextChild();
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
if (aFirstNewContent->GetPreviousSibling() &&
|
|
GetParentType(frameType) == eTypeBlock && haveNoShadowDOM) {
|
|
// If there's a text node in the normal content list just before the new
|
|
// items, and it has no frame, make a frame construction item for it. If it
|
|
// doesn't need a frame, ConstructFramesFromItemList below won't give it
|
|
// one. No need to do all this if our parent type is not block, though,
|
|
// since WipeContainingBlock already handles that situation.
|
|
//
|
|
// Because we're appending, we don't need to worry about any text
|
|
// after the appended content; there can only be generated content
|
|
// (and bare text nodes are not generated). Native anonymous content
|
|
// generated by frames never participates in inline layout.
|
|
AddTextItemIfNeeded(state, *parentStyle, insertion,
|
|
aFirstNewContent->GetPreviousSibling(), items);
|
|
}
|
|
for (nsIContent* child = aFirstNewContent; child;
|
|
child = child->GetNextSibling()) {
|
|
AddFrameConstructionItems(state, child, false, *parentStyle, insertion,
|
|
items);
|
|
}
|
|
|
|
nsIFrame* prevSibling = ::FindAppendPrevSibling(parentFrame, nextSibling);
|
|
|
|
// Perform special check for diddling around with the frames in
|
|
// a ib-split inline frame.
|
|
// If we're appending before :after content, then we're not really
|
|
// appending, so let WipeContainingBlock know that.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (WipeContainingBlock(state, containingBlock, parentFrame, items, true,
|
|
prevSibling)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
// If the parent is a block frame, and we're not in a special case
|
|
// where frames can be moved around, determine if the list is for the
|
|
// start or end of the block.
|
|
if (parentFrame->IsBlockFrameOrSubclass() && !haveFirstLetterStyle &&
|
|
!haveFirstLineStyle && !IsFramePartOfIBSplit(parentFrame)) {
|
|
items.SetLineBoundaryAtStart(!prevSibling ||
|
|
!prevSibling->IsInlineOutside() ||
|
|
prevSibling->IsBrFrame());
|
|
// :after content can't be <br> so no need to check it
|
|
//
|
|
// FIXME(emilio): A display: contents sibling could! Write a test-case and
|
|
// fix.
|
|
items.SetLineBoundaryAtEnd(!nextSibling || !nextSibling->IsInlineOutside());
|
|
}
|
|
// To suppress whitespace-only text frames, we have to verify that
|
|
// our container's DOM child list matches its flattened tree child list.
|
|
items.SetParentHasNoShadowDOM(haveNoShadowDOM);
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
state.MaybePushFloatContainingBlock(parentFrame, floatSaveState);
|
|
|
|
nsFrameList frameList;
|
|
ConstructFramesFromItemList(state, items, parentFrame,
|
|
ParentIsWrapperAnonBox(parentFrame), frameList);
|
|
|
|
for (nsIContent* child = aFirstNewContent; child;
|
|
child = child->GetNextSibling()) {
|
|
// Invalidate now instead of before the WipeContainingBlock call, just in
|
|
// case we do wipe; in that case we don't need to do this walk at all.
|
|
// XXXbz does that matter? Would it make more sense to save some virtual
|
|
// GetChildAt_Deprecated calls instead and do this during construction of
|
|
// our FrameConstructionItemList?
|
|
InvalidateCanvasIfNeeded(mPresShell, child);
|
|
}
|
|
|
|
// If the container is a table and a caption was appended, it needs to be put
|
|
// in the table wrapper frame's additional child list.
|
|
nsFrameList captionList;
|
|
if (LayoutFrameType::Table == frameType) {
|
|
// Pull out the captions. Note that we don't want to do that as we go,
|
|
// because processing a single caption can add a whole bunch of things to
|
|
// the frame items due to pseudoframe processing. So we'd have to pull
|
|
// captions from a list anyway; might as well do that here.
|
|
// XXXbz this is no longer true; we could pull captions directly out of the
|
|
// FrameConstructionItemList now.
|
|
PullOutCaptionFrames(frameList, captionList);
|
|
}
|
|
|
|
if (haveFirstLineStyle && parentFrame == containingBlock) {
|
|
// It's possible that some of the new frames go into a
|
|
// first-line frame. Look at them and see...
|
|
AppendFirstLineFrames(state, containingBlock->GetContent(), containingBlock,
|
|
frameList);
|
|
// That moved things into line frames as needed, reparenting their
|
|
// styles. Nothing else needs to be done.
|
|
} else if (parentFrame->Style()->HasPseudoElementData()) {
|
|
// parentFrame might be inside a ::first-line frame. Check whether it is,
|
|
// and if so fix up our styles.
|
|
CheckForFirstLineInsertion(parentFrame, frameList);
|
|
CheckForFirstLineInsertion(parentFrame, captionList);
|
|
}
|
|
|
|
// Notify the parent frame passing it the list of new frames
|
|
// Append the flowed frames to the principal child list; captions
|
|
// need special treatment
|
|
if (captionList.NotEmpty()) { // append the caption to the table wrapper
|
|
NS_ASSERTION(LayoutFrameType::Table == frameType, "how did that happen?");
|
|
nsContainerFrame* outerTable = parentFrame->GetParent();
|
|
captionList.ApplySetParent(outerTable);
|
|
AppendFrames(outerTable, FrameChildListID::Caption, std::move(captionList));
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (MaybeRecreateForColumnSpan(state, parentFrame, frameList, prevSibling)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
if (frameList.NotEmpty()) { // append the in-flow kids
|
|
AppendFramesToParent(state, parentFrame, frameList, prevSibling);
|
|
}
|
|
|
|
// Recover first-letter frames
|
|
if (haveFirstLetterStyle) {
|
|
RecoverLetterFrames(containingBlock);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentAppended: resulting frame model:\n");
|
|
parentFrame->List(stdout);
|
|
}
|
|
#endif
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = GetAccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aFirstNewContent, nullptr);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ContentInserted(nsIContent* aChild,
|
|
InsertionKind aInsertionKind) {
|
|
ContentRangeInserted(aChild, aChild->GetNextSibling(), aInsertionKind);
|
|
}
|
|
|
|
// ContentRangeInserted handles creating frames for a range of nodes that
|
|
// aren't at the end of their childlist. ContentRangeInserted isn't a real
|
|
// content notification, but rather it handles regular ContentInserted calls
|
|
// for a single node as well as the lazy construction of frames for a range of
|
|
// nodes when called from CreateNeededFrames. For a range of nodes to be
|
|
// suitable to have its frames constructed all at once they must meet the same
|
|
// conditions that ContentAppended imposes (GetRangeInsertionPoint checks
|
|
// these), plus more. Namely when finding the insertion prevsibling we must not
|
|
// need to consult something specific to any one node in the range, so that the
|
|
// insertion prevsibling would be the same for each node in the range. So we
|
|
// pass the first node in the range to GetInsertionPrevSibling, and if
|
|
// IsValidSibling (the only place GetInsertionPrevSibling might look at the
|
|
// passed in node itself) needs to resolve style on the node we record this and
|
|
// return that this range needs to be split up and inserted separately. Table
|
|
// captions need extra attention as we need to determine where to insert them
|
|
// in the caption list, while skipping any nodes in the range being inserted
|
|
// (because when we treat the caption frames the other nodes have had their
|
|
// frames constructed but not yet inserted into the frame tree).
|
|
void nsCSSFrameConstructor::ContentRangeInserted(nsIContent* aStartChild,
|
|
nsIContent* aEndChild,
|
|
InsertionKind aInsertionKind) {
|
|
MOZ_ASSERT(aInsertionKind == InsertionKind::Sync ||
|
|
!RestyleManager()->IsInStyleRefresh());
|
|
|
|
AUTO_PROFILER_LABEL_HOT("nsCSSFrameConstructor::ContentRangeInserted",
|
|
LAYOUT_FrameConstruction);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
|
|
MOZ_ASSERT(aStartChild, "must always pass a child");
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf(
|
|
"nsCSSFrameConstructor::ContentRangeInserted container=%p "
|
|
"start-child=%p end-child=%p lazy=%d\n",
|
|
aStartChild->GetParent(), aStartChild, aEndChild,
|
|
aInsertionKind == InsertionKind::Async);
|
|
if (gReallyNoisyContentUpdates) {
|
|
if (aStartChild->GetParent()) {
|
|
aStartChild->GetParent()->List(stdout, 0);
|
|
} else {
|
|
aStartChild->List(stdout, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
// XXX the GetContent() != child check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
NS_ASSERTION(
|
|
!child->GetPrimaryFrame() ||
|
|
child->GetPrimaryFrame()->GetContent() != child,
|
|
"asked to construct a frame for a node that already has a frame");
|
|
}
|
|
#endif
|
|
|
|
bool isSingleInsert = (aStartChild->GetNextSibling() == aEndChild);
|
|
NS_ASSERTION(isSingleInsert || aInsertionKind == InsertionKind::Sync,
|
|
"range insert shouldn't be lazy");
|
|
NS_ASSERTION(isSingleInsert || aEndChild,
|
|
"range should not include all nodes after aStartChild");
|
|
|
|
// If we have a null parent, then this must be the document element being
|
|
// inserted, or some other child of the document in the DOM (might be a PI,
|
|
// say).
|
|
if (!aStartChild->GetParent()) {
|
|
MOZ_ASSERT(isSingleInsert,
|
|
"root node insertion should be a single insertion");
|
|
Element* docElement = mDocument->GetRootElement();
|
|
if (aStartChild != docElement) {
|
|
// Not the root element; just bail out
|
|
return;
|
|
}
|
|
|
|
MOZ_ASSERT(!mRootElementFrame, "root element frame already created");
|
|
if (aInsertionKind == InsertionKind::Async) {
|
|
docElement->SetFlags(NODE_NEEDS_FRAME);
|
|
LazilyStyleNewChildRange(docElement, nullptr);
|
|
return;
|
|
}
|
|
|
|
// Create frames for the document element and its child elements
|
|
if (ConstructDocElementFrame(docElement)) {
|
|
InvalidateCanvasIfNeeded(mPresShell, aStartChild);
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf(
|
|
"nsCSSFrameConstructor::ContentRangeInserted: resulting frame "
|
|
"model:\n");
|
|
mRootElementFrame->List(stdout);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = GetAccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aStartChild, aEndChild);
|
|
}
|
|
#endif
|
|
|
|
return;
|
|
}
|
|
|
|
InsertionPoint insertion;
|
|
if (isSingleInsert) {
|
|
// See if we have a Shadow DOM insertion point. If so, then that's our real
|
|
// parent frame; if not, then the frame hasn't been built yet and we just
|
|
// bail.
|
|
insertion = GetInsertionPoint(aStartChild);
|
|
} else {
|
|
// Get our insertion point. If we need to issue single ContentInserteds
|
|
// GetRangeInsertionPoint will take care of that for us.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
insertion = GetRangeInsertionPoint(aStartChild, aEndChild, aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
|
|
if (!insertion.mParentFrame) {
|
|
// We're punting on frame construction because there's no container frame.
|
|
// The Servo-backed style system handles this case like the lazy frame
|
|
// construction case, except when we're already constructing frames, in
|
|
// which case we shouldn't need to do anything else.
|
|
if (aInsertionKind == InsertionKind::Async) {
|
|
LazilyStyleNewChildRange(aStartChild, aEndChild);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (aInsertionKind == InsertionKind::Async) {
|
|
ConstructLazily(CONTENTINSERT, aStartChild);
|
|
LazilyStyleNewChildRange(aStartChild, aEndChild);
|
|
return;
|
|
}
|
|
|
|
bool isAppend, isRangeInsertSafe;
|
|
nsIFrame* prevSibling = GetInsertionPrevSibling(
|
|
&insertion, aStartChild, &isAppend, &isRangeInsertSafe);
|
|
|
|
// check if range insert is safe
|
|
if (!isSingleInsert && !isRangeInsertSafe) {
|
|
// must fall back to a single ContertInserted for each child in the range
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
IssueSingleInsertNofications(aStartChild, aEndChild, InsertionKind::Sync);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
LayoutFrameType frameType = insertion.mParentFrame->Type();
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (MaybeRecreateForFrameset(insertion.mParentFrame, aStartChild,
|
|
aEndChild)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
// We should only get here with fieldsets when doing a single insert, because
|
|
// fieldsets have multiple insertion points.
|
|
NS_ASSERTION(isSingleInsert || frameType != LayoutFrameType::FieldSet,
|
|
"Unexpected parent");
|
|
// Note that this check is insufficient if aStartChild is not a legend with
|
|
// display::contents that contains a legend. We'll catch that case in
|
|
// WipeContainingBlock. (That code would also catch this case, but handling
|
|
// this early is slightly faster.)
|
|
// XXXmats we should be able to optimize this when the fieldset doesn't
|
|
// currently have a rendered legend. ContentRangeInserted needs to be fixed
|
|
// to use the inner frame as the content insertion frame in that case.
|
|
if (GetFieldSetFrameFor(insertion.mParentFrame) &&
|
|
aStartChild->NodeInfo()->NameAtom() == nsGkAtoms::legend) {
|
|
// Just reframe the parent, since figuring out whether this
|
|
// should be the new legend and then handling it is too complex.
|
|
// We could do a little better here --- check if the fieldset already
|
|
// has a legend which occurs earlier in its child list than this node,
|
|
// and if so, proceed. But we'd have to extend nsFieldSetFrame
|
|
// to locate this legend in the inserted frames and extract it.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(insertion.mParentFrame->GetContent(),
|
|
InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
// Don't construct kids of leaves
|
|
if (insertion.mParentFrame->IsLeaf()) {
|
|
// Clear lazy bits so we don't try to construct again.
|
|
ClearLazyBits(aStartChild, aEndChild);
|
|
return;
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (WipeInsertionParent(insertion.mParentFrame)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
nsFrameConstructorState state(
|
|
mPresShell, GetAbsoluteContainingBlock(insertion.mParentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(insertion.mParentFrame, ABS_POS),
|
|
GetFloatContainingBlock(insertion.mParentFrame),
|
|
do_AddRef(mFrameTreeState));
|
|
|
|
// Recover state for the containing block - we need to know if
|
|
// it has :first-letter or :first-line style applied to it. The
|
|
// reason we care is that the internal structure in these cases
|
|
// is not the normal structure and requires custom updating
|
|
// logic.
|
|
nsContainerFrame* containingBlock = state.mFloatedList.mContainingBlock;
|
|
bool haveFirstLetterStyle = false;
|
|
bool haveFirstLineStyle = false;
|
|
|
|
// In order to shave off some cycles, we only dig up the
|
|
// containing block haveFirst* flags if the parent frame where
|
|
// the insertion/append is occurring is an inline or block
|
|
// container. For other types of containers this isn't relevant.
|
|
StyleDisplayInside parentDisplayInside =
|
|
insertion.mParentFrame->StyleDisplay()->DisplayInside();
|
|
|
|
// Examine the insertion.mParentFrame where the insertion is taking
|
|
// place. If it's a certain kind of container then some special
|
|
// processing is done.
|
|
if (StyleDisplayInside::Flow == parentDisplayInside) {
|
|
// Recover the special style flags for the containing block
|
|
if (containingBlock) {
|
|
haveFirstLetterStyle = HasFirstLetterStyle(containingBlock);
|
|
haveFirstLineStyle = ShouldHaveFirstLineStyle(
|
|
containingBlock->GetContent(), containingBlock->Style());
|
|
}
|
|
|
|
if (haveFirstLetterStyle) {
|
|
// If our current insertion.mParentFrame is a Letter frame, use its parent
|
|
// as our new parent hint
|
|
if (insertion.mParentFrame->IsLetterFrame()) {
|
|
// If insertion.mParentFrame is out of flow, then we actually want the
|
|
// parent of the placeholder frame.
|
|
if (insertion.mParentFrame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
|
|
nsPlaceholderFrame* placeholderFrame =
|
|
insertion.mParentFrame->GetPlaceholderFrame();
|
|
NS_ASSERTION(placeholderFrame, "No placeholder for out-of-flow?");
|
|
insertion.mParentFrame = placeholderFrame->GetParent();
|
|
} else {
|
|
insertion.mParentFrame = insertion.mParentFrame->GetParent();
|
|
}
|
|
}
|
|
|
|
// Remove the old letter frames before doing the insertion
|
|
RemoveLetterFrames(mPresShell, state.mFloatedList.mContainingBlock);
|
|
|
|
// Removing the letterframes messes around with the frame tree, removing
|
|
// and creating frames. We need to reget our prevsibling, parent frame,
|
|
// etc.
|
|
prevSibling = GetInsertionPrevSibling(&insertion, aStartChild, &isAppend,
|
|
&isRangeInsertSafe);
|
|
|
|
// Need check whether a range insert is still safe.
|
|
if (!isSingleInsert && !isRangeInsertSafe) {
|
|
// Need to recover the letter frames first.
|
|
RecoverLetterFrames(state.mFloatedList.mContainingBlock);
|
|
|
|
// must fall back to a single ContertInserted for each child in the
|
|
// range
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
IssueSingleInsertNofications(aStartChild, aEndChild,
|
|
InsertionKind::Sync);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
frameType = insertion.mParentFrame->Type();
|
|
}
|
|
}
|
|
|
|
// This handles fallback to 'list-style-type' when a 'list-style-image' fails
|
|
// to load.
|
|
if (aStartChild->IsInNativeAnonymousSubtree() &&
|
|
aStartChild->IsHTMLElement(nsGkAtoms::mozgeneratedcontentimage)) {
|
|
MOZ_ASSERT(isSingleInsert);
|
|
MOZ_ASSERT(insertion.mParentFrame->Style()->GetPseudoType() ==
|
|
PseudoStyleType::marker,
|
|
"we can only handle ::marker fallback for now");
|
|
nsIContent* const nextSibling = aStartChild->GetNextSibling();
|
|
MOZ_ASSERT(nextSibling && nextSibling->IsText(),
|
|
"expected a text node after the list-style-image image");
|
|
DestroyContext context(mPresShell);
|
|
RemoveFrame(context, FrameChildListID::Principal,
|
|
nextSibling->GetPrimaryFrame());
|
|
auto* const container = aStartChild->GetParent()->AsElement();
|
|
nsIContent* firstNewChild = nullptr;
|
|
auto InsertChild = [this, container, nextSibling,
|
|
&firstNewChild](RefPtr<nsIContent>&& aChild) {
|
|
// We don't strictly have to set NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE
|
|
// here; it would get set under AppendChildTo. But AppendChildTo might
|
|
// think that we're going from not being anonymous to being anonymous and
|
|
// do some extra work; setting the flag here avoids that.
|
|
aChild->SetFlags(NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE);
|
|
container->InsertChildBefore(aChild, nextSibling, false, IgnoreErrors());
|
|
if (auto* childElement = Element::FromNode(aChild)) {
|
|
// If we created any children elements, Servo needs to traverse them,
|
|
// but the root is already set up.
|
|
mPresShell->StyleSet()->StyleNewSubtree(childElement);
|
|
}
|
|
if (!firstNewChild) {
|
|
firstNewChild = aChild;
|
|
}
|
|
};
|
|
CreateGeneratedContentFromListStyleType(
|
|
state, *insertion.mContainer->AsElement(),
|
|
*insertion.mParentFrame->Style(), InsertChild);
|
|
if (!firstNewChild) {
|
|
// No fallback content - we're done.
|
|
return;
|
|
}
|
|
aStartChild = firstNewChild;
|
|
MOZ_ASSERT(firstNewChild->GetNextSibling() == nextSibling,
|
|
"list-style-type should only create one child");
|
|
}
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
RefPtr<ComputedStyle> parentStyle =
|
|
ResolveComputedStyle(insertion.mContainer);
|
|
ParentType parentType = GetParentType(frameType);
|
|
FlattenedChildIterator iter(insertion.mContainer);
|
|
const bool haveNoShadowDOM =
|
|
!iter.ShadowDOMInvolved() || !iter.GetNextChild();
|
|
if (aStartChild->GetPreviousSibling() && parentType == eTypeBlock &&
|
|
haveNoShadowDOM) {
|
|
// If there's a text node in the normal content list just before the
|
|
// new nodes, and it has no frame, make a frame construction item for
|
|
// it, because it might need a frame now. No need to do this if our
|
|
// parent type is not block, though, since WipeContainingBlock
|
|
// already handles that situation.
|
|
AddTextItemIfNeeded(state, *parentStyle, insertion,
|
|
aStartChild->GetPreviousSibling(), items);
|
|
}
|
|
|
|
if (isSingleInsert) {
|
|
AddFrameConstructionItems(state, aStartChild,
|
|
aStartChild->IsRootOfNativeAnonymousSubtree(),
|
|
*parentStyle, insertion, items);
|
|
} else {
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
AddFrameConstructionItems(state, child, false, *parentStyle, insertion,
|
|
items);
|
|
}
|
|
}
|
|
|
|
if (aEndChild && parentType == eTypeBlock && haveNoShadowDOM) {
|
|
// If there's a text node in the normal content list just after the
|
|
// new nodes, and it has no frame, make a frame construction item for
|
|
// it, because it might need a frame now. No need to do this if our
|
|
// parent type is not block, though, since WipeContainingBlock
|
|
// already handles that situation.
|
|
AddTextItemIfNeeded(state, *parentStyle, insertion, aEndChild, items);
|
|
}
|
|
|
|
// Perform special check for diddling around with the frames in
|
|
// a special inline frame.
|
|
// If we're appending before :after content, then we're not really
|
|
// appending, so let WipeContainingBlock know that.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (WipeContainingBlock(state, containingBlock, insertion.mParentFrame, items,
|
|
isAppend, prevSibling)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
state.MaybePushFloatContainingBlock(insertion.mParentFrame, floatSaveState);
|
|
|
|
if (state.mPresContext->IsPaginated()) {
|
|
// Because this function can be called outside frame construction, we need
|
|
// to set state.mAutoPageNameValue based on what the parent frame's auto
|
|
// value is.
|
|
// Calling this from outside the frame constructor can violate many of the
|
|
// expectations in AutoFrameConstructionPageName, and unlike during frame
|
|
// construction we already have an auto value from parentFrame, so we do
|
|
// not use AutoFrameConstructionPageName here.
|
|
state.mAutoPageNameValue = insertion.mParentFrame->GetAutoPageValue();
|
|
#ifdef DEBUG
|
|
insertion.mParentFrame->mWasVisitedByAutoFrameConstructionPageName = true;
|
|
#endif
|
|
}
|
|
|
|
// If the container is a table and a caption will be appended, it needs to be
|
|
// put in the table wrapper frame's additional child list.
|
|
// We make no attempt here to set flags to indicate whether the list
|
|
// will be at the start or end of a block. It doesn't seem worthwhile.
|
|
nsFrameList frameList, captionList;
|
|
ConstructFramesFromItemList(state, items, insertion.mParentFrame,
|
|
ParentIsWrapperAnonBox(insertion.mParentFrame),
|
|
frameList);
|
|
|
|
if (frameList.NotEmpty()) {
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
InvalidateCanvasIfNeeded(mPresShell, child);
|
|
}
|
|
|
|
if (LayoutFrameType::Table == frameType ||
|
|
LayoutFrameType::TableWrapper == frameType) {
|
|
PullOutCaptionFrames(frameList, captionList);
|
|
if (prevSibling && prevSibling->IsTableCaption()) {
|
|
// This can happen, but only if the table is empty (otherwise
|
|
// SafeToInsertPseudoNeedingChildren bails).
|
|
prevSibling = nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (haveFirstLineStyle && insertion.mParentFrame == containingBlock &&
|
|
isAppend) {
|
|
// It's possible that the new frame goes into a first-line
|
|
// frame. Look at it and see...
|
|
AppendFirstLineFrames(state, containingBlock->GetContent(), containingBlock,
|
|
frameList);
|
|
} else if (insertion.mParentFrame->Style()->HasPseudoElementData()) {
|
|
CheckForFirstLineInsertion(insertion.mParentFrame, frameList);
|
|
CheckForFirstLineInsertion(insertion.mParentFrame, captionList);
|
|
}
|
|
|
|
// We might have captions; put them into the caption list of the
|
|
// table wrapper frame.
|
|
if (captionList.NotEmpty()) {
|
|
NS_ASSERTION(LayoutFrameType::Table == frameType ||
|
|
LayoutFrameType::TableWrapper == frameType,
|
|
"parent for caption is not table?");
|
|
// We need to determine where to put the caption items; start with the
|
|
// the parent frame that has already been determined and get the insertion
|
|
// prevsibling of the first caption item.
|
|
bool captionIsAppend;
|
|
nsIFrame* captionPrevSibling = nullptr;
|
|
|
|
// aIsRangeInsertSafe is ignored on purpose because it is irrelevant here.
|
|
bool ignored;
|
|
InsertionPoint captionInsertion = insertion;
|
|
if (isSingleInsert) {
|
|
captionPrevSibling = GetInsertionPrevSibling(
|
|
&captionInsertion, aStartChild, &captionIsAppend, &ignored);
|
|
} else {
|
|
nsIContent* firstCaption = captionList.FirstChild()->GetContent();
|
|
// It is very important here that we skip the children in
|
|
// [aStartChild,aEndChild) when looking for a
|
|
// prevsibling.
|
|
captionPrevSibling = GetInsertionPrevSibling(
|
|
&captionInsertion, firstCaption, &captionIsAppend, &ignored,
|
|
aStartChild, aEndChild);
|
|
}
|
|
|
|
nsContainerFrame* outerTable =
|
|
captionInsertion.mParentFrame->IsTableFrame()
|
|
? captionInsertion.mParentFrame->GetParent()
|
|
: captionInsertion.mParentFrame;
|
|
|
|
// If the parent is not a table wrapper frame we will try to add frames
|
|
// to a named child list that the parent does not honor and the frames
|
|
// will get lost.
|
|
MOZ_ASSERT(outerTable->IsTableWrapperFrame(),
|
|
"Pseudo frame construction failure; "
|
|
"a caption can be only a child of a table wrapper frame");
|
|
|
|
// If the parent of our current prevSibling is different from the frame
|
|
// we'll actually use as the parent, then the calculated insertion
|
|
// point is now invalid (bug 341382).
|
|
if (captionPrevSibling && captionPrevSibling->GetParent() != outerTable) {
|
|
captionPrevSibling = nullptr;
|
|
}
|
|
|
|
captionList.ApplySetParent(outerTable);
|
|
if (captionIsAppend) {
|
|
AppendFrames(outerTable, FrameChildListID::Caption,
|
|
std::move(captionList));
|
|
} else {
|
|
InsertFrames(outerTable, FrameChildListID::Caption, captionPrevSibling,
|
|
std::move(captionList));
|
|
}
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (MaybeRecreateForColumnSpan(state, insertion.mParentFrame, frameList,
|
|
prevSibling)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
if (frameList.NotEmpty()) {
|
|
// Notify the parent frame
|
|
if (isAppend) {
|
|
AppendFramesToParent(state, insertion.mParentFrame, frameList,
|
|
prevSibling);
|
|
} else {
|
|
InsertFrames(insertion.mParentFrame, FrameChildListID::Principal,
|
|
prevSibling, std::move(frameList));
|
|
}
|
|
}
|
|
|
|
if (haveFirstLetterStyle) {
|
|
// Recover the letter frames for the containing block when
|
|
// it has first-letter style.
|
|
RecoverLetterFrames(state.mFloatedList.mContainingBlock);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates && insertion.mParentFrame) {
|
|
printf(
|
|
"nsCSSFrameConstructor::ContentRangeInserted: resulting frame "
|
|
"model:\n");
|
|
insertion.mParentFrame->List(stdout);
|
|
}
|
|
#endif
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = GetAccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aStartChild, aEndChild);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static bool IsWhitespaceFrame(const nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame, "invalid argument");
|
|
return aFrame->IsTextFrame() && aFrame->GetContent()->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
static bool IsSyntheticColGroup(const nsIFrame* aFrame) {
|
|
return aFrame->IsTableColGroupFrame() &&
|
|
static_cast<const nsTableColGroupFrame*>(aFrame)->IsSynthetic();
|
|
}
|
|
|
|
static bool IsOnlyNonWhitespaceFrameInList(const nsFrameList& aFrameList,
|
|
const nsIFrame* aFrame) {
|
|
for (const nsIFrame* f : aFrameList) {
|
|
if (f == aFrame) {
|
|
// If we have continuations, ignore them too.
|
|
aFrame = aFrame->GetNextContinuation();
|
|
} else if (!IsWhitespaceFrame(f) && !IsSyntheticColGroup(f)) {
|
|
// Synthetic colgroups get created unconditionally, so let's not consider
|
|
// them as giving us a non-whitespace frame.
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool AllChildListsAreEffectivelyEmpty(nsIFrame* aFrame) {
|
|
for (auto& [list, listID] : aFrame->ChildLists()) {
|
|
if (list.IsEmpty()) {
|
|
continue;
|
|
}
|
|
// We have some existing frame, usually that would be considered as making
|
|
// this list nonempty. But let's make an exception for the synthetic
|
|
// colgroup that tables have, since that gets created unconditionally.
|
|
if (listID == FrameChildListID::ColGroup) {
|
|
if (nsIFrame* f = list.OnlyChild(); f && IsSyntheticColGroup(f)) {
|
|
continue;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool SafeToInsertPseudoNeedingChildren(nsIFrame* aFrame) {
|
|
return AllChildListsAreEffectivelyEmpty(aFrame);
|
|
}
|
|
|
|
// Returns true if aFrame is the only meaningful child of aParent (which is
|
|
// known to be a wrapper-pseudo). This lets us determine whether aParent can be
|
|
// removed, as a result of aFrame being removed.
|
|
static bool IsOnlyMeaningfulChildOfWrapperPseudo(nsIFrame* aFrame,
|
|
nsIFrame* aParent) {
|
|
MOZ_ASSERT(IsWrapperPseudo(aParent));
|
|
// Handle a few special cases with tables and colgroups / captions.
|
|
if (aParent->IsTableFrame()) {
|
|
auto* wrapper = aParent->GetParent();
|
|
MOZ_ASSERT(wrapper);
|
|
MOZ_ASSERT(wrapper->IsTableWrapperFrame());
|
|
MOZ_ASSERT(!aFrame->IsTableCaption(),
|
|
"Caption parent should be the wrapper");
|
|
// We can't remove the table if there are any captions present (captions are
|
|
// never anonymous themselves), because table wrapper always relies on
|
|
// having a table frame.
|
|
if (!wrapper->GetChildList(FrameChildListID::Caption).IsEmpty()) {
|
|
return false;
|
|
}
|
|
// Similarly we can't remove the table if there's still a non-anonymous col
|
|
// group (unless aFrame _is_ the non-anonymous colgroup).
|
|
if (aFrame->IsTableColGroupFrame()) {
|
|
return aParent->PrincipalChildList().IsEmpty() &&
|
|
IsOnlyNonWhitespaceFrameInList(
|
|
aParent->GetChildList(FrameChildListID::ColGroup), aFrame);
|
|
}
|
|
const auto& colGroupList =
|
|
aParent->GetChildList(FrameChildListID::ColGroup);
|
|
if (!colGroupList.IsEmpty()) {
|
|
nsIFrame* f = colGroupList.OnlyChild();
|
|
if (!f || !IsSyntheticColGroup(f)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
if (aFrame->IsTableCaption()) {
|
|
MOZ_ASSERT(aParent->IsTableWrapperFrame());
|
|
MOZ_ASSERT(aParent->PrincipalChildList().OnlyChild());
|
|
MOZ_ASSERT(aParent->PrincipalChildList().OnlyChild()->IsTableFrame());
|
|
return IsOnlyNonWhitespaceFrameInList(
|
|
aParent->GetChildList(FrameChildListID::Caption), aFrame) &&
|
|
// This checks for both colgroups and the principal list of the table
|
|
// frame.
|
|
AllChildListsAreEffectivelyEmpty(
|
|
aParent->PrincipalChildList().OnlyChild());
|
|
}
|
|
MOZ_ASSERT(!aFrame->IsTableColGroupFrame());
|
|
return IsOnlyNonWhitespaceFrameInList(aParent->PrincipalChildList(), aFrame);
|
|
}
|
|
|
|
static bool CanRemoveWrapperPseudoForChildRemoval(nsIFrame* aFrame,
|
|
nsIFrame* aParent) {
|
|
if (!IsOnlyMeaningfulChildOfWrapperPseudo(aFrame, aParent)) {
|
|
return false;
|
|
}
|
|
if (aParent->IsRubyBaseContainerFrame()) {
|
|
// We can't remove the first ruby base container of a ruby frame unless
|
|
// it has no siblings. See CreateNeededPseudoSiblings.
|
|
return aParent->GetPrevSibling() || !aParent->GetNextSibling();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::ContentRemoved(nsIContent* aChild,
|
|
nsIContent* aOldNextSibling,
|
|
RemoveFlags aFlags) {
|
|
MOZ_ASSERT(aChild);
|
|
MOZ_ASSERT(!aChild->IsRootOfNativeAnonymousSubtree() || !aOldNextSibling,
|
|
"Anonymous roots don't have siblings");
|
|
AUTO_PROFILER_LABEL_HOT("nsCSSFrameConstructor::ContentRemoved",
|
|
LAYOUT_FrameConstruction);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
nsPresContext* presContext = mPresShell->GetPresContext();
|
|
MOZ_ASSERT(presContext, "Our presShell should have a valid presContext");
|
|
|
|
// We want to detect when the viewport override element stored in the
|
|
// prescontext is in the subtree being removed. Except in fullscreen cases
|
|
// (which are handled in Element::UnbindFromTree and do not get stored on the
|
|
// prescontext), the override element is always either the root element or a
|
|
// <body> child of the root element. So we can only be removing the stored
|
|
// override element if the thing being removed is either the override element
|
|
// itself or the root element (which can be a parent of the override element).
|
|
if (aChild == presContext->GetViewportScrollStylesOverrideElement() ||
|
|
(aChild->IsElement() && !aChild->GetParent())) {
|
|
// We might be removing the element that we propagated viewport scrollbar
|
|
// styles from. Recompute those. (This clause covers two of the three
|
|
// possible scrollbar-propagation sources: the <body> [as aChild or a
|
|
// descendant] and the root node. The other possible scrollbar-propagation
|
|
// source is a fullscreen element, and we have code elsewhere to update
|
|
// scrollbars after fullscreen elements are removed -- specifically, it's
|
|
// part of the fullscreen cleanup code called by Element::UnbindFromTree.
|
|
// We don't handle the fullscreen case here, because it doesn't change the
|
|
// scrollbar styles override element stored on the prescontext.)
|
|
Element* newOverrideElement =
|
|
presContext->UpdateViewportScrollStylesOverride();
|
|
|
|
// If aChild is the root, then we don't need to do any reframing of
|
|
// newOverrideElement, because we're about to tear down the whole frame tree
|
|
// anyway. And we need to make sure we don't do any such reframing, because
|
|
// reframing the <body> can trigger a reframe of the <html> and then reenter
|
|
// here.
|
|
//
|
|
// But if aChild is not the root, and if newOverrideElement is not
|
|
// the root and isn't aChild (which it could be if all we're doing
|
|
// here is reframing the current override element), it needs
|
|
// reframing. In particular, it used to have a scrollframe
|
|
// (because its overflow was not "visible"), but now it will
|
|
// propagate its overflow to the viewport, so it should not need a
|
|
// scrollframe anymore.
|
|
if (aChild->GetParent() && newOverrideElement &&
|
|
newOverrideElement->GetParent() && newOverrideElement != aChild) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(newOverrideElement, InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf(
|
|
"nsCSSFrameConstructor::ContentRemoved container=%p child=%p "
|
|
"old-next-sibling=%p\n",
|
|
aChild->GetParent(), aChild, aOldNextSibling);
|
|
if (gReallyNoisyContentUpdates) {
|
|
aChild->GetParent()->List(stdout, 0);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
nsIFrame* childFrame = aChild->GetPrimaryFrame();
|
|
if (!childFrame || childFrame->GetContent() != aChild) {
|
|
// XXXbz the GetContent() != aChild check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
childFrame = nullptr;
|
|
}
|
|
|
|
// If we're removing the root, then make sure to remove things starting at
|
|
// the viewport's child instead of the primary frame (which might even be
|
|
// null if the root was display:none, even though the frames above it got
|
|
// created). Detecting removal of a root is a little exciting; in particular,
|
|
// having no parent is necessary but NOT sufficient.
|
|
//
|
|
// Due to how we process reframes, the content node might not even be in our
|
|
// document by now. So explicitly check whether the viewport's first kid's
|
|
// content node is aChild.
|
|
//
|
|
// FIXME(emilio): I think the "might not be in our document" bit is impossible
|
|
// now.
|
|
bool isRoot = false;
|
|
if (!aChild->GetParent()) {
|
|
if (nsIFrame* viewport = GetRootFrame()) {
|
|
nsIFrame* firstChild = viewport->PrincipalChildList().FirstChild();
|
|
if (firstChild && firstChild->GetContent() == aChild) {
|
|
isRoot = true;
|
|
childFrame = firstChild;
|
|
NS_ASSERTION(!childFrame->GetNextSibling(), "How did that happen?");
|
|
}
|
|
}
|
|
}
|
|
|
|
// We need to be conservative about when to determine whether something has
|
|
// display: contents or not because at this point our actual display may be
|
|
// different.
|
|
//
|
|
// Consider the case of:
|
|
//
|
|
// <div id="A" style="display: contents"><div id="B"></div></div>
|
|
//
|
|
// If we reconstruct A because its display changed to "none", we still need to
|
|
// cleanup the frame on B, but A's display is now "none", so we can't poke at
|
|
// the style of it.
|
|
//
|
|
// FIXME(emilio, bug 1450366): We can make this faster without adding much
|
|
// complexity for the display: none -> other case, which right now
|
|
// unnecessarily walks the content tree down.
|
|
auto CouldHaveBeenDisplayContents = [aFlags](nsIContent* aContent) -> bool {
|
|
return aFlags == REMOVE_FOR_RECONSTRUCTION || IsDisplayContents(aContent);
|
|
};
|
|
|
|
if (!childFrame && CouldHaveBeenDisplayContents(aChild)) {
|
|
// NOTE(emilio): We may iterate through ::before and ::after here and they
|
|
// may be gone after the respective ContentRemoved call. Right now
|
|
// StyleChildrenIterator handles that properly, so it's not an issue.
|
|
StyleChildrenIterator iter(aChild);
|
|
for (nsIContent* c = iter.GetNextChild(); c; c = iter.GetNextChild()) {
|
|
if (c->GetPrimaryFrame() || CouldHaveBeenDisplayContents(c)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
bool didReconstruct = ContentRemoved(c, nullptr, aFlags);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
if (didReconstruct) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (childFrame) {
|
|
if (aFlags == REMOVE_FOR_RECONSTRUCTION) {
|
|
// Before removing the frames associated with the content object,
|
|
// ask them to save their state onto our state object.
|
|
CaptureStateForFramesOf(aChild, mFrameTreeState);
|
|
}
|
|
|
|
InvalidateCanvasIfNeeded(mPresShell, aChild);
|
|
|
|
// See whether we need to remove more than just childFrame
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
if (MaybeRecreateContainerForFrameRemoval(childFrame)) {
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
|
|
// Get the childFrame's parent frame
|
|
nsIFrame* parentFrame = childFrame->GetParent();
|
|
LayoutFrameType parentType = parentFrame->Type();
|
|
|
|
if (parentType == LayoutFrameType::FrameSet &&
|
|
IsSpecialFramesetChild(aChild)) {
|
|
// Just reframe the parent, since framesets are weird like that.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(parentFrame->GetContent(), InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
|
|
// If we're a child of MathML, then we should reframe the MathML content.
|
|
// If we're non-MathML, then we would be wrapped in a block so we need to
|
|
// check our grandparent in that case.
|
|
nsIFrame* possibleMathMLAncestor = parentType == LayoutFrameType::Block
|
|
? parentFrame->GetParent()
|
|
: parentFrame;
|
|
if (possibleMathMLAncestor->IsMathMLFrame()) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(parentFrame->GetContent(), InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (aFlags != REMOVE_FOR_RECONSTRUCTION) {
|
|
if (nsAccessibilityService* accService = GetAccService()) {
|
|
accService->ContentRemoved(mPresShell, aChild);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Examine the containing-block for the removed content and see if
|
|
// :first-letter style applies.
|
|
nsIFrame* inflowChild = childFrame;
|
|
if (childFrame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
|
|
inflowChild = childFrame->GetPlaceholderFrame();
|
|
NS_ASSERTION(inflowChild, "No placeholder for out-of-flow?");
|
|
}
|
|
nsContainerFrame* containingBlock =
|
|
GetFloatContainingBlock(inflowChild->GetParent());
|
|
bool haveFLS = containingBlock && HasFirstLetterStyle(containingBlock);
|
|
if (haveFLS) {
|
|
// Trap out to special routine that handles adjusting a blocks
|
|
// frame tree when first-letter style is present.
|
|
#ifdef NOISY_FIRST_LETTER
|
|
printf("ContentRemoved: containingBlock=");
|
|
containingBlock->ListTag(stdout);
|
|
printf(" parentFrame=");
|
|
parentFrame->ListTag(stdout);
|
|
printf(" childFrame=");
|
|
childFrame->ListTag(stdout);
|
|
printf("\n");
|
|
#endif
|
|
|
|
// First update the containing blocks structure by removing the
|
|
// existing letter frames. This makes the subsequent logic
|
|
// simpler.
|
|
RemoveLetterFrames(mPresShell, containingBlock);
|
|
|
|
// Recover childFrame and parentFrame
|
|
childFrame = aChild->GetPrimaryFrame();
|
|
if (!childFrame || childFrame->GetContent() != aChild) {
|
|
// XXXbz the GetContent() != aChild check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
return false;
|
|
}
|
|
parentFrame = childFrame->GetParent();
|
|
parentType = parentFrame->Type();
|
|
|
|
#ifdef NOISY_FIRST_LETTER
|
|
printf(" ==> revised parentFrame=");
|
|
parentFrame->ListTag(stdout);
|
|
printf(" childFrame=");
|
|
childFrame->ListTag(stdout);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentRemoved: childFrame=");
|
|
childFrame->ListTag(stdout);
|
|
putchar('\n');
|
|
parentFrame->List(stdout);
|
|
}
|
|
#endif
|
|
|
|
// Notify the parent frame that it should delete the frame
|
|
if (childFrame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)) {
|
|
childFrame = childFrame->GetPlaceholderFrame();
|
|
NS_ASSERTION(childFrame, "Missing placeholder frame for out of flow.");
|
|
parentFrame = childFrame->GetParent();
|
|
}
|
|
|
|
// Take care of wrapper anonymous boxes that we might need to remove while
|
|
// at it. Note that MaybeRecreateContainerForFrameRemoval takes care of
|
|
// harder cases (merging sibling anonymous boxes etc).
|
|
while (IsWrapperPseudo(parentFrame) &&
|
|
CanRemoveWrapperPseudoForChildRemoval(childFrame, parentFrame)) {
|
|
childFrame = parentFrame;
|
|
parentFrame = childFrame->GetParent();
|
|
}
|
|
|
|
DestroyContext context(mPresShell);
|
|
RemoveFrame(context, nsLayoutUtils::GetChildListNameFor(childFrame),
|
|
childFrame);
|
|
|
|
// NOTE(emilio): aChild could be dead here already if it is a ::before or
|
|
// ::after pseudo-element (since in that case it was owned by childFrame,
|
|
// which we just destroyed).
|
|
|
|
if (isRoot) {
|
|
mRootElementFrame = nullptr;
|
|
mRootElementStyleFrame = nullptr;
|
|
mDocElementContainingBlock = nullptr;
|
|
mCanvasFrame = nullptr;
|
|
mPageSequenceFrame = nullptr;
|
|
}
|
|
|
|
if (haveFLS && mRootElementFrame) {
|
|
RecoverLetterFrames(containingBlock);
|
|
}
|
|
|
|
// If we're just reconstructing frames for the element, then the
|
|
// following ContentInserted notification on the element will
|
|
// take care of fixing up any adjacent text nodes.
|
|
if (aOldNextSibling && aFlags == REMOVE_CONTENT) {
|
|
MOZ_ASSERT(aChild->GetParentNode(),
|
|
"How did we have a sibling without a parent?");
|
|
// Adjacent whitespace-only text nodes might have been suppressed if
|
|
// this node does not have inline ends. Create frames for them now
|
|
// if necessary.
|
|
// Reframe any text node just before the node being removed, if there is
|
|
// one, and if it's not the last child or the first child. If a whitespace
|
|
// textframe was being suppressed and it's now the last child or first
|
|
// child then it can stay suppressed since the parent must be a block
|
|
// and hence it's adjacent to a block end.
|
|
// If aOldNextSibling is null, then the text node before the node being
|
|
// removed is the last node, and we don't need to worry about it.
|
|
nsIContent* prevSibling = aOldNextSibling->GetPreviousSibling();
|
|
if (prevSibling && prevSibling->GetPreviousSibling()) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
ReframeTextIfNeeded(prevSibling);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
// Reframe any text node just after the node being removed, if there is
|
|
// one, and if it's not the last child or the first child.
|
|
if (aOldNextSibling->GetNextSibling() &&
|
|
aOldNextSibling->GetPreviousSibling()) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
ReframeTextIfNeeded(aOldNextSibling);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates && parentFrame) {
|
|
printf("nsCSSFrameConstructor::ContentRemoved: resulting frame model:\n");
|
|
parentFrame->List(stdout);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* This method invalidates the canvas when frames are removed or added for a
|
|
* node that might have its background propagated to the canvas, i.e., a
|
|
* document root node or an HTML BODY which is a child of the root node.
|
|
*
|
|
* @param aFrame a frame for a content node about to be removed or a frame that
|
|
* was just created for a content node that was inserted.
|
|
*/
|
|
static void InvalidateCanvasIfNeeded(PresShell* aPresShell, nsIContent* aNode) {
|
|
MOZ_ASSERT(aPresShell->GetRootFrame(), "What happened here?");
|
|
MOZ_ASSERT(aPresShell->GetPresContext(), "Say what?");
|
|
|
|
// Note that both in ContentRemoved and ContentInserted the content node
|
|
// will still have the right parent pointer, so looking at that is ok.
|
|
|
|
nsIContent* parent = aNode->GetParent();
|
|
if (parent) {
|
|
// Has a parent; might not be what we want
|
|
nsIContent* grandParent = parent->GetParent();
|
|
if (grandParent) {
|
|
// Has a grandparent, so not what we want
|
|
return;
|
|
}
|
|
|
|
// Check whether it's an HTML body
|
|
if (!aNode->IsHTMLElement(nsGkAtoms::body)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// At this point the node has no parent or it's an HTML <body> child of the
|
|
// root. We might not need to invalidate in this case (eg we might be in
|
|
// XHTML or something), but chances are we want to. Play it safe.
|
|
// Invalidate the viewport.
|
|
|
|
nsIFrame* rootFrame = aPresShell->GetRootFrame();
|
|
rootFrame->InvalidateFrameSubtree();
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::EnsureFrameForTextNodeIsCreatedAfterFlush(
|
|
CharacterData* aContent) {
|
|
if (!aContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE)) {
|
|
return false;
|
|
}
|
|
|
|
if (mAlwaysCreateFramesForIgnorableWhitespace) {
|
|
return false;
|
|
}
|
|
|
|
// Text frame may have been suppressed. Disable suppression and signal that a
|
|
// flush should be performed. We do this on a document-wide basis so that
|
|
// pages that repeatedly query metrics for collapsed-whitespace text nodes
|
|
// don't trigger pathological behavior.
|
|
mAlwaysCreateFramesForIgnorableWhitespace = true;
|
|
Element* root = mDocument->GetRootElement();
|
|
if (!root) {
|
|
return false;
|
|
}
|
|
|
|
RestyleManager()->PostRestyleEvent(root, RestyleHint{0},
|
|
nsChangeHint_ReconstructFrame);
|
|
return true;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CharacterDataChanged(
|
|
nsIContent* aContent, const CharacterDataChangeInfo& aInfo) {
|
|
AUTO_PROFILER_LABEL_HOT("nsCSSFrameConstructor::CharacterDataChanged",
|
|
LAYOUT_FrameConstruction);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
|
|
if ((aContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE) &&
|
|
!aContent->TextIsOnlyWhitespace()) ||
|
|
(aContent->HasFlag(NS_REFRAME_IF_WHITESPACE) &&
|
|
aContent->TextIsOnlyWhitespace())) {
|
|
#ifdef DEBUG
|
|
nsIFrame* frame = aContent->GetPrimaryFrame();
|
|
NS_ASSERTION(!frame || !frame->IsGeneratedContentFrame(),
|
|
"Bit should never be set on generated content");
|
|
#endif
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(aContent, InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
// It's possible the frame whose content changed isn't inserted into the
|
|
// frame hierarchy yet, or that there is no frame that maps the content
|
|
if (nsIFrame* frame = aContent->GetPrimaryFrame()) {
|
|
#if 0
|
|
NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
|
|
("nsCSSFrameConstructor::CharacterDataChanged: content=%p[%s] subcontent=%p frame=%p",
|
|
aContent, ContentTag(aContent, 0),
|
|
aSubContent, frame));
|
|
#endif
|
|
|
|
if (frame->HasAnyStateBits(NS_FRAME_IS_IN_SINGLE_CHAR_MI)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(aContent, InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
// Special check for text content that is a child of a letter frame. If
|
|
// this happens, we should remove the letter frame, do whatever we're
|
|
// planning to do with this notification, then put the letter frame back.
|
|
// Note that this is basically what RecreateFramesForContent ends up doing;
|
|
// the reason we dont' want to call that here is that our text content
|
|
// could be native anonymous, in which case RecreateFramesForContent would
|
|
// completely barf on it. And recreating the non-anonymous ancestor would
|
|
// just lead us to come back into this notification (e.g. if quotes or
|
|
// counters are involved), leading to a loop.
|
|
nsContainerFrame* block = GetFloatContainingBlock(frame);
|
|
bool haveFirstLetterStyle = false;
|
|
if (block) {
|
|
// See if the block has first-letter style applied to it.
|
|
haveFirstLetterStyle = HasFirstLetterStyle(block);
|
|
if (haveFirstLetterStyle) {
|
|
RemoveLetterFrames(mPresShell, block);
|
|
// Reget |frame|, since we might have killed it.
|
|
// Do we really need to call CharacterDataChanged in this case, though?
|
|
frame = aContent->GetPrimaryFrame();
|
|
NS_ASSERTION(frame, "Should have frame here!");
|
|
}
|
|
}
|
|
|
|
// Notify the first frame that maps the content. It will generate a reflow
|
|
// command
|
|
frame->CharacterDataChanged(aInfo);
|
|
|
|
if (haveFirstLetterStyle) {
|
|
RecoverLetterFrames(block);
|
|
}
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::RecalcQuotesAndCounters() {
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
|
|
if (mQuotesDirty) {
|
|
mQuotesDirty = false;
|
|
mContainStyleScopeManager.RecalcAllQuotes();
|
|
}
|
|
|
|
if (mCountersDirty) {
|
|
mCountersDirty = false;
|
|
mContainStyleScopeManager.RecalcAllCounters();
|
|
}
|
|
|
|
NS_ASSERTION(!mQuotesDirty, "Quotes updates will be lost");
|
|
NS_ASSERTION(!mCountersDirty, "Counter updates will be lost");
|
|
}
|
|
|
|
void nsCSSFrameConstructor::NotifyCounterStylesAreDirty() {
|
|
mContainStyleScopeManager.SetAllCountersDirty();
|
|
CountersDirty();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::WillDestroyFrameTree() {
|
|
#if defined(DEBUG_dbaron_off)
|
|
mContainStyleScopeManager.DumpCounters();
|
|
#endif
|
|
|
|
// Prevent frame tree destruction from being O(N^2)
|
|
mContainStyleScopeManager.Clear();
|
|
nsFrameManager::Destroy();
|
|
}
|
|
|
|
// STATIC
|
|
|
|
// XXXbz I'd really like this method to go away. Once we have inline-block and
|
|
// I can just use that for sized broken images, that can happen, maybe.
|
|
//
|
|
// NOTE(emilio): This needs to match MozAltContent handling.
|
|
void nsCSSFrameConstructor::GetAlternateTextFor(const Element& aElement,
|
|
nsAString& aAltText) {
|
|
// The "alt" attribute specifies alternate text that is rendered
|
|
// when the image can not be displayed.
|
|
if (aElement.GetAttr(nsGkAtoms::alt, aAltText)) {
|
|
return;
|
|
}
|
|
|
|
if (aElement.IsHTMLElement(nsGkAtoms::input)) {
|
|
// If there's no "alt" attribute, and aElement is an input element, then use
|
|
// the value of the "value" attribute.
|
|
if (aElement.GetAttr(nsGkAtoms::value, aAltText)) {
|
|
return;
|
|
}
|
|
|
|
// If there's no "value" attribute either, then use the localized string for
|
|
// "Submit" as the alternate text.
|
|
nsContentUtils::GetMaybeLocalizedString(nsContentUtils::eFORMS_PROPERTIES,
|
|
"Submit", aElement.OwnerDoc(),
|
|
aAltText);
|
|
}
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::CreateContinuingOuterTableFrame(
|
|
nsIFrame* aFrame, nsContainerFrame* aParentFrame, nsIContent* aContent,
|
|
ComputedStyle* aComputedStyle) {
|
|
nsTableWrapperFrame* newFrame =
|
|
NS_NewTableWrapperFrame(mPresShell, aComputedStyle);
|
|
|
|
newFrame->Init(aContent, aParentFrame, aFrame);
|
|
|
|
// Create a continuing inner table frame, and if there's a caption then
|
|
// replicate the caption
|
|
nsFrameList newChildFrames;
|
|
|
|
nsIFrame* childFrame = aFrame->PrincipalChildList().FirstChild();
|
|
if (childFrame) {
|
|
nsIFrame* continuingTableFrame =
|
|
CreateContinuingFrame(childFrame, newFrame);
|
|
newChildFrames.AppendFrame(nullptr, continuingTableFrame);
|
|
|
|
NS_ASSERTION(!childFrame->GetNextSibling(),
|
|
"there can be only one inner table frame");
|
|
}
|
|
|
|
// Set the table wrapper's initial child list
|
|
newFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(newChildFrames));
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::CreateContinuingTableFrame(
|
|
nsIFrame* aFrame, nsContainerFrame* aParentFrame, nsIContent* aContent,
|
|
ComputedStyle* aComputedStyle) {
|
|
nsTableFrame* newFrame = NS_NewTableFrame(mPresShell, aComputedStyle);
|
|
|
|
newFrame->Init(aContent, aParentFrame, aFrame);
|
|
|
|
// Replicate any header/footer frames
|
|
nsFrameList childFrames;
|
|
for (nsIFrame* childFrame : aFrame->PrincipalChildList()) {
|
|
// See if it's a header/footer, possibly wrapped in a scroll frame.
|
|
nsTableRowGroupFrame* rowGroupFrame =
|
|
static_cast<nsTableRowGroupFrame*>(childFrame);
|
|
// If the row group was continued, then don't replicate it.
|
|
nsIFrame* rgNextInFlow = rowGroupFrame->GetNextInFlow();
|
|
if (rgNextInFlow) {
|
|
rowGroupFrame->SetRepeatable(false);
|
|
} else if (rowGroupFrame->IsRepeatable()) {
|
|
// Replicate the header/footer frame.
|
|
nsTableRowGroupFrame* headerFooterFrame;
|
|
nsFrameList childList;
|
|
|
|
nsFrameConstructorState state(
|
|
mPresShell, GetAbsoluteContainingBlock(newFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(newFrame, ABS_POS), nullptr);
|
|
state.mCreatingExtraFrames = true;
|
|
|
|
ComputedStyle* const headerFooterComputedStyle = rowGroupFrame->Style();
|
|
headerFooterFrame = static_cast<nsTableRowGroupFrame*>(
|
|
NS_NewTableRowGroupFrame(mPresShell, headerFooterComputedStyle));
|
|
|
|
nsIContent* headerFooter = rowGroupFrame->GetContent();
|
|
headerFooterFrame->Init(headerFooter, newFrame, nullptr);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlock(state, absoluteSaveState,
|
|
headerFooterFrame);
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
state.MaybePushFloatContainingBlock(headerFooterFrame, floatSaveState);
|
|
|
|
ProcessChildren(state, headerFooter, rowGroupFrame->Style(),
|
|
headerFooterFrame, true, childList, false, nullptr);
|
|
NS_ASSERTION(state.mFloatedList.IsEmpty(), "unexpected floated element");
|
|
headerFooterFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
headerFooterFrame->SetRepeatable(true);
|
|
|
|
// Table specific initialization
|
|
headerFooterFrame->InitRepeatedFrame(rowGroupFrame);
|
|
|
|
// XXX Deal with absolute and fixed frames...
|
|
childFrames.AppendFrame(nullptr, headerFooterFrame);
|
|
}
|
|
}
|
|
|
|
// Set the table frame's initial child list
|
|
newFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childFrames));
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::CreateContinuingFrame(
|
|
nsIFrame* aFrame, nsContainerFrame* aParentFrame, bool aIsFluid) {
|
|
ComputedStyle* computedStyle = aFrame->Style();
|
|
nsIFrame* newFrame = nullptr;
|
|
nsIFrame* nextContinuation = aFrame->GetNextContinuation();
|
|
nsIFrame* nextInFlow = aFrame->GetNextInFlow();
|
|
|
|
// Use the frame type to determine what type of frame to create
|
|
LayoutFrameType frameType = aFrame->Type();
|
|
nsIContent* content = aFrame->GetContent();
|
|
|
|
if (LayoutFrameType::Text == frameType) {
|
|
newFrame = NS_NewContinuingTextFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Inline == frameType) {
|
|
newFrame = NS_NewInlineFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Block == frameType) {
|
|
MOZ_ASSERT(!aFrame->IsTableCaption(),
|
|
"no support for fragmenting table captions yet");
|
|
newFrame = NS_NewBlockFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::ColumnSetWrapper == frameType) {
|
|
newFrame =
|
|
NS_NewColumnSetWrapperFrame(mPresShell, computedStyle, nsFrameState(0));
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::ColumnSet == frameType) {
|
|
MOZ_ASSERT(!aFrame->IsTableCaption(),
|
|
"no support for fragmenting table captions yet");
|
|
newFrame = NS_NewColumnSetFrame(mPresShell, computedStyle, nsFrameState(0));
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::PrintedSheet == frameType) {
|
|
newFrame = ConstructPrintedSheetFrame(mPresShell, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Page == frameType) {
|
|
nsCanvasFrame* canvasFrame; // (unused outparam for ConstructPageFrame)
|
|
newFrame =
|
|
ConstructPageFrame(mPresShell, aParentFrame, aFrame, canvasFrame);
|
|
} else if (LayoutFrameType::TableWrapper == frameType) {
|
|
newFrame = CreateContinuingOuterTableFrame(aFrame, aParentFrame, content,
|
|
computedStyle);
|
|
} else if (LayoutFrameType::Table == frameType) {
|
|
newFrame = CreateContinuingTableFrame(aFrame, aParentFrame, content,
|
|
computedStyle);
|
|
} else if (LayoutFrameType::TableRowGroup == frameType) {
|
|
newFrame = NS_NewTableRowGroupFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::TableRow == frameType) {
|
|
nsTableRowFrame* rowFrame = NS_NewTableRowFrame(mPresShell, computedStyle);
|
|
|
|
rowFrame->Init(content, aParentFrame, aFrame);
|
|
|
|
// Create a continuing frame for each table cell frame
|
|
nsFrameList newChildList;
|
|
nsIFrame* cellFrame = aFrame->PrincipalChildList().FirstChild();
|
|
while (cellFrame) {
|
|
// See if it's a table cell frame
|
|
if (cellFrame->IsTableCellFrame()) {
|
|
nsIFrame* continuingCellFrame =
|
|
CreateContinuingFrame(cellFrame, rowFrame);
|
|
newChildList.AppendFrame(nullptr, continuingCellFrame);
|
|
}
|
|
cellFrame = cellFrame->GetNextSibling();
|
|
}
|
|
|
|
rowFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(newChildList));
|
|
newFrame = rowFrame;
|
|
|
|
} else if (LayoutFrameType::TableCell == frameType) {
|
|
// Warning: If you change this and add a wrapper frame around table cell
|
|
// frames, make sure Bug 368554 doesn't regress!
|
|
// See IsInAutoWidthTableCellForQuirk() in nsImageFrame.cpp.
|
|
nsTableFrame* tableFrame =
|
|
static_cast<nsTableRowFrame*>(aParentFrame)->GetTableFrame();
|
|
nsTableCellFrame* cellFrame =
|
|
NS_NewTableCellFrame(mPresShell, computedStyle, tableFrame);
|
|
|
|
cellFrame->Init(content, aParentFrame, aFrame);
|
|
|
|
// Create a continuing area frame
|
|
nsIFrame* blockFrame = aFrame->PrincipalChildList().FirstChild();
|
|
nsIFrame* continuingBlockFrame =
|
|
CreateContinuingFrame(blockFrame, cellFrame);
|
|
|
|
SetInitialSingleChild(cellFrame, continuingBlockFrame);
|
|
newFrame = cellFrame;
|
|
} else if (LayoutFrameType::Line == frameType) {
|
|
newFrame = NS_NewFirstLineFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Letter == frameType) {
|
|
newFrame = NS_NewFirstLetterFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Image == frameType) {
|
|
auto* imageFrame = static_cast<nsImageFrame*>(aFrame);
|
|
newFrame = imageFrame->CreateContinuingFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::ImageControl == frameType) {
|
|
newFrame = NS_NewImageControlFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::FieldSet == frameType) {
|
|
newFrame = NS_NewFieldSetFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::FlexContainer == frameType) {
|
|
newFrame = NS_NewFlexContainerFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::GridContainer == frameType) {
|
|
newFrame = NS_NewGridContainerFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Ruby == frameType) {
|
|
newFrame = NS_NewRubyFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::RubyBaseContainer == frameType) {
|
|
newFrame = NS_NewRubyBaseContainerFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::RubyTextContainer == frameType) {
|
|
newFrame = NS_NewRubyTextContainerFrame(mPresShell, computedStyle);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else {
|
|
MOZ_CRASH("unexpected frame type");
|
|
}
|
|
|
|
// Init() set newFrame to be a fluid continuation of aFrame.
|
|
// If we want a non-fluid continuation, we need to call SetPrevContinuation()
|
|
// to reset NS_FRAME_IS_FLUID_CONTINUATION.
|
|
if (!aIsFluid) {
|
|
newFrame->SetPrevContinuation(aFrame);
|
|
}
|
|
|
|
// If a continuing frame needs to carry frame state bits from its previous
|
|
// continuation or parent, set them in nsIFrame::Init(), or in any derived
|
|
// frame class's Init() if the bits are belong to specific group.
|
|
|
|
if (nextInFlow) {
|
|
nextInFlow->SetPrevInFlow(newFrame);
|
|
newFrame->SetNextInFlow(nextInFlow);
|
|
} else if (nextContinuation) {
|
|
nextContinuation->SetPrevContinuation(newFrame);
|
|
newFrame->SetNextContinuation(nextContinuation);
|
|
}
|
|
|
|
// aFrame cannot be a dynamic reflow root because it has a continuation now.
|
|
aFrame->RemoveStateBits(NS_FRAME_DYNAMIC_REFLOW_ROOT);
|
|
|
|
MOZ_ASSERT(!newFrame->GetNextSibling(), "unexpected sibling");
|
|
return newFrame;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::MaybeSetNextPageContentFramePageName(
|
|
const nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame, "Frame should not be null");
|
|
// No parent means the root frame, which isn't what this funciton is for.
|
|
MOZ_ASSERT(aFrame->GetParent(),
|
|
"Frame should be the first child placed on a new page, not the "
|
|
"root frame.");
|
|
if (mNextPageContentFramePageName) {
|
|
return;
|
|
}
|
|
const nsAtom* const autoValue = aFrame->GetParent()->GetAutoPageValue();
|
|
mNextPageContentFramePageName = aFrame->ComputePageValue(autoValue);
|
|
}
|
|
|
|
nsresult nsCSSFrameConstructor::ReplicateFixedFrames(
|
|
nsPageContentFrame* aParentFrame) {
|
|
// Now deal with fixed-pos things.... They should appear on all pages,
|
|
// so we want to move over the placeholders when processing the child
|
|
// of the pageContentFrame.
|
|
|
|
nsIFrame* prevPageContentFrame = aParentFrame->GetPrevInFlow();
|
|
if (!prevPageContentFrame) {
|
|
return NS_OK;
|
|
}
|
|
nsContainerFrame* canvasFrame =
|
|
do_QueryFrame(aParentFrame->PrincipalChildList().FirstChild());
|
|
nsIFrame* prevCanvasFrame =
|
|
prevPageContentFrame->PrincipalChildList().FirstChild();
|
|
if (!canvasFrame || !prevCanvasFrame) {
|
|
// document's root element frame missing
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsFrameList fixedPlaceholders;
|
|
nsIFrame* firstFixed =
|
|
prevPageContentFrame->GetChildList(FrameChildListID::Fixed).FirstChild();
|
|
if (!firstFixed) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Don't allow abs-pos descendants of the fixed content to escape the content.
|
|
// This should not normally be possible (because fixed-pos elements should
|
|
// be absolute containers) but fixed-pos tables currently aren't abs-pos
|
|
// containers.
|
|
nsFrameConstructorState state(mPresShell, aParentFrame, nullptr,
|
|
mRootElementFrame);
|
|
state.mCreatingExtraFrames = true;
|
|
|
|
// We can't use an ancestor filter here, because we're not going to
|
|
// be usefully recurring down the tree. This means that other
|
|
// places in frame construction can't assume a filter is
|
|
// initialized!
|
|
|
|
// Iterate across fixed frames and replicate each whose placeholder is a
|
|
// descendant of aFrame. (We don't want to explicitly copy placeholders that
|
|
// are within fixed frames, because that would cause duplicates on the new
|
|
// page - bug 389619)
|
|
for (nsIFrame* fixed = firstFixed; fixed; fixed = fixed->GetNextSibling()) {
|
|
nsIFrame* prevPlaceholder = fixed->GetPlaceholderFrame();
|
|
if (prevPlaceholder && nsLayoutUtils::IsProperAncestorFrame(
|
|
prevCanvasFrame, prevPlaceholder)) {
|
|
// We want to use the same style as the primary style frame for
|
|
// our content
|
|
nsIContent* content = fixed->GetContent();
|
|
ComputedStyle* computedStyle =
|
|
nsLayoutUtils::GetStyleFrame(content)->Style();
|
|
AutoFrameConstructionItemList items(this);
|
|
AddFrameConstructionItemsInternal(state, content, canvasFrame, true,
|
|
computedStyle,
|
|
{ItemFlag::AllowPageBreak}, items);
|
|
ConstructFramesFromItemList(state, items, canvasFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
fixedPlaceholders);
|
|
}
|
|
}
|
|
|
|
// Add the placeholders to our primary child list.
|
|
// XXXbz this is a little screwed up, since the fixed frames will have
|
|
// broken auto-positioning. Oh, well.
|
|
NS_ASSERTION(!canvasFrame->PrincipalChildList().FirstChild(),
|
|
"leaking frames; doc root continuation must be empty");
|
|
canvasFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(fixedPlaceholders));
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCSSFrameConstructor::InsertionPoint nsCSSFrameConstructor::GetInsertionPoint(
|
|
nsIContent* aChild) {
|
|
MOZ_ASSERT(aChild);
|
|
nsIContent* insertionElement = aChild->GetFlattenedTreeParent();
|
|
if (!insertionElement) {
|
|
// The element doesn't belong in the flattened tree, and thus we don't want
|
|
// to render it.
|
|
return {};
|
|
}
|
|
|
|
return {GetContentInsertionFrameFor(insertionElement), insertionElement};
|
|
}
|
|
|
|
// Capture state for the frame tree rooted at the frame associated with the
|
|
// content object, aContent
|
|
void nsCSSFrameConstructor::CaptureStateForFramesOf(
|
|
nsIContent* aContent, nsILayoutHistoryState* aHistoryState) {
|
|
if (!aHistoryState) {
|
|
return;
|
|
}
|
|
nsIFrame* frame = aContent->GetPrimaryFrame();
|
|
if (frame == mRootElementFrame) {
|
|
frame = mRootElementFrame
|
|
? GetAbsoluteContainingBlock(mRootElementFrame, FIXED_POS)
|
|
: GetRootFrame();
|
|
}
|
|
for (; frame;
|
|
frame = nsLayoutUtils::GetNextContinuationOrIBSplitSibling(frame)) {
|
|
CaptureFrameState(frame, aHistoryState);
|
|
}
|
|
}
|
|
|
|
static bool IsWhitespaceFrame(nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aFrame, "invalid argument");
|
|
return aFrame->IsTextFrame() && aFrame->GetContent()->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
static nsIFrame* FindNextNonWhitespaceSibling(nsIFrame* aFrame) {
|
|
nsIFrame* f = aFrame;
|
|
do {
|
|
f = f->GetNextSibling();
|
|
} while (f && IsWhitespaceFrame(f));
|
|
return f;
|
|
}
|
|
|
|
static nsIFrame* FindPreviousNonWhitespaceSibling(nsIFrame* aFrame) {
|
|
nsIFrame* f = aFrame;
|
|
do {
|
|
f = f->GetPrevSibling();
|
|
} while (f && IsWhitespaceFrame(f));
|
|
return f;
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval(
|
|
nsIFrame* aFrame) {
|
|
#define TRACE(reason) \
|
|
PROFILER_MARKER("MaybeRecreateContainerForFrameRemoval: " reason, LAYOUT, \
|
|
{}, Tracing, "Layout")
|
|
MOZ_ASSERT(aFrame, "Must have a frame");
|
|
MOZ_ASSERT(aFrame->GetParent(), "Frame shouldn't be root");
|
|
MOZ_ASSERT(aFrame == aFrame->FirstContinuation(),
|
|
"aFrame not the result of GetPrimaryFrame()?");
|
|
|
|
nsIFrame* inFlowFrame = aFrame->HasAnyStateBits(NS_FRAME_OUT_OF_FLOW)
|
|
? aFrame->GetPlaceholderFrame()
|
|
: aFrame;
|
|
MOZ_ASSERT(inFlowFrame, "How did that happen?");
|
|
MOZ_ASSERT(inFlowFrame == inFlowFrame->FirstContinuation(),
|
|
"placeholder for primary frame has previous continuations?");
|
|
nsIFrame* parent = inFlowFrame->GetParent();
|
|
|
|
if (inFlowFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
nsIFrame* grandparent = parent->GetParent();
|
|
MOZ_ASSERT(grandparent);
|
|
|
|
bool needsReframe =
|
|
// 1. Removing a column-span may lead to an empty
|
|
// ::-moz-column-span-wrapper.
|
|
inFlowFrame->IsColumnSpan() ||
|
|
// 2. Removing a frame which has any column-span siblings may also
|
|
// lead to an empty ::-moz-column-span-wrapper subtree. The
|
|
// column-span siblings were the frame's children, but later become
|
|
// the frame's siblings after CreateColumnSpanSiblings().
|
|
inFlowFrame->HasColumnSpanSiblings() ||
|
|
// 3. Removing the only child of a ::-moz-column-content, whose
|
|
// ColumnSet grandparent has a previous column-span sibling, requires
|
|
// reframing since we might connect the ColumnSet's next column-span
|
|
// sibling (if there's one). Note that this isn't actually needed if
|
|
// the ColumnSet is at the end of ColumnSetWrapper since we create
|
|
// empty ones at the end anyway, but we're not worried about
|
|
// optimizing that case.
|
|
(parent->Style()->GetPseudoType() == PseudoStyleType::columnContent &&
|
|
// The only child in ::-moz-column-content (might be tall enough to
|
|
// split across columns)
|
|
!inFlowFrame->GetPrevSibling() && !inFlowFrame->GetNextSibling() &&
|
|
// That ::-moz-column-content is the first column.
|
|
!parent->GetPrevInFlow() &&
|
|
// The ColumnSet grandparent has a previous sibling that is a
|
|
// column-span.
|
|
grandparent->GetPrevSibling());
|
|
|
|
if (needsReframe) {
|
|
nsContainerFrame* containingBlock =
|
|
GetMultiColumnContainingBlockFor(inFlowFrame);
|
|
|
|
#ifdef DEBUG
|
|
if (IsFramePartOfIBSplit(inFlowFrame)) {
|
|
nsIFrame* ibContainingBlock = GetIBContainingBlockFor(inFlowFrame);
|
|
MOZ_ASSERT(containingBlock == ibContainingBlock ||
|
|
nsLayoutUtils::IsProperAncestorFrame(containingBlock,
|
|
ibContainingBlock),
|
|
"Multi-column containing block should be equal to or be the "
|
|
"ancestor of the IB containing block!");
|
|
}
|
|
#endif
|
|
|
|
TRACE("Multi-column");
|
|
RecreateFramesForContent(containingBlock->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (IsFramePartOfIBSplit(aFrame)) {
|
|
// The removal functions can't handle removal of an {ib} split directly; we
|
|
// need to rebuild the containing block.
|
|
TRACE("IB split removal");
|
|
ReframeContainingBlock(aFrame);
|
|
return true;
|
|
}
|
|
|
|
if (inFlowFrame->IsRenderedLegend()) {
|
|
TRACE("Fieldset / Legend");
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Might need to reconstruct things if this frame's nextSibling is a table
|
|
// or ruby pseudo, since removal of this frame might mean that this pseudo
|
|
// needs to get merged with the frame's prevSibling if that's also a table
|
|
// or ruby pseudo.
|
|
nsIFrame* nextSibling =
|
|
FindNextNonWhitespaceSibling(inFlowFrame->LastContinuation());
|
|
NS_ASSERTION(!IsWrapperPseudo(inFlowFrame),
|
|
"Shouldn't happen here (we start removals from primary frames)");
|
|
// Effectively, for the ruby pseudo sibling case, only pseudo <ruby> frame
|
|
// need to be checked here, since all other types of such frames will have
|
|
// a ruby container parent, and be catched by "Check ruby containers" below.
|
|
if (nextSibling && IsWrapperPseudo(nextSibling)) {
|
|
nsIFrame* prevSibling = FindPreviousNonWhitespaceSibling(inFlowFrame);
|
|
if (prevSibling && IsWrapperPseudo(prevSibling)) {
|
|
TRACE("Pseudo sibling");
|
|
// Good enough to recreate frames for aFrame's parent's content; even if
|
|
// aFrame's parent is a pseudo, that'll be the right content node.
|
|
// FIXME(emilio): Consider doing a more subtle check here like, only if
|
|
// prevSibling and nextSibling share frame type. Or even consider just
|
|
// moving the frames around and destroying nextSibling?
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Check ruby containers
|
|
LayoutFrameType parentType = parent->Type();
|
|
if (parentType == LayoutFrameType::Ruby ||
|
|
RubyUtils::IsRubyContainerBox(parentType)) {
|
|
// In ruby containers, pseudo frames may be created from
|
|
// whitespaces or even nothing. There are two cases we actually
|
|
// need to handle here, but hard to check exactly:
|
|
// 1. Status of spaces beside the frame may vary, and related
|
|
// frames may be constructed or destroyed accordingly.
|
|
// 2. The type of the first child of a ruby frame determines
|
|
// whether a pseudo ruby base container should exist.
|
|
TRACE("Ruby container");
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Reconstruct if inflowFrame is parent's only child, and parent is, or has,
|
|
// a non-fluid continuation, i.e. it was split by bidi resolution
|
|
if (!inFlowFrame->GetPrevSibling() && !inFlowFrame->GetNextSibling() &&
|
|
((parent->GetPrevContinuation() && !parent->GetPrevInFlow()) ||
|
|
(parent->GetNextContinuation() && !parent->GetNextInFlow()))) {
|
|
TRACE("Removing last child of non-fluid split parent");
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// We might still need to reconstruct things if the parent of inFlowFrame is
|
|
// ib-split, since in that case the removal of aFrame might affect the
|
|
// splitting of its parent.
|
|
if (!IsFramePartOfIBSplit(parent)) {
|
|
return false;
|
|
}
|
|
|
|
// If inFlowFrame is not the only in-flow child of |parent|, then removing
|
|
// it will change nothing about the {ib} split.
|
|
if (inFlowFrame != parent->PrincipalChildList().FirstChild() ||
|
|
inFlowFrame->LastContinuation()->GetNextSibling()) {
|
|
return false;
|
|
}
|
|
|
|
// If the parent is the first or last part of the {ib} split, then
|
|
// removing one of its kids will have no effect on the splitting.
|
|
// Get the first continuation up front so we don't have to do it twice.
|
|
nsIFrame* parentFirstContinuation = parent->FirstContinuation();
|
|
if (!GetIBSplitSibling(parentFirstContinuation) ||
|
|
!GetIBSplitPrevSibling(parentFirstContinuation)) {
|
|
return false;
|
|
}
|
|
|
|
TRACE("IB split parent");
|
|
ReframeContainingBlock(parent);
|
|
return true;
|
|
#undef TRACE
|
|
}
|
|
|
|
void nsCSSFrameConstructor::UpdateTableCellSpans(nsIContent* aContent) {
|
|
nsTableCellFrame* cellFrame = do_QueryFrame(aContent->GetPrimaryFrame());
|
|
|
|
// It's possible that this warning could fire if some other style change
|
|
// simultaneously changes the 'display' of the element and makes it no
|
|
// longer be a table cell.
|
|
NS_WARNING_ASSERTION(cellFrame, "Hint should only be posted on table cells!");
|
|
|
|
if (cellFrame) {
|
|
cellFrame->GetTableFrame()->RowOrColSpanChanged(cellFrame);
|
|
}
|
|
}
|
|
|
|
static nsIContent* GetTopmostMathMLElement(nsIContent* aMathMLContent) {
|
|
MOZ_ASSERT(aMathMLContent->IsMathMLElement());
|
|
MOZ_ASSERT(aMathMLContent->GetPrimaryFrame());
|
|
MOZ_ASSERT(aMathMLContent->GetPrimaryFrame()->IsMathMLFrame());
|
|
nsIContent* root = aMathMLContent;
|
|
|
|
for (nsIContent* parent = aMathMLContent->GetFlattenedTreeParent(); parent;
|
|
parent = parent->GetFlattenedTreeParent()) {
|
|
nsIFrame* frame = parent->GetPrimaryFrame();
|
|
if (!frame || !frame->IsMathMLFrame()) {
|
|
break;
|
|
}
|
|
root = parent;
|
|
}
|
|
|
|
return root;
|
|
}
|
|
|
|
// We don't know how to re-insert an anonymous subtree root, so recreate the
|
|
// closest non-generated ancestor instead, except for a few special cases...
|
|
static bool ShouldRecreateContainerForNativeAnonymousContentRoot(
|
|
nsIContent* aContent) {
|
|
if (!aContent->IsRootOfNativeAnonymousSubtree()) {
|
|
return false;
|
|
}
|
|
if (ManualNACPtr::IsManualNAC(aContent)) {
|
|
// Editor NAC, would enter an infinite loop, and we sorta get away with it
|
|
// because it's all abspos.
|
|
return false;
|
|
}
|
|
if (auto* el = Element::FromNode(aContent)) {
|
|
if (auto* classes = el->GetClasses()) {
|
|
if (classes->Contains(nsGkAtoms::mozCustomContentContainer,
|
|
eCaseMatters)) {
|
|
// Canvas anonymous content (like the custom content container) is also
|
|
// fine, because its only sibling is a tooltip which is also abspos, so
|
|
// relative insertion order doesn't really matter.
|
|
//
|
|
// This is important because the inspector uses it, and we don't want
|
|
// inspecting the page to change behavior heavily (and reframing
|
|
// unfortunately has side-effects sometimes, even though they're bugs).
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::RecreateFramesForContent(
|
|
nsIContent* aContent, InsertionKind aInsertionKind) {
|
|
MOZ_ASSERT(aContent);
|
|
|
|
// If there is no document, we don't want to recreate frames for it. (You
|
|
// shouldn't generally be giving this method content without a document
|
|
// anyway).
|
|
// Rebuilding the frame tree can have bad effects, especially if it's the
|
|
// frame tree for chrome (see bug 157322).
|
|
if (NS_WARN_IF(!aContent->GetComposedDoc())) {
|
|
return;
|
|
}
|
|
|
|
// TODO(emilio): We technically can find the right insertion point nowadays
|
|
// using StyleChildrenIterator rather than FlattenedChildIterator. But we'd
|
|
// need to tweak the setup to insert into replaced elements to filter which
|
|
// anonymous roots can be allowed, and which can't.
|
|
//
|
|
// TODO(emilio, 2022): Is this true? If we have a replaced element we wouldn't
|
|
// have generated e.g., a ::before/::after pseudo-element to begin with (which
|
|
// is what this code is about, so maybe we can just remove this piece of code
|
|
// altogether).
|
|
if (ShouldRecreateContainerForNativeAnonymousContentRoot(aContent)) {
|
|
do {
|
|
aContent = aContent->GetParent();
|
|
} while (ShouldRecreateContainerForNativeAnonymousContentRoot(aContent));
|
|
return RecreateFramesForContent(aContent, InsertionKind::Async);
|
|
}
|
|
|
|
nsIFrame* frame = aContent->GetPrimaryFrame();
|
|
if (frame && frame->IsMathMLFrame()) {
|
|
// Reframe the topmost MathML element to prevent exponential blowup
|
|
// (see bug 397518).
|
|
aContent = GetTopmostMathMLElement(aContent);
|
|
frame = aContent->GetPrimaryFrame();
|
|
}
|
|
|
|
if (frame) {
|
|
nsIFrame* parent = frame->GetParent();
|
|
nsIContent* parentContent = parent ? parent->GetContent() : nullptr;
|
|
// If the parent frame is a leaf then the subsequent insert will fail to
|
|
// create a frame, so we need to recreate the parent content. This happens
|
|
// with native anonymous content from the editor.
|
|
if (parent && parent->IsLeaf() && parentContent &&
|
|
parentContent != aContent) {
|
|
return RecreateFramesForContent(parentContent, InsertionKind::Async);
|
|
}
|
|
}
|
|
|
|
if (frame && MaybeRecreateContainerForFrameRemoval(frame)) {
|
|
return;
|
|
}
|
|
|
|
MOZ_ASSERT(aContent->GetParentNode());
|
|
|
|
// Remove the frames associated with the content object.
|
|
nsIContent* nextSibling = aContent->IsRootOfNativeAnonymousSubtree()
|
|
? nullptr
|
|
: aContent->GetNextSibling();
|
|
bool didReconstruct =
|
|
ContentRemoved(aContent, nextSibling, REMOVE_FOR_RECONSTRUCTION);
|
|
|
|
if (!didReconstruct) {
|
|
if (aInsertionKind == InsertionKind::Async && aContent->IsElement()) {
|
|
// FIXME(emilio, bug 1397239): There's nothing removing the frame state
|
|
// for elements that go away before we come back to the frame
|
|
// constructor.
|
|
//
|
|
// Also, it'd be nice to just use the `ContentRangeInserted` path for
|
|
// both elements and non-elements, but we need to make lazy frame
|
|
// construction to apply to all elements first.
|
|
RestyleManager()->PostRestyleEvent(aContent->AsElement(), RestyleHint{0},
|
|
nsChangeHint_ReconstructFrame);
|
|
} else {
|
|
// Now, recreate the frames associated with this content object. If
|
|
// ContentRemoved triggered reconstruction, then we don't need to do this
|
|
// because the frames will already have been built.
|
|
ContentRangeInserted(aContent, aContent->GetNextSibling(),
|
|
aInsertionKind);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::DestroyFramesFor(nsIContent* aContent) {
|
|
MOZ_ASSERT(aContent && aContent->GetParentNode());
|
|
|
|
nsIContent* nextSibling = aContent->IsRootOfNativeAnonymousSubtree()
|
|
? nullptr
|
|
: aContent->GetNextSibling();
|
|
|
|
return ContentRemoved(aContent, nextSibling, REMOVE_FOR_RECONSTRUCTION);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// Block frame construction code
|
|
|
|
already_AddRefed<ComputedStyle> nsCSSFrameConstructor::GetFirstLetterStyle(
|
|
nsIContent* aContent, ComputedStyle* aComputedStyle) {
|
|
if (aContent) {
|
|
return mPresShell->StyleSet()->ResolvePseudoElementStyle(
|
|
*aContent->AsElement(), PseudoStyleType::firstLetter, nullptr,
|
|
aComputedStyle);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
already_AddRefed<ComputedStyle> nsCSSFrameConstructor::GetFirstLineStyle(
|
|
nsIContent* aContent, ComputedStyle* aComputedStyle) {
|
|
if (aContent) {
|
|
return mPresShell->StyleSet()->ResolvePseudoElementStyle(
|
|
*aContent->AsElement(), PseudoStyleType::firstLine, nullptr,
|
|
aComputedStyle);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// Predicate to see if a given content (block element) has
|
|
// first-letter style applied to it.
|
|
bool nsCSSFrameConstructor::ShouldHaveFirstLetterStyle(
|
|
nsIContent* aContent, ComputedStyle* aComputedStyle) {
|
|
return nsLayoutUtils::HasPseudoStyle(aContent, aComputedStyle,
|
|
PseudoStyleType::firstLetter,
|
|
mPresShell->GetPresContext());
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::HasFirstLetterStyle(nsIFrame* aBlockFrame) {
|
|
MOZ_ASSERT(aBlockFrame, "Need a frame");
|
|
NS_ASSERTION(aBlockFrame->IsBlockFrameOrSubclass(), "Not a block frame?");
|
|
|
|
return aBlockFrame->HasAnyStateBits(NS_BLOCK_HAS_FIRST_LETTER_STYLE);
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::ShouldHaveFirstLineStyle(
|
|
nsIContent* aContent, ComputedStyle* aComputedStyle) {
|
|
bool hasFirstLine = nsLayoutUtils::HasPseudoStyle(
|
|
aContent, aComputedStyle, PseudoStyleType::firstLine,
|
|
mPresShell->GetPresContext());
|
|
return hasFirstLine && !aContent->IsHTMLElement(nsGkAtoms::fieldset);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ShouldHaveSpecialBlockStyle(
|
|
nsIContent* aContent, ComputedStyle* aComputedStyle,
|
|
bool* aHaveFirstLetterStyle, bool* aHaveFirstLineStyle) {
|
|
*aHaveFirstLetterStyle = ShouldHaveFirstLetterStyle(aContent, aComputedStyle);
|
|
*aHaveFirstLineStyle = ShouldHaveFirstLineStyle(aContent, aComputedStyle);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::PseudoParentData
|
|
nsCSSFrameConstructor::sPseudoParentData[eParentTypeCount] = {
|
|
// Cell
|
|
{{&nsCSSFrameConstructor::ConstructTableCell,
|
|
FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRow)},
|
|
PseudoStyleType::tableCell},
|
|
// Row
|
|
{{&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRowGroup)},
|
|
PseudoStyleType::tableRow},
|
|
// Row group
|
|
{{&nsCSSFrameConstructor::ConstructTableRowOrRowGroup,
|
|
FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable)},
|
|
PseudoStyleType::tableRowGroup},
|
|
// Column group
|
|
{{ToCreationFunc(NS_NewTableColGroupFrame),
|
|
FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET |
|
|
FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_SKIP_ABSPOS_PUSH |
|
|
// Not FCDATA_IS_WRAPPER_ANON_BOX, because we don't need to
|
|
// restyle these: they have non-inheriting styles.
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable)},
|
|
PseudoStyleType::tableColGroup},
|
|
// Table
|
|
{{&nsCSSFrameConstructor::ConstructTable,
|
|
FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX},
|
|
PseudoStyleType::table},
|
|
// Ruby
|
|
{{ToCreationFunc(NS_NewRubyFrame),
|
|
FCDATA_IS_LINE_PARTICIPANT | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX | FCDATA_SKIP_FRAMESET},
|
|
PseudoStyleType::ruby},
|
|
// Ruby Base
|
|
{{ToCreationFunc(NS_NewRubyBaseFrame),
|
|
FCDATA_USE_CHILD_ITEMS | FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyBaseContainer) |
|
|
FCDATA_SKIP_FRAMESET},
|
|
PseudoStyleType::rubyBase},
|
|
// Ruby Base Container
|
|
{{ToCreationFunc(NS_NewRubyBaseContainerFrame),
|
|
FCDATA_USE_CHILD_ITEMS | FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby) |
|
|
FCDATA_SKIP_FRAMESET},
|
|
PseudoStyleType::rubyBaseContainer},
|
|
// Ruby Text
|
|
{{ToCreationFunc(NS_NewRubyTextFrame),
|
|
FCDATA_USE_CHILD_ITEMS | FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyTextContainer) |
|
|
FCDATA_SKIP_FRAMESET},
|
|
PseudoStyleType::rubyText},
|
|
// Ruby Text Container
|
|
{{ToCreationFunc(NS_NewRubyTextContainerFrame),
|
|
FCDATA_USE_CHILD_ITEMS | FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby) |
|
|
FCDATA_SKIP_FRAMESET},
|
|
PseudoStyleType::rubyTextContainer}};
|
|
|
|
void nsCSSFrameConstructor::CreateNeededAnonFlexOrGridItems(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems,
|
|
nsIFrame* aParentFrame) {
|
|
if (aItems.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
if (!aParentFrame->IsFlexOrGridContainer()) {
|
|
return;
|
|
}
|
|
|
|
const bool isLegacyWebKitBox =
|
|
IsFlexContainerForLegacyWebKitBox(aParentFrame);
|
|
FCItemIterator iter(aItems);
|
|
do {
|
|
// Advance iter past children that don't want to be wrapped
|
|
if (iter.SkipItemsThatDontNeedAnonFlexOrGridItem(aState,
|
|
isLegacyWebKitBox)) {
|
|
// Hit the end of the items without finding any remaining children that
|
|
// need to be wrapped. We're finished!
|
|
return;
|
|
}
|
|
|
|
// If our next potentially-wrappable child is whitespace, then see if
|
|
// there's anything wrappable immediately after it. If not, we just drop
|
|
// the whitespace and move on. (We're not supposed to create any anonymous
|
|
// flex/grid items that _only_ contain whitespace).
|
|
// (BUT if this is generated content, then we don't give whitespace nodes
|
|
// any special treatment, because they're probably not really whitespace --
|
|
// they're just temporarily empty, waiting for their generated text.)
|
|
// XXXdholbert If this node's generated text will *actually end up being
|
|
// entirely whitespace*, then we technically should still skip over it, per
|
|
// the CSS grid & flexbox specs. I'm not bothering with that at this point,
|
|
// since it's a pretty extreme edge case.
|
|
if (!aParentFrame->IsGeneratedContentFrame() &&
|
|
iter.item().IsWhitespace(aState)) {
|
|
FCItemIterator afterWhitespaceIter(iter);
|
|
bool hitEnd = afterWhitespaceIter.SkipWhitespace(aState);
|
|
bool nextChildNeedsAnonItem =
|
|
!hitEnd && afterWhitespaceIter.item().NeedsAnonFlexOrGridItem(
|
|
aState, isLegacyWebKitBox);
|
|
|
|
if (!nextChildNeedsAnonItem) {
|
|
// There's nothing after the whitespace that we need to wrap, so we
|
|
// just drop this run of whitespace.
|
|
iter.DeleteItemsTo(this, afterWhitespaceIter);
|
|
if (hitEnd) {
|
|
// Nothing left to do -- we're finished!
|
|
return;
|
|
}
|
|
// else, we have a next child and it does not want to be wrapped. So,
|
|
// we jump back to the beginning of the loop to skip over that child
|
|
// (and anything else non-wrappable after it)
|
|
MOZ_ASSERT(!iter.IsDone() && !iter.item().NeedsAnonFlexOrGridItem(
|
|
aState, isLegacyWebKitBox),
|
|
"hitEnd and/or nextChildNeedsAnonItem lied");
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// Now |iter| points to the first child that needs to be wrapped in an
|
|
// anonymous flex/grid item. Now we see how many children after it also want
|
|
// to be wrapped in an anonymous flex/grid item.
|
|
FCItemIterator endIter(iter); // iterator to find the end of the group
|
|
endIter.SkipItemsThatNeedAnonFlexOrGridItem(aState, isLegacyWebKitBox);
|
|
|
|
NS_ASSERTION(iter != endIter,
|
|
"Should've had at least one wrappable child to seek past");
|
|
|
|
// Now, we create the anonymous flex or grid item to contain the children
|
|
// between |iter| and |endIter|.
|
|
nsIContent* parentContent = aParentFrame->GetContent();
|
|
RefPtr<ComputedStyle> wrapperStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::anonymousItem, aParentFrame->Style());
|
|
|
|
static constexpr FrameConstructionData sBlockFCData(
|
|
ToCreationFunc(NS_NewBlockFrame), FCDATA_SKIP_FRAMESET |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX);
|
|
|
|
// Use the content of our parent frame
|
|
auto* newItem = new (this) FrameConstructionItem(
|
|
&sBlockFCData, parentContent, wrapperStyle.forget(), true);
|
|
|
|
newItem->mIsAllInline =
|
|
newItem->mComputedStyle->StyleDisplay()->IsInlineOutsideStyle();
|
|
newItem->mIsBlock = !newItem->mIsAllInline;
|
|
|
|
MOZ_ASSERT(!newItem->mIsAllInline && newItem->mIsBlock,
|
|
"expecting anonymous flex/grid items to be block-level "
|
|
"(this will make a difference when we encounter "
|
|
"'align-items: baseline')");
|
|
|
|
// Anonymous flex and grid items induce line boundaries around their
|
|
// contents.
|
|
newItem->mChildItems.SetLineBoundaryAtStart(true);
|
|
newItem->mChildItems.SetLineBoundaryAtEnd(true);
|
|
// The parent of the items in aItems is also the parent of the items
|
|
// in mChildItems
|
|
newItem->mChildItems.SetParentHasNoShadowDOM(aItems.ParentHasNoShadowDOM());
|
|
|
|
// Eat up all items between |iter| and |endIter| and put them in our
|
|
// wrapper. This advances |iter| to point to |endIter|.
|
|
iter.AppendItemsToList(this, endIter, newItem->mChildItems);
|
|
|
|
iter.InsertItem(newItem);
|
|
} while (!iter.IsDone());
|
|
}
|
|
|
|
/* static */ nsCSSFrameConstructor::RubyWhitespaceType
|
|
nsCSSFrameConstructor::ComputeRubyWhitespaceType(StyleDisplay aPrevDisplay,
|
|
StyleDisplay aNextDisplay) {
|
|
MOZ_ASSERT(aPrevDisplay.IsRuby() && aNextDisplay.IsRuby());
|
|
if (aPrevDisplay == aNextDisplay &&
|
|
(aPrevDisplay == StyleDisplay::RubyBase ||
|
|
aPrevDisplay == StyleDisplay::RubyText)) {
|
|
return eRubyInterLeafWhitespace;
|
|
}
|
|
if (aNextDisplay == StyleDisplay::RubyText ||
|
|
aNextDisplay == StyleDisplay::RubyTextContainer) {
|
|
return eRubyInterLevelWhitespace;
|
|
}
|
|
return eRubyInterSegmentWhitespace;
|
|
}
|
|
|
|
/**
|
|
* This function checks the content from |aStartIter| to |aEndIter|,
|
|
* determines whether it contains only whitespace, and if yes,
|
|
* interprets the type of whitespace. This method does not change
|
|
* any of the iters.
|
|
*/
|
|
/* static */ nsCSSFrameConstructor::RubyWhitespaceType
|
|
nsCSSFrameConstructor::InterpretRubyWhitespace(nsFrameConstructorState& aState,
|
|
const FCItemIterator& aStartIter,
|
|
const FCItemIterator& aEndIter) {
|
|
if (!aStartIter.item().IsWhitespace(aState)) {
|
|
return eRubyNotWhitespace;
|
|
}
|
|
|
|
FCItemIterator spaceEndIter(aStartIter);
|
|
spaceEndIter.SkipWhitespace(aState);
|
|
if (spaceEndIter != aEndIter) {
|
|
return eRubyNotWhitespace;
|
|
}
|
|
|
|
// Any leading or trailing whitespace in non-pseudo ruby box
|
|
// should have been trimmed, hence there should not be any
|
|
// whitespace at the start or the end.
|
|
MOZ_ASSERT(!aStartIter.AtStart() && !aEndIter.IsDone());
|
|
FCItemIterator prevIter(aStartIter);
|
|
prevIter.Prev();
|
|
return ComputeRubyWhitespaceType(
|
|
prevIter.item().mComputedStyle->StyleDisplay()->mDisplay,
|
|
aEndIter.item().mComputedStyle->StyleDisplay()->mDisplay);
|
|
}
|
|
|
|
/**
|
|
* This function eats up consecutive items which do not want the current
|
|
* parent into either a ruby base box or a ruby text box. When it
|
|
* returns, |aIter| points to the first item it doesn't wrap.
|
|
*/
|
|
void nsCSSFrameConstructor::WrapItemsInPseudoRubyLeafBox(
|
|
FCItemIterator& aIter, ComputedStyle* aParentStyle,
|
|
nsIContent* aParentContent) {
|
|
StyleDisplay parentDisplay = aParentStyle->StyleDisplay()->mDisplay;
|
|
ParentType parentType, wrapperType;
|
|
if (parentDisplay == StyleDisplay::RubyTextContainer) {
|
|
parentType = eTypeRubyTextContainer;
|
|
wrapperType = eTypeRubyText;
|
|
} else {
|
|
MOZ_ASSERT(parentDisplay == StyleDisplay::RubyBaseContainer);
|
|
parentType = eTypeRubyBaseContainer;
|
|
wrapperType = eTypeRubyBase;
|
|
}
|
|
|
|
MOZ_ASSERT(aIter.item().DesiredParentType() != parentType,
|
|
"Should point to something needs to be wrapped.");
|
|
|
|
FCItemIterator endIter(aIter);
|
|
endIter.SkipItemsNotWantingParentType(parentType);
|
|
|
|
WrapItemsInPseudoParent(aParentContent, aParentStyle, wrapperType, aIter,
|
|
endIter);
|
|
}
|
|
|
|
/**
|
|
* This function eats up consecutive items into a ruby level container.
|
|
* It may create zero or one level container. When it returns, |aIter|
|
|
* points to the first item it doesn't wrap.
|
|
*/
|
|
void nsCSSFrameConstructor::WrapItemsInPseudoRubyLevelContainer(
|
|
nsFrameConstructorState& aState, FCItemIterator& aIter,
|
|
ComputedStyle* aParentStyle, nsIContent* aParentContent) {
|
|
MOZ_ASSERT(aIter.item().DesiredParentType() != eTypeRuby,
|
|
"Pointing to a level container?");
|
|
|
|
FrameConstructionItem& firstItem = aIter.item();
|
|
ParentType wrapperType = firstItem.DesiredParentType();
|
|
if (wrapperType != eTypeRubyTextContainer) {
|
|
// If the first item is not ruby text,
|
|
// it should be in a base container.
|
|
wrapperType = eTypeRubyBaseContainer;
|
|
}
|
|
|
|
FCItemIterator endIter(aIter);
|
|
do {
|
|
if (endIter.SkipItemsWantingParentType(wrapperType) ||
|
|
// If the skipping above stops at some item which wants a
|
|
// different ruby parent, then we have finished.
|
|
IsRubyParentType(endIter.item().DesiredParentType())) {
|
|
// No more items need to be wrapped in this level container.
|
|
break;
|
|
}
|
|
|
|
FCItemIterator contentEndIter(endIter);
|
|
contentEndIter.SkipItemsNotWantingRubyParent();
|
|
// endIter must be on something doesn't want a ruby parent.
|
|
MOZ_ASSERT(contentEndIter != endIter);
|
|
|
|
// InterpretRubyWhitespace depends on the fact that any leading or
|
|
// trailing whitespace described in the spec have been trimmed at
|
|
// this point. With this precondition, it is safe not to check
|
|
// whether contentEndIter has been done.
|
|
RubyWhitespaceType whitespaceType =
|
|
InterpretRubyWhitespace(aState, endIter, contentEndIter);
|
|
if (whitespaceType == eRubyInterLevelWhitespace) {
|
|
// Remove inter-level whitespace.
|
|
bool atStart = (aIter == endIter);
|
|
endIter.DeleteItemsTo(this, contentEndIter);
|
|
if (atStart) {
|
|
aIter = endIter;
|
|
}
|
|
} else if (whitespaceType == eRubyInterSegmentWhitespace) {
|
|
// If this level container starts with inter-segment whitespaces,
|
|
// wrap them. Break at contentEndIter. Otherwise, leave it here.
|
|
// Break at endIter. They will be wrapped when we are here again.
|
|
if (aIter == endIter) {
|
|
MOZ_ASSERT(wrapperType == eTypeRubyBaseContainer,
|
|
"Inter-segment whitespace should be wrapped in rbc");
|
|
endIter = contentEndIter;
|
|
}
|
|
break;
|
|
} else if (wrapperType == eTypeRubyTextContainer &&
|
|
whitespaceType != eRubyInterLeafWhitespace) {
|
|
// Misparented inline content that's not inter-annotation
|
|
// whitespace doesn't belong in a pseudo ruby text container.
|
|
// Break at endIter.
|
|
break;
|
|
} else {
|
|
endIter = contentEndIter;
|
|
}
|
|
} while (!endIter.IsDone());
|
|
|
|
// It is possible that everything our parent wants us to wrap is
|
|
// simply an inter-level whitespace, which has been trimmed, or
|
|
// an inter-segment whitespace, which will be wrapped later.
|
|
// In those cases, don't create anything.
|
|
if (aIter != endIter) {
|
|
WrapItemsInPseudoParent(aParentContent, aParentStyle, wrapperType, aIter,
|
|
endIter);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function trims leading and trailing whitespaces
|
|
* in the given item list.
|
|
*/
|
|
void nsCSSFrameConstructor::TrimLeadingAndTrailingWhitespaces(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems) {
|
|
FCItemIterator iter(aItems);
|
|
if (!iter.IsDone() && iter.item().IsWhitespace(aState)) {
|
|
FCItemIterator spaceEndIter(iter);
|
|
spaceEndIter.SkipWhitespace(aState);
|
|
iter.DeleteItemsTo(this, spaceEndIter);
|
|
}
|
|
|
|
iter.SetToEnd();
|
|
if (!iter.AtStart()) {
|
|
FCItemIterator spaceEndIter(iter);
|
|
do {
|
|
iter.Prev();
|
|
if (iter.AtStart()) {
|
|
// It's fine to not check the first item, because we
|
|
// should have trimmed leading whitespaces above.
|
|
break;
|
|
}
|
|
} while (iter.item().IsWhitespace(aState));
|
|
iter.Next();
|
|
if (iter != spaceEndIter) {
|
|
iter.DeleteItemsTo(this, spaceEndIter);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function walks through the child list (aItems) and creates
|
|
* needed pseudo ruby boxes to wrap misparented children.
|
|
*/
|
|
void nsCSSFrameConstructor::CreateNeededPseudoInternalRubyBoxes(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems,
|
|
nsIFrame* aParentFrame) {
|
|
const ParentType ourParentType = GetParentType(aParentFrame);
|
|
if (!IsRubyParentType(ourParentType) ||
|
|
aItems.AllWantParentType(ourParentType)) {
|
|
return;
|
|
}
|
|
|
|
if (!IsRubyPseudo(aParentFrame) ||
|
|
ourParentType == eTypeRuby /* for 'display:block ruby' */) {
|
|
// Normally, ruby pseudo frames start from and end at some elements,
|
|
// which means they don't have leading and trailing whitespaces at
|
|
// all. But there are two cases where they do actually have leading
|
|
// or trailing whitespaces:
|
|
// 1. It is an inter-segment whitespace which in an individual ruby
|
|
// base container.
|
|
// 2. The pseudo frame starts from or ends at consecutive inline
|
|
// content, which is not pure whitespace, but includes some.
|
|
// In either case, the whitespaces are not the leading or trailing
|
|
// whitespaces defined in the spec, and thus should not be trimmed.
|
|
TrimLeadingAndTrailingWhitespaces(aState, aItems);
|
|
}
|
|
|
|
FCItemIterator iter(aItems);
|
|
nsIContent* parentContent = aParentFrame->GetContent();
|
|
ComputedStyle* parentStyle = aParentFrame->Style();
|
|
while (!iter.IsDone()) {
|
|
if (!iter.SkipItemsWantingParentType(ourParentType)) {
|
|
if (ourParentType == eTypeRuby) {
|
|
WrapItemsInPseudoRubyLevelContainer(aState, iter, parentStyle,
|
|
parentContent);
|
|
} else {
|
|
WrapItemsInPseudoRubyLeafBox(iter, parentStyle, parentContent);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* This function works as follows: we walk through the child list (aItems) and
|
|
* find items that cannot have aParentFrame as their parent. We wrap
|
|
* continuous runs of such items into a FrameConstructionItem for a frame that
|
|
* gets them closer to their desired parents. For example, a run of non-row
|
|
* children of a row-group will get wrapped in a row. When we later construct
|
|
* the frame for this wrapper (in this case for the row), it'll be the correct
|
|
* parent for the cells in the set of items we wrapped or we'll wrap cells
|
|
* around everything else. At the end of this method, aItems is guaranteed to
|
|
* contain only items for frames that can be direct kids of aParentFrame.
|
|
*/
|
|
void nsCSSFrameConstructor::CreateNeededPseudoContainers(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems,
|
|
nsIFrame* aParentFrame) {
|
|
ParentType ourParentType = GetParentType(aParentFrame);
|
|
if (IsRubyParentType(ourParentType) ||
|
|
aItems.AllWantParentType(ourParentType)) {
|
|
// Nothing to do here
|
|
return;
|
|
}
|
|
|
|
FCItemIterator iter(aItems);
|
|
do {
|
|
if (iter.SkipItemsWantingParentType(ourParentType)) {
|
|
// Nothing else to do here; we're finished
|
|
return;
|
|
}
|
|
|
|
// Now we're pointing to the first child that wants a different parent
|
|
// type.
|
|
|
|
// Now try to figure out what kids we can group together. We can generally
|
|
// group everything that has a different desired parent type from us. Two
|
|
// exceptions to this:
|
|
// 1) If our parent type is table, we can't group columns with anything
|
|
// else other than whitespace.
|
|
// 2) Whitespace that lies between two things we can group which both want
|
|
// a non-block parent should be dropped, even if we can't group them
|
|
// with each other and even if the whitespace wants a parent of
|
|
// ourParentType. Ends of the list count as things that don't want a
|
|
// block parent (so that for example we'll drop a whitespace-only list).
|
|
|
|
FCItemIterator endIter(iter); /* iterator to find the end of the group */
|
|
ParentType groupingParentType = endIter.item().DesiredParentType();
|
|
if (aItems.AllWantParentType(groupingParentType) &&
|
|
groupingParentType != eTypeBlock) {
|
|
// Just group them all and be done with it. We need the check for
|
|
// eTypeBlock here to catch the "all the items are whitespace" case
|
|
// described above.
|
|
endIter.SetToEnd();
|
|
} else {
|
|
// Locate the end of the group.
|
|
|
|
// Keep track of the type the previous item wanted, in case we have to
|
|
// deal with whitespace. Start it off with ourParentType, since that's
|
|
// the last thing |iter| would have skipped over.
|
|
ParentType prevParentType = ourParentType;
|
|
do {
|
|
// Walk an iterator past any whitespace that we might be able to drop
|
|
// from the list
|
|
FCItemIterator spaceEndIter(endIter);
|
|
if (prevParentType != eTypeBlock &&
|
|
!aParentFrame->IsGeneratedContentFrame() &&
|
|
spaceEndIter.item().IsWhitespace(aState)) {
|
|
bool trailingSpaces = spaceEndIter.SkipWhitespace(aState);
|
|
|
|
// We drop the whitespace in the following cases:
|
|
// 1) If these are not trailing spaces and the next item wants a table
|
|
// or table-part parent
|
|
// 2) If these are trailing spaces and aParentFrame is a
|
|
// tabular container according to rule 1.3 of CSS 2.1 Sec 17.2.1.
|
|
// (Being a tabular container pretty much means ourParentType is
|
|
// not eTypeBlock besides the eTypeColGroup case, which won't
|
|
// reach here.)
|
|
if ((!trailingSpaces &&
|
|
IsTableParentType(spaceEndIter.item().DesiredParentType())) ||
|
|
(trailingSpaces && ourParentType != eTypeBlock)) {
|
|
bool updateStart = (iter == endIter);
|
|
endIter.DeleteItemsTo(this, spaceEndIter);
|
|
NS_ASSERTION(trailingSpaces == endIter.IsDone(),
|
|
"These should match");
|
|
|
|
if (updateStart) {
|
|
iter = endIter;
|
|
}
|
|
|
|
if (trailingSpaces) {
|
|
break; /* Found group end */
|
|
}
|
|
|
|
if (updateStart) {
|
|
// Update groupingParentType, since it might have been eTypeBlock
|
|
// just because of the whitespace.
|
|
groupingParentType = iter.item().DesiredParentType();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now endIter points to a non-whitespace item or a non-droppable
|
|
// whitespace item. In the latter case, if this is the end of the group
|
|
// we'll traverse this whitespace again. But it'll all just be quick
|
|
// DesiredParentType() checks which will match ourParentType (that's
|
|
// what it means that this is the group end), so it's OK.
|
|
// However, when we are grouping a ruby parent, and endIter points to
|
|
// a non-droppable whitespace, if the next non-whitespace item also
|
|
// wants a ruby parent, the whitespace should also be included into
|
|
// the current ruby container.
|
|
prevParentType = endIter.item().DesiredParentType();
|
|
if (prevParentType == ourParentType &&
|
|
(endIter == spaceEndIter || spaceEndIter.IsDone() ||
|
|
!IsRubyParentType(groupingParentType) ||
|
|
!IsRubyParentType(spaceEndIter.item().DesiredParentType()))) {
|
|
// End the group at endIter.
|
|
break;
|
|
}
|
|
|
|
if (ourParentType == eTypeTable &&
|
|
(prevParentType == eTypeColGroup) !=
|
|
(groupingParentType == eTypeColGroup)) {
|
|
// Either we started with columns and now found something else, or
|
|
// vice versa. In any case, end the grouping.
|
|
break;
|
|
}
|
|
|
|
// If we have some whitespace that we were not able to drop and there is
|
|
// an item after the whitespace that is already properly parented, then
|
|
// make sure to include the spaces in our group but stop the group after
|
|
// that.
|
|
if (spaceEndIter != endIter && !spaceEndIter.IsDone() &&
|
|
ourParentType == spaceEndIter.item().DesiredParentType()) {
|
|
endIter = spaceEndIter;
|
|
break;
|
|
}
|
|
|
|
// Include the whitespace we didn't drop (if any) in the group.
|
|
endIter = spaceEndIter;
|
|
prevParentType = endIter.item().DesiredParentType();
|
|
|
|
endIter.Next();
|
|
} while (!endIter.IsDone());
|
|
}
|
|
|
|
if (iter == endIter) {
|
|
// Nothing to wrap here; just skipped some whitespace
|
|
continue;
|
|
}
|
|
|
|
// Now group together all the items between iter and endIter. The right
|
|
// parent type to use depends on ourParentType.
|
|
ParentType wrapperType;
|
|
switch (ourParentType) {
|
|
case eTypeRow:
|
|
// The parent type for a cell is eTypeBlock, since that's what a cell
|
|
// looks like to its kids.
|
|
wrapperType = eTypeBlock;
|
|
break;
|
|
case eTypeRowGroup:
|
|
wrapperType = eTypeRow;
|
|
break;
|
|
case eTypeTable:
|
|
// Either colgroup or rowgroup, depending on what we're grouping.
|
|
wrapperType =
|
|
groupingParentType == eTypeColGroup ? eTypeColGroup : eTypeRowGroup;
|
|
break;
|
|
case eTypeColGroup:
|
|
MOZ_CRASH("Colgroups should be suppresing non-col child items");
|
|
default:
|
|
NS_ASSERTION(ourParentType == eTypeBlock, "Unrecognized parent type");
|
|
if (IsRubyParentType(groupingParentType)) {
|
|
wrapperType = eTypeRuby;
|
|
} else {
|
|
NS_ASSERTION(IsTableParentType(groupingParentType),
|
|
"groupingParentType should be either Ruby or table");
|
|
wrapperType = eTypeTable;
|
|
}
|
|
}
|
|
|
|
ComputedStyle* parentStyle = aParentFrame->Style();
|
|
WrapItemsInPseudoParent(aParentFrame->GetContent(), parentStyle,
|
|
wrapperType, iter, endIter);
|
|
|
|
// Now |iter| points to the item that was the first one we didn't wrap;
|
|
// loop and see whether we need to skip it or wrap it in something
|
|
// different.
|
|
} while (!iter.IsDone());
|
|
}
|
|
|
|
/**
|
|
* This method wraps frame construction item from |aIter| to
|
|
* |aEndIter|. After it returns, aIter points to the first item
|
|
* after the wrapper.
|
|
*/
|
|
void nsCSSFrameConstructor::WrapItemsInPseudoParent(
|
|
nsIContent* aParentContent, ComputedStyle* aParentStyle,
|
|
ParentType aWrapperType, FCItemIterator& aIter,
|
|
const FCItemIterator& aEndIter) {
|
|
const PseudoParentData& pseudoData = sPseudoParentData[aWrapperType];
|
|
PseudoStyleType pseudoType = pseudoData.mPseudoType;
|
|
auto& parentDisplay = *aParentStyle->StyleDisplay();
|
|
auto parentDisplayInside = parentDisplay.DisplayInside();
|
|
|
|
// XXXmats should we use IsInlineInsideStyle() here instead? seems odd to
|
|
// exclude RubyBaseContainer/RubyTextContainer...
|
|
if (pseudoType == PseudoStyleType::table &&
|
|
(parentDisplay.IsInlineFlow() ||
|
|
parentDisplayInside == StyleDisplayInside::RubyBase ||
|
|
parentDisplayInside == StyleDisplayInside::RubyText)) {
|
|
pseudoType = PseudoStyleType::inlineTable;
|
|
}
|
|
|
|
RefPtr<ComputedStyle> wrapperStyle;
|
|
if (pseudoData.mFCData.mBits & FCDATA_IS_WRAPPER_ANON_BOX) {
|
|
wrapperStyle = mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
pseudoType, aParentStyle);
|
|
} else {
|
|
wrapperStyle =
|
|
mPresShell->StyleSet()->ResolveNonInheritingAnonymousBoxStyle(
|
|
pseudoType);
|
|
}
|
|
|
|
// Use the content of our parent frame
|
|
auto* newItem = new (this) FrameConstructionItem(
|
|
&pseudoData.mFCData, aParentContent, wrapperStyle.forget(), true);
|
|
|
|
const nsStyleDisplay* disp = newItem->mComputedStyle->StyleDisplay();
|
|
// Here we're cheating a tad... technically, table-internal items should be
|
|
// inline if aParentFrame is inline, but they'll get wrapped in an
|
|
// inline-table in the end, so it'll all work out. In any case, arguably
|
|
// we don't need to maintain this state at this point... but it's better
|
|
// to, I guess.
|
|
newItem->mIsAllInline = disp->IsInlineOutsideStyle();
|
|
|
|
bool isRuby = disp->IsRubyDisplayType();
|
|
if (!isRuby) {
|
|
// Table pseudo frames always induce line boundaries around their
|
|
// contents.
|
|
newItem->mChildItems.SetLineBoundaryAtStart(true);
|
|
newItem->mChildItems.SetLineBoundaryAtEnd(true);
|
|
}
|
|
// The parent of the items in aItems is also the parent of the items
|
|
// in mChildItems
|
|
newItem->mChildItems.SetParentHasNoShadowDOM(
|
|
aIter.List()->ParentHasNoShadowDOM());
|
|
|
|
// Eat up all items between |aIter| and |aEndIter| and put them in our
|
|
// wrapper Advances |aIter| to point to |aEndIter|.
|
|
aIter.AppendItemsToList(this, aEndIter, newItem->mChildItems);
|
|
|
|
aIter.InsertItem(newItem);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CreateNeededPseudoSiblings(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems,
|
|
nsIFrame* aParentFrame) {
|
|
if (aItems.IsEmpty() || GetParentType(aParentFrame) != eTypeRuby) {
|
|
return;
|
|
}
|
|
|
|
FCItemIterator iter(aItems);
|
|
StyleDisplay firstDisplay =
|
|
iter.item().mComputedStyle->StyleDisplay()->mDisplay;
|
|
if (firstDisplay == StyleDisplay::RubyBaseContainer) {
|
|
return;
|
|
}
|
|
NS_ASSERTION(firstDisplay == StyleDisplay::RubyTextContainer,
|
|
"Child of ruby frame should either a rbc or a rtc");
|
|
|
|
const PseudoParentData& pseudoData =
|
|
sPseudoParentData[eTypeRubyBaseContainer];
|
|
RefPtr<ComputedStyle> pseudoStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
pseudoData.mPseudoType, aParentFrame->Style());
|
|
FrameConstructionItem* newItem = new (this) FrameConstructionItem(
|
|
&pseudoData.mFCData,
|
|
// Use the content of the parent frame
|
|
aParentFrame->GetContent(), pseudoStyle.forget(), true);
|
|
newItem->mIsAllInline = true;
|
|
newItem->mChildItems.SetParentHasNoShadowDOM(true);
|
|
iter.InsertItem(newItem);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
/**
|
|
* Returns true iff aFrame should be wrapped in an anonymous flex/grid item,
|
|
* rather than being a direct child of aContainerFrame.
|
|
*
|
|
* NOTE: aContainerFrame must be a flex or grid container - this function is
|
|
* purely for sanity-checking the children of these container types.
|
|
* NOTE: See also NeedsAnonFlexOrGridItem(), for the non-debug version of this
|
|
* logic (which operates a bit earlier, on FCData instead of frames).
|
|
*/
|
|
static bool FrameWantsToBeInAnonymousItem(const nsIFrame* aContainerFrame,
|
|
const nsIFrame* aFrame) {
|
|
MOZ_ASSERT(aContainerFrame->IsFlexOrGridContainer());
|
|
|
|
// Any line-participant frames (e.g. text) definitely want to be wrapped in
|
|
// an anonymous flex/grid item.
|
|
if (aFrame->IsLineParticipant()) {
|
|
return true;
|
|
}
|
|
|
|
// If the container is a -webkit-{inline-}box container, then placeholders
|
|
// also need to be wrapped, for compatibility.
|
|
if (IsFlexContainerForLegacyWebKitBox(aContainerFrame) &&
|
|
aFrame->IsPlaceholderFrame()) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
static void VerifyGridFlexContainerChildren(nsIFrame* aParentFrame,
|
|
const nsFrameList& aChildren) {
|
|
#ifdef DEBUG
|
|
if (!aParentFrame->IsFlexOrGridContainer()) {
|
|
return;
|
|
}
|
|
|
|
bool prevChildWasAnonItem = false;
|
|
for (const nsIFrame* child : aChildren) {
|
|
MOZ_ASSERT(!FrameWantsToBeInAnonymousItem(aParentFrame, child),
|
|
"frame wants to be inside an anonymous item, but it isn't");
|
|
if (IsAnonymousItem(child)) {
|
|
AssertAnonymousFlexOrGridItemParent(child, aParentFrame);
|
|
MOZ_ASSERT(!prevChildWasAnonItem, "two anon items in a row");
|
|
nsIFrame* firstWrappedChild = child->PrincipalChildList().FirstChild();
|
|
MOZ_ASSERT(firstWrappedChild, "anonymous item shouldn't be empty");
|
|
prevChildWasAnonItem = true;
|
|
} else {
|
|
prevChildWasAnonItem = false;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static bool FrameHasOnlyPlaceholderPrevSiblings(const nsIFrame* aFrame) {
|
|
// Check for prev siblings, ignoring placeholder frames.
|
|
MOZ_ASSERT(aFrame, "frame must not be null");
|
|
const nsIFrame* prevSibling = aFrame;
|
|
do {
|
|
prevSibling = prevSibling->GetPrevSibling();
|
|
} while (prevSibling && prevSibling->IsPlaceholderFrame());
|
|
return !prevSibling;
|
|
}
|
|
|
|
static bool FrameHasOnlyPlaceholderNextSiblings(const nsIFrame* aFrame) {
|
|
// Check for next siblings, ignoring placeholder frames.
|
|
MOZ_ASSERT(aFrame, "frame must not be null");
|
|
const nsIFrame* nextSibling = aFrame;
|
|
do {
|
|
nextSibling = nextSibling->GetNextSibling();
|
|
} while (nextSibling && nextSibling->IsPlaceholderFrame());
|
|
return !nextSibling;
|
|
}
|
|
|
|
static void SetPageValues(nsIFrame* const aFrame,
|
|
const nsAtom* const aAutoValue,
|
|
const nsAtom* const aStartValue,
|
|
const nsAtom* const aEndValue) {
|
|
MOZ_ASSERT(aAutoValue, "Auto page value should never be null");
|
|
MOZ_ASSERT(aStartValue || aEndValue, "Should not have called with no values");
|
|
nsIFrame::PageValues* pageValues =
|
|
aFrame->GetProperty(nsIFrame::PageValuesProperty());
|
|
|
|
if (aStartValue) {
|
|
if (aStartValue == aAutoValue) {
|
|
// If the page value struct already exists, set the start value to null
|
|
// to indicate the auto value.
|
|
if (pageValues) {
|
|
pageValues->mStartPageValue = nullptr;
|
|
}
|
|
} else {
|
|
// The start value is not auto, so we need to store it, creating the
|
|
// page values struct if it does not already exist.
|
|
if (!pageValues) {
|
|
pageValues = new nsIFrame::PageValues();
|
|
aFrame->SetProperty(nsIFrame::PageValuesProperty(), pageValues);
|
|
}
|
|
pageValues->mStartPageValue = aStartValue;
|
|
}
|
|
}
|
|
if (aEndValue) {
|
|
if (aEndValue == aAutoValue) {
|
|
// If the page value struct already exists, set the end value to null
|
|
// to indicate the auto value.
|
|
if (pageValues) {
|
|
pageValues->mEndPageValue = nullptr;
|
|
}
|
|
} else {
|
|
// The end value is not auto, so we need to store it, creating the
|
|
// page values struct if it does not already exist.
|
|
if (!pageValues) {
|
|
pageValues = new nsIFrame::PageValues();
|
|
aFrame->SetProperty(nsIFrame::PageValuesProperty(), pageValues);
|
|
}
|
|
pageValues->mEndPageValue = aEndValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
inline void nsCSSFrameConstructor::ConstructFramesFromItemList(
|
|
nsFrameConstructorState& aState, FrameConstructionItemList& aItems,
|
|
nsContainerFrame* aParentFrame, bool aParentIsWrapperAnonBox,
|
|
nsFrameList& aFrameList) {
|
|
#ifdef DEBUG
|
|
if (aParentFrame->StyleContent()->mContent.IsNone() &&
|
|
StaticPrefs::layout_css_element_content_none_enabled()) {
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
MOZ_ASSERT(iter.item().mContent->IsInNativeAnonymousSubtree() ||
|
|
iter.item().mComputedStyle->IsPseudoOrAnonBox());
|
|
}
|
|
}
|
|
|
|
// The assertion condition should match the logic in
|
|
// MaybePushFloatContainingBlock().
|
|
MOZ_ASSERT(!(ShouldSuppressFloatingOfDescendants(aParentFrame) ||
|
|
aParentFrame->IsFloatContainingBlock()) ||
|
|
aState.mFloatCBCandidate == aParentFrame,
|
|
"Our caller or ProcessChildren()'s caller should call "
|
|
"MaybePushFloatContainingBlock() to handle the float containing "
|
|
"block candidate!");
|
|
aState.mFloatCBCandidate = nullptr;
|
|
#endif
|
|
|
|
// Ensure aParentIsWrapperAnonBox is correct. We _could_ compute it directly,
|
|
// but it would be a bit slow, which is why we pass it from callers, who have
|
|
// that information offhand in many cases.
|
|
MOZ_ASSERT(ParentIsWrapperAnonBox(aParentFrame) == aParentIsWrapperAnonBox);
|
|
|
|
// Note: we explicitly exclude TableColGroupFrame because it doesn't
|
|
// have the FCDATA_IS_WRAPPER_ANON_BOX on pseudos so aParentIsWrapperAnonBox
|
|
// is false for such pseudos (see sPseudoParentData below).
|
|
if (!aParentIsWrapperAnonBox && aState.mHasRenderedLegend &&
|
|
aParentFrame->GetContent()->IsHTMLElement(nsGkAtoms::fieldset) &&
|
|
!aParentFrame->IsTableColGroupFrame()) {
|
|
DebugOnly<bool> found = false;
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
if (iter.item().mIsRenderedLegend) {
|
|
// This makes the rendered legend the first frame in the fieldset child
|
|
// list which makes keyboard traversal follow the visual order.
|
|
nsFieldSetFrame* fieldSetFrame = GetFieldSetFrameFor(aParentFrame);
|
|
nsFrameList renderedLegend;
|
|
ConstructFramesFromItem(aState, iter, fieldSetFrame, renderedLegend);
|
|
MOZ_ASSERT(renderedLegend.OnlyChild(),
|
|
"a rendered legend should have exactly one frame");
|
|
fieldSetFrame->InsertFrames(FrameChildListID::Principal, nullptr,
|
|
nullptr, std::move(renderedLegend));
|
|
FCItemIterator next = iter;
|
|
next.Next();
|
|
iter.DeleteItemsTo(this, next);
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
MOZ_ASSERT(found, "should have found our rendered legend");
|
|
}
|
|
|
|
CreateNeededPseudoContainers(aState, aItems, aParentFrame);
|
|
CreateNeededAnonFlexOrGridItems(aState, aItems, aParentFrame);
|
|
CreateNeededPseudoInternalRubyBoxes(aState, aItems, aParentFrame);
|
|
CreateNeededPseudoSiblings(aState, aItems, aParentFrame);
|
|
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
MOZ_ASSERT(!iter.item().mIsRenderedLegend,
|
|
"Only one item can be the rendered legend, "
|
|
"and it should've been handled above");
|
|
NS_ASSERTION(iter.item().DesiredParentType() == GetParentType(aParentFrame),
|
|
"Needed pseudos didn't get created; expect bad things");
|
|
ConstructFramesFromItem(aState, iter, aParentFrame, aFrameList);
|
|
}
|
|
|
|
VerifyGridFlexContainerChildren(aParentFrame, aFrameList);
|
|
|
|
// Calculate and propagate page-name values for each frame in the frame list.
|
|
// We do not want to compute and propagate page-name values from frames that
|
|
// are children of any subclasses of block frames, but not actually a block
|
|
// frame. The page-name property does not apply to frames which cannot create
|
|
// class A breakpoints (currently no subclass of BlockFrame can). Because the
|
|
// property does not apply, those children also cannot propagate page-name
|
|
// values.
|
|
// This assumption helps avoid unnecessarily handling page-names for frames
|
|
// such as form controls, which also avoids bug 1819468.
|
|
if (aState.mPresContext->IsPaginated() && aParentFrame->IsBlockFrame()) {
|
|
// Set the start/end page values while iterating the frame list, to walk
|
|
// up the frame tree only once after iterating the frame list.
|
|
// This also avoids extra property lookups on these frames.
|
|
MOZ_ASSERT(aState.mAutoPageNameValue == aParentFrame->GetAutoPageValue(),
|
|
"aState.mAutoPageNameValue should have been equivalent to "
|
|
"the auto value stored on our parent frame.");
|
|
// Even though we store null for page values that equal the "auto" resolved
|
|
// value on frames, we always want startPageValue/endPageValue to be the
|
|
// actual atoms reflecting the start/end values. This is because when we
|
|
// propagate the values up the frame tree, we will need to compare them to
|
|
// the auto value for each ancestor. This value might be different than the
|
|
// auto value for this frame.
|
|
const nsAtom* startPageValue = nullptr;
|
|
const nsAtom* endPageValue = nullptr;
|
|
for (nsIFrame* f : aFrameList) {
|
|
if (f->IsPlaceholderFrame()) {
|
|
continue;
|
|
}
|
|
// Resolve auto against the parent frame's used page name, which has been
|
|
// determined and set on aState.mAutoPageNameValue. If this item is not
|
|
// block-level then we use the value that auto resolves to.
|
|
//
|
|
// This is to achieve the propagation behavior described in the spec:
|
|
//
|
|
// "A start page value and end page value is determined for each box as
|
|
// the value (if any) propagated from its first or last child box
|
|
// (respectively), else the used value on the box itself."
|
|
//
|
|
// "A child propagates its own start or end page value if and only if the
|
|
// page property applies to it."
|
|
//
|
|
// The page property only applies to "boxes that create class A break
|
|
// points". When taken together, this means that non block-level children
|
|
// do not propagate start/end page values, and instead we use "the used
|
|
// value on the box itself", the "box itself" being aParentFrame. This
|
|
// value has been determined and saved as aState.mAutoPageNameValue
|
|
//
|
|
// https://www.w3.org/TR/css-page-3/#using-named-pages
|
|
// https://www.w3.org/TR/css-break-3/#btw-blocks
|
|
const StylePageName& pageName = f->StylePage()->mPage;
|
|
const nsAtom* const pageNameAtom =
|
|
(pageName.IsPageName() && f->IsBlockOutside())
|
|
? pageName.AsPageName().AsAtom()
|
|
: aState.mAutoPageNameValue;
|
|
nsIFrame::PageValues* pageValues =
|
|
f->GetProperty(nsIFrame::PageValuesProperty());
|
|
// If this frame has any children, it will already have had its page
|
|
// values set at this point. However, if no page values have been set,
|
|
// we must ensure that the appropriate PageValuesProperty value has been
|
|
// set.
|
|
// If the page name is equal to the auto value, then PageValuesProperty
|
|
// should remain null to indicate that the start/end values are both
|
|
// equal to the auto value.
|
|
if (pageNameAtom != aState.mAutoPageNameValue && !pageValues) {
|
|
pageValues = new nsIFrame::PageValues{pageNameAtom, pageNameAtom};
|
|
f->SetProperty(nsIFrame::PageValuesProperty(), pageValues);
|
|
}
|
|
// We don't want to use GetStartPageValue() or GetEndPageValue(), as each
|
|
// requires a property lookup which we can avoid here.
|
|
if (!startPageValue) {
|
|
startPageValue = (pageValues && pageValues->mStartPageValue)
|
|
? pageValues->mStartPageValue.get()
|
|
: aState.mAutoPageNameValue;
|
|
}
|
|
endPageValue = (pageValues && pageValues->mEndPageValue)
|
|
? pageValues->mEndPageValue.get()
|
|
: aState.mAutoPageNameValue;
|
|
MOZ_ASSERT(startPageValue && endPageValue,
|
|
"Should have found start/end page value");
|
|
}
|
|
MOZ_ASSERT(!startPageValue == !endPageValue,
|
|
"Should have set both or neither page values");
|
|
if (startPageValue) {
|
|
// Walk up the frame tree from our parent frame, propagating start and
|
|
// end page values.
|
|
// As we go, if we find that, for a frame, we are not contributing one of
|
|
// the start/end page values, then our subtree will not contribute this
|
|
// value from that frame onward. startPageValue/endPageValue are set to
|
|
// null to indicate this.
|
|
// Stop iterating when we are not contributing either start or end
|
|
// values, when we hit the root frame (no parent), or when we find a
|
|
// frame that is not a block frame.
|
|
for (nsContainerFrame* ancestorFrame = aParentFrame;
|
|
(startPageValue || endPageValue) && ancestorFrame &&
|
|
ancestorFrame->IsBlockFrame();
|
|
ancestorFrame = ancestorFrame->GetParent()) {
|
|
MOZ_ASSERT(!ancestorFrame->GetPrevInFlow(),
|
|
"Should not have fragmentation yet");
|
|
MOZ_ASSERT(ancestorFrame->mWasVisitedByAutoFrameConstructionPageName,
|
|
"Frame should have been visited by "
|
|
"AutoFrameConstructionPageName");
|
|
{
|
|
// Get what the auto value is, based on this frame's parent.
|
|
// For the root frame, `auto` resolves to the empty atom.
|
|
const nsContainerFrame* const parent = ancestorFrame->GetParent();
|
|
const nsAtom* const parentAuto = MOZ_LIKELY(parent)
|
|
? parent->GetAutoPageValue()
|
|
: nsGkAtoms::_empty;
|
|
SetPageValues(ancestorFrame, parentAuto, startPageValue,
|
|
endPageValue);
|
|
}
|
|
// Once we stop contributing start/end values, we know there is a
|
|
// sibling subtree that contributed that value to our shared parent
|
|
// instead of our starting frame's subtree. This means once
|
|
// startPageValue/endPageValue becomes null, indicating that we are no
|
|
// longer contributing that page value, it should stay null and we no
|
|
// longer need to check for siblings in that direction.
|
|
if (startPageValue &&
|
|
!FrameHasOnlyPlaceholderPrevSiblings(ancestorFrame)) {
|
|
startPageValue = nullptr;
|
|
}
|
|
if (endPageValue &&
|
|
!FrameHasOnlyPlaceholderNextSiblings(ancestorFrame)) {
|
|
endPageValue = nullptr;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aParentIsWrapperAnonBox) {
|
|
for (nsIFrame* f : aFrameList) {
|
|
f->SetParentIsWrapperAnonBox();
|
|
}
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AddFCItemsForAnonymousContent(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aFrame,
|
|
const nsTArray<nsIAnonymousContentCreator::ContentInfo>& aAnonymousItems,
|
|
FrameConstructionItemList& aItemsToConstruct,
|
|
const AutoFrameConstructionPageName&) {
|
|
for (const auto& info : aAnonymousItems) {
|
|
nsIContent* content = info.mContent;
|
|
// Gecko-styled nodes should have no pending restyle flags.
|
|
// Assert some things about this content
|
|
MOZ_ASSERT(!(content->GetFlags() &
|
|
(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME)),
|
|
"Should not be marked as needing frames");
|
|
MOZ_ASSERT(!content->GetPrimaryFrame(), "Should have no existing frame");
|
|
MOZ_ASSERT(!content->IsComment() && !content->IsProcessingInstruction(),
|
|
"Why is someone creating garbage anonymous content");
|
|
|
|
// Make sure we eagerly performed the servo cascade when the anonymous
|
|
// nodes were created.
|
|
MOZ_ASSERT(!content->IsElement() || content->AsElement()->HasServoData());
|
|
|
|
RefPtr<ComputedStyle> computedStyle = ResolveComputedStyle(content);
|
|
|
|
AddFrameConstructionItemsInternal(aState, content, aFrame, true,
|
|
computedStyle, {ItemFlag::AllowPageBreak},
|
|
aItemsToConstruct);
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ProcessChildren(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
ComputedStyle* aComputedStyle, nsContainerFrame* aFrame,
|
|
const bool aCanHaveGeneratedContent, nsFrameList& aFrameList,
|
|
const bool aAllowBlockStyles, nsIFrame* aPossiblyLeafFrame) {
|
|
MOZ_ASSERT(aFrame, "Must have parent frame here");
|
|
MOZ_ASSERT(aFrame->GetContentInsertionFrame() == aFrame,
|
|
"Parent frame in ProcessChildren should be its own "
|
|
"content insertion frame");
|
|
|
|
const uint32_t kMaxDepth = 2 * MAX_REFLOW_DEPTH;
|
|
static_assert(kMaxDepth <= UINT16_MAX, "mCurrentDepth type is too narrow");
|
|
AutoRestore<uint16_t> savedDepth(mCurrentDepth);
|
|
if (mCurrentDepth != UINT16_MAX) {
|
|
++mCurrentDepth;
|
|
}
|
|
|
|
if (!aPossiblyLeafFrame) {
|
|
aPossiblyLeafFrame = aFrame;
|
|
}
|
|
|
|
// XXXbz ideally, this would do all the pushing of various
|
|
// containing blocks as needed, so callers don't have to do it...
|
|
|
|
// Check that our parent frame is a block before allowing ::first-letter/line.
|
|
// E.g. <button style="display:grid"> should not allow it.
|
|
const bool allowFirstPseudos =
|
|
aAllowBlockStyles && aFrame->IsBlockFrameOrSubclass();
|
|
bool haveFirstLetterStyle = false, haveFirstLineStyle = false;
|
|
if (allowFirstPseudos) {
|
|
ShouldHaveSpecialBlockStyle(aContent, aComputedStyle, &haveFirstLetterStyle,
|
|
&haveFirstLineStyle);
|
|
}
|
|
|
|
AutoFrameConstructionItemList itemsToConstruct(this);
|
|
AutoFrameConstructionPageName pageNameTracker(aState, aFrame);
|
|
|
|
// If we have first-letter or first-line style then frames can get
|
|
// moved around so don't set these flags.
|
|
if (allowFirstPseudos && !haveFirstLetterStyle && !haveFirstLineStyle) {
|
|
itemsToConstruct.SetLineBoundaryAtStart(true);
|
|
itemsToConstruct.SetLineBoundaryAtEnd(true);
|
|
}
|
|
|
|
// Create any anonymous frames we need here.
|
|
AutoTArray<nsIAnonymousContentCreator::ContentInfo, 4> anonymousItems;
|
|
GetAnonymousContent(aContent, aPossiblyLeafFrame, anonymousItems);
|
|
#ifdef DEBUG
|
|
for (uint32_t i = 0; i < anonymousItems.Length(); ++i) {
|
|
MOZ_ASSERT(anonymousItems[i].mContent->IsRootOfNativeAnonymousSubtree(),
|
|
"Content should know it's an anonymous subtree");
|
|
}
|
|
#endif
|
|
AddFCItemsForAnonymousContent(aState, aFrame, anonymousItems,
|
|
itemsToConstruct, pageNameTracker);
|
|
|
|
nsBlockFrame* listItem = nullptr;
|
|
bool isOutsideMarker = false;
|
|
if (!aPossiblyLeafFrame->IsLeaf()) {
|
|
// :before/:after content should have the same style parent as normal kids.
|
|
//
|
|
// Note that we don't use this style for looking up things like special
|
|
// block styles because in some cases involving table pseudo-frames it has
|
|
// nothing to do with the parent frame's desired behavior.
|
|
auto* styleParentFrame =
|
|
nsIFrame::CorrectStyleParentFrame(aFrame, PseudoStyleType::NotPseudo);
|
|
ComputedStyle* computedStyle = styleParentFrame->Style();
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
if (computedStyle->StyleDisplay()->IsListItem() &&
|
|
(listItem = do_QueryFrame(aFrame)) &&
|
|
!styleParentFrame->IsFieldSetFrame()) {
|
|
isOutsideMarker = computedStyle->StyleList()->mListStylePosition ==
|
|
StyleListStylePosition::Outside;
|
|
ItemFlags extraFlags;
|
|
if (isOutsideMarker) {
|
|
extraFlags += ItemFlag::IsForOutsideMarker;
|
|
}
|
|
CreateGeneratedContentItem(aState, aFrame, *aContent->AsElement(),
|
|
*computedStyle, PseudoStyleType::marker,
|
|
itemsToConstruct, extraFlags);
|
|
}
|
|
// Probe for generated content before
|
|
CreateGeneratedContentItem(aState, aFrame, *aContent->AsElement(),
|
|
*computedStyle, PseudoStyleType::before,
|
|
itemsToConstruct);
|
|
}
|
|
|
|
const bool addChildItems = MOZ_LIKELY(mCurrentDepth < kMaxDepth);
|
|
if (!addChildItems) {
|
|
NS_WARNING("ProcessChildren max depth exceeded");
|
|
}
|
|
|
|
FlattenedChildIterator iter(aContent);
|
|
const InsertionPoint insertion(aFrame, aContent);
|
|
for (nsIContent* child = iter.GetNextChild(); child;
|
|
child = iter.GetNextChild()) {
|
|
MOZ_ASSERT(insertion.mContainer == GetInsertionPoint(child).mContainer,
|
|
"GetInsertionPoint should agree with us");
|
|
if (addChildItems) {
|
|
AddFrameConstructionItems(aState, child, iter.ShadowDOMInvolved(),
|
|
*computedStyle, insertion, itemsToConstruct);
|
|
} else {
|
|
ClearLazyBits(child, child->GetNextSibling());
|
|
}
|
|
}
|
|
itemsToConstruct.SetParentHasNoShadowDOM(!iter.ShadowDOMInvolved());
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
// Probe for generated content after
|
|
CreateGeneratedContentItem(aState, aFrame, *aContent->AsElement(),
|
|
*computedStyle, PseudoStyleType::after,
|
|
itemsToConstruct);
|
|
}
|
|
} else {
|
|
ClearLazyBits(aContent->GetFirstChild(), nullptr);
|
|
}
|
|
|
|
ConstructFramesFromItemList(aState, itemsToConstruct, aFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
aFrameList);
|
|
|
|
if (listItem) {
|
|
if (auto* markerFrame = nsLayoutUtils::GetMarkerFrame(aContent)) {
|
|
for (auto* childFrame : aFrameList) {
|
|
if (markerFrame == childFrame) {
|
|
if (isOutsideMarker) {
|
|
// SetMarkerFrameForListItem will add childFrame to the
|
|
// FrameChildListID::Bullet
|
|
aFrameList.RemoveFrame(childFrame);
|
|
auto* grandParent = listItem->GetParent()->GetParent();
|
|
if (listItem->Style()->GetPseudoType() ==
|
|
PseudoStyleType::columnContent &&
|
|
grandParent && grandParent->IsColumnSetWrapperFrame()) {
|
|
listItem = do_QueryFrame(grandParent);
|
|
MOZ_ASSERT(listItem,
|
|
"ColumnSetWrapperFrame is expected to be "
|
|
"a nsBlockFrame subclass");
|
|
childFrame->SetParent(listItem);
|
|
}
|
|
}
|
|
listItem->SetMarkerFrameForListItem(childFrame);
|
|
MOZ_ASSERT(listItem->HasOutsideMarker() == isOutsideMarker);
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = GetAccService()) {
|
|
auto* marker = markerFrame->GetContent();
|
|
accService->ContentRangeInserted(mPresShell, marker, nullptr);
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (haveFirstLetterStyle) {
|
|
WrapFramesInFirstLetterFrame(aFrame, aFrameList);
|
|
}
|
|
if (haveFirstLineStyle) {
|
|
WrapFramesInFirstLineFrame(aState, aContent, aFrame, nullptr, aFrameList);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Support for :first-line style
|
|
|
|
// Special routine to handle placing a list of frames into a block
|
|
// frame that has first-line style. The routine ensures that the first
|
|
// collection of inline frames end up in a first-line frame.
|
|
// NOTE: aState may have containing block information related to a
|
|
// different part of the frame tree than where the first line occurs.
|
|
// In particular aState may be set up for where ContentInserted or
|
|
// ContentAppended is inserting content, which may be some
|
|
// non-first-in-flow continuation of the block to which the first-line
|
|
// belongs. So this function needs to be careful about how it uses
|
|
// aState.
|
|
void nsCSSFrameConstructor::WrapFramesInFirstLineFrame(
|
|
nsFrameConstructorState& aState, nsIContent* aBlockContent,
|
|
nsContainerFrame* aBlockFrame, nsFirstLineFrame* aLineFrame,
|
|
nsFrameList& aFrameList) {
|
|
// Extract any initial inline frames from aFrameList so we can put them
|
|
// in the first-line.
|
|
nsFrameList firstLineChildren =
|
|
aFrameList.Split([](nsIFrame* f) { return !f->IsInlineOutside(); });
|
|
|
|
if (firstLineChildren.IsEmpty()) {
|
|
// Nothing is supposed to go into the first-line; nothing to do
|
|
return;
|
|
}
|
|
|
|
if (!aLineFrame) {
|
|
// Create line frame
|
|
ComputedStyle* parentStyle = nsIFrame::CorrectStyleParentFrame(
|
|
aBlockFrame, PseudoStyleType::firstLine)
|
|
->Style();
|
|
RefPtr<ComputedStyle> firstLineStyle =
|
|
GetFirstLineStyle(aBlockContent, parentStyle);
|
|
|
|
aLineFrame = NS_NewFirstLineFrame(mPresShell, firstLineStyle);
|
|
|
|
// Initialize the line frame
|
|
InitAndRestoreFrame(aState, aBlockContent, aBlockFrame, aLineFrame);
|
|
|
|
// The lineFrame will be the block's first child; the rest of the
|
|
// frame list (after lastInlineFrame) will be the second and
|
|
// subsequent children; insert lineFrame into aFrameList.
|
|
aFrameList.InsertFrame(nullptr, nullptr, aLineFrame);
|
|
|
|
NS_ASSERTION(aLineFrame->Style() == firstLineStyle,
|
|
"Bogus style on line frame");
|
|
}
|
|
|
|
// Give the inline frames to the lineFrame <b>after</b> reparenting them
|
|
ReparentFrames(this, aLineFrame, firstLineChildren, true);
|
|
if (aLineFrame->PrincipalChildList().IsEmpty() &&
|
|
aLineFrame->HasAnyStateBits(NS_FRAME_FIRST_REFLOW)) {
|
|
aLineFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(firstLineChildren));
|
|
} else {
|
|
AppendFrames(aLineFrame, FrameChildListID::Principal,
|
|
std::move(firstLineChildren));
|
|
}
|
|
}
|
|
|
|
// Special routine to handle appending a new frame to a block frame's
|
|
// child list. Takes care of placing the new frame into the right
|
|
// place when first-line style is present.
|
|
void nsCSSFrameConstructor::AppendFirstLineFrames(
|
|
nsFrameConstructorState& aState, nsIContent* aBlockContent,
|
|
nsContainerFrame* aBlockFrame, nsFrameList& aFrameList) {
|
|
// It's possible that aBlockFrame needs to have a first-line frame
|
|
// created because it doesn't currently have any children.
|
|
const nsFrameList& blockKids = aBlockFrame->PrincipalChildList();
|
|
if (blockKids.IsEmpty()) {
|
|
WrapFramesInFirstLineFrame(aState, aBlockContent, aBlockFrame, nullptr,
|
|
aFrameList);
|
|
return;
|
|
}
|
|
|
|
// Examine the last block child - if it's a first-line frame then
|
|
// appended frames need special treatment.
|
|
nsIFrame* lastBlockKid = blockKids.LastChild();
|
|
if (!lastBlockKid->IsLineFrame()) {
|
|
// No first-line frame at the end of the list, therefore there is
|
|
// an intervening block between any first-line frame the frames
|
|
// we are appending. Therefore, we don't need any special
|
|
// treatment of the appended frames.
|
|
return;
|
|
}
|
|
|
|
nsFirstLineFrame* lineFrame = static_cast<nsFirstLineFrame*>(lastBlockKid);
|
|
WrapFramesInFirstLineFrame(aState, aBlockContent, aBlockFrame, lineFrame,
|
|
aFrameList);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CheckForFirstLineInsertion(
|
|
nsIFrame* aParentFrame, nsFrameList& aFrameList) {
|
|
MOZ_ASSERT(aParentFrame->Style()->HasPseudoElementData(),
|
|
"Why were we called?");
|
|
|
|
if (aFrameList.IsEmpty()) {
|
|
// Happens often enough, with the caption stuff. No need to do the ancestor
|
|
// walk here.
|
|
return;
|
|
}
|
|
|
|
class RestyleManager* restyleManager = RestyleManager();
|
|
|
|
// Check whether there's a ::first-line on the path up from aParentFrame.
|
|
// Note that we can't stop until we've run out of ancestors with
|
|
// pseudo-element data, because the first-letter might be somewhere way up the
|
|
// tree; in particular it might be past our containing block.
|
|
nsIFrame* ancestor = aParentFrame;
|
|
while (ancestor) {
|
|
if (!ancestor->Style()->HasPseudoElementData()) {
|
|
// We know we won't find a ::first-line now.
|
|
return;
|
|
}
|
|
|
|
if (!ancestor->IsLineFrame()) {
|
|
ancestor = ancestor->GetParent();
|
|
continue;
|
|
}
|
|
|
|
if (!ancestor->Style()->IsPseudoElement()) {
|
|
// This is a continuation lineframe, not the first line; no need to do
|
|
// anything to the styles.
|
|
return;
|
|
}
|
|
|
|
// Fix up the styles of aFrameList for ::first-line.
|
|
for (nsIFrame* f : aFrameList) {
|
|
restyleManager->ReparentComputedStyleForFirstLine(f);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// First-letter support
|
|
|
|
// Determine how many characters in the text fragment apply to the
|
|
// first letter
|
|
static int32_t FirstLetterCount(const nsTextFragment* aFragment) {
|
|
int32_t count = 0;
|
|
int32_t firstLetterLength = 0;
|
|
|
|
const uint32_t n = aFragment->GetLength();
|
|
for (uint32_t i = 0; i < n; i++) {
|
|
const char16_t ch = aFragment->CharAt(i);
|
|
// FIXME: take content language into account when deciding whitespace.
|
|
if (dom::IsSpaceCharacter(ch)) {
|
|
if (firstLetterLength) {
|
|
break;
|
|
}
|
|
count++;
|
|
continue;
|
|
}
|
|
// XXX I18n
|
|
if ((ch == '\'') || (ch == '\"')) {
|
|
if (firstLetterLength) {
|
|
break;
|
|
}
|
|
// keep looping
|
|
firstLetterLength = 1;
|
|
} else {
|
|
count++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
static bool NeedFirstLetterContinuation(Text* aText) {
|
|
MOZ_ASSERT(aText, "null ptr");
|
|
int32_t flc = FirstLetterCount(&aText->TextFragment());
|
|
int32_t tl = aText->TextDataLength();
|
|
return flc < tl;
|
|
}
|
|
|
|
static bool IsFirstLetterContent(Text* aText) {
|
|
return aText->TextDataLength() && !aText->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
/**
|
|
* Create a letter frame, only make it a floating frame.
|
|
*/
|
|
nsFirstLetterFrame* nsCSSFrameConstructor::CreateFloatingLetterFrame(
|
|
nsFrameConstructorState& aState, Text* aTextContent, nsIFrame* aTextFrame,
|
|
nsContainerFrame* aParentFrame, ComputedStyle* aParentStyle,
|
|
ComputedStyle* aComputedStyle, nsFrameList& aResult) {
|
|
MOZ_ASSERT(aParentStyle);
|
|
|
|
nsFirstLetterFrame* letterFrame =
|
|
NS_NewFloatingFirstLetterFrame(mPresShell, aComputedStyle);
|
|
// We don't want to use a text content for a non-text frame (because we want
|
|
// its primary frame to be a text frame).
|
|
nsIContent* letterContent = aParentFrame->GetContent();
|
|
nsContainerFrame* containingBlock =
|
|
aState.GetGeometricParent(*aComputedStyle->StyleDisplay(), aParentFrame);
|
|
InitAndRestoreFrame(aState, letterContent, containingBlock, letterFrame);
|
|
|
|
// Init the text frame to refer to the letter frame.
|
|
//
|
|
// Make sure we get a proper style for it (the one passed in is for the letter
|
|
// frame and will have the float property set on it; the text frame shouldn't
|
|
// have that set).
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet();
|
|
RefPtr<ComputedStyle> textSC =
|
|
styleSet->ResolveStyleForText(aTextContent, aComputedStyle);
|
|
aTextFrame->SetComputedStyleWithoutNotification(textSC);
|
|
InitAndRestoreFrame(aState, aTextContent, letterFrame, aTextFrame);
|
|
|
|
// And then give the text frame to the letter frame
|
|
SetInitialSingleChild(letterFrame, aTextFrame);
|
|
|
|
// See if we will need to continue the text frame (does it contain
|
|
// more than just the first-letter text or not?) If it does, then we
|
|
// create (in advance) a continuation frame for it.
|
|
nsIFrame* nextTextFrame = nullptr;
|
|
if (NeedFirstLetterContinuation(aTextContent)) {
|
|
// Create continuation
|
|
nextTextFrame = CreateContinuingFrame(aTextFrame, aParentFrame);
|
|
RefPtr<ComputedStyle> newSC =
|
|
styleSet->ResolveStyleForText(aTextContent, aParentStyle);
|
|
nextTextFrame->SetComputedStyle(newSC);
|
|
}
|
|
|
|
NS_ASSERTION(aResult.IsEmpty(), "aResult should be an empty nsFrameList!");
|
|
// Put the new float before any of the floats in the block we're doing
|
|
// first-letter for, that is, before any floats whose parent is
|
|
// containingBlock.
|
|
nsIFrame* prevSibling = nullptr;
|
|
for (nsIFrame* f : aState.mFloatedList) {
|
|
if (f->GetParent() == containingBlock) {
|
|
break;
|
|
}
|
|
prevSibling = f;
|
|
}
|
|
|
|
aState.AddChild(letterFrame, aResult, letterContent, aParentFrame, false,
|
|
true, true, prevSibling);
|
|
|
|
if (nextTextFrame) {
|
|
aResult.AppendFrame(nullptr, nextTextFrame);
|
|
}
|
|
|
|
return letterFrame;
|
|
}
|
|
|
|
/**
|
|
* Create a new letter frame for aTextFrame. The letter frame will be
|
|
* a child of aParentFrame.
|
|
*/
|
|
void nsCSSFrameConstructor::CreateLetterFrame(
|
|
nsContainerFrame* aBlockFrame, nsContainerFrame* aBlockContinuation,
|
|
Text* aTextContent, nsContainerFrame* aParentFrame, nsFrameList& aResult) {
|
|
NS_ASSERTION(aBlockFrame->IsBlockFrameOrSubclass(), "Not a block frame?");
|
|
|
|
// Get a ComputedStyle for the first-letter-frame.
|
|
//
|
|
// Keep this in sync with nsBlockFrame::UpdatePseudoElementStyles.
|
|
nsIFrame* parentFrame = nsIFrame::CorrectStyleParentFrame(
|
|
aParentFrame, PseudoStyleType::firstLetter);
|
|
|
|
ComputedStyle* parentComputedStyle = parentFrame->Style();
|
|
ComputedStyle* parentComputedStyleIgnoringFirstLine = parentComputedStyle;
|
|
if (parentFrame->IsLineFrame()) {
|
|
parentComputedStyleIgnoringFirstLine =
|
|
nsIFrame::CorrectStyleParentFrame(aBlockFrame,
|
|
PseudoStyleType::firstLetter)
|
|
->Style();
|
|
}
|
|
|
|
// Use content from containing block so that we can actually
|
|
// find a matching style rule.
|
|
nsIContent* blockContent = aBlockFrame->GetContent();
|
|
|
|
// Create first-letter style rule, ignoring first line. If we already have a
|
|
// first-line we'll reparent the style below.
|
|
RefPtr<ComputedStyle> sc =
|
|
GetFirstLetterStyle(blockContent, parentComputedStyleIgnoringFirstLine);
|
|
|
|
if (sc) {
|
|
if (parentComputedStyleIgnoringFirstLine != parentComputedStyle) {
|
|
sc = mPresShell->StyleSet()->ReparentComputedStyle(
|
|
sc, parentComputedStyle, parentComputedStyle,
|
|
blockContent->AsElement());
|
|
}
|
|
|
|
RefPtr<ComputedStyle> textSC =
|
|
mPresShell->StyleSet()->ResolveStyleForText(aTextContent, sc);
|
|
|
|
// Create a new text frame (the original one will be discarded)
|
|
// pass a temporary stylecontext, the correct one will be set
|
|
// later. Start off by unsetting the primary frame for
|
|
// aTextContent, so it's no longer pointing to the to-be-destroyed
|
|
// frame.
|
|
// XXXbz it would be really nice to destroy the old frame _first_,
|
|
// then create the new one, so we could avoid this hack.
|
|
aTextContent->SetPrimaryFrame(nullptr);
|
|
nsIFrame* textFrame = NS_NewTextFrame(mPresShell, textSC);
|
|
|
|
NS_ASSERTION(aBlockContinuation == GetFloatContainingBlock(aParentFrame),
|
|
"Containing block is confused");
|
|
nsFrameConstructorState state(
|
|
mPresShell, GetAbsoluteContainingBlock(aParentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(aParentFrame, ABS_POS), aBlockContinuation);
|
|
|
|
// Create the right type of first-letter frame
|
|
const nsStyleDisplay* display = sc->StyleDisplay();
|
|
nsFirstLetterFrame* letterFrame;
|
|
if (display->IsFloatingStyle() && !aParentFrame->IsInSVGTextSubtree()) {
|
|
// Make a floating first-letter frame
|
|
letterFrame = CreateFloatingLetterFrame(state, aTextContent, textFrame,
|
|
aParentFrame, parentComputedStyle,
|
|
sc, aResult);
|
|
} else {
|
|
// Make an inflow first-letter frame
|
|
letterFrame = NS_NewFirstLetterFrame(mPresShell, sc);
|
|
|
|
// Initialize the first-letter-frame. We don't want to use a text
|
|
// content for a non-text frame (because we want its primary frame to
|
|
// be a text frame).
|
|
nsIContent* letterContent = aParentFrame->GetContent();
|
|
letterFrame->Init(letterContent, aParentFrame, nullptr);
|
|
|
|
InitAndRestoreFrame(state, aTextContent, letterFrame, textFrame);
|
|
|
|
SetInitialSingleChild(letterFrame, textFrame);
|
|
aResult.Clear();
|
|
aResult.AppendFrame(nullptr, letterFrame);
|
|
NS_ASSERTION(!aBlockFrame->GetPrevContinuation(),
|
|
"should have the first continuation here");
|
|
aBlockFrame->AddStateBits(NS_BLOCK_HAS_FIRST_LETTER_CHILD);
|
|
}
|
|
MOZ_ASSERT(
|
|
!aBlockFrame->GetPrevContinuation(),
|
|
"Setting up a first-letter frame on a non-first block continuation?");
|
|
auto parent =
|
|
static_cast<nsContainerFrame*>(aParentFrame->FirstContinuation());
|
|
if (MOZ_UNLIKELY(parent->IsLineFrame())) {
|
|
parent = static_cast<nsContainerFrame*>(
|
|
parent->GetParent()->FirstContinuation());
|
|
}
|
|
parent->SetHasFirstLetterChild();
|
|
aBlockFrame->SetProperty(nsContainerFrame::FirstLetterProperty(),
|
|
letterFrame);
|
|
aTextContent->SetPrimaryFrame(textFrame);
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
|
nsContainerFrame* aBlockFrame, nsFrameList& aBlockFrames) {
|
|
aBlockFrame->AddStateBits(NS_BLOCK_HAS_FIRST_LETTER_STYLE);
|
|
|
|
nsContainerFrame* parentFrame = nullptr;
|
|
nsIFrame* textFrame = nullptr;
|
|
nsIFrame* prevFrame = nullptr;
|
|
nsFrameList letterFrames;
|
|
bool stopLooking = false;
|
|
WrapFramesInFirstLetterFrame(
|
|
aBlockFrame, aBlockFrame, aBlockFrame, aBlockFrames.FirstChild(),
|
|
&parentFrame, &textFrame, &prevFrame, letterFrames, &stopLooking);
|
|
if (!parentFrame) {
|
|
return;
|
|
}
|
|
DestroyContext context(mPresShell);
|
|
if (parentFrame == aBlockFrame) {
|
|
// Take textFrame out of the block's frame list and substitute the
|
|
// letter frame(s) instead.
|
|
aBlockFrames.DestroyFrame(context, textFrame);
|
|
aBlockFrames.InsertFrames(nullptr, prevFrame, std::move(letterFrames));
|
|
} else {
|
|
// Take the old textFrame out of the inline parent's child list
|
|
RemoveFrame(context, FrameChildListID::Principal, textFrame);
|
|
|
|
// Insert in the letter frame(s)
|
|
parentFrame->InsertFrames(FrameChildListID::Principal, prevFrame, nullptr,
|
|
std::move(letterFrames));
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
|
nsContainerFrame* aBlockFrame, nsContainerFrame* aBlockContinuation,
|
|
nsContainerFrame* aParentFrame, nsIFrame* aParentFrameList,
|
|
nsContainerFrame** aModifiedParent, nsIFrame** aTextFrame,
|
|
nsIFrame** aPrevFrame, nsFrameList& aLetterFrames, bool* aStopLooking) {
|
|
nsIFrame* prevFrame = nullptr;
|
|
nsIFrame* frame = aParentFrameList;
|
|
|
|
// This loop attempts to implement "Finding the First Letter":
|
|
// https://drafts.csswg.org/css-pseudo-4/#application-in-css
|
|
// FIXME: we don't handle nested blocks correctly yet though (bug 214004)
|
|
while (frame) {
|
|
nsIFrame* nextFrame = frame->GetNextSibling();
|
|
|
|
// Skip all ::markers and placeholders.
|
|
if (frame->Style()->GetPseudoType() == PseudoStyleType::marker ||
|
|
frame->IsPlaceholderFrame()) {
|
|
prevFrame = frame;
|
|
frame = nextFrame;
|
|
continue;
|
|
}
|
|
LayoutFrameType frameType = frame->Type();
|
|
if (LayoutFrameType::Text == frameType) {
|
|
// Wrap up first-letter content in a letter frame
|
|
Text* textContent = frame->GetContent()->AsText();
|
|
if (IsFirstLetterContent(textContent)) {
|
|
// Create letter frame to wrap up the text
|
|
CreateLetterFrame(aBlockFrame, aBlockContinuation, textContent,
|
|
aParentFrame, aLetterFrames);
|
|
|
|
// Provide adjustment information for parent
|
|
*aModifiedParent = aParentFrame;
|
|
*aTextFrame = frame;
|
|
*aPrevFrame = prevFrame;
|
|
*aStopLooking = true;
|
|
return;
|
|
}
|
|
} else if (IsInlineFrame(frame) && frameType != LayoutFrameType::Br) {
|
|
nsIFrame* kids = frame->PrincipalChildList().FirstChild();
|
|
WrapFramesInFirstLetterFrame(aBlockFrame, aBlockContinuation,
|
|
static_cast<nsContainerFrame*>(frame), kids,
|
|
aModifiedParent, aTextFrame, aPrevFrame,
|
|
aLetterFrames, aStopLooking);
|
|
if (*aStopLooking) {
|
|
return;
|
|
}
|
|
} else {
|
|
// This will stop us looking to create more letter frames. For
|
|
// example, maybe the frame-type is "letterFrame" or
|
|
// "placeholderFrame". This keeps us from creating extra letter
|
|
// frames, and also prevents us from creating letter frames when
|
|
// the first real content child of a block is not text (e.g. an
|
|
// image, hr, etc.)
|
|
*aStopLooking = true;
|
|
break;
|
|
}
|
|
|
|
prevFrame = frame;
|
|
frame = nextFrame;
|
|
}
|
|
}
|
|
|
|
static nsIFrame* FindFirstLetterFrame(nsIFrame* aFrame,
|
|
FrameChildListID aListID) {
|
|
for (nsIFrame* f : aFrame->GetChildList(aListID)) {
|
|
if (f->IsLetterFrame()) {
|
|
return f;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
static void ClearHasFirstLetterChildFrom(nsContainerFrame* aParentFrame) {
|
|
MOZ_ASSERT(aParentFrame);
|
|
auto* parent =
|
|
static_cast<nsContainerFrame*>(aParentFrame->FirstContinuation());
|
|
if (MOZ_UNLIKELY(parent->IsLineFrame())) {
|
|
MOZ_ASSERT(!parent->HasFirstLetterChild());
|
|
parent = static_cast<nsContainerFrame*>(
|
|
parent->GetParent()->FirstContinuation());
|
|
}
|
|
MOZ_ASSERT(parent->HasFirstLetterChild());
|
|
parent->ClearHasFirstLetterChild();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::RemoveFloatingFirstLetterFrames(
|
|
PresShell* aPresShell, nsIFrame* aBlockFrame) {
|
|
// Look for the first letter frame on the FrameChildListID::Float, then
|
|
// FrameChildListID::PushedFloats.
|
|
nsIFrame* floatFrame =
|
|
::FindFirstLetterFrame(aBlockFrame, FrameChildListID::Float);
|
|
if (!floatFrame) {
|
|
floatFrame =
|
|
::FindFirstLetterFrame(aBlockFrame, FrameChildListID::PushedFloats);
|
|
if (!floatFrame) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Take the text frame away from the letter frame (so it isn't
|
|
// destroyed when we destroy the letter frame).
|
|
nsIFrame* textFrame = floatFrame->PrincipalChildList().FirstChild();
|
|
if (!textFrame) {
|
|
return;
|
|
}
|
|
|
|
// Discover the placeholder frame for the letter frame
|
|
nsPlaceholderFrame* placeholderFrame = floatFrame->GetPlaceholderFrame();
|
|
if (!placeholderFrame) {
|
|
// Somethings really wrong
|
|
return;
|
|
}
|
|
nsContainerFrame* parentFrame = placeholderFrame->GetParent();
|
|
if (!parentFrame) {
|
|
// Somethings really wrong
|
|
return;
|
|
}
|
|
|
|
ClearHasFirstLetterChildFrom(parentFrame);
|
|
|
|
// Create a new text frame with the right style that maps all of the content
|
|
// that was previously part of the letter frame (and probably continued
|
|
// elsewhere).
|
|
ComputedStyle* parentSC = parentFrame->Style();
|
|
nsIContent* textContent = textFrame->GetContent();
|
|
if (!textContent) {
|
|
return;
|
|
}
|
|
RefPtr<ComputedStyle> newSC =
|
|
aPresShell->StyleSet()->ResolveStyleForText(textContent, parentSC);
|
|
nsIFrame* newTextFrame = NS_NewTextFrame(aPresShell, newSC);
|
|
newTextFrame->Init(textContent, parentFrame, nullptr);
|
|
|
|
// Destroy the old text frame's continuations (the old text frame
|
|
// will be destroyed when its letter frame is destroyed).
|
|
nsIFrame* frameToDelete = textFrame->LastContinuation();
|
|
DestroyContext context(mPresShell);
|
|
while (frameToDelete != textFrame) {
|
|
nsIFrame* nextFrameToDelete = frameToDelete->GetPrevContinuation();
|
|
RemoveFrame(context, FrameChildListID::Principal, frameToDelete);
|
|
frameToDelete = nextFrameToDelete;
|
|
}
|
|
|
|
nsIFrame* prevSibling = placeholderFrame->GetPrevSibling();
|
|
|
|
// Now that everything is set...
|
|
#ifdef NOISY_FIRST_LETTER
|
|
printf(
|
|
"RemoveFloatingFirstLetterFrames: textContent=%p oldTextFrame=%p "
|
|
"newTextFrame=%p\n",
|
|
textContent.get(), textFrame, newTextFrame);
|
|
#endif
|
|
|
|
// Remove placeholder frame and the float
|
|
RemoveFrame(context, FrameChildListID::Principal, placeholderFrame);
|
|
|
|
// Now that the old frames are gone, we can start pointing to our
|
|
// new primary frame.
|
|
textContent->SetPrimaryFrame(newTextFrame);
|
|
|
|
// Wallpaper bug 822910.
|
|
bool offsetsNeedFixing = prevSibling && prevSibling->IsTextFrame();
|
|
if (offsetsNeedFixing) {
|
|
prevSibling->AddStateBits(TEXT_OFFSETS_NEED_FIXING);
|
|
}
|
|
|
|
// Insert text frame in its place
|
|
InsertFrames(parentFrame, FrameChildListID::Principal, prevSibling,
|
|
nsFrameList(newTextFrame, newTextFrame));
|
|
|
|
if (offsetsNeedFixing) {
|
|
prevSibling->RemoveStateBits(TEXT_OFFSETS_NEED_FIXING);
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::RemoveFirstLetterFrames(
|
|
PresShell* aPresShell, nsContainerFrame* aFrame,
|
|
nsContainerFrame* aBlockFrame, bool* aStopLooking) {
|
|
nsIFrame* prevSibling = nullptr;
|
|
nsIFrame* kid = aFrame->PrincipalChildList().FirstChild();
|
|
|
|
while (kid) {
|
|
if (kid->IsLetterFrame()) {
|
|
ClearHasFirstLetterChildFrom(aFrame);
|
|
nsIFrame* textFrame = kid->PrincipalChildList().FirstChild();
|
|
if (!textFrame) {
|
|
break;
|
|
}
|
|
|
|
// Create a new textframe
|
|
ComputedStyle* parentSC = aFrame->Style();
|
|
if (!parentSC) {
|
|
break;
|
|
}
|
|
nsIContent* textContent = textFrame->GetContent();
|
|
if (!textContent) {
|
|
break;
|
|
}
|
|
RefPtr<ComputedStyle> newSC =
|
|
aPresShell->StyleSet()->ResolveStyleForText(textContent, parentSC);
|
|
textFrame = NS_NewTextFrame(aPresShell, newSC);
|
|
textFrame->Init(textContent, aFrame, nullptr);
|
|
|
|
DestroyContext context(mPresShell);
|
|
|
|
// Next rip out the kid and replace it with the text frame
|
|
RemoveFrame(context, FrameChildListID::Principal, kid);
|
|
|
|
// Now that the old frames are gone, we can start pointing to our
|
|
// new primary frame.
|
|
textContent->SetPrimaryFrame(textFrame);
|
|
|
|
// Wallpaper bug 822910.
|
|
bool offsetsNeedFixing = prevSibling && prevSibling->IsTextFrame();
|
|
if (offsetsNeedFixing) {
|
|
prevSibling->AddStateBits(TEXT_OFFSETS_NEED_FIXING);
|
|
}
|
|
|
|
// Insert text frame in its place
|
|
InsertFrames(aFrame, FrameChildListID::Principal, prevSibling,
|
|
nsFrameList(textFrame, textFrame));
|
|
|
|
if (offsetsNeedFixing) {
|
|
prevSibling->RemoveStateBits(TEXT_OFFSETS_NEED_FIXING);
|
|
}
|
|
|
|
*aStopLooking = true;
|
|
NS_ASSERTION(!aBlockFrame->GetPrevContinuation(),
|
|
"should have the first continuation here");
|
|
aBlockFrame->RemoveStateBits(NS_BLOCK_HAS_FIRST_LETTER_CHILD);
|
|
break;
|
|
}
|
|
if (IsInlineFrame(kid)) {
|
|
nsContainerFrame* kidAsContainerFrame = do_QueryFrame(kid);
|
|
if (kidAsContainerFrame) {
|
|
// Look inside child inline frame for the letter frame.
|
|
RemoveFirstLetterFrames(aPresShell, kidAsContainerFrame, aBlockFrame,
|
|
aStopLooking);
|
|
if (*aStopLooking) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
prevSibling = kid;
|
|
kid = kid->GetNextSibling();
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::RemoveLetterFrames(PresShell* aPresShell,
|
|
nsContainerFrame* aBlockFrame) {
|
|
aBlockFrame =
|
|
static_cast<nsContainerFrame*>(aBlockFrame->FirstContinuation());
|
|
aBlockFrame->RemoveProperty(nsContainerFrame::FirstLetterProperty());
|
|
nsContainerFrame* continuation = aBlockFrame;
|
|
|
|
bool stopLooking = false;
|
|
do {
|
|
RemoveFloatingFirstLetterFrames(aPresShell, continuation);
|
|
RemoveFirstLetterFrames(aPresShell, continuation, aBlockFrame,
|
|
&stopLooking);
|
|
if (stopLooking) {
|
|
break;
|
|
}
|
|
continuation =
|
|
static_cast<nsContainerFrame*>(continuation->GetNextContinuation());
|
|
} while (continuation);
|
|
}
|
|
|
|
// Fixup the letter frame situation for the given block
|
|
void nsCSSFrameConstructor::RecoverLetterFrames(nsContainerFrame* aBlockFrame) {
|
|
aBlockFrame =
|
|
static_cast<nsContainerFrame*>(aBlockFrame->FirstContinuation());
|
|
nsContainerFrame* continuation = aBlockFrame;
|
|
|
|
nsContainerFrame* parentFrame = nullptr;
|
|
nsIFrame* textFrame = nullptr;
|
|
nsIFrame* prevFrame = nullptr;
|
|
nsFrameList letterFrames;
|
|
bool stopLooking = false;
|
|
do {
|
|
// XXX shouldn't this bit be set already (bug 408493), assert instead?
|
|
continuation->AddStateBits(NS_BLOCK_HAS_FIRST_LETTER_STYLE);
|
|
WrapFramesInFirstLetterFrame(
|
|
aBlockFrame, continuation, continuation,
|
|
continuation->PrincipalChildList().FirstChild(), &parentFrame,
|
|
&textFrame, &prevFrame, letterFrames, &stopLooking);
|
|
if (stopLooking) {
|
|
break;
|
|
}
|
|
continuation =
|
|
static_cast<nsContainerFrame*>(continuation->GetNextContinuation());
|
|
} while (continuation);
|
|
|
|
if (!parentFrame) {
|
|
return;
|
|
}
|
|
// Take the old textFrame out of the parent's child list
|
|
DestroyContext context(mPresShell);
|
|
RemoveFrame(context, FrameChildListID::Principal, textFrame);
|
|
|
|
// Insert in the letter frame(s)
|
|
parentFrame->InsertFrames(FrameChildListID::Principal, prevFrame, nullptr,
|
|
std::move(letterFrames));
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void nsCSSFrameConstructor::ConstructBlock(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, nsContainerFrame* aContentParentFrame,
|
|
ComputedStyle* aComputedStyle, nsContainerFrame** aNewFrame,
|
|
nsFrameList& aFrameList, nsIFrame* aPositionedFrameForAbsPosContainer) {
|
|
// clang-format off
|
|
//
|
|
// If a block frame is in a multi-column subtree, its children may need to
|
|
// be chopped into runs of blocks containing column-spans and runs of
|
|
// blocks containing no column-spans. Each run containing column-spans
|
|
// will be wrapped by an anonymous block. See CreateColumnSpanSiblings() for
|
|
// the implementation.
|
|
//
|
|
// If a block frame is a multi-column container, its children will need to
|
|
// be processed as above. Moreover, it creates a ColumnSetWrapperFrame as
|
|
// its outermost frame, and its children which have no
|
|
// -moz-column-span-wrapper pseudo will be wrapped in ColumnSetFrames. See
|
|
// FinishBuildingColumns() for the implementation.
|
|
//
|
|
// The multi-column subtree maintains the following invariants:
|
|
//
|
|
// 1) All the frames have the frame state bit
|
|
// NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR set, except for top-level
|
|
// ColumnSetWrapperFrame and those children in the column-span subtrees.
|
|
//
|
|
// 2) The first and last frame under ColumnSetWrapperFrame are always
|
|
// ColumnSetFrame.
|
|
//
|
|
// 3) ColumnSetFrames are linked together as continuations.
|
|
//
|
|
// 4) Those column-span wrappers are *not* linked together with themselves nor
|
|
// with the original block frame. The continuation chain consists of the
|
|
// original block frame and the original block's continuations wrapping
|
|
// non-column-spans.
|
|
//
|
|
// For example, this HTML
|
|
// <div id="x" style="column-count: 2;">
|
|
// <div style="column-span: all">a</div>
|
|
// <div id="y">
|
|
// b
|
|
// <div style="column-span: all">c</div>
|
|
// <div style="column-span: all">d</div>
|
|
// e
|
|
// </div>
|
|
// </div>
|
|
// <div style="column-span: all">f</div>
|
|
//
|
|
// yields the following frame tree.
|
|
//
|
|
// A) ColumnSetWrapper (original style)
|
|
// B) ColumnSet (-moz-column-set) <-- always created by BeginBuildingColumns
|
|
// C) Block (-moz-column-content)
|
|
// D) Block (-moz-column-span-wrapper, created by x)
|
|
// E) Block (div)
|
|
// F) Text ("a")
|
|
// G) ColumnSet (-moz-column-set)
|
|
// H) Block (-moz-column-content, created by x)
|
|
// I) Block (div, y)
|
|
// J) Text ("b")
|
|
// K) Block (-moz-column-span-wrapper, created by x)
|
|
// L) Block (-moz-column-span-wrapper, created by y)
|
|
// M) Block (div, new BFC)
|
|
// N) Text ("c")
|
|
// O) Block (div, new BFC)
|
|
// P) Text ("d")
|
|
// Q) ColumnSet (-moz-column-set)
|
|
// R) Block (-moz-column-content, created by x)
|
|
// S) Block (div, y)
|
|
// T) Text ("e")
|
|
// U) Block (div, new BFC) <-- not in multi-column hierarchy
|
|
// V) Text ("f")
|
|
//
|
|
// ColumnSet linkage described in 3): B -> G -> Q
|
|
//
|
|
// Block linkage described in 4): C -> H -> R and I -> S
|
|
//
|
|
// clang-format on
|
|
|
|
nsBlockFrame* blockFrame = do_QueryFrame(*aNewFrame);
|
|
MOZ_ASSERT(blockFrame && blockFrame->IsBlockFrame(), "not a block frame?");
|
|
|
|
// Create column hierarchy if necessary.
|
|
const bool needsColumn =
|
|
aComputedStyle->StyleColumn()->IsColumnContainerStyle();
|
|
if (needsColumn) {
|
|
*aNewFrame = BeginBuildingColumns(aState, aContent, aParentFrame,
|
|
blockFrame, aComputedStyle);
|
|
|
|
if (aPositionedFrameForAbsPosContainer == blockFrame) {
|
|
aPositionedFrameForAbsPosContainer = *aNewFrame;
|
|
}
|
|
} else {
|
|
// No need to create column hierarchy. Initialize block frame.
|
|
blockFrame->SetComputedStyleWithoutNotification(aComputedStyle);
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, blockFrame);
|
|
}
|
|
|
|
aState.AddChild(*aNewFrame, aFrameList, aContent,
|
|
aContentParentFrame ? aContentParentFrame : aParentFrame);
|
|
if (!mRootElementFrame) {
|
|
mRootElementFrame = *aNewFrame;
|
|
}
|
|
|
|
// We should make the outer frame be the absolute containing block,
|
|
// if one is required. We have to do this because absolute
|
|
// positioning must be computed with respect to the CSS dimensions
|
|
// of the element, which are the dimensions of the outer block. But
|
|
// we can't really do that because only blocks can have absolute
|
|
// children. So use the block and try to compensate with hacks
|
|
// in nsBlockFrame::CalculateContainingBlockSizeForAbsolutes.
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
(*aNewFrame)->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (aPositionedFrameForAbsPosContainer) {
|
|
aState.PushAbsoluteContainingBlock(
|
|
*aNewFrame, aPositionedFrameForAbsPosContainer, absoluteSaveState);
|
|
}
|
|
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.MaybePushFloatContainingBlock(blockFrame, floatSaveState);
|
|
|
|
if (aParentFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR) &&
|
|
!ShouldSuppressColumnSpanDescendants(aParentFrame)) {
|
|
blockFrame->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
}
|
|
|
|
// Process the child content
|
|
nsFrameList childList;
|
|
ProcessChildren(aState, aContent, aComputedStyle, blockFrame, true, childList,
|
|
true);
|
|
|
|
if (!MayNeedToCreateColumnSpanSiblings(blockFrame, childList)) {
|
|
// No need to create column-span siblings.
|
|
blockFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
return;
|
|
}
|
|
|
|
// Extract any initial non-column-span kids, and put them in block frame's
|
|
// child list.
|
|
nsFrameList initialNonColumnSpanKids =
|
|
childList.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
blockFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(initialNonColumnSpanKids));
|
|
|
|
if (childList.IsEmpty()) {
|
|
// No more kids to process (there weren't any column-span kids).
|
|
return;
|
|
}
|
|
|
|
nsFrameList columnSpanSiblings = CreateColumnSpanSiblings(
|
|
aState, blockFrame, childList,
|
|
// If we're constructing a column container, pass nullptr as
|
|
// aPositionedFrame to forbid reparenting absolute/fixed positioned frames
|
|
// to column contents or column-span wrappers.
|
|
needsColumn ? nullptr : aPositionedFrameForAbsPosContainer);
|
|
|
|
if (needsColumn) {
|
|
// We're constructing a column container; need to finish building it.
|
|
FinishBuildingColumns(aState, *aNewFrame, blockFrame, columnSpanSiblings);
|
|
} else {
|
|
// We're constructing a normal block which has column-span children in a
|
|
// column hierarchy such as "x" in the following example.
|
|
//
|
|
// <div style="column-count: 2">
|
|
// <div id="x">
|
|
// <div>normal child</div>
|
|
// <div style="column-span">spanner</div>
|
|
// </div>
|
|
// </div>
|
|
aFrameList.AppendFrames(nullptr, std::move(columnSpanSiblings));
|
|
}
|
|
|
|
MOZ_ASSERT(columnSpanSiblings.IsEmpty(),
|
|
"The column-span siblings should be moved to the proper place!");
|
|
}
|
|
|
|
nsBlockFrame* nsCSSFrameConstructor::BeginBuildingColumns(
|
|
nsFrameConstructorState& aState, nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame, nsContainerFrame* aColumnContent,
|
|
ComputedStyle* aComputedStyle) {
|
|
MOZ_ASSERT(aColumnContent->IsBlockFrame(),
|
|
"aColumnContent should be a block frame.");
|
|
MOZ_ASSERT(aComputedStyle->StyleColumn()->IsColumnContainerStyle(),
|
|
"No need to build a column hierarchy!");
|
|
|
|
// The initial column hierarchy looks like this:
|
|
//
|
|
// ColumnSetWrapper (original style)
|
|
// ColumnSet (-moz-column-set)
|
|
// Block (-moz-column-content)
|
|
//
|
|
nsBlockFrame* columnSetWrapper = NS_NewColumnSetWrapperFrame(
|
|
mPresShell, aComputedStyle, nsFrameState(NS_FRAME_OWNS_ANON_BOXES));
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, columnSetWrapper);
|
|
if (aParentFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR) &&
|
|
!ShouldSuppressColumnSpanDescendants(aParentFrame)) {
|
|
columnSetWrapper->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
}
|
|
|
|
AutoFrameConstructionPageName pageNameTracker(aState, columnSetWrapper);
|
|
RefPtr<ComputedStyle> columnSetStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::columnSet, aComputedStyle);
|
|
nsContainerFrame* columnSet = NS_NewColumnSetFrame(
|
|
mPresShell, columnSetStyle, nsFrameState(NS_FRAME_OWNS_ANON_BOXES));
|
|
InitAndRestoreFrame(aState, aContent, columnSetWrapper, columnSet);
|
|
columnSet->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
|
|
RefPtr<ComputedStyle> blockStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::columnContent, columnSetStyle);
|
|
aColumnContent->SetComputedStyleWithoutNotification(blockStyle);
|
|
InitAndRestoreFrame(aState, aContent, columnSet, aColumnContent);
|
|
aColumnContent->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
|
|
// Set up the parent-child chain.
|
|
SetInitialSingleChild(columnSetWrapper, columnSet);
|
|
SetInitialSingleChild(columnSet, aColumnContent);
|
|
|
|
return columnSetWrapper;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FinishBuildingColumns(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aColumnSetWrapper,
|
|
nsContainerFrame* aColumnContent, nsFrameList& aColumnContentSiblings) {
|
|
nsContainerFrame* prevColumnSet = aColumnContent->GetParent();
|
|
|
|
MOZ_ASSERT(prevColumnSet->IsColumnSetFrame() &&
|
|
prevColumnSet->GetParent() == aColumnSetWrapper,
|
|
"Should have established column hierarchy!");
|
|
|
|
// Tag the first ColumnSet to have column-span siblings so that the bit can
|
|
// propagate to all the continuations. We don't want the last ColumnSet to
|
|
// have this bit, so we will unset the bit for it at the end of this function.
|
|
prevColumnSet->SetHasColumnSpanSiblings(true);
|
|
|
|
nsFrameList finalList;
|
|
while (aColumnContentSiblings.NotEmpty()) {
|
|
nsIFrame* f = aColumnContentSiblings.RemoveFirstChild();
|
|
if (f->IsColumnSpan()) {
|
|
// Do nothing for column-span wrappers. Just move it to the final
|
|
// items.
|
|
finalList.AppendFrame(aColumnSetWrapper, f);
|
|
} else {
|
|
auto* continuingColumnSet = static_cast<nsContainerFrame*>(
|
|
CreateContinuingFrame(prevColumnSet, aColumnSetWrapper, false));
|
|
MOZ_ASSERT(continuingColumnSet->HasColumnSpanSiblings(),
|
|
"The bit should propagate to the next continuation!");
|
|
|
|
f->SetParent(continuingColumnSet);
|
|
SetInitialSingleChild(continuingColumnSet, f);
|
|
finalList.AppendFrame(aColumnSetWrapper, continuingColumnSet);
|
|
prevColumnSet = continuingColumnSet;
|
|
}
|
|
}
|
|
|
|
// Unset the bit because the last ColumnSet has no column-span siblings.
|
|
prevColumnSet->SetHasColumnSpanSiblings(false);
|
|
|
|
aColumnSetWrapper->AppendFrames(FrameChildListID::Principal,
|
|
std::move(finalList));
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::MayNeedToCreateColumnSpanSiblings(
|
|
nsContainerFrame* aBlockFrame, const nsFrameList& aChildList) {
|
|
if (!aBlockFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
// The block frame isn't in a multi-column block formatting context.
|
|
return false;
|
|
}
|
|
|
|
if (ShouldSuppressColumnSpanDescendants(aBlockFrame)) {
|
|
// No need to create column-span siblings for a frame that suppresses them.
|
|
return false;
|
|
}
|
|
|
|
if (aChildList.IsEmpty()) {
|
|
// No child needs to be processed.
|
|
return false;
|
|
}
|
|
|
|
// Need to actually look into the child list.
|
|
return true;
|
|
}
|
|
|
|
nsFrameList nsCSSFrameConstructor::CreateColumnSpanSiblings(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aInitialBlock,
|
|
nsFrameList& aChildList, nsIFrame* aPositionedFrame) {
|
|
MOZ_ASSERT(aInitialBlock->IsBlockFrameOrSubclass());
|
|
MOZ_ASSERT(!aPositionedFrame || aPositionedFrame->IsAbsPosContainingBlock());
|
|
|
|
nsIContent* const content = aInitialBlock->GetContent();
|
|
nsContainerFrame* const parentFrame = aInitialBlock->GetParent();
|
|
const bool isInitialBlockFloatCB = aInitialBlock->IsFloatContainingBlock();
|
|
|
|
nsFrameList siblings;
|
|
nsContainerFrame* lastNonColumnSpanWrapper = aInitialBlock;
|
|
|
|
// Tag the first non-column-span wrapper to have column-span siblings so that
|
|
// the bit can propagate to all the continuations. We don't want the last
|
|
// wrapper to have this bit, so we will unset the bit for it at the end of
|
|
// this function.
|
|
lastNonColumnSpanWrapper->SetHasColumnSpanSiblings(true);
|
|
do {
|
|
MOZ_ASSERT(aChildList.NotEmpty(), "Why call this if child list is empty?");
|
|
MOZ_ASSERT(aChildList.FirstChild()->IsColumnSpan(),
|
|
"Must have the child starting with column-span!");
|
|
|
|
// Grab the consecutive column-span kids, and reparent them into a
|
|
// block frame.
|
|
RefPtr<ComputedStyle> columnSpanWrapperStyle =
|
|
mPresShell->StyleSet()->ResolveNonInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::columnSpanWrapper);
|
|
nsBlockFrame* columnSpanWrapper =
|
|
NS_NewBlockFrame(mPresShell, columnSpanWrapperStyle);
|
|
InitAndRestoreFrame(aState, content, parentFrame, columnSpanWrapper, false);
|
|
columnSpanWrapper->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR |
|
|
NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
|
|
nsFrameList columnSpanKids =
|
|
aChildList.Split([](nsIFrame* f) { return !f->IsColumnSpan(); });
|
|
columnSpanKids.ApplySetParent(columnSpanWrapper);
|
|
columnSpanWrapper->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(columnSpanKids));
|
|
if (aPositionedFrame) {
|
|
aState.ReparentAbsoluteItems(columnSpanWrapper);
|
|
}
|
|
|
|
siblings.AppendFrame(nullptr, columnSpanWrapper);
|
|
|
|
// Grab the consecutive non-column-span kids, and reparent them into a new
|
|
// continuation of the last non-column-span wrapper frame.
|
|
auto* nonColumnSpanWrapper = static_cast<nsContainerFrame*>(
|
|
CreateContinuingFrame(lastNonColumnSpanWrapper, parentFrame, false));
|
|
nonColumnSpanWrapper->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR |
|
|
NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
MOZ_ASSERT(nonColumnSpanWrapper->HasColumnSpanSiblings(),
|
|
"The bit should propagate to the next continuation!");
|
|
|
|
if (aChildList.NotEmpty()) {
|
|
nsFrameList nonColumnSpanKids =
|
|
aChildList.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
|
|
nonColumnSpanKids.ApplySetParent(nonColumnSpanWrapper);
|
|
nonColumnSpanWrapper->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(nonColumnSpanKids));
|
|
if (aPositionedFrame) {
|
|
aState.ReparentAbsoluteItems(nonColumnSpanWrapper);
|
|
}
|
|
if (isInitialBlockFloatCB) {
|
|
aState.ReparentFloats(nonColumnSpanWrapper);
|
|
}
|
|
}
|
|
|
|
siblings.AppendFrame(nullptr, nonColumnSpanWrapper);
|
|
|
|
lastNonColumnSpanWrapper = nonColumnSpanWrapper;
|
|
} while (aChildList.NotEmpty());
|
|
|
|
// Unset the bit because the last non-column-span wrapper has no column-span
|
|
// siblings.
|
|
lastNonColumnSpanWrapper->SetHasColumnSpanSiblings(false);
|
|
|
|
return siblings;
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::MaybeRecreateForColumnSpan(
|
|
nsFrameConstructorState& aState, nsContainerFrame* aParentFrame,
|
|
nsFrameList& aFrameList, nsIFrame* aPrevSibling) {
|
|
if (!aParentFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
return false;
|
|
}
|
|
|
|
if (aFrameList.IsEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
MOZ_ASSERT(!IsFramePartOfIBSplit(aParentFrame),
|
|
"We should have wiped aParentFrame in WipeContainingBlock if it's "
|
|
"part of IB split!");
|
|
|
|
nsIFrame* nextSibling = ::GetInsertNextSibling(aParentFrame, aPrevSibling);
|
|
if (!nextSibling && IsLastContinuationForColumnContent(aParentFrame)) {
|
|
// We are appending a list of frames to the last continuation of a
|
|
// ::-moz-column-content. This is the case where we can fix the frame tree
|
|
// instead of reframing the containing block. Return false and let
|
|
// AppendFramesToParent() deal with this.
|
|
return false;
|
|
}
|
|
|
|
auto HasColumnSpan = [](const nsFrameList& aList) {
|
|
for (nsIFrame* f : aList) {
|
|
if (f->IsColumnSpan()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
if (HasColumnSpan(aFrameList)) {
|
|
// If any frame in the frame list has "column-span:all" style, i.e. a
|
|
// -moz-column-span-wrapper frame, we need to reframe the multi-column
|
|
// containing block.
|
|
//
|
|
// We can only be here if none of the new inserted nsIContent* nodes (via
|
|
// ContentAppended or ContentRangeInserted) have column-span:all style, yet
|
|
// some of them have column-span:all descendants. Sadly, there's no way to
|
|
// detect this by checking FrameConstructionItems in WipeContainingBlock().
|
|
// Otherwise, we would have already wiped the multi-column containing block.
|
|
PROFILER_MARKER("Reframe multi-column after constructing frame list",
|
|
LAYOUT, {}, Tracing, "Layout");
|
|
|
|
// aFrameList can contain placeholder frames. In order to destroy their
|
|
// associated out-of-flow frames properly, we need to manually flush all the
|
|
// out-of-flow frames in aState to their container frames.
|
|
aState.ProcessFrameInsertionsForAllLists();
|
|
DestroyContext context(mPresShell);
|
|
aFrameList.DestroyFrames(context);
|
|
RecreateFramesForContent(
|
|
GetMultiColumnContainingBlockFor(aParentFrame)->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
nsIFrame* nsCSSFrameConstructor::ConstructInline(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame, const nsStyleDisplay* aDisplay,
|
|
nsFrameList& aFrameList) {
|
|
// If an inline frame has non-inline kids, then we chop up the child list
|
|
// into runs of blocks and runs of inlines, create anonymous block frames to
|
|
// contain the runs of blocks, inline frames with our style for the runs of
|
|
// inlines, and put all these frames, in order, into aFrameList.
|
|
//
|
|
// When there are column-span blocks in a run of blocks, instead of creating
|
|
// an anonymous block to wrap them, we create multiple anonymous blocks,
|
|
// wrapping runs of non-column-spans and runs of column-spans.
|
|
//
|
|
// We return the the first one. The whole setup is called an {ib}
|
|
// split; in what follows "frames in the split" refers to the anonymous blocks
|
|
// and inlines that contain our children.
|
|
//
|
|
// {ib} splits maintain the following invariants:
|
|
// 1) All frames in the split have the NS_FRAME_PART_OF_IBSPLIT bit
|
|
// set.
|
|
//
|
|
// 2) Each frame in the split has the nsIFrame::IBSplitSibling
|
|
// property pointing to the next frame in the split, except for the last
|
|
// one, which does not have it set.
|
|
//
|
|
// 3) Each frame in the split has the nsIFrame::IBSplitPrevSibling
|
|
// property pointing to the previous frame in the split, except for the
|
|
// first one, which does not have it set.
|
|
//
|
|
// 4) The first and last frame in the split are always inlines.
|
|
//
|
|
// 5) The frames wrapping runs of non-column-spans are linked together as
|
|
// continuations. The frames wrapping runs of column-spans are *not*
|
|
// linked with each other nor with other non-column-span wrappers.
|
|
//
|
|
// 6) The first and last frame in the chains of blocks are always wrapping
|
|
// non-column-spans. Both of them are created even if they're empty.
|
|
//
|
|
// An invariant that is NOT maintained is that the wrappers are actually
|
|
// linked via GetNextSibling linkage. A simple example is an inline
|
|
// containing an inline that contains a block. The three parts of the inner
|
|
// inline end up with three different parents.
|
|
//
|
|
// For example, this HTML:
|
|
// <span>
|
|
// <div>a</div>
|
|
// <span>
|
|
// b
|
|
// <div>c</div>
|
|
// </span>
|
|
// d
|
|
// <div>e</div>
|
|
// f
|
|
// </span>
|
|
// Gives the following frame tree:
|
|
//
|
|
// Inline (outer span)
|
|
// Block (anonymous, outer span)
|
|
// Block (div)
|
|
// Text("a")
|
|
// Inline (outer span)
|
|
// Inline (inner span)
|
|
// Text("b")
|
|
// Block (anonymous, outer span)
|
|
// Block (anonymous, inner span)
|
|
// Block (div)
|
|
// Text("c")
|
|
// Inline (outer span)
|
|
// Inline (inner span)
|
|
// Text("d")
|
|
// Block (anonymous, outer span)
|
|
// Block (div)
|
|
// Text("e")
|
|
// Inline (outer span)
|
|
// Text("f")
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
ComputedStyle* const computedStyle = aItem.mComputedStyle;
|
|
|
|
nsInlineFrame* newFrame = NS_NewInlineFrame(mPresShell, computedStyle);
|
|
|
|
// Initialize the frame
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newFrame);
|
|
|
|
// definition cannot be inside next block because the object's destructor is
|
|
// significant. this is part of the fix for bug 42372
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
bool isAbsPosCB = newFrame->IsAbsPosContainingBlock();
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (isAbsPosCB) {
|
|
// Relatively positioned frames becomes a container for child
|
|
// frames that are positioned
|
|
aState.PushAbsoluteContainingBlock(newFrame, newFrame, absoluteSaveState);
|
|
}
|
|
|
|
if (aParentFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR) &&
|
|
!ShouldSuppressColumnSpanDescendants(aParentFrame)) {
|
|
newFrame->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
}
|
|
|
|
// Process the child content
|
|
nsFrameList childList;
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems, newFrame,
|
|
/* aParentIsWrapperAnonBox = */ false, childList);
|
|
|
|
nsIFrame* firstBlock = nullptr;
|
|
if (!aItem.mIsAllInline) {
|
|
for (nsIFrame* f : childList) {
|
|
if (f->IsBlockOutside()) {
|
|
firstBlock = f;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aItem.mIsAllInline || !firstBlock) {
|
|
// This part is easy. We either already know we have no non-inline kids,
|
|
// or haven't found any when constructing actual frames (the latter can
|
|
// happen only if out-of-flows that we thought had no containing block
|
|
// acquired one when ancestor inline frames and {ib} splits got
|
|
// constructed). Just put all the kids into the single inline frame and
|
|
// bail.
|
|
newFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(childList));
|
|
aState.AddChild(newFrame, aFrameList, content, aParentFrame);
|
|
return newFrame;
|
|
}
|
|
|
|
// This inline frame contains several types of children. Therefore this frame
|
|
// has to be chopped into several pieces, as described above.
|
|
|
|
// Grab the first inline's kids
|
|
nsFrameList firstInlineKids = childList.TakeFramesBefore(firstBlock);
|
|
newFrame->SetInitialChildList(FrameChildListID::Principal,
|
|
std::move(firstInlineKids));
|
|
|
|
aFrameList.AppendFrame(nullptr, newFrame);
|
|
|
|
newFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
CreateIBSiblings(aState, newFrame, isAbsPosCB, childList, aFrameList);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CreateIBSiblings(nsFrameConstructorState& aState,
|
|
nsContainerFrame* aInitialInline,
|
|
bool aIsAbsPosCB,
|
|
nsFrameList& aChildList,
|
|
nsFrameList& aSiblings) {
|
|
MOZ_ASSERT(aIsAbsPosCB == aInitialInline->IsAbsPosContainingBlock());
|
|
|
|
nsIContent* content = aInitialInline->GetContent();
|
|
ComputedStyle* computedStyle = aInitialInline->Style();
|
|
nsContainerFrame* parentFrame = aInitialInline->GetParent();
|
|
|
|
// Resolve the right style for our anonymous blocks.
|
|
//
|
|
// The distinction in styles is needed because of CSS 2.1, section
|
|
// 9.2.1.1, which says:
|
|
//
|
|
// When such an inline box is affected by relative positioning, any
|
|
// resulting translation also affects the block-level box contained
|
|
// in the inline box.
|
|
RefPtr<ComputedStyle> blockSC =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(
|
|
PseudoStyleType::mozBlockInsideInlineWrapper, computedStyle);
|
|
|
|
nsContainerFrame* lastNewInline =
|
|
static_cast<nsContainerFrame*>(aInitialInline->FirstContinuation());
|
|
do {
|
|
// On entry to this loop aChildList is not empty and the first frame in it
|
|
// is block-level.
|
|
MOZ_ASSERT(aChildList.NotEmpty(), "Should have child items");
|
|
MOZ_ASSERT(aChildList.FirstChild()->IsBlockOutside(),
|
|
"Must have list starting with block");
|
|
|
|
// The initial run of blocks belongs to an anonymous block that we create
|
|
// right now. The anonymous block will be the parent of these block
|
|
// children of the inline.
|
|
nsBlockFrame* blockFrame = NS_NewBlockFrame(mPresShell, blockSC);
|
|
InitAndRestoreFrame(aState, content, parentFrame, blockFrame, false);
|
|
if (aInitialInline->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
blockFrame->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
}
|
|
|
|
// Find the first non-block child which defines the end of our block kids
|
|
// and the start of our next inline's kids
|
|
nsFrameList blockKids =
|
|
aChildList.Split([](nsIFrame* f) { return !f->IsBlockOutside(); });
|
|
|
|
if (!aInitialInline->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
MoveChildrenTo(aInitialInline, blockFrame, blockKids);
|
|
|
|
SetFrameIsIBSplit(lastNewInline, blockFrame);
|
|
aSiblings.AppendFrame(nullptr, blockFrame);
|
|
} else {
|
|
// Extract any initial non-column-span frames, and put them in
|
|
// blockFrame's child list.
|
|
nsFrameList initialNonColumnSpanKids =
|
|
blockKids.Split([](nsIFrame* f) { return f->IsColumnSpan(); });
|
|
MoveChildrenTo(aInitialInline, blockFrame, initialNonColumnSpanKids);
|
|
|
|
SetFrameIsIBSplit(lastNewInline, blockFrame);
|
|
aSiblings.AppendFrame(nullptr, blockFrame);
|
|
|
|
if (blockKids.NotEmpty()) {
|
|
// Although SetFrameIsIBSplit() will add NS_FRAME_PART_OF_IBSPLIT for
|
|
// blockFrame later, we manually add the bit earlier here to make all
|
|
// the continuations of blockFrame created in
|
|
// CreateColumnSpanSiblings(), i.e. non-column-span wrappers, have the
|
|
// bit via nsIFrame::Init().
|
|
blockFrame->AddStateBits(NS_FRAME_PART_OF_IBSPLIT);
|
|
|
|
nsFrameList columnSpanSiblings =
|
|
CreateColumnSpanSiblings(aState, blockFrame, blockKids,
|
|
aIsAbsPosCB ? aInitialInline : nullptr);
|
|
aSiblings.AppendFrames(nullptr, std::move(columnSpanSiblings));
|
|
}
|
|
}
|
|
|
|
// Now grab the initial inlines in aChildList and put them into an inline
|
|
// frame.
|
|
nsInlineFrame* inlineFrame = NS_NewInlineFrame(mPresShell, computedStyle);
|
|
InitAndRestoreFrame(aState, content, parentFrame, inlineFrame, false);
|
|
inlineFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (aInitialInline->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
inlineFrame->AddStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR);
|
|
}
|
|
|
|
if (aIsAbsPosCB) {
|
|
inlineFrame->MarkAsAbsoluteContainingBlock();
|
|
}
|
|
|
|
if (aChildList.NotEmpty()) {
|
|
nsFrameList inlineKids =
|
|
aChildList.Split([](nsIFrame* f) { return f->IsBlockOutside(); });
|
|
MoveChildrenTo(aInitialInline, inlineFrame, inlineKids);
|
|
}
|
|
|
|
SetFrameIsIBSplit(blockFrame, inlineFrame);
|
|
aSiblings.AppendFrame(nullptr, inlineFrame);
|
|
lastNewInline = inlineFrame;
|
|
} while (aChildList.NotEmpty());
|
|
|
|
SetFrameIsIBSplit(lastNewInline, nullptr);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::BuildInlineChildItems(
|
|
nsFrameConstructorState& aState, FrameConstructionItem& aParentItem,
|
|
bool aItemIsWithinSVGText, bool aItemAllowsTextPathChild) {
|
|
ComputedStyle* const parentComputedStyle = aParentItem.mComputedStyle;
|
|
nsIContent* const parentContent = aParentItem.mContent;
|
|
|
|
if (!aItemIsWithinSVGText) {
|
|
if (parentComputedStyle->StyleDisplay()->IsListItem()) {
|
|
CreateGeneratedContentItem(aState, nullptr, *parentContent->AsElement(),
|
|
*parentComputedStyle, PseudoStyleType::marker,
|
|
aParentItem.mChildItems);
|
|
}
|
|
// Probe for generated content before
|
|
CreateGeneratedContentItem(aState, nullptr, *parentContent->AsElement(),
|
|
*parentComputedStyle, PseudoStyleType::before,
|
|
aParentItem.mChildItems);
|
|
}
|
|
|
|
ItemFlags flags;
|
|
if (aItemIsWithinSVGText) {
|
|
flags += ItemFlag::IsWithinSVGText;
|
|
}
|
|
if (aItemAllowsTextPathChild &&
|
|
aParentItem.mContent->IsSVGElement(nsGkAtoms::a)) {
|
|
flags += ItemFlag::AllowTextPathChild;
|
|
}
|
|
|
|
FlattenedChildIterator iter(parentContent);
|
|
for (nsIContent* content = iter.GetNextChild(); content;
|
|
content = iter.GetNextChild()) {
|
|
AddFrameConstructionItems(aState, content, iter.ShadowDOMInvolved(),
|
|
*parentComputedStyle, InsertionPoint(),
|
|
aParentItem.mChildItems, flags);
|
|
}
|
|
|
|
if (!aItemIsWithinSVGText) {
|
|
// Probe for generated content after
|
|
CreateGeneratedContentItem(aState, nullptr, *parentContent->AsElement(),
|
|
*parentComputedStyle, PseudoStyleType::after,
|
|
aParentItem.mChildItems);
|
|
}
|
|
|
|
aParentItem.mIsAllInline = aParentItem.mChildItems.AreAllItemsInline();
|
|
}
|
|
|
|
// return whether it's ok to append (in the AppendFrames sense) to
|
|
// aParentFrame if our nextSibling is aNextSibling. aParentFrame must
|
|
// be an ib-split inline.
|
|
static bool IsSafeToAppendToIBSplitInline(nsIFrame* aParentFrame,
|
|
nsIFrame* aNextSibling) {
|
|
MOZ_ASSERT(IsInlineFrame(aParentFrame), "Must have an inline parent here");
|
|
|
|
do {
|
|
NS_ASSERTION(IsFramePartOfIBSplit(aParentFrame),
|
|
"How is this not part of an ib-split?");
|
|
if (aNextSibling || aParentFrame->GetNextContinuation() ||
|
|
GetIBSplitSibling(aParentFrame)) {
|
|
return false;
|
|
}
|
|
|
|
aNextSibling = aParentFrame->GetNextSibling();
|
|
aParentFrame = aParentFrame->GetParent();
|
|
} while (IsInlineFrame(aParentFrame));
|
|
|
|
return true;
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::WipeInsertionParent(nsContainerFrame* aFrame) {
|
|
#define TRACE(reason) \
|
|
PROFILER_MARKER("WipeInsertionParent: " reason, LAYOUT, {}, Tracing, \
|
|
"Layout");
|
|
|
|
const LayoutFrameType frameType = aFrame->Type();
|
|
|
|
// FIXME(emilio): This looks terribly inefficient if you insert elements deep
|
|
// in a MathML subtree.
|
|
if (aFrame->IsMathMLFrame()) {
|
|
TRACE("MathML");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// A ruby-related frame that's getting new children.
|
|
// The situation for ruby is complex, especially when interacting with
|
|
// spaces. It contains these two special cases apart from tables:
|
|
// 1) There are effectively three types of white spaces in ruby frames
|
|
// we handle differently: leading/tailing/inter-level space,
|
|
// inter-base/inter-annotation space, and inter-segment space.
|
|
// These three types of spaces can be converted to each other when
|
|
// their sibling changes.
|
|
// 2) The first effective child of a ruby frame must always be a ruby
|
|
// base container. It should be created or destroyed accordingly.
|
|
if (IsRubyPseudo(aFrame) || frameType == LayoutFrameType::Ruby ||
|
|
RubyUtils::IsRubyContainerBox(frameType)) {
|
|
// We want to optimize it better, and avoid reframing as much as
|
|
// possible. But given the cases above, and the fact that a ruby
|
|
// usually won't be very large, it should be fine to reframe it.
|
|
TRACE("Ruby");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Reframe the multi-column container whenever elements insert/append
|
|
// into it because we need to reconstruct column-span split.
|
|
if (aFrame->IsColumnSetWrapperFrame()) {
|
|
TRACE("Multi-column");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
|
|
#undef TRACE
|
|
}
|
|
|
|
bool nsCSSFrameConstructor::WipeContainingBlock(
|
|
nsFrameConstructorState& aState, nsIFrame* aContainingBlock,
|
|
nsIFrame* aFrame, FrameConstructionItemList& aItems, bool aIsAppend,
|
|
nsIFrame* aPrevSibling) {
|
|
#define TRACE(reason) \
|
|
PROFILER_MARKER("WipeContainingBlock: " reason, LAYOUT, {}, Tracing, \
|
|
"Layout");
|
|
|
|
if (aItems.IsEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
// Before we go and append the frames, we must check for several
|
|
// special situations.
|
|
|
|
if (aFrame->GetContent() == mDocument->GetRootElement()) {
|
|
// Situation #1 is when we insert content that becomes the canonical body
|
|
// element, and its used WritingMode is different from the root element's
|
|
// used WritingMode.
|
|
// We need to reframe the root element so that the root element's frames has
|
|
// the correct writing-mode propagated from body element. (See
|
|
// nsCSSFrameConstructor::ConstructDocElementFrame.)
|
|
//
|
|
// Bug 1594297: When inserting a new <body>, we may need to reframe the old
|
|
// <body> which has a "overflow" value other than simple "visible". But it's
|
|
// tricky, see bug 1593752.
|
|
nsIContent* bodyElement = mDocument->GetBodyElement();
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
const WritingMode bodyWM(iter.item().mComputedStyle);
|
|
if (iter.item().mContent == bodyElement &&
|
|
bodyWM != aFrame->GetWritingMode()) {
|
|
TRACE("Root");
|
|
RecreateFramesForContent(mDocument->GetRootElement(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
nsIFrame* nextSibling = ::GetInsertNextSibling(aFrame, aPrevSibling);
|
|
|
|
// Situation #2 is a flex / grid container frame into which we're inserting
|
|
// new inline non-replaced children, adjacent to an existing anonymous flex or
|
|
// grid item.
|
|
if (aFrame->IsFlexOrGridContainer()) {
|
|
FCItemIterator iter(aItems);
|
|
|
|
// Check if we're adding to-be-wrapped content right *after* an existing
|
|
// anonymous flex or grid item (which would need to absorb this content).
|
|
const bool isLegacyWebKitBox = IsFlexContainerForLegacyWebKitBox(aFrame);
|
|
if (aPrevSibling && IsAnonymousItem(aPrevSibling) &&
|
|
iter.item().NeedsAnonFlexOrGridItem(aState, isLegacyWebKitBox)) {
|
|
TRACE("Inserting inline after anon flex or grid item");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Check if we're adding to-be-wrapped content right *before* an existing
|
|
// anonymous flex or grid item (which would need to absorb this content).
|
|
if (nextSibling && IsAnonymousItem(nextSibling)) {
|
|
// Jump to the last entry in the list
|
|
iter.SetToEnd();
|
|
iter.Prev();
|
|
if (iter.item().NeedsAnonFlexOrGridItem(aState, isLegacyWebKitBox)) {
|
|
TRACE("Inserting inline before anon flex or grid item");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Situation #3 is an anonymous flex or grid item that's getting new children
|
|
// who don't want to be wrapped.
|
|
if (IsAnonymousItem(aFrame)) {
|
|
AssertAnonymousFlexOrGridItemParent(aFrame, aFrame->GetParent());
|
|
|
|
// We need to push a null float containing block to be sure that
|
|
// "NeedsAnonFlexOrGridItem" will know we're not honoring floats for this
|
|
// inserted content. (In particular, this is necessary in order for
|
|
// its "GetGeometricParent" call to return the correct result.)
|
|
// We're not honoring floats on this content because it has the
|
|
// _flex/grid container_ as its parent in the content tree.
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
aState.PushFloatContainingBlock(nullptr, floatSaveState);
|
|
|
|
FCItemIterator iter(aItems);
|
|
// Skip over things that _do_ need an anonymous flex item, because
|
|
// they're perfectly happy to go here -- they won't cause a reframe.
|
|
nsIFrame* containerFrame = aFrame->GetParent();
|
|
const bool isLegacyWebKitBox =
|
|
IsFlexContainerForLegacyWebKitBox(containerFrame);
|
|
if (!iter.SkipItemsThatNeedAnonFlexOrGridItem(aState, isLegacyWebKitBox)) {
|
|
// We hit something that _doesn't_ need an anonymous flex item!
|
|
// Rebuild the flex container to bust it out.
|
|
TRACE("Inserting non-inlines inside anon flex or grid item");
|
|
RecreateFramesForContent(containerFrame->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// If we get here, then everything in |aItems| needs to be wrapped in
|
|
// an anonymous flex or grid item. That's where it's already going - good!
|
|
}
|
|
|
|
// Situation #4 is a case when table pseudo-frames don't work out right
|
|
ParentType parentType = GetParentType(aFrame);
|
|
// If all the kids want a parent of the type that aFrame is, then we're all
|
|
// set to go. Indeed, there won't be any table pseudo-frames created between
|
|
// aFrame and the kids, so those won't need to be merged with any table
|
|
// pseudo-frames that might already be kids of aFrame. If aFrame itself is a
|
|
// table pseudo-frame, then all the kids in this list would have wanted a
|
|
// frame of that type wrapping them anyway, so putting them inside it is ok.
|
|
if (!aItems.AllWantParentType(parentType)) {
|
|
// Don't give up yet. If parentType is not eTypeBlock and the parent is
|
|
// not a generated content frame, then try filtering whitespace out of the
|
|
// list.
|
|
if (parentType != eTypeBlock && !aFrame->IsGeneratedContentFrame()) {
|
|
// For leading whitespace followed by a kid that wants our parent type,
|
|
// there are four cases:
|
|
// 1) We have a previous sibling which is not a table pseudo. That means
|
|
// that previous sibling wanted a (non-block) parent of the type we're
|
|
// looking at. Then the whitespace comes between two table-internal
|
|
// elements, so should be collapsed out.
|
|
// 2) We have a previous sibling which is a table pseudo. It might have
|
|
// kids who want this whitespace, so we need to reframe.
|
|
// 3) We have no previous sibling and our parent frame is not a table
|
|
// pseudo. That means that we'll be at the beginning of our actual
|
|
// non-block-type parent, and the whitespace is OK to collapse out.
|
|
// If something is ever inserted before us, it'll find our own parent
|
|
// as its parent and if it's something that would care about the
|
|
// whitespace it'll want a block parent, so it'll trigger a reframe at
|
|
// that point.
|
|
// 4) We have no previous sibling and our parent frame is a table pseudo.
|
|
// Need to reframe.
|
|
// All that is predicated on finding the correct previous sibling. We
|
|
// might have to walk backwards along continuations from aFrame to do so.
|
|
//
|
|
// It's always OK to drop whitespace between any two items that want a
|
|
// parent of type parentType.
|
|
//
|
|
// For trailing whitespace preceded by a kid that wants our parent type,
|
|
// there are four cases:
|
|
// 1) We have a next sibling which is not a table pseudo. That means
|
|
// that next sibling wanted a (non-block) parent of the type we're
|
|
// looking at. Then the whitespace comes between two table-internal
|
|
// elements, so should be collapsed out.
|
|
// 2) We have a next sibling which is a table pseudo. It might have
|
|
// kids who want this whitespace, so we need to reframe.
|
|
// 3) We have no next sibling and our parent frame is not a table
|
|
// pseudo. That means that we'll be at the end of our actual
|
|
// non-block-type parent, and the whitespace is OK to collapse out.
|
|
// If something is ever inserted after us, it'll find our own parent
|
|
// as its parent and if it's something that would care about the
|
|
// whitespace it'll want a block parent, so it'll trigger a reframe at
|
|
// that point.
|
|
// 4) We have no next sibling and our parent frame is a table pseudo.
|
|
// Need to reframe.
|
|
// All that is predicated on finding the correct next sibling. We might
|
|
// have to walk forward along continuations from aFrame to do so. That
|
|
// said, in the case when nextSibling is null at this point and aIsAppend
|
|
// is true, we know we're in case 3. Furthermore, in that case we don't
|
|
// even have to worry about the table pseudo situation; we know our
|
|
// parent is not a table pseudo there.
|
|
FCItemIterator iter(aItems);
|
|
FCItemIterator start(iter);
|
|
do {
|
|
if (iter.SkipItemsWantingParentType(parentType)) {
|
|
break;
|
|
}
|
|
|
|
// iter points to an item that wants a different parent. If it's not
|
|
// whitespace, we're done; no more point scanning the list.
|
|
if (!iter.item().IsWhitespace(aState)) {
|
|
break;
|
|
}
|
|
|
|
if (iter == start) {
|
|
// Leading whitespace. How to handle this depends on our
|
|
// previous sibling and aFrame. See the long comment above.
|
|
nsIFrame* prevSibling = aPrevSibling;
|
|
if (!prevSibling) {
|
|
// Try to find one after all
|
|
nsIFrame* parentPrevCont = aFrame->GetPrevContinuation();
|
|
while (parentPrevCont) {
|
|
prevSibling = parentPrevCont->PrincipalChildList().LastChild();
|
|
if (prevSibling) {
|
|
break;
|
|
}
|
|
parentPrevCont = parentPrevCont->GetPrevContinuation();
|
|
}
|
|
};
|
|
if (prevSibling) {
|
|
if (IsTablePseudo(prevSibling)) {
|
|
// need to reframe
|
|
break;
|
|
}
|
|
} else if (IsTablePseudo(aFrame)) {
|
|
// need to reframe
|
|
break;
|
|
}
|
|
}
|
|
|
|
FCItemIterator spaceEndIter(iter);
|
|
// Advance spaceEndIter past any whitespace
|
|
bool trailingSpaces = spaceEndIter.SkipWhitespace(aState);
|
|
|
|
bool okToDrop;
|
|
if (trailingSpaces) {
|
|
// Trailing whitespace. How to handle this depeds on aIsAppend, our
|
|
// next sibling and aFrame. See the long comment above.
|
|
okToDrop = aIsAppend && !nextSibling;
|
|
if (!okToDrop) {
|
|
if (!nextSibling) {
|
|
// Try to find one after all
|
|
nsIFrame* parentNextCont = aFrame->GetNextContinuation();
|
|
while (parentNextCont) {
|
|
nextSibling = parentNextCont->PrincipalChildList().FirstChild();
|
|
if (nextSibling) {
|
|
break;
|
|
}
|
|
parentNextCont = parentNextCont->GetNextContinuation();
|
|
}
|
|
}
|
|
|
|
okToDrop = (nextSibling && !IsTablePseudo(nextSibling)) ||
|
|
(!nextSibling && !IsTablePseudo(aFrame));
|
|
}
|
|
#ifdef DEBUG
|
|
else {
|
|
NS_ASSERTION(!IsTablePseudo(aFrame), "How did that happen?");
|
|
}
|
|
#endif
|
|
} else {
|
|
okToDrop = (spaceEndIter.item().DesiredParentType() == parentType);
|
|
}
|
|
|
|
if (okToDrop) {
|
|
iter.DeleteItemsTo(this, spaceEndIter);
|
|
} else {
|
|
// We're done: we don't want to drop the whitespace, and it has the
|
|
// wrong parent type.
|
|
break;
|
|
}
|
|
|
|
// Now loop, since |iter| points to item right after the whitespace we
|
|
// removed.
|
|
} while (!iter.IsDone());
|
|
}
|
|
|
|
// We might be able to figure out some sort of optimizations here, but they
|
|
// would have to depend on having a correct aPrevSibling and a correct next
|
|
// sibling. For example, we can probably avoid reframing if none of
|
|
// aFrame, aPrevSibling, and next sibling are table pseudo-frames. But it
|
|
// doesn't seem worth it to worry about that for now, especially since we
|
|
// in fact do not have a reliable aPrevSibling, nor any next sibling, in
|
|
// this method.
|
|
|
|
// aItems might have changed, so recheck the parent type thing. In fact,
|
|
// it might be empty, so recheck that too.
|
|
if (aItems.IsEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
// If aFrame is empty, the insertion process will be able to take care of
|
|
// creating any needed pseudo-parents.
|
|
if (!aItems.AllWantParentType(parentType) &&
|
|
!SafeToInsertPseudoNeedingChildren(aFrame)) {
|
|
// Reframing aFrame->GetContent() is good enough, since the content of
|
|
// table pseudo-frames is the ancestor content.
|
|
TRACE("Pseudo-frames going wrong");
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Situation #5 is a frame in multicol subtree that's getting new children.
|
|
if (aFrame->HasAnyStateBits(NS_FRAME_HAS_MULTI_COLUMN_ANCESTOR)) {
|
|
bool anyColumnSpanItems = false;
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
if (iter.item().mComputedStyle->StyleColumn()->IsColumnSpanStyle()) {
|
|
anyColumnSpanItems = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool needsReframe =
|
|
// 1. Insert / append any column-span children.
|
|
anyColumnSpanItems ||
|
|
// 2. GetInsertionPrevSibling() modifies insertion parent. If the prev
|
|
// sibling is a column-span, aFrame ends up being the
|
|
// column-span-wrapper.
|
|
aFrame->Style()->GetPseudoType() ==
|
|
PseudoStyleType::columnSpanWrapper ||
|
|
// 3. Append into {ib} split container. There might be room for
|
|
// optimization, but let's reframe for correctness...
|
|
IsFramePartOfIBSplit(aFrame);
|
|
|
|
if (needsReframe) {
|
|
TRACE("Multi-column");
|
|
RecreateFramesForContent(
|
|
GetMultiColumnContainingBlockFor(aFrame)->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// If we get here, then we need further check for {ib} split to decide
|
|
// whether to reframe. For example, appending a block into an empty inline
|
|
// that is not part of an {ib} split, but should become an {ib} split.
|
|
}
|
|
|
|
// A <fieldset> may need to pick up a new rendered legend from aItems.
|
|
// We currently can't handle this case without recreating frames for
|
|
// the fieldset.
|
|
// XXXmats we should be able to optimize this when the fieldset doesn't
|
|
// currently have a rendered legend. ContentRangeInserted needs to be fixed
|
|
// to use the inner frame as the content insertion frame in that case.
|
|
if (const auto* fieldset = GetFieldSetFrameFor(aFrame)) {
|
|
// Check if any item is eligible to be a rendered legend.
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
const auto& item = iter.item();
|
|
if (!item.mContent->IsHTMLElement(nsGkAtoms::legend)) {
|
|
continue;
|
|
}
|
|
const auto* display = item.mComputedStyle->StyleDisplay();
|
|
if (display->IsFloatingStyle() ||
|
|
display->IsAbsolutelyPositionedStyle()) {
|
|
continue;
|
|
}
|
|
TRACE("Fieldset with rendered legend");
|
|
RecreateFramesForContent(fieldset->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Now we have several cases involving {ib} splits. Put them all in a
|
|
// do/while with breaks to take us to the "go and reconstruct" code.
|
|
do {
|
|
if (IsInlineFrame(aFrame)) {
|
|
if (aItems.AreAllItemsInline()) {
|
|
// We can just put the kids in.
|
|
return false;
|
|
}
|
|
|
|
if (!IsFramePartOfIBSplit(aFrame)) {
|
|
// Need to go ahead and reconstruct.
|
|
break;
|
|
}
|
|
|
|
// Now we're adding kids including some blocks to an inline part of an
|
|
// {ib} split. If we plan to call AppendFrames, and don't have a next
|
|
// sibling for the new frames, and our parent is the last continuation of
|
|
// the last part of the {ib} split, and the same is true of all our
|
|
// ancestor inlines (they have no following continuations and they're the
|
|
// last part of their {ib} splits and we'd be adding to the end for all
|
|
// of them), then AppendFrames will handle things for us. Bail out in
|
|
// that case.
|
|
if (aIsAppend && IsSafeToAppendToIBSplitInline(aFrame, nextSibling)) {
|
|
return false;
|
|
}
|
|
|
|
// Need to reconstruct.
|
|
break;
|
|
}
|
|
|
|
// Now we know we have a block parent. If it's not part of an
|
|
// ib-split, we're all set.
|
|
if (!IsFramePartOfIBSplit(aFrame)) {
|
|
return false;
|
|
}
|
|
|
|
// We're adding some kids to a block part of an {ib} split. If all the
|
|
// kids are blocks, we don't need to reconstruct.
|
|
if (aItems.AreAllItemsBlock()) {
|
|
return false;
|
|
}
|
|
|
|
// We might have some inline kids for this block. Just fall out of the
|
|
// loop and reconstruct.
|
|
} while (false);
|
|
|
|
// If we don't have a containing block, start with aFrame and look for one.
|
|
if (!aContainingBlock) {
|
|
aContainingBlock = aFrame;
|
|
}
|
|
|
|
// To find the right block to reframe, just walk up the tree until we find a
|
|
// frame that is:
|
|
// 1) Not part of an IB split
|
|
// 2) Not a pseudo-frame
|
|
// 3) Not an inline frame
|
|
// We're guaranteed to find one, since ComputedStyle::ApplyStyleFixups
|
|
// enforces that the root is display:none, display:table, or display:block.
|
|
// Note that walking up "too far" is OK in terms of correctness, even if it
|
|
// might be a little inefficient. This is why we walk out of all
|
|
// pseudo-frames -- telling which ones are or are not OK to walk out of is
|
|
// too hard (and I suspect that we do in fact need to walk out of all of
|
|
// them).
|
|
while (IsFramePartOfIBSplit(aContainingBlock) ||
|
|
aContainingBlock->IsInlineOutside() ||
|
|
aContainingBlock->Style()->IsPseudoOrAnonBox()) {
|
|
aContainingBlock = aContainingBlock->GetParent();
|
|
NS_ASSERTION(aContainingBlock,
|
|
"Must have non-inline, non-ib-split, non-pseudo frame as "
|
|
"root (or child of root, for a table root)!");
|
|
}
|
|
|
|
// Tell parent of the containing block to reformulate the
|
|
// entire block. This is painful and definitely not optimal
|
|
// but it will *always* get the right answer.
|
|
|
|
nsIContent* blockContent = aContainingBlock->GetContent();
|
|
TRACE("IB splits");
|
|
RecreateFramesForContent(blockContent, InsertionKind::Async);
|
|
return true;
|
|
#undef TRACE
|
|
}
|
|
|
|
void nsCSSFrameConstructor::ReframeContainingBlock(nsIFrame* aFrame) {
|
|
// XXXbz how exactly would we get here while isReflowing anyway? Should this
|
|
// whole test be ifdef DEBUG?
|
|
if (mPresShell->IsReflowLocked()) {
|
|
// don't ReframeContainingBlock, this will result in a crash
|
|
// if we remove a tree that's in reflow - see bug 121368 for testcase
|
|
NS_ERROR(
|
|
"Atemptted to nsCSSFrameConstructor::ReframeContainingBlock during a "
|
|
"Reflow!!!");
|
|
return;
|
|
}
|
|
|
|
// Get the first "normal" ancestor of the target frame.
|
|
nsIFrame* containingBlock = GetIBContainingBlockFor(aFrame);
|
|
if (containingBlock) {
|
|
// From here we look for the containing block in case the target
|
|
// frame is already a block (which can happen when an inline frame
|
|
// wraps some of its content in an anonymous block; see
|
|
// ConstructInline)
|
|
|
|
// NOTE: We used to get the FloatContainingBlock here, but it was often
|
|
// wrong. GetIBContainingBlock works much better and provides the correct
|
|
// container in all cases so GetFloatContainingBlock(aFrame) has been
|
|
// removed
|
|
|
|
// And get the containingBlock's content
|
|
if (nsIContent* blockContent = containingBlock->GetContent()) {
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf(" ==> blockContent=%p\n", blockContent);
|
|
}
|
|
#endif
|
|
RecreateFramesForContent(blockContent, InsertionKind::Async);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// If we get here, we're screwed!
|
|
RecreateFramesForContent(mPresShell->GetDocument()->GetRootElement(),
|
|
InsertionKind::Async);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////
|
|
// nsCSSFrameConstructor::FrameConstructionItem methods //
|
|
//////////////////////////////////////////////////////////
|
|
bool nsCSSFrameConstructor::FrameConstructionItem::IsWhitespace(
|
|
nsFrameConstructorState& aState) const {
|
|
MOZ_ASSERT(aState.mCreatingExtraFrames || !mContent->GetPrimaryFrame(),
|
|
"How did that happen?");
|
|
if (!mIsText) {
|
|
return false;
|
|
}
|
|
mContent->SetFlags(NS_CREATE_FRAME_IF_NON_WHITESPACE |
|
|
NS_REFRAME_IF_WHITESPACE);
|
|
return mContent->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////
|
|
// nsCSSFrameConstructor::FrameConstructionItemList methods //
|
|
//////////////////////////////////////////////////////////////
|
|
void nsCSSFrameConstructor::FrameConstructionItemList::AdjustCountsForItem(
|
|
FrameConstructionItem* aItem, int32_t aDelta) {
|
|
MOZ_ASSERT(aDelta == 1 || aDelta == -1, "Unexpected delta");
|
|
mItemCount += aDelta;
|
|
if (aItem->mIsAllInline) {
|
|
mInlineCount += aDelta;
|
|
}
|
|
if (aItem->mIsBlock) {
|
|
mBlockCount += aDelta;
|
|
}
|
|
mDesiredParentCounts[aItem->DesiredParentType()] += aDelta;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// nsCSSFrameConstructor::FrameConstructionItemList::Iterator methods //
|
|
////////////////////////////////////////////////////////////////////////
|
|
inline bool nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
SkipItemsWantingParentType(ParentType aParentType) {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
while (item().DesiredParentType() == aParentType) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
SkipItemsNotWantingParentType(ParentType aParentType) {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
while (item().DesiredParentType() != aParentType) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Note: we implement -webkit-{inline-}box using nsFlexContainerFrame, but we
|
|
// use different rules for what gets wrapped in an anonymous flex item.
|
|
bool nsCSSFrameConstructor::FrameConstructionItem::NeedsAnonFlexOrGridItem(
|
|
const nsFrameConstructorState& aState, bool aIsLegacyWebKitBox) {
|
|
if (mFCData->mBits & FCDATA_IS_LINE_PARTICIPANT) {
|
|
// This will be an inline non-replaced box.
|
|
return true;
|
|
}
|
|
|
|
if (aIsLegacyWebKitBox) {
|
|
if (mComputedStyle->StyleDisplay()->IsInlineOutsideStyle()) {
|
|
// In an emulated legacy box, all inline-level content gets wrapped in an
|
|
// anonymous flex item.
|
|
return true;
|
|
}
|
|
if (mIsPopup ||
|
|
(!(mFCData->mBits & FCDATA_DISALLOW_OUT_OF_FLOW) &&
|
|
aState.GetGeometricParent(*mComputedStyle->StyleDisplay(), nullptr))) {
|
|
// We're abspos or fixedpos (or a XUL popup), which means we'll spawn a
|
|
// placeholder which (because our container is an emulated legacy box)
|
|
// we'll need to wrap in an anonymous flex item. So, we just treat
|
|
// _this_ frame as if _it_ needs to be wrapped in an anonymous flex item,
|
|
// and then when we spawn the placeholder, it'll end up in the right
|
|
// spot.
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
inline bool nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
SkipItemsThatNeedAnonFlexOrGridItem(const nsFrameConstructorState& aState,
|
|
bool aIsLegacyWebKitBox) {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
while (item().NeedsAnonFlexOrGridItem(aState, aIsLegacyWebKitBox)) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
SkipItemsThatDontNeedAnonFlexOrGridItem(
|
|
const nsFrameConstructorState& aState, bool aIsLegacyWebKitBox) {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
while (!(item().NeedsAnonFlexOrGridItem(aState, aIsLegacyWebKitBox))) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
SkipItemsNotWantingRubyParent() {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
while (!IsRubyParentType(item().DesiredParentType())) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool
|
|
nsCSSFrameConstructor::FrameConstructionItemList::Iterator::SkipWhitespace(
|
|
nsFrameConstructorState& aState) {
|
|
MOZ_ASSERT(!IsDone(), "Shouldn't be done yet");
|
|
MOZ_ASSERT(item().IsWhitespace(aState), "Not pointing to whitespace?");
|
|
do {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
} while (item().IsWhitespace(aState));
|
|
|
|
return false;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
AppendItemToList(FrameConstructionItemList& aTargetList) {
|
|
NS_ASSERTION(&aTargetList != &mList, "Unexpected call");
|
|
MOZ_ASSERT(!IsDone(), "should not be done");
|
|
|
|
FrameConstructionItem* item = mCurrent;
|
|
Next();
|
|
item->remove();
|
|
aTargetList.mItems.insertBack(item);
|
|
|
|
mList.AdjustCountsForItem(item, -1);
|
|
aTargetList.AdjustCountsForItem(item, 1);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FrameConstructionItemList::Iterator::
|
|
AppendItemsToList(nsCSSFrameConstructor* aFCtor, const Iterator& aEnd,
|
|
FrameConstructionItemList& aTargetList) {
|
|
NS_ASSERTION(&aTargetList != &mList, "Unexpected call");
|
|
MOZ_ASSERT(&mList == &aEnd.mList, "End iterator for some other list?");
|
|
|
|
// We can't just move our guts to the other list if it already has
|
|
// some information or if we're not moving our entire list.
|
|
if (!AtStart() || !aEnd.IsDone() || !aTargetList.IsEmpty()) {
|
|
do {
|
|
AppendItemToList(aTargetList);
|
|
} while (*this != aEnd);
|
|
return;
|
|
}
|
|
|
|
// Move our entire list of items into the empty target list.
|
|
aTargetList.mItems = std::move(mList.mItems);
|
|
|
|
// Copy over the various counters
|
|
aTargetList.mInlineCount = mList.mInlineCount;
|
|
aTargetList.mBlockCount = mList.mBlockCount;
|
|
aTargetList.mItemCount = mList.mItemCount;
|
|
memcpy(aTargetList.mDesiredParentCounts, mList.mDesiredParentCounts,
|
|
sizeof(aTargetList.mDesiredParentCounts));
|
|
|
|
// reset mList
|
|
mList.Reset(aFCtor);
|
|
|
|
// Point ourselves to aEnd, as advertised
|
|
SetToEnd();
|
|
MOZ_ASSERT(*this == aEnd, "How did that happen?");
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FrameConstructionItemList::Iterator::InsertItem(
|
|
FrameConstructionItem* aItem) {
|
|
if (IsDone()) {
|
|
mList.mItems.insertBack(aItem);
|
|
} else {
|
|
// Just insert the item before us. There's no magic here.
|
|
mCurrent->setPrevious(aItem);
|
|
}
|
|
mList.AdjustCountsForItem(aItem, 1);
|
|
|
|
MOZ_ASSERT(aItem->getNext() == mCurrent, "How did that happen?");
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FrameConstructionItemList::Iterator::DeleteItemsTo(
|
|
nsCSSFrameConstructor* aFCtor, const Iterator& aEnd) {
|
|
MOZ_ASSERT(&mList == &aEnd.mList, "End iterator for some other list?");
|
|
MOZ_ASSERT(*this != aEnd, "Shouldn't be at aEnd yet");
|
|
|
|
do {
|
|
NS_ASSERTION(!IsDone(), "Ran off end of list?");
|
|
FrameConstructionItem* item = mCurrent;
|
|
Next();
|
|
item->remove();
|
|
mList.AdjustCountsForItem(item, -1);
|
|
item->Delete(aFCtor);
|
|
} while (*this != aEnd);
|
|
}
|
|
|
|
void nsCSSFrameConstructor::QuotesDirty() {
|
|
mQuotesDirty = true;
|
|
mPresShell->SetNeedLayoutFlush();
|
|
}
|
|
|
|
void nsCSSFrameConstructor::CountersDirty() {
|
|
mCountersDirty = true;
|
|
mPresShell->SetNeedLayoutFlush();
|
|
}
|
|
|
|
void* nsCSSFrameConstructor::AllocateFCItem() {
|
|
void* item;
|
|
if (mFirstFreeFCItem) {
|
|
item = mFirstFreeFCItem;
|
|
mFirstFreeFCItem = mFirstFreeFCItem->mNext;
|
|
} else {
|
|
item = mFCItemPool.Allocate(sizeof(FrameConstructionItem));
|
|
}
|
|
++mFCItemsInUse;
|
|
return item;
|
|
}
|
|
|
|
void nsCSSFrameConstructor::FreeFCItem(FrameConstructionItem* aItem) {
|
|
MOZ_ASSERT(mFCItemsInUse != 0);
|
|
if (--mFCItemsInUse == 0) {
|
|
// The arena is now unused - clear it but retain one chunk.
|
|
mFirstFreeFCItem = nullptr;
|
|
mFCItemPool.Clear();
|
|
} else {
|
|
// Prepend it to the list of free items.
|
|
FreeFCItemLink* item = reinterpret_cast<FreeFCItemLink*>(aItem);
|
|
item->mNext = mFirstFreeFCItem;
|
|
mFirstFreeFCItem = item;
|
|
}
|
|
}
|
|
|
|
void nsCSSFrameConstructor::AddSizeOfIncludingThis(
|
|
nsWindowSizes& aSizes) const {
|
|
if (nsIFrame* rootFrame = GetRootFrame()) {
|
|
rootFrame->AddSizeOfExcludingThisForTree(aSizes);
|
|
if (RetainedDisplayListBuilder* builder =
|
|
rootFrame->GetProperty(RetainedDisplayListBuilder::Cached())) {
|
|
builder->AddSizeOfIncludingThis(aSizes);
|
|
}
|
|
}
|
|
|
|
// This must be done after measuring from the frame tree, since frame
|
|
// manager will measure sizes of staled computed values and style
|
|
// structs, which only make sense after we know what are being used.
|
|
nsFrameManager::AddSizeOfIncludingThis(aSizes);
|
|
|
|
// Measurement of the following members may be added later if DMD finds it
|
|
// is worthwhile:
|
|
// - mFCItemPool
|
|
// - mContainStyleScopeManager
|
|
}
|