зеркало из https://github.com/mozilla/gecko-dev.git
13290 строки
525 KiB
C++
13290 строки
525 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 "mozilla/AutoRestore.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "mozilla/ErrorResult.h"
|
|
#include "mozilla/dom/HTMLDetailsElement.h"
|
|
#include "mozilla/dom/HTMLSelectElement.h"
|
|
#include "mozilla/dom/HTMLSummaryElement.h"
|
|
#include "mozilla/EventStates.h"
|
|
#include "mozilla/Likely.h"
|
|
#include "mozilla/LinkedList.h"
|
|
#include "mozilla/PresShell.h"
|
|
#include "mozilla/ServoBindings.h"
|
|
#include "nsAbsoluteContainingBlock.h"
|
|
#include "nsCSSPseudoElements.h"
|
|
#include "nsAtom.h"
|
|
#include "nsIFrameInlines.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsPresContext.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsIDocumentInlines.h"
|
|
#include "nsTableFrame.h"
|
|
#include "nsTableColFrame.h"
|
|
#include "nsTableRowFrame.h"
|
|
#include "nsTableCellFrame.h"
|
|
#include "nsIDOMHTMLDocument.h"
|
|
#include "nsHTMLParts.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "mozilla/StyleSetHandle.h"
|
|
#include "mozilla/StyleSetHandleInlines.h"
|
|
#include "nsViewManager.h"
|
|
#include "nsStyleConsts.h"
|
|
#ifdef MOZ_XUL
|
|
#include "nsXULElement.h"
|
|
#include "mozilla/dom/BoxObject.h"
|
|
#endif // MOZ_XUL
|
|
#include "nsContainerFrame.h"
|
|
#include "nsNameSpaceManager.h"
|
|
#include "nsIComboboxControlFrame.h"
|
|
#include "nsComboboxControlFrame.h"
|
|
#include "nsIListControlFrame.h"
|
|
#include "nsIDOMCharacterData.h"
|
|
#include "nsPlaceholderFrame.h"
|
|
#include "nsTableRowGroupFrame.h"
|
|
#include "nsIFormControl.h"
|
|
#include "nsCSSAnonBoxes.h"
|
|
#include "nsTextFragment.h"
|
|
#include "nsIAnonymousContentCreator.h"
|
|
#include "nsBindingManager.h"
|
|
#include "nsXBLBinding.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsIScriptError.h"
|
|
#ifdef XP_MACOSX
|
|
#include "nsIDocShell.h"
|
|
#endif
|
|
#include "ChildIterator.h"
|
|
#include "nsError.h"
|
|
#include "nsLayoutUtils.h"
|
|
#include "nsAutoPtr.h"
|
|
#include "nsBoxFrame.h"
|
|
#include "nsBoxLayout.h"
|
|
#include "nsFlexContainerFrame.h"
|
|
#include "nsGridContainerFrame.h"
|
|
#include "RubyUtils.h"
|
|
#include "nsRubyFrame.h"
|
|
#include "nsRubyBaseFrame.h"
|
|
#include "nsRubyBaseContainerFrame.h"
|
|
#include "nsRubyTextFrame.h"
|
|
#include "nsRubyTextContainerFrame.h"
|
|
#include "nsImageFrame.h"
|
|
#include "nsIObjectLoadingContent.h"
|
|
#include "nsTArray.h"
|
|
#include "nsGenericDOMDataNode.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/ElementInlines.h"
|
|
#include "nsAutoLayoutPhase.h"
|
|
#include "nsStyleStructInlines.h"
|
|
#include "nsPageContentFrame.h"
|
|
#include "mozilla/GeckoStyleContext.h"
|
|
#include "mozilla/RestyleManager.h"
|
|
#include "mozilla/RestyleManagerInlines.h"
|
|
#include "StickyScrollContainer.h"
|
|
#include "nsFieldSetFrame.h"
|
|
#include "nsInlineFrame.h"
|
|
#include "nsBlockFrame.h"
|
|
#include "nsCanvasFrame.h"
|
|
#include "nsFirstLetterFrame.h"
|
|
#include "nsGfxScrollFrame.h"
|
|
#include "nsPageFrame.h"
|
|
#include "nsSimplePageSequenceFrame.h"
|
|
#include "nsTableWrapperFrame.h"
|
|
#include "nsIScrollableFrame.h"
|
|
#include "nsBackdropFrame.h"
|
|
#include "nsTransitionManager.h"
|
|
#include "DetailsFrame.h"
|
|
#include "nsThemeConstants.h"
|
|
|
|
#ifdef MOZ_XUL
|
|
#include "nsIRootBox.h"
|
|
#endif
|
|
#ifdef ACCESSIBILITY
|
|
#include "nsAccessibilityService.h"
|
|
#endif
|
|
|
|
#include "nsXBLService.h"
|
|
|
|
#undef NOISY_FIRST_LETTER
|
|
|
|
#include "nsMathMLParts.h"
|
|
#include "mozilla/dom/SVGTests.h"
|
|
#include "nsSVGUtils.h"
|
|
|
|
#include "nsRefreshDriver.h"
|
|
#include "nsRuleProcessorData.h"
|
|
#include "nsTextNode.h"
|
|
#include "ActiveLayerTracker.h"
|
|
#include "nsIPresShellInlines.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
// An alias for convenience.
|
|
static const nsIFrame::ChildListID kPrincipalList = nsIFrame::kPrincipalList;
|
|
|
|
nsIFrame*
|
|
NS_NewHTMLCanvasFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewHTMLVideoFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsContainerFrame*
|
|
NS_NewSVGOuterSVGFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsContainerFrame*
|
|
NS_NewSVGOuterSVGAnonChildFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGInnerSVGFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGGeometryFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGGFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGGenericContainerFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsContainerFrame*
|
|
NS_NewSVGForeignObjectFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGAFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGSwitchFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGSymbolFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGTextFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGContainerFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGUseFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGViewFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
extern nsIFrame*
|
|
NS_NewSVGLinearGradientFrame(nsIPresShell *aPresShell, nsStyleContext* aContext);
|
|
extern nsIFrame*
|
|
NS_NewSVGRadialGradientFrame(nsIPresShell *aPresShell, nsStyleContext* aContext);
|
|
extern nsIFrame*
|
|
NS_NewSVGStopFrame(nsIPresShell *aPresShell, nsStyleContext* aContext);
|
|
nsContainerFrame*
|
|
NS_NewSVGMarkerFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsContainerFrame*
|
|
NS_NewSVGMarkerAnonChildFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
extern nsIFrame*
|
|
NS_NewSVGImageFrame(nsIPresShell *aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGClipPathFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGFilterFrame(nsIPresShell *aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGPatternFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGMaskFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGFEContainerFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGFELeafFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGFEImageFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewSVGFEUnstyledLeafFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
#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
|
|
|
|
|
|
#ifdef MOZ_XUL
|
|
#include "nsMenuFrame.h"
|
|
#include "nsPopupSetFrame.h"
|
|
#include "nsTreeColFrame.h"
|
|
#include "nsIBoxObject.h"
|
|
#include "nsPIListBoxObject.h"
|
|
#include "nsListBoxBodyFrame.h"
|
|
#include "nsListItemFrame.h"
|
|
#include "nsXULLabelFrame.h"
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
nsIFrame*
|
|
NS_NewAutoRepeatBoxFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsContainerFrame*
|
|
NS_NewRootBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsContainerFrame*
|
|
NS_NewDocElementBoxFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewDeckFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewLeafBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewStackFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewProgressMeterFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewRangeFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewImageBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewTextBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewGroupBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewButtonBoxFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewSplitterFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewMenuPopupFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewPopupSetFrame(nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewMenuFrame (nsIPresShell* aPresShell, nsStyleContext* aContext, uint32_t aFlags);
|
|
|
|
nsIFrame*
|
|
NS_NewMenuBarFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewTreeBodyFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
// grid
|
|
nsresult
|
|
NS_NewGridLayout2 ( nsIPresShell* aPresShell, nsBoxLayout** aNewLayout );
|
|
nsIFrame*
|
|
NS_NewGridRowLeafFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
nsIFrame*
|
|
NS_NewGridRowGroupFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
// end grid
|
|
|
|
nsIFrame*
|
|
NS_NewTitleBarFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewResizerFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
|
|
#endif
|
|
|
|
nsHTMLScrollFrame*
|
|
NS_NewHTMLScrollFrame(nsIPresShell* aPresShell, nsStyleContext* aContext, bool aIsRoot);
|
|
|
|
nsXULScrollFrame*
|
|
NS_NewXULScrollFrame(nsIPresShell* aPresShell, nsStyleContext* aContext,
|
|
bool aIsRoot, bool aClipAllDescendants);
|
|
|
|
nsIFrame*
|
|
NS_NewSliderFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewScrollbarFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
nsIFrame*
|
|
NS_NewScrollbarButtonFrame (nsIPresShell* aPresShell, nsStyleContext* aContext);
|
|
|
|
|
|
#ifdef NOISY_FINDFRAME
|
|
static int32_t FFWC_totalCount=0;
|
|
static int32_t FFWC_doLoop=0;
|
|
static int32_t FFWC_doSibling=0;
|
|
static int32_t FFWC_recursions=0;
|
|
static int32_t FFWC_nextInFlows=0;
|
|
#endif
|
|
|
|
// Wrapper class to handle stack-construction a TreeMatchContext only if we're
|
|
// using the Gecko style system.
|
|
class MOZ_STACK_CLASS TreeMatchContextHolder
|
|
{
|
|
public:
|
|
explicit TreeMatchContextHolder(nsIDocument* aDocument)
|
|
{
|
|
if (!aDocument->IsStyledByServo()) {
|
|
mMaybeTreeMatchContext.emplace(aDocument,
|
|
TreeMatchContext::ForFrameConstruction);
|
|
}
|
|
}
|
|
|
|
bool Exists() const { return mMaybeTreeMatchContext.isSome(); }
|
|
operator TreeMatchContext*() { return mMaybeTreeMatchContext.ptrOr(nullptr); }
|
|
|
|
TreeMatchContext* operator ->()
|
|
{
|
|
MOZ_ASSERT(mMaybeTreeMatchContext.isSome());
|
|
return mMaybeTreeMatchContext.ptr();
|
|
}
|
|
|
|
private:
|
|
Maybe<TreeMatchContext> mMaybeTreeMatchContext;
|
|
};
|
|
|
|
// Returns true if aFrame is an anonymous flex/grid item.
|
|
static inline bool
|
|
IsAnonymousFlexOrGridItem(const nsIFrame* aFrame)
|
|
{
|
|
const nsAtom* pseudoType = aFrame->StyleContext()->GetPseudo();
|
|
return pseudoType == nsCSSAnonBoxes::anonymousFlexItem ||
|
|
pseudoType == nsCSSAnonBoxes::anonymousGridItem;
|
|
}
|
|
|
|
// Returns true if aFrame is a flex/grid container.
|
|
static inline bool
|
|
IsFlexOrGridContainer(const nsIFrame* aFrame)
|
|
{
|
|
const LayoutFrameType t = aFrame->Type();
|
|
return t == LayoutFrameType::FlexContainer ||
|
|
t == LayoutFrameType::GridContainer;
|
|
}
|
|
|
|
// Returns true IFF the given nsIFrame is a nsFlexContainerFrame and
|
|
// represents a -webkit-{inline-}box container.
|
|
static inline bool
|
|
IsFlexContainerForLegacyBox(const nsIFrame* aFrame)
|
|
{
|
|
return aFrame->IsFlexContainerFrame() &&
|
|
aFrame->HasAnyStateBits(NS_STATE_FLEX_IS_LEGACY_WEBKIT_BOX);
|
|
}
|
|
|
|
#if DEBUG
|
|
static void
|
|
AssertAnonymousFlexOrGridItemParent(const nsIFrame* aChild,
|
|
const nsIFrame* aParent)
|
|
{
|
|
MOZ_ASSERT(IsAnonymousFlexOrGridItem(aChild),
|
|
"expected an anonymous flex or grid item child frame");
|
|
MOZ_ASSERT(aParent, "expected a parent frame");
|
|
const nsAtom* pseudoType = aChild->StyleContext()->GetPseudo();
|
|
if (pseudoType == nsCSSAnonBoxes::anonymousFlexItem) {
|
|
MOZ_ASSERT(aParent->IsFlexContainerFrame(),
|
|
"anonymous flex items should only exist as children "
|
|
"of flex container frames");
|
|
} else {
|
|
MOZ_ASSERT(aParent->IsGridContainerFrame(),
|
|
"anonymous grid items should only exist as children "
|
|
"of grid container frames");
|
|
}
|
|
}
|
|
#else
|
|
#define AssertAnonymousFlexOrGridItemParent(x, y) do { /* nothing */ } while(0)
|
|
#endif
|
|
|
|
static inline nsContainerFrame*
|
|
GetFieldSetBlockFrame(nsIFrame* aFieldsetFrame)
|
|
{
|
|
// Depends on the fieldset child frame order - see ConstructFieldSetFrame() below.
|
|
nsIFrame* firstChild = aFieldsetFrame->PrincipalChildList().FirstChild();
|
|
nsIFrame* inner = firstChild && firstChild->GetNextSibling() ? firstChild->GetNextSibling() : firstChild;
|
|
return inner ? inner->GetContentInsertionFrame() : nullptr;
|
|
}
|
|
|
|
#define FCDATA_DECL(_flags, _func) \
|
|
{ _flags, { (FrameCreationFunc)_func }, nullptr, nullptr }
|
|
#define FCDATA_WITH_WRAPPING_BLOCK(_flags, _func, _anon_box) \
|
|
{ _flags | FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS, \
|
|
{ (FrameCreationFunc)_func }, nullptr, &_anon_box }
|
|
|
|
#define UNREACHABLE_FCDATA() \
|
|
{ 0, { (FrameCreationFunc)nullptr }, nullptr, nullptr }
|
|
//----------------------------------------------------------------------
|
|
|
|
/**
|
|
* 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->IsFrameOfType(nsIFrame::eLineParticipant);
|
|
}
|
|
|
|
/**
|
|
* 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->IsFrameOfType(nsIFrame::eSVG) ||
|
|
nsSVGUtils::IsInSVGTextSubtree(aFrame);
|
|
}
|
|
|
|
/**
|
|
* 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->IsFrameOfType(nsIFrame::eMathML) ||
|
|
aFrame->IsXULBoxFrame() ||
|
|
::IsFlexOrGridContainer(aFrame);
|
|
}
|
|
|
|
/**
|
|
* If any children require a block parent, return the first such child.
|
|
* Otherwise return null.
|
|
*/
|
|
static nsIContent*
|
|
AnyKidsNeedBlockParent(nsIFrame *aFrameList)
|
|
{
|
|
for (nsIFrame *k = aFrameList; k; k = k->GetNextSibling()) {
|
|
// Line participants, such as text and inline frames, can't be
|
|
// directly inside a XUL box; they must be wrapped in an
|
|
// intermediate block.
|
|
if (k->IsFrameOfType(nsIFrame::eLineParticipant)) {
|
|
return k->GetContent();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// 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.
|
|
//
|
|
// The latter shouldn't affect any styles in practice, so only needs style
|
|
// context reparenting in the Gecko backend, to make our style context tree
|
|
// assertions happy. The former passes aForceStyleReparent == true.
|
|
if (aForceStyleReparent || aRestyleManager->IsGecko()) {
|
|
aRestyleManager->ReparentStyleContext(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->GetStateBits() & NS_FRAME_PART_OF_IBSPLIT) != 0;
|
|
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)
|
|
{
|
|
NS_PRECONDITION(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)
|
|
{
|
|
NS_PRECONDITION(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, bool aReturnEmptyTrailingInline)
|
|
{
|
|
for (nsIFrame *frame = aFrame, *next; ; frame = next) {
|
|
next = GetIBSplitSibling(frame);
|
|
if (!next ||
|
|
(!aReturnEmptyTrailingInline && !next->PrincipalChildList().FirstChild() &&
|
|
!GetIBSplitSibling(next))) {
|
|
NS_ASSERTION(!next || !frame->IsInlineOutside(),
|
|
"Should have a block here!");
|
|
return static_cast<nsContainerFrame*>(frame);
|
|
}
|
|
}
|
|
NS_NOTREACHED("unreachable code");
|
|
return nullptr;
|
|
}
|
|
|
|
static void
|
|
SetFrameIsIBSplit(nsContainerFrame* aFrame, nsContainerFrame* aIBSplitSibling)
|
|
{
|
|
NS_PRECONDITION(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)
|
|
{
|
|
NS_PRECONDITION(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
|
|
// style context -- 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->StyleContext()->GetPseudo())
|
|
break;
|
|
|
|
aFrame = parentFrame;
|
|
} while (1);
|
|
|
|
// 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;
|
|
}
|
|
|
|
// This is a bit slow, but sometimes we need it.
|
|
static bool
|
|
ParentIsWrapperAnonBox(nsIFrame* aParent)
|
|
{
|
|
nsIFrame* maybeAnonBox = aParent;
|
|
if (maybeAnonBox->StyleContext()->GetPseudo() ==
|
|
nsCSSAnonBoxes::cellContent) {
|
|
// The thing that would maybe be a wrapper anon box is the cell.
|
|
maybeAnonBox = maybeAnonBox->GetParent();
|
|
}
|
|
return maybeAnonBox->StyleContext()->IsWrapperAnonBox();
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// 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).
|
|
|
|
// After this function returns, aLink is pointing to the first link at or
|
|
// after its starting position for which the next frame is a block. If there
|
|
// is no such link, it points to the end of the list.
|
|
static void
|
|
FindFirstBlock(nsFrameList::FrameLinkEnumerator& aLink)
|
|
{
|
|
for ( ; !aLink.AtEnd(); aLink.Next()) {
|
|
if (!aLink.NextFrame()->IsInlineOutside()) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// This function returns a frame link enumerator pointing to the first link in
|
|
// the list for which the next frame is not block. If there is no such link,
|
|
// it points to the end of the list.
|
|
static nsFrameList::FrameLinkEnumerator
|
|
FindFirstNonBlock(const nsFrameList& aList)
|
|
{
|
|
nsFrameList::FrameLinkEnumerator link(aList);
|
|
for (; !link.AtEnd(); link.Next()) {
|
|
if (link.NextFrame()->IsInlineOutside()) {
|
|
break;
|
|
}
|
|
}
|
|
return link;
|
|
}
|
|
|
|
inline void
|
|
SetInitialSingleChild(nsContainerFrame* aParent, nsIFrame* aFrame)
|
|
{
|
|
NS_PRECONDITION(!aFrame->GetNextSibling(), "Should be using a frame list");
|
|
nsFrameList temp(aFrame, aFrame);
|
|
aParent->SetInitialChildList(kPrincipalList, temp);
|
|
}
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
// Structure used when constructing formatting object trees.
|
|
struct nsFrameItems : public nsFrameList
|
|
{
|
|
// Appends the frame to the end of the list
|
|
void AddChild(nsIFrame* aChild);
|
|
};
|
|
|
|
void
|
|
nsFrameItems::AddChild(nsIFrame* aChild)
|
|
{
|
|
NS_PRECONDITION(aChild, "nsFrameItems::AddChild");
|
|
|
|
// It'd be really nice if we could just AppendFrames(kPrincipalList, aChild) here,
|
|
// but some of our callers put frames that have different
|
|
// parents (caption, I'm looking at you) on the same framelist, and
|
|
// nsFrameList asserts if you try to do that.
|
|
if (IsEmpty()) {
|
|
SetFrames(aChild);
|
|
}
|
|
else {
|
|
NS_ASSERTION(aChild != mLastChild,
|
|
"Same frame being added to frame list twice?");
|
|
mLastChild->SetNextSibling(aChild);
|
|
mLastChild = nsLayoutUtils::GetLastSibling(aChild);
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
// Structure used when constructing formatting object trees. Contains
|
|
// state information needed for absolutely positioned elements
|
|
struct nsAbsoluteItems : nsFrameItems {
|
|
// containing block for absolutely positioned elements
|
|
nsContainerFrame* containingBlock;
|
|
|
|
explicit nsAbsoluteItems(nsContainerFrame* aContainingBlock);
|
|
#ifdef DEBUG
|
|
// XXXbz Does this need a debug-only assignment operator that nulls out the
|
|
// childList in the nsAbsoluteItems we're copying? Introducing a difference
|
|
// between debug and non-debug behavior seems bad, so I guess not...
|
|
~nsAbsoluteItems() {
|
|
NS_ASSERTION(!FirstChild(),
|
|
"Dangling child list. Someone forgot to insert it?");
|
|
}
|
|
#endif
|
|
|
|
// Appends the frame to the end of the list
|
|
void AddChild(nsIFrame* aChild);
|
|
};
|
|
|
|
nsAbsoluteItems::nsAbsoluteItems(nsContainerFrame* aContainingBlock)
|
|
: containingBlock(aContainingBlock)
|
|
{
|
|
}
|
|
|
|
// Additional behavior is that it sets the frame's NS_FRAME_OUT_OF_FLOW flag
|
|
void
|
|
nsAbsoluteItems::AddChild(nsIFrame* aChild)
|
|
{
|
|
aChild->AddStateBits(NS_FRAME_OUT_OF_FLOW);
|
|
NS_ASSERTION(aChild->GetPlaceholderFrame(),
|
|
"Child without placeholder being added to nsAbsoluteItems?");
|
|
nsFrameItems::AddChild(aChild);
|
|
}
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
// 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:
|
|
typedef nsIFrame::ChildListID ChildListID;
|
|
nsFrameConstructorSaveState();
|
|
~nsFrameConstructorSaveState();
|
|
|
|
private:
|
|
nsAbsoluteItems* mItems; // pointer to struct whose data we save/restore
|
|
nsAbsoluteItems mSavedItems; // copy of original data
|
|
|
|
// The name of the child list in which our frames would belong
|
|
ChildListID mChildListID;
|
|
nsFrameConstructorState* mState;
|
|
|
|
// State used only when we're saving the abs-pos state for a transformed
|
|
// element.
|
|
nsAbsoluteItems mSavedFixedItems;
|
|
|
|
bool mSavedFixedPosIsAbsPos;
|
|
|
|
friend class nsFrameConstructorState;
|
|
};
|
|
|
|
// Structure used to keep track of a list of bindings we need to call
|
|
// AddToAttachedQueue on. These should be in post-order depth-first
|
|
// flattened tree traversal order.
|
|
struct PendingBinding : public LinkedListElement<PendingBinding>
|
|
{
|
|
#ifdef NS_BUILD_REFCNT_LOGGING
|
|
PendingBinding() {
|
|
MOZ_COUNT_CTOR(PendingBinding);
|
|
}
|
|
~PendingBinding() {
|
|
MOZ_COUNT_DTOR(PendingBinding);
|
|
}
|
|
#endif
|
|
|
|
RefPtr<nsXBLBinding> mBinding;
|
|
};
|
|
|
|
// Structure used for maintaining state information during the
|
|
// frame construction process
|
|
class MOZ_STACK_CLASS nsFrameConstructorState {
|
|
public:
|
|
typedef nsIFrame::ChildListID ChildListID;
|
|
|
|
nsPresContext *mPresContext;
|
|
nsIPresShell *mPresShell;
|
|
nsFrameManager *mFrameManager;
|
|
|
|
#ifdef MOZ_XUL
|
|
// Frames destined for the kPopupList.
|
|
nsAbsoluteItems mPopupItems;
|
|
#endif
|
|
|
|
// Containing block information for out-of-flow frames.
|
|
nsAbsoluteItems mFixedItems;
|
|
nsAbsoluteItems mAbsoluteItems;
|
|
nsAbsoluteItems mFloatedItems;
|
|
// The containing block of a frame in the top layer is defined by the
|
|
// spec: fixed-positioned frames are children of the viewport frame,
|
|
// and absolutely-positioned frames are children of the initial
|
|
// containing block. They would not be caught by any other containing
|
|
// block, e.g. frames with transform or filter.
|
|
nsAbsoluteItems mTopLayerFixedItems;
|
|
nsAbsoluteItems mTopLayerAbsoluteItems;
|
|
|
|
nsCOMPtr<nsILayoutHistoryState> mFrameState;
|
|
// These bits will be added to the state bits of any frame we construct
|
|
// using this state.
|
|
nsFrameState mAdditionalStateBits;
|
|
|
|
// When working with the transform and filter properties, we want to hook
|
|
// the abs-pos and fixed-pos lists together, since such
|
|
// elements are fixed-pos containing blocks. This flag determines
|
|
// whether or not we want to wire the fixed-pos and abs-pos lists
|
|
// together.
|
|
bool mFixedPosIsAbsPos;
|
|
|
|
// A boolean to indicate whether we have a "pending" popupgroup. That is, we
|
|
// have already created the FrameConstructionItem for the root popupgroup but
|
|
// we have not yet created the relevant frame.
|
|
bool mHavePendingPopupgroup;
|
|
|
|
// 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;
|
|
|
|
nsCOMArray<nsIContent> mGeneratedTextNodesWithInitializer;
|
|
|
|
// Selector matching context for. This is null when we're using the Servo style
|
|
// system.
|
|
TreeMatchContext* mTreeMatchContext;
|
|
|
|
// Constructor
|
|
// Use the passed-in history state.
|
|
//
|
|
// aTreeMatchContext is null when we're using the Servo style system.
|
|
nsFrameConstructorState(
|
|
nsIPresShell* aPresShell,
|
|
TreeMatchContext* aTreeMatchContext,
|
|
nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock,
|
|
already_AddRefed<nsILayoutHistoryState> aHistoryState);
|
|
// Get the history state from the pres context's pres shell.
|
|
nsFrameConstructorState(nsIPresShell* aPresShell,
|
|
TreeMatchContext* aTreeMatchContext,
|
|
nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock);
|
|
|
|
~nsFrameConstructorState();
|
|
|
|
bool HasAncestorFilter()
|
|
{
|
|
return mTreeMatchContext && mTreeMatchContext->mAncestorFilter.HasFilter();
|
|
}
|
|
|
|
// 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 push the existing float containing block state and
|
|
// create a new scope. Code that uses this function should get matching
|
|
// logic in GetFloatContainingBlock.
|
|
// Pushing a null float containing block forbids any frames from being
|
|
// floated until a new float containing block is pushed.
|
|
// XXX we should get rid of null float containing blocks and teach the
|
|
// various frame classes to deal with floats instead.
|
|
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;
|
|
|
|
/**
|
|
* 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 aFrameItems the list to add in-flow frames to
|
|
* @param aContent the content pointer for aNewFrame
|
|
* @param aStyleContext the style context resolved for aContent
|
|
* @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
|
|
* @param aIsOutOfFlowPopup pass true if the frame is an out-of-flow popup
|
|
* (XUL-only)
|
|
*/
|
|
void AddChild(nsIFrame* aNewFrame,
|
|
nsFrameItems& aFrameItems,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
nsContainerFrame* aParentFrame,
|
|
bool aCanBePositioned = true,
|
|
bool aCanBeFloated = true,
|
|
bool aIsOutOfFlowPopup = false,
|
|
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.
|
|
*/
|
|
nsAbsoluteItems& GetFixedItems()
|
|
{
|
|
return mFixedPosIsAbsPos ? mAbsoluteItems : mFixedItems;
|
|
}
|
|
const nsAbsoluteItems& GetFixedItems() const
|
|
{
|
|
return mFixedPosIsAbsPos ? mAbsoluteItems : mFixedItems;
|
|
}
|
|
|
|
|
|
/**
|
|
* class to automatically push and pop a pending binding in the frame
|
|
* constructor state. See nsCSSFrameConstructor::FrameConstructionItem
|
|
* mPendingBinding documentation.
|
|
*/
|
|
class PendingBindingAutoPusher;
|
|
friend class PendingBindingAutoPusher;
|
|
class MOZ_STACK_CLASS PendingBindingAutoPusher {
|
|
public:
|
|
PendingBindingAutoPusher(nsFrameConstructorState& aState,
|
|
PendingBinding* aPendingBinding) :
|
|
mState(aState),
|
|
mPendingBinding(aState.mCurrentPendingBindingInsertionPoint)
|
|
{
|
|
if (aPendingBinding) {
|
|
aState.mCurrentPendingBindingInsertionPoint = aPendingBinding;
|
|
}
|
|
}
|
|
|
|
~PendingBindingAutoPusher()
|
|
{
|
|
mState.mCurrentPendingBindingInsertionPoint = mPendingBinding;
|
|
}
|
|
|
|
private:
|
|
nsFrameConstructorState& mState;
|
|
PendingBinding* mPendingBinding;
|
|
};
|
|
|
|
/**
|
|
* Add a new pending binding to the list
|
|
*/
|
|
void AddPendingBinding(PendingBinding* aPendingBinding) {
|
|
if (mCurrentPendingBindingInsertionPoint) {
|
|
mCurrentPendingBindingInsertionPoint->setPrevious(aPendingBinding);
|
|
} else {
|
|
mPendingBindings.insertBack(aPendingBinding);
|
|
}
|
|
}
|
|
|
|
protected:
|
|
friend class nsFrameConstructorSaveState;
|
|
|
|
/**
|
|
* ProcessFrameInsertions takes the frames in aFrameItems and adds them as
|
|
* kids to the aChildListID child list of |aFrameItems.containingBlock|.
|
|
*/
|
|
void ProcessFrameInsertions(nsAbsoluteItems& aFrameItems,
|
|
ChildListID aChildListID);
|
|
|
|
/**
|
|
* GetOutOfFlowFrameItems 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.
|
|
*/
|
|
nsAbsoluteItems* GetOutOfFlowFrameItems(nsIFrame* aNewFrame,
|
|
bool aCanBePositioned,
|
|
bool aCanBeFloated,
|
|
bool aIsOutOfFlowPopup,
|
|
nsFrameState* aPlaceholderType);
|
|
|
|
void ConstructBackdropFrameFor(nsIContent* aContent, nsIFrame* aFrame);
|
|
|
|
// Our list of all pending bindings. When we're done, we need to call
|
|
// AddToAttachedQueue on all of them, in order.
|
|
LinkedList<PendingBinding> mPendingBindings;
|
|
|
|
PendingBinding* mCurrentPendingBindingInsertionPoint;
|
|
};
|
|
|
|
nsFrameConstructorState::nsFrameConstructorState(
|
|
nsIPresShell* aPresShell,
|
|
TreeMatchContext* aTreeMatchContext,
|
|
nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock,
|
|
already_AddRefed<nsILayoutHistoryState> aHistoryState)
|
|
: mPresContext(aPresShell->GetPresContext()),
|
|
mPresShell(aPresShell),
|
|
mFrameManager(aPresShell->FrameManager()),
|
|
#ifdef MOZ_XUL
|
|
mPopupItems(nullptr),
|
|
#endif
|
|
mFixedItems(aFixedContainingBlock),
|
|
mAbsoluteItems(aAbsoluteContainingBlock),
|
|
mFloatedItems(aFloatContainingBlock),
|
|
mTopLayerFixedItems(
|
|
static_cast<nsContainerFrame*>(mFrameManager->GetRootFrame())),
|
|
mTopLayerAbsoluteItems(
|
|
aPresShell->FrameConstructor()->GetDocElementContainingBlock()),
|
|
// See PushAbsoluteContaningBlock below
|
|
mFrameState(aHistoryState),
|
|
mAdditionalStateBits(nsFrameState(0)),
|
|
// If the fixed-pos containing block is equal to the abs-pos containing
|
|
// block, use the abs-pos containing block's abs-pos list for fixed-pos
|
|
// frames.
|
|
mFixedPosIsAbsPos(aFixedContainingBlock == aAbsoluteContainingBlock),
|
|
mHavePendingPopupgroup(false),
|
|
mCreatingExtraFrames(false),
|
|
mTreeMatchContext(aTreeMatchContext),
|
|
mCurrentPendingBindingInsertionPoint(nullptr)
|
|
{
|
|
#ifdef MOZ_XUL
|
|
nsIRootBox* rootBox = nsIRootBox::GetRootBox(aPresShell);
|
|
if (rootBox) {
|
|
mPopupItems.containingBlock = rootBox->GetPopupSetFrame();
|
|
}
|
|
#endif
|
|
MOZ_COUNT_CTOR(nsFrameConstructorState);
|
|
}
|
|
|
|
nsFrameConstructorState::nsFrameConstructorState(nsIPresShell* aPresShell,
|
|
TreeMatchContext* aTreeMatchContext,
|
|
nsContainerFrame* aFixedContainingBlock,
|
|
nsContainerFrame* aAbsoluteContainingBlock,
|
|
nsContainerFrame* aFloatContainingBlock)
|
|
: nsFrameConstructorState(aPresShell,
|
|
aTreeMatchContext,
|
|
aFixedContainingBlock,
|
|
aAbsoluteContainingBlock,
|
|
aFloatContainingBlock,
|
|
aPresShell->GetDocument()->GetLayoutHistoryState())
|
|
{
|
|
}
|
|
|
|
nsFrameConstructorState::~nsFrameConstructorState()
|
|
{
|
|
MOZ_COUNT_DTOR(nsFrameConstructorState);
|
|
ProcessFrameInsertions(mTopLayerFixedItems, nsIFrame::kFixedList);
|
|
ProcessFrameInsertions(mTopLayerAbsoluteItems, nsIFrame::kAbsoluteList);
|
|
ProcessFrameInsertions(mFloatedItems, nsIFrame::kFloatList);
|
|
ProcessFrameInsertions(mAbsoluteItems, nsIFrame::kAbsoluteList);
|
|
ProcessFrameInsertions(mFixedItems, nsIFrame::kFixedList);
|
|
#ifdef MOZ_XUL
|
|
ProcessFrameInsertions(mPopupItems, nsIFrame::kPopupList);
|
|
#endif
|
|
for (int32_t i = mGeneratedTextNodesWithInitializer.Count() - 1; i >= 0; --i) {
|
|
mGeneratedTextNodesWithInitializer[i]->
|
|
DeleteProperty(nsGkAtoms::genConInitializerProperty);
|
|
}
|
|
if (!mPendingBindings.isEmpty()) {
|
|
nsBindingManager* bindingManager = mPresShell->GetDocument()->BindingManager();
|
|
do {
|
|
nsAutoPtr<PendingBinding> pendingBinding;
|
|
pendingBinding = mPendingBindings.popFirst();
|
|
bindingManager->AddToAttachedQueue(pendingBinding->mBinding);
|
|
} while (!mPendingBindings.isEmpty());
|
|
mCurrentPendingBindingInsertionPoint = nullptr;
|
|
}
|
|
}
|
|
|
|
static nsContainerFrame*
|
|
AdjustAbsoluteContainingBlock(nsContainerFrame* aContainingBlockIn)
|
|
{
|
|
if (!aContainingBlockIn) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Always use the container's first continuation. (Inline frames can have
|
|
// non-fluid bidi continuations...)
|
|
return static_cast<nsContainerFrame*>(aContainingBlockIn->FirstContinuation());
|
|
}
|
|
|
|
void
|
|
nsFrameConstructorState::PushAbsoluteContainingBlock(nsContainerFrame* aNewAbsoluteContainingBlock,
|
|
nsIFrame* aPositionedFrame,
|
|
nsFrameConstructorSaveState& aSaveState)
|
|
{
|
|
aSaveState.mItems = &mAbsoluteItems;
|
|
aSaveState.mSavedItems = mAbsoluteItems;
|
|
aSaveState.mChildListID = nsIFrame::kAbsoluteList;
|
|
aSaveState.mState = this;
|
|
aSaveState.mSavedFixedPosIsAbsPos = mFixedPosIsAbsPos;
|
|
|
|
if (mFixedPosIsAbsPos) {
|
|
// Since we're going to replace mAbsoluteItems, we need to save it into
|
|
// mFixedItems now (and save the current value of mFixedItems).
|
|
aSaveState.mSavedFixedItems = mFixedItems;
|
|
mFixedItems = mAbsoluteItems;
|
|
}
|
|
|
|
mAbsoluteItems =
|
|
nsAbsoluteItems(AdjustAbsoluteContainingBlock(aNewAbsoluteContainingBlock));
|
|
|
|
/* See if we're wiring the fixed-pos and abs-pos lists together. This happens iff
|
|
* we're a transformed element.
|
|
*/
|
|
mFixedPosIsAbsPos = aPositionedFrame &&
|
|
aPositionedFrame->IsFixedPosContainingBlock();
|
|
|
|
if (aNewAbsoluteContainingBlock) {
|
|
aNewAbsoluteContainingBlock->MarkAsAbsoluteContainingBlock();
|
|
}
|
|
}
|
|
|
|
void
|
|
nsFrameConstructorState::PushFloatContainingBlock(nsContainerFrame* aNewFloatContainingBlock,
|
|
nsFrameConstructorSaveState& aSaveState)
|
|
{
|
|
NS_PRECONDITION(!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.mItems = &mFloatedItems;
|
|
aSaveState.mSavedItems = mFloatedItems;
|
|
aSaveState.mChildListID = nsIFrame::kFloatList;
|
|
aSaveState.mState = this;
|
|
mFloatedItems = nsAbsoluteItems(aNewFloatContainingBlock);
|
|
}
|
|
|
|
nsContainerFrame*
|
|
nsFrameConstructorState::GetGeometricParent(const nsStyleDisplay* aStyleDisplay,
|
|
nsContainerFrame* aContentParentFrame) const
|
|
{
|
|
NS_PRECONDITION(aStyleDisplay, "Must have display struct!");
|
|
|
|
// 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 &&
|
|
nsSVGUtils::IsInSVGTextSubtree(aContentParentFrame)) {
|
|
return aContentParentFrame;
|
|
}
|
|
|
|
if (aStyleDisplay->IsFloatingStyle() && mFloatedItems.containingBlock) {
|
|
NS_ASSERTION(!aStyleDisplay->IsAbsolutelyPositionedStyle(),
|
|
"Absolutely positioned _and_ floating?");
|
|
return mFloatedItems.containingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay->mTopLayer != NS_STYLE_TOP_LAYER_NONE) {
|
|
MOZ_ASSERT(aStyleDisplay->mTopLayer == NS_STYLE_TOP_LAYER_TOP,
|
|
"-moz-top-layer should be either none or top");
|
|
MOZ_ASSERT(aStyleDisplay->IsAbsolutelyPositionedStyle(),
|
|
"Top layer items should always be absolutely positioned");
|
|
if (aStyleDisplay->mPosition == NS_STYLE_POSITION_FIXED) {
|
|
MOZ_ASSERT(mTopLayerFixedItems.containingBlock, "No root frame?");
|
|
return mTopLayerFixedItems.containingBlock;
|
|
}
|
|
MOZ_ASSERT(aStyleDisplay->mPosition == NS_STYLE_POSITION_ABSOLUTE);
|
|
MOZ_ASSERT(mTopLayerAbsoluteItems.containingBlock);
|
|
return mTopLayerAbsoluteItems.containingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay->mPosition == NS_STYLE_POSITION_ABSOLUTE &&
|
|
mAbsoluteItems.containingBlock) {
|
|
return mAbsoluteItems.containingBlock;
|
|
}
|
|
|
|
if (aStyleDisplay->mPosition == NS_STYLE_POSITION_FIXED &&
|
|
GetFixedItems().containingBlock) {
|
|
return GetFixedItems().containingBlock;
|
|
}
|
|
|
|
return aContentParentFrame;
|
|
}
|
|
|
|
nsAbsoluteItems*
|
|
nsFrameConstructorState::GetOutOfFlowFrameItems(nsIFrame* aNewFrame,
|
|
bool aCanBePositioned,
|
|
bool aCanBeFloated,
|
|
bool aIsOutOfFlowPopup,
|
|
nsFrameState* aPlaceholderType)
|
|
{
|
|
#ifdef MOZ_XUL
|
|
if (MOZ_UNLIKELY(aIsOutOfFlowPopup)) {
|
|
MOZ_ASSERT(mPopupItems.containingBlock, "Must have a popup set frame!");
|
|
*aPlaceholderType = PLACEHOLDER_FOR_POPUP;
|
|
return &mPopupItems;
|
|
}
|
|
#endif // MOZ_XUL
|
|
if (aCanBeFloated && aNewFrame->IsFloating()) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_FLOAT;
|
|
return &mFloatedItems;
|
|
}
|
|
|
|
if (aCanBePositioned) {
|
|
const nsStyleDisplay* disp = aNewFrame->StyleDisplay();
|
|
if (disp->mTopLayer != NS_STYLE_TOP_LAYER_NONE) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_TOPLAYER;
|
|
if (disp->mPosition == NS_STYLE_POSITION_FIXED) {
|
|
*aPlaceholderType |= PLACEHOLDER_FOR_FIXEDPOS;
|
|
return &mTopLayerFixedItems;
|
|
}
|
|
*aPlaceholderType |= PLACEHOLDER_FOR_ABSPOS;
|
|
return &mTopLayerAbsoluteItems;
|
|
}
|
|
if (disp->mPosition == NS_STYLE_POSITION_ABSOLUTE) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_ABSPOS;
|
|
return &mAbsoluteItems;
|
|
}
|
|
if (disp->mPosition == NS_STYLE_POSITION_FIXED) {
|
|
*aPlaceholderType = PLACEHOLDER_FOR_FIXEDPOS;
|
|
return &GetFixedItems();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
nsFrameConstructorState::ConstructBackdropFrameFor(nsIContent* aContent,
|
|
nsIFrame* aFrame)
|
|
{
|
|
MOZ_ASSERT(aFrame->StyleDisplay()->mTopLayer == NS_STYLE_TOP_LAYER_TOP);
|
|
nsContainerFrame* frame = do_QueryFrame(aFrame);
|
|
if (!frame) {
|
|
NS_WARNING("Cannot create backdrop frame for non-container frame");
|
|
return;
|
|
}
|
|
|
|
RefPtr<nsStyleContext> style = mPresShell->StyleSet()->
|
|
ResolvePseudoElementStyle(aContent->AsElement(),
|
|
CSSPseudoElementType::backdrop,
|
|
/* aParentStyleContext */ nullptr,
|
|
/* aPseudoElement */ nullptr);
|
|
MOZ_ASSERT(style->StyleDisplay()->mTopLayer == NS_STYLE_TOP_LAYER_TOP);
|
|
nsContainerFrame* parentFrame =
|
|
GetGeometricParent(style->StyleDisplay(), nullptr);
|
|
|
|
nsBackdropFrame* backdropFrame = new (mPresShell) nsBackdropFrame(style);
|
|
backdropFrame->Init(aContent, parentFrame, nullptr);
|
|
|
|
nsFrameState placeholderType;
|
|
nsAbsoluteItems* frameItems = GetOutOfFlowFrameItems(backdropFrame,
|
|
true, true, false,
|
|
&placeholderType);
|
|
MOZ_ASSERT(placeholderType & PLACEHOLDER_FOR_TOPLAYER);
|
|
|
|
nsIFrame* placeholder = nsCSSFrameConstructor::
|
|
CreatePlaceholderFrameFor(mPresShell, aContent, backdropFrame,
|
|
frame, nullptr, placeholderType);
|
|
nsFrameList temp(placeholder, placeholder);
|
|
frame->SetInitialChildList(nsIFrame::kBackdropList, temp);
|
|
|
|
frameItems->AddChild(backdropFrame);
|
|
}
|
|
|
|
void
|
|
nsFrameConstructorState::AddChild(nsIFrame* aNewFrame,
|
|
nsFrameItems& aFrameItems,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
nsContainerFrame* aParentFrame,
|
|
bool aCanBePositioned,
|
|
bool aCanBeFloated,
|
|
bool aIsOutOfFlowPopup,
|
|
bool aInsertAfter,
|
|
nsIFrame* aInsertAfterFrame)
|
|
{
|
|
NS_PRECONDITION(!aNewFrame->GetNextSibling(), "Shouldn't happen");
|
|
|
|
nsFrameState placeholderType;
|
|
nsAbsoluteItems* outOfFlowFrameItems =
|
|
GetOutOfFlowFrameItems(aNewFrame, aCanBePositioned, aCanBeFloated,
|
|
aIsOutOfFlowPopup, &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.
|
|
nsFrameItems* frameItems;
|
|
if (outOfFlowFrameItems && outOfFlowFrameItems->containingBlock) {
|
|
MOZ_ASSERT(aNewFrame->GetParent() == outOfFlowFrameItems->containingBlock,
|
|
"Parent of the frame is not the containing block?");
|
|
frameItems = outOfFlowFrameItems;
|
|
} else {
|
|
frameItems = &aFrameItems;
|
|
placeholderType = nsFrameState(0);
|
|
}
|
|
|
|
if (placeholderType) {
|
|
NS_ASSERTION(frameItems != &aFrameItems,
|
|
"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
|
|
aFrameItems.AddChild(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) {
|
|
frameItems->InsertFrame(nullptr, aInsertAfterFrame, aNewFrame);
|
|
} else {
|
|
frameItems->AddChild(aNewFrame);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsFrameConstructorState::ProcessFrameInsertions(nsAbsoluteItems& aFrameItems,
|
|
ChildListID aChildListID)
|
|
{
|
|
#define NS_NONXUL_LIST_TEST (&aFrameItems == &mFloatedItems && \
|
|
aChildListID == nsIFrame::kFloatList) || \
|
|
((&aFrameItems == &mAbsoluteItems || \
|
|
&aFrameItems == &mTopLayerAbsoluteItems) && \
|
|
aChildListID == nsIFrame::kAbsoluteList) || \
|
|
((&aFrameItems == &mFixedItems || \
|
|
&aFrameItems == &mTopLayerFixedItems) && \
|
|
aChildListID == nsIFrame::kFixedList)
|
|
#ifdef MOZ_XUL
|
|
NS_PRECONDITION(NS_NONXUL_LIST_TEST ||
|
|
(&aFrameItems == &mPopupItems &&
|
|
aChildListID == nsIFrame::kPopupList),
|
|
"Unexpected aFrameItems/aChildListID combination");
|
|
#else
|
|
NS_PRECONDITION(NS_NONXUL_LIST_TEST,
|
|
"Unexpected aFrameItems/aChildListID combination");
|
|
#endif
|
|
|
|
if (aFrameItems.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
nsContainerFrame* containingBlock = aFrameItems.containingBlock;
|
|
|
|
NS_ASSERTION(containingBlock,
|
|
"Child list without containing block?");
|
|
|
|
if (aChildListID == nsIFrame::kFixedList) {
|
|
// 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->GetStateBits() & 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, aFrameItems);
|
|
} else {
|
|
containingBlock->SetInitialChildList(aChildListID, aFrameItems);
|
|
}
|
|
} else if (aChildListID == nsIFrame::kFixedList ||
|
|
aChildListID == nsIFrame::kAbsoluteList) {
|
|
// The order is not important for abs-pos/fixed-pos frame list, just
|
|
// append the frame items to the list directly.
|
|
mFrameManager->AppendFrames(containingBlock, aChildListID, aFrameItems);
|
|
} 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 = aFrameItems.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
|
|
mFrameManager->AppendFrames(containingBlock, aChildListID, aFrameItems);
|
|
} 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;
|
|
}
|
|
}
|
|
mFrameManager->InsertFrames(containingBlock, aChildListID,
|
|
insertionPoint, aFrameItems);
|
|
}
|
|
}
|
|
|
|
MOZ_ASSERT(aFrameItems.IsEmpty(), "How did that happen?");
|
|
}
|
|
|
|
|
|
nsFrameConstructorSaveState::nsFrameConstructorSaveState()
|
|
: mItems(nullptr),
|
|
mSavedItems(nullptr),
|
|
mChildListID(kPrincipalList),
|
|
mState(nullptr),
|
|
mSavedFixedItems(nullptr),
|
|
mSavedFixedPosIsAbsPos(false)
|
|
{
|
|
}
|
|
|
|
nsFrameConstructorSaveState::~nsFrameConstructorSaveState()
|
|
{
|
|
// Restore the state
|
|
if (mItems) {
|
|
NS_ASSERTION(mState, "Can't have mItems set without having a state!");
|
|
mState->ProcessFrameInsertions(*mItems, mChildListID);
|
|
*mItems = mSavedItems;
|
|
#ifdef DEBUG
|
|
// We've transferred the child list, so drop the pointer we held to it.
|
|
// Note that this only matters for the assert in ~nsAbsoluteItems.
|
|
mSavedItems.Clear();
|
|
#endif
|
|
if (mItems == &mState->mAbsoluteItems) {
|
|
mState->mFixedPosIsAbsPos = mSavedFixedPosIsAbsPos;
|
|
if (mSavedFixedPosIsAbsPos) {
|
|
// mAbsoluteItems was moved to mFixedItems, so move mFixedItems back
|
|
// and repair the old mFixedItems now.
|
|
mState->mAbsoluteItems = mState->mFixedItems;
|
|
mState->mFixedItems = mSavedFixedItems;
|
|
#ifdef DEBUG
|
|
mSavedFixedItems.Clear();
|
|
#endif
|
|
}
|
|
}
|
|
NS_ASSERTION(!mItems->LastChild() || !mItems->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.
|
|
* nsFrame::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 contexts.
|
|
*/
|
|
// 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)
|
|
{
|
|
bool sameGrandParent = aOldParent->GetParent() == aNewParent->GetParent();
|
|
|
|
if (aNewParent->HasView() || aOldParent->HasView() || !sameGrandParent) {
|
|
// Move the frames into the new view
|
|
nsContainerFrame::ReparentFrameViewList(aFrameList, aOldParent, aNewParent);
|
|
}
|
|
|
|
for (nsFrameList::Enumerator e(aFrameList); !e.AtEnd(); e.Next()) {
|
|
e.get()->SetParent(aNewParent);
|
|
}
|
|
|
|
if (aNewParent->PrincipalChildList().IsEmpty() &&
|
|
(aNewParent->GetStateBits() & NS_FRAME_FIRST_REFLOW)) {
|
|
aNewParent->SetInitialChildList(kPrincipalList, aFrameList);
|
|
} else {
|
|
aNewParent->AppendFrames(kPrincipalList, aFrameList);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
nsCSSFrameConstructor::nsCSSFrameConstructor(nsIDocument* aDocument,
|
|
nsIPresShell* aPresShell)
|
|
: nsFrameManager(aPresShell)
|
|
, mDocument(aDocument)
|
|
, mRootElementFrame(nullptr)
|
|
, mRootElementStyleFrame(nullptr)
|
|
, mDocElementContainingBlock(nullptr)
|
|
, mPageSequenceFrame(nullptr)
|
|
, mFirstFreeFCItem(nullptr)
|
|
, mFCItemsInUse(0)
|
|
, mCurrentDepth(0)
|
|
#ifdef DEBUG
|
|
, mUpdateCount(0)
|
|
#endif
|
|
, mQuotesDirty(false)
|
|
, mCountersDirty(false)
|
|
, mIsDestroyingFrameTree(false)
|
|
, mHasRootAbsPosContainingBlock(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 = PL_strchr(flags, ',');
|
|
if (comma)
|
|
*comma = '\0';
|
|
|
|
bool found = false;
|
|
FrameCtorDebugFlags* flag = gFlags;
|
|
FrameCtorDebugFlags* limit = gFlags + NUM_DEBUG_FLAGS;
|
|
while (flag < limit) {
|
|
if (PL_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)
|
|
{
|
|
NS_PRECONDITION(mUpdateCount != 0,
|
|
"Should be in an update while destroying frames");
|
|
|
|
if (aFrame->GetStateBits() & NS_FRAME_GENERATED_CONTENT) {
|
|
if (mQuoteList.DestroyNodesFor(aFrame))
|
|
QuotesDirty();
|
|
}
|
|
|
|
if (aFrame->HasAnyStateBits(NS_FRAME_HAS_CSS_COUNTER_STYLE) &&
|
|
mCounterManager.DestroyNodesFor(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);
|
|
|
|
nsFrameManager::NotifyDestroyingFrame(aFrame);
|
|
}
|
|
|
|
static bool
|
|
HasGeneratedContent(const nsIContent* aChild)
|
|
{
|
|
if (!aChild->MayHaveAnonymousChildren()) {
|
|
return false;
|
|
}
|
|
|
|
return nsLayoutUtils::GetBeforeFrame(aChild) ||
|
|
nsLayoutUtils::GetAfterFrame(aChild);
|
|
}
|
|
|
|
struct nsGenConInitializer {
|
|
nsAutoPtr<nsGenConNode> mNode;
|
|
nsGenConList* mList;
|
|
void (nsCSSFrameConstructor::*mDirtyAll)();
|
|
|
|
nsGenConInitializer(nsGenConNode* aNode, nsGenConList* aList,
|
|
void (nsCSSFrameConstructor::*aDirtyAll)())
|
|
: mNode(aNode), mList(aList), mDirtyAll(aDirtyAll) {}
|
|
};
|
|
|
|
already_AddRefed<nsIContent>
|
|
nsCSSFrameConstructor::CreateGenConTextNode(nsFrameConstructorState& aState,
|
|
const nsString& aString,
|
|
RefPtr<nsTextNode>* aText,
|
|
nsGenConInitializer* aInitializer)
|
|
{
|
|
RefPtr<nsTextNode> content = new nsTextNode(mDocument->NodeInfoManager());
|
|
content->SetText(aString, false);
|
|
if (aText) {
|
|
*aText = content;
|
|
}
|
|
if (aInitializer) {
|
|
content->SetProperty(nsGkAtoms::genConInitializerProperty, aInitializer,
|
|
nsINode::DeleteProperty<nsGenConInitializer>);
|
|
aState.mGeneratedTextNodesWithInitializer.AppendObject(content);
|
|
}
|
|
return content.forget();
|
|
}
|
|
|
|
already_AddRefed<nsIContent>
|
|
nsCSSFrameConstructor::CreateGeneratedContent(nsFrameConstructorState& aState,
|
|
nsIContent* aParentContent,
|
|
nsStyleContext* aStyleContext,
|
|
uint32_t aContentIndex)
|
|
{
|
|
// Get the content value
|
|
const nsStyleContentData &data =
|
|
aStyleContext->StyleContent()->ContentAt(aContentIndex);
|
|
nsStyleContentType type = data.GetType();
|
|
|
|
switch (type) {
|
|
case eStyleContentType_Image: {
|
|
imgRequestProxy* image = data.GetImage();
|
|
if (!image) {
|
|
// CSS had something specified that couldn't be converted to an
|
|
// image object
|
|
return nullptr;
|
|
}
|
|
|
|
// Create an image content object and pass it the image request.
|
|
// XXX Check if it's an image type we can handle...
|
|
|
|
RefPtr<NodeInfo> nodeInfo;
|
|
nodeInfo = mDocument->NodeInfoManager()->
|
|
GetNodeInfo(nsGkAtoms::mozgeneratedcontentimage, nullptr,
|
|
kNameSpaceID_XHTML, nsIDOMNode::ELEMENT_NODE);
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewGenConImageContent(getter_AddRefs(content), nodeInfo.forget(),
|
|
image);
|
|
return content.forget();
|
|
}
|
|
|
|
case eStyleContentType_String:
|
|
return CreateGenConTextNode(aState,
|
|
nsDependentString(data.GetString()),
|
|
nullptr, nullptr);
|
|
|
|
case eStyleContentType_Attr: {
|
|
RefPtr<nsAtom> attrName;
|
|
int32_t attrNameSpace = kNameSpaceID_None;
|
|
nsAutoString contentString(data.GetString());
|
|
|
|
int32_t barIndex = contentString.FindChar('|'); // CSS namespace delimiter
|
|
if (-1 != barIndex) {
|
|
nsAutoString nameSpaceVal;
|
|
contentString.Left(nameSpaceVal, barIndex);
|
|
nsresult error;
|
|
attrNameSpace = nameSpaceVal.ToInteger(&error);
|
|
contentString.Cut(0, barIndex + 1);
|
|
if (contentString.Length()) {
|
|
if (mDocument->IsHTMLDocument() && aParentContent->IsHTMLElement()) {
|
|
ToLowerCase(contentString);
|
|
}
|
|
attrName = NS_Atomize(contentString);
|
|
}
|
|
}
|
|
else {
|
|
if (mDocument->IsHTMLDocument() && aParentContent->IsHTMLElement()) {
|
|
ToLowerCase(contentString);
|
|
}
|
|
attrName = NS_Atomize(contentString);
|
|
}
|
|
|
|
if (!attrName) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(),
|
|
attrNameSpace, attrName, getter_AddRefs(content));
|
|
return content.forget();
|
|
}
|
|
|
|
case eStyleContentType_Counter:
|
|
case eStyleContentType_Counters: {
|
|
nsStyleContentData::CounterFunction* counters = data.GetCounters();
|
|
nsCounterList* counterList =
|
|
mCounterManager.CounterListFor(counters->mIdent);
|
|
|
|
nsCounterUseNode* node =
|
|
new nsCounterUseNode(counters, aContentIndex,
|
|
type == eStyleContentType_Counters);
|
|
|
|
nsGenConInitializer* initializer =
|
|
new nsGenConInitializer(node, counterList,
|
|
&nsCSSFrameConstructor::CountersDirty);
|
|
return CreateGenConTextNode(aState, EmptyString(), &node->mText,
|
|
initializer);
|
|
}
|
|
|
|
case eStyleContentType_OpenQuote:
|
|
case eStyleContentType_CloseQuote:
|
|
case eStyleContentType_NoOpenQuote:
|
|
case eStyleContentType_NoCloseQuote: {
|
|
nsQuoteNode* node =
|
|
new nsQuoteNode(type, aContentIndex);
|
|
|
|
nsGenConInitializer* initializer =
|
|
new nsGenConInitializer(node, &mQuoteList,
|
|
&nsCSSFrameConstructor::QuotesDirty);
|
|
return CreateGenConTextNode(aState, EmptyString(), &node->mText,
|
|
initializer);
|
|
}
|
|
|
|
case eStyleContentType_AltContent: {
|
|
// 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 (aParentContent->HasAttr(kNameSpaceID_None, nsGkAtoms::alt)) {
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(),
|
|
kNameSpaceID_None, nsGkAtoms::alt, getter_AddRefs(content));
|
|
return content.forget();
|
|
}
|
|
|
|
if (aParentContent->IsHTMLElement() &&
|
|
aParentContent->NodeInfo()->Equals(nsGkAtoms::input)) {
|
|
if (aParentContent->HasAttr(kNameSpaceID_None, nsGkAtoms::value)) {
|
|
nsCOMPtr<nsIContent> content;
|
|
NS_NewAttributeContent(mDocument->NodeInfoManager(),
|
|
kNameSpaceID_None, nsGkAtoms::value, getter_AddRefs(content));
|
|
return content.forget();
|
|
}
|
|
|
|
nsAutoString temp;
|
|
nsContentUtils::GetLocalizedString(nsContentUtils::eFORMS_PROPERTIES,
|
|
"Submit", temp);
|
|
return CreateGenConTextNode(aState, temp, nullptr, nullptr);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case eStyleContentType_Uninitialized:
|
|
NS_NOTREACHED("uninitialized content type");
|
|
return nullptr;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
/*
|
|
* 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 or
|
|
* _moz_generated_content_after) 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 or
|
|
* ::after style.
|
|
*/
|
|
void
|
|
nsCSSFrameConstructor::CreateGeneratedContentItem(nsFrameConstructorState& aState,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIContent* aParentContent,
|
|
nsStyleContext* aStyleContext,
|
|
CSSPseudoElementType aPseudoElement,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
MOZ_ASSERT(aPseudoElement == CSSPseudoElementType::before ||
|
|
aPseudoElement == CSSPseudoElementType::after,
|
|
"unexpected aPseudoElement");
|
|
|
|
MOZ_ASSERT(aParentContent->IsElement());
|
|
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
|
|
// Probe for the existence of the pseudo-element
|
|
RefPtr<nsStyleContext> pseudoStyleContext;
|
|
pseudoStyleContext =
|
|
styleSet->ProbePseudoElementStyle(aParentContent->AsElement(),
|
|
aPseudoElement,
|
|
aStyleContext,
|
|
aState.mTreeMatchContext);
|
|
if (!pseudoStyleContext)
|
|
return;
|
|
|
|
bool isBefore = aPseudoElement == CSSPseudoElementType::before;
|
|
|
|
// |ProbePseudoStyleFor| checked the 'display' property and the
|
|
// |ContentCount()| of the 'content' property for us.
|
|
RefPtr<NodeInfo> nodeInfo;
|
|
nsAtom* elemName = isBefore ?
|
|
nsGkAtoms::mozgeneratedcontentbefore : nsGkAtoms::mozgeneratedcontentafter;
|
|
nodeInfo = mDocument->NodeInfoManager()->GetNodeInfo(elemName, nullptr,
|
|
kNameSpaceID_None,
|
|
nsIDOMNode::ELEMENT_NODE);
|
|
nsCOMPtr<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.
|
|
nsAtom* property = isBefore
|
|
? nsGkAtoms::beforePseudoProperty : nsGkAtoms::afterPseudoProperty;
|
|
aParentContent->SetProperty(property, container.get());
|
|
|
|
container->SetIsNativeAnonymousRoot();
|
|
container->SetPseudoElementType(aPseudoElement);
|
|
|
|
// If the parent is in a shadow tree, make sure we don't
|
|
// bind with a document because shadow roots and its descendants
|
|
// are not in document.
|
|
nsIDocument* bindDocument =
|
|
aParentContent->HasFlag(NODE_IS_IN_SHADOW_TREE) ? nullptr : mDocument;
|
|
rv = container->BindToTree(bindDocument, aParentContent, aParentContent, true);
|
|
if (NS_FAILED(rv)) {
|
|
container->UnbindFromTree();
|
|
return;
|
|
}
|
|
|
|
// 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.
|
|
auto* servoStyle = pseudoStyleContext->GetAsServo();
|
|
if (servoStyle) {
|
|
bool hasServoAnimations =
|
|
Servo_ComputedValues_SpecifiesAnimationsOrTransitions(servoStyle);
|
|
if (!hasServoAnimations) {
|
|
Servo_SetExplicitStyle(container, servoStyle);
|
|
} 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()->AsServo()->StyleNewSubtree(container);
|
|
pseudoStyleContext =
|
|
styleSet->AsServo()->ResolveServoStyle(container);
|
|
}
|
|
} else {
|
|
mozilla::GeckoRestyleManager* geckoRM = RestyleManager()->AsGecko();
|
|
GeckoRestyleManager::ReframingStyleContexts* rsc =
|
|
geckoRM->GetReframingStyleContexts();
|
|
|
|
if (rsc) {
|
|
RefPtr<GeckoStyleContext> newContext =
|
|
GeckoStyleContext::TakeRef(pseudoStyleContext.forget());
|
|
if (auto* oldStyleContext = rsc->Get(container, aPseudoElement)) {
|
|
GeckoRestyleManager::TryInitiatingTransition(aState.mPresContext,
|
|
container,
|
|
oldStyleContext,
|
|
&newContext);
|
|
} else {
|
|
aState.mPresContext->TransitionManager()->
|
|
PruneCompletedTransitions(aParentContent->AsElement(),
|
|
aPseudoElement, newContext);
|
|
}
|
|
pseudoStyleContext = newContext.forget();
|
|
}
|
|
}
|
|
|
|
uint32_t contentCount = pseudoStyleContext->StyleContent()->ContentCount();
|
|
bool createdChildElement = false;
|
|
for (uint32_t contentIndex = 0; contentIndex < contentCount; contentIndex++) {
|
|
nsCOMPtr<nsIContent> content =
|
|
CreateGeneratedContent(aState, aParentContent, pseudoStyleContext,
|
|
contentIndex);
|
|
if (content) {
|
|
container->AppendChildTo(content, false);
|
|
if (content->IsElement()) {
|
|
createdChildElement = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// We may need to do a synchronous servo traversal in various uncommon cases.
|
|
if (servoStyle) {
|
|
if (createdChildElement) {
|
|
// If we created any children elements, Servo needs to traverse them, but
|
|
// the root is already set up.
|
|
mPresShell->StyleSet()->AsServo()->StyleNewChildren(container);
|
|
}
|
|
}
|
|
|
|
AddFrameConstructionItemsInternal(aState, container, aParentFrame, elemName,
|
|
kNameSpaceID_None, true,
|
|
pseudoStyleContext,
|
|
ITEM_IS_GENERATED_CONTENT, nullptr,
|
|
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)
|
|
{
|
|
nsAtom* pseudoType = aFrame->StyleContext()->GetPseudo();
|
|
return pseudoType &&
|
|
(pseudoType == nsCSSAnonBoxes::table ||
|
|
pseudoType == nsCSSAnonBoxes::inlineTable ||
|
|
pseudoType == nsCSSAnonBoxes::tableColGroup ||
|
|
pseudoType == nsCSSAnonBoxes::tableRowGroup ||
|
|
pseudoType == nsCSSAnonBoxes::tableRow ||
|
|
pseudoType == nsCSSAnonBoxes::tableCell ||
|
|
(pseudoType == nsCSSAnonBoxes::cellContent &&
|
|
aFrame->GetParent()->StyleContext()->GetPseudo() ==
|
|
nsCSSAnonBoxes::tableCell) ||
|
|
(pseudoType == nsCSSAnonBoxes::tableWrapper &&
|
|
(aFrame->PrincipalChildList().FirstChild()->StyleContext()->GetPseudo() ==
|
|
nsCSSAnonBoxes::table ||
|
|
aFrame->PrincipalChildList().FirstChild()->StyleContext()->GetPseudo() ==
|
|
nsCSSAnonBoxes::inlineTable)));
|
|
}
|
|
|
|
static bool
|
|
IsRubyPseudo(nsIFrame* aFrame)
|
|
{
|
|
return RubyUtils::IsRubyPseudo(aFrame->StyleContext()->GetPseudo());
|
|
}
|
|
|
|
static bool
|
|
IsTableOrRubyPseudo(nsIFrame* aFrame)
|
|
{
|
|
return IsTablePseudo(aFrame) || IsRubyPseudo(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;
|
|
}
|
|
|
|
static nsContainerFrame*
|
|
AdjustCaptionParentFrame(nsContainerFrame* aParentFrame)
|
|
{
|
|
if (aParentFrame->IsTableFrame()) {
|
|
return aParentFrame->GetParent();
|
|
}
|
|
return aParentFrame;
|
|
}
|
|
|
|
/**
|
|
* If the parent frame is a |tableFrame| and the child is a
|
|
* |captionFrame|, then we want to insert the frames beneath the
|
|
* |tableFrame|'s parent frame. Returns |true| if the parent frame
|
|
* needed to be fixed up.
|
|
*/
|
|
static bool
|
|
GetCaptionAdjustedParent(nsContainerFrame* aParentFrame,
|
|
const nsIFrame* aChildFrame,
|
|
nsContainerFrame** aAdjParentFrame)
|
|
{
|
|
*aAdjParentFrame = aParentFrame;
|
|
bool haveCaption = false;
|
|
|
|
if (aChildFrame->IsTableCaption()) {
|
|
haveCaption = true;
|
|
*aAdjParentFrame = ::AdjustCaptionParentFrame(aParentFrame);
|
|
}
|
|
return haveCaption;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::AdjustParentFrame(nsContainerFrame** aParentFrame,
|
|
const FrameConstructionData* aFCData,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
NS_PRECONDITION(aStyleContext, "Must have child's style context");
|
|
NS_PRECONDITION(aFCData, "Must have frame construction data");
|
|
|
|
bool tablePart = ((aFCData->mBits & FCDATA_IS_TABLE_PART) != 0);
|
|
|
|
if (tablePart && aStyleContext->StyleDisplay()->mDisplay ==
|
|
StyleDisplay::TableCaption) {
|
|
*aParentFrame = ::AdjustCaptionParentFrame(*aParentFrame);
|
|
}
|
|
}
|
|
|
|
// Pull all the captions present in aItems out into aCaptions
|
|
static void
|
|
PullOutCaptionFrames(nsFrameItems& aItems, nsFrameItems& aCaptions)
|
|
{
|
|
nsIFrame* child = aItems.FirstChild();
|
|
while (child) {
|
|
nsIFrame* nextSibling = child->GetNextSibling();
|
|
if (child->IsTableCaption()) {
|
|
aItems.RemoveFrame(child);
|
|
aCaptions.AddChild(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,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
NS_PRECONDITION(aDisplay->mDisplay == StyleDisplay::Table ||
|
|
aDisplay->mDisplay == StyleDisplay::InlineTable,
|
|
"Unexpected call");
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
const uint32_t nameSpaceID = aItem.mNameSpaceID;
|
|
|
|
// create the pseudo SC for the table wrapper as a child of the inner SC
|
|
RefPtr<nsStyleContext> outerStyleContext;
|
|
outerStyleContext = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::tableWrapper,
|
|
styleContext);
|
|
|
|
// Create the table wrapper frame which holds the caption and inner table frame
|
|
nsContainerFrame* newFrame;
|
|
if (kNameSpaceID_MathML == nameSpaceID)
|
|
newFrame = NS_NewMathMLmtableOuterFrame(mPresShell, outerStyleContext);
|
|
else
|
|
newFrame = NS_NewTableWrapperFrame(mPresShell, outerStyleContext);
|
|
|
|
nsContainerFrame* geometricParent =
|
|
aState.GetGeometricParent(outerStyleContext->StyleDisplay(),
|
|
aParentFrame);
|
|
|
|
// Init the table wrapper frame
|
|
InitAndRestoreFrame(aState, content, geometricParent, newFrame);
|
|
|
|
// Create the inner table frame
|
|
nsContainerFrame* innerFrame;
|
|
if (kNameSpaceID_MathML == nameSpaceID)
|
|
innerFrame = NS_NewMathMLmtableFrame(mPresShell, styleContext);
|
|
else
|
|
innerFrame = NS_NewTableFrame(mPresShell, styleContext);
|
|
|
|
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, aFrameItems, content, styleContext, aParentFrame);
|
|
|
|
if (!mRootElementFrame) {
|
|
// The frame we're constructing will be the root element frame.
|
|
// Set mRootElementFrame before processing children.
|
|
mRootElementFrame = newFrame;
|
|
}
|
|
|
|
nsFrameItems childItems;
|
|
|
|
// Process children
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
const nsStyleDisplay* display = outerStyleContext->StyleDisplay();
|
|
|
|
// Mark the table frame as an absolute container if needed
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (display->IsAbsPosContainingBlock(newFrame)) {
|
|
aState.PushAbsoluteContainingBlock(newFrame, newFrame, absoluteSaveState);
|
|
}
|
|
NS_ASSERTION(aItem.mAnonChildren.IsEmpty(),
|
|
"nsIAnonymousContentCreator::CreateAnonymousContent "
|
|
"implementations for table frames are not currently expected "
|
|
"to output a list where the items have their own children");
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems,
|
|
innerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX,
|
|
childItems);
|
|
} else {
|
|
ProcessChildren(aState, content, styleContext, innerFrame,
|
|
true, childItems, false, aItem.mPendingBinding);
|
|
}
|
|
|
|
nsFrameItems captionItems;
|
|
PullOutCaptionFrames(childItems, captionItems);
|
|
|
|
// Set the inner table frame's initial primary list
|
|
innerFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
|
|
// Set the table wrapper frame's secondary childlist lists
|
|
if (captionItems.NotEmpty()) {
|
|
newFrame->SetInitialChildList(nsIFrame::kCaptionList, captionItems);
|
|
}
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
static void
|
|
MakeTablePartAbsoluteContainingBlockIfNeeded(nsFrameConstructorState& aState,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameConstructorSaveState& aAbsSaveState,
|
|
nsContainerFrame* aFrame)
|
|
{
|
|
// If we're positioned, then we need to become an absolute containing block
|
|
// for any absolutely positioned children and register for post-reflow fixup.
|
|
//
|
|
// Note that usually if a frame type can be an absolute containing block, we
|
|
// always set NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN, whether it actually is or not.
|
|
// However, in this case flag serves the additional purpose of indicating that
|
|
// the frame was registered with its table frame. This allows us to avoid the
|
|
// overhead of unregistering the frame in most cases.
|
|
if (aDisplay->IsAbsPosContainingBlock(aFrame)) {
|
|
aFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
aState.PushAbsoluteContainingBlock(aFrame, aFrame, aAbsSaveState);
|
|
nsTableFrame::RegisterPositionedTablePart(aFrame);
|
|
}
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructTableRowOrRowGroup(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
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.mStyleContext->StyleDisplay() == aDisplay,
|
|
"Display style doesn't match style context");
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
const uint32_t nameSpaceID = aItem.mNameSpaceID;
|
|
|
|
nsContainerFrame* newFrame;
|
|
if (aDisplay->mDisplay == StyleDisplay::TableRow) {
|
|
if (kNameSpaceID_MathML == nameSpaceID)
|
|
newFrame = NS_NewMathMLmtrFrame(mPresShell, styleContext);
|
|
else
|
|
newFrame = NS_NewTableRowFrame(mPresShell, styleContext);
|
|
} else {
|
|
newFrame = NS_NewTableRowGroupFrame(mPresShell, styleContext);
|
|
}
|
|
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newFrame);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlockIfNeeded(aState, aDisplay,
|
|
absoluteSaveState,
|
|
newFrame);
|
|
|
|
nsFrameItems childItems;
|
|
NS_ASSERTION(aItem.mAnonChildren.IsEmpty(),
|
|
"nsIAnonymousContentCreator::CreateAnonymousContent "
|
|
"implementations for table frames are not currently expected "
|
|
"to output a list where the items have their own children");
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems, newFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX,
|
|
childItems);
|
|
} else {
|
|
ProcessChildren(aState, content, styleContext, newFrame,
|
|
true, childItems, false, aItem.mPendingBinding);
|
|
}
|
|
|
|
newFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
aFrameItems.AddChild(newFrame);
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructTableCol(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
nsTableColFrame* colFrame = NS_NewTableColFrame(mPresShell, styleContext);
|
|
InitAndRestoreFrame(aState, content, aParentFrame, colFrame);
|
|
|
|
NS_ASSERTION(colFrame->StyleContext() == styleContext,
|
|
"Unexpected style context");
|
|
|
|
aFrameItems.AddChild(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, styleContext);
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newCol, false);
|
|
aFrameItems.LastChild()->SetNextContinuation(newCol);
|
|
newCol->SetPrevContinuation(aFrameItems.LastChild());
|
|
aFrameItems.AddChild(newCol);
|
|
newCol->SetColType(eColAnonymousCol);
|
|
}
|
|
|
|
return colFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructTableCell(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
MOZ_ASSERT(aDisplay->mDisplay == StyleDisplay::TableCell,
|
|
"Unexpected call");
|
|
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
const uint32_t nameSpaceID = aItem.mNameSpaceID;
|
|
|
|
nsTableFrame* tableFrame =
|
|
static_cast<nsTableRowFrame*>(aParentFrame)->GetTableFrame();
|
|
nsContainerFrame* newFrame;
|
|
// <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 (kNameSpaceID_MathML == nameSpaceID && !tableFrame->IsBorderCollapse()) {
|
|
newFrame = NS_NewMathMLmtdFrame(mPresShell, styleContext, 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.
|
|
newFrame = NS_NewTableCellFrame(mPresShell, styleContext, tableFrame);
|
|
}
|
|
|
|
// Initialize the table cell frame
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newFrame);
|
|
newFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Resolve pseudo style and initialize the body cell frame
|
|
RefPtr<nsStyleContext> innerPseudoStyle;
|
|
innerPseudoStyle = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::cellContent,
|
|
styleContext);
|
|
|
|
// Create a block frame that will format the cell's content
|
|
bool isBlock;
|
|
nsContainerFrame* cellInnerFrame;
|
|
if (kNameSpaceID_MathML == nameSpaceID) {
|
|
cellInnerFrame = NS_NewMathMLmtdInnerFrame(mPresShell, innerPseudoStyle);
|
|
isBlock = false;
|
|
} else {
|
|
cellInnerFrame = NS_NewBlockFormattingContext(mPresShell, innerPseudoStyle);
|
|
isBlock = true;
|
|
}
|
|
|
|
InitAndRestoreFrame(aState, content, newFrame, cellInnerFrame);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlockIfNeeded(aState, aDisplay,
|
|
absoluteSaveState,
|
|
newFrame);
|
|
|
|
nsFrameItems childItems;
|
|
NS_ASSERTION(aItem.mAnonChildren.IsEmpty(),
|
|
"nsIAnonymousContentCreator::CreateAnonymousContent "
|
|
"implementations for table frames are not currently expected "
|
|
"to output a list where the items have their own children");
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
// Need to push ourselves as a float containing block.
|
|
// XXXbz it might be nice to work on getting the parent
|
|
// FrameConstructionItem down into ProcessChildren and just making use of
|
|
// the push there, but that's a bit of work.
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
if (!isBlock) { /* MathML case */
|
|
aState.PushFloatContainingBlock(nullptr, floatSaveState);
|
|
} else {
|
|
aState.PushFloatContainingBlock(cellInnerFrame, floatSaveState);
|
|
}
|
|
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems, cellInnerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX,
|
|
childItems);
|
|
} else {
|
|
// Process the child content
|
|
ProcessChildren(aState, content, styleContext, cellInnerFrame,
|
|
true, childItems, isBlock, aItem.mPendingBinding);
|
|
}
|
|
|
|
cellInnerFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
SetInitialSingleChild(newFrame, cellInnerFrame);
|
|
aFrameItems.AddChild(newFrame);
|
|
return newFrame;
|
|
}
|
|
|
|
static inline bool
|
|
NeedFrameFor(const nsFrameConstructorState& aState,
|
|
nsIFrame* aParentFrame,
|
|
nsIContent* aChildContent)
|
|
{
|
|
// XXX the GetContent() != aChildContent check is needed due to bug 135040.
|
|
// Remove it once that's fixed.
|
|
NS_PRECONDITION(!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 of
|
|
// eExcludesIgnorableWhitespace frames, where we know we'll be dropping them
|
|
// all anyway, and involve an extra walk down the frame construction item
|
|
// list.
|
|
if ((aParentFrame &&
|
|
(!aParentFrame->IsFrameOfType(nsIFrame::eExcludesIgnorableWhitespace) ||
|
|
aParentFrame->IsGeneratedContentFrame())) ||
|
|
!aChildContent->IsNodeOfType(nsINode::eTEXT)) {
|
|
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,
|
|
nsILayoutHistoryState* aFrameState)
|
|
{
|
|
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 context for the viewport since it may be affected
|
|
// by a new root element style (e.g. a propagated 'direction').
|
|
// @see nsStyleContext::ApplyStyleFixups
|
|
{
|
|
RefPtr<nsStyleContext> sc = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::viewport, nullptr);
|
|
GetRootFrame()->SetStyleContextWithoutNotification(sc);
|
|
}
|
|
|
|
// Make sure to call UpdateViewportScrollbarStylesOverride before
|
|
// SetUpDocElementContainingBlock, since it sets up our scrollbar state
|
|
// properly.
|
|
DebugOnly<nsIContent*> propagatedScrollFrom;
|
|
if (nsPresContext* presContext = mPresShell->GetPresContext()) {
|
|
propagatedScrollFrom = presContext->UpdateViewportScrollbarStylesOverride();
|
|
}
|
|
|
|
SetUpDocElementContainingBlock(aDocElement);
|
|
|
|
NS_ASSERTION(mDocElementContainingBlock, "Should have parent by now");
|
|
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
// Initialize the ancestor filter with null for now; we'll push
|
|
// aDocElement once we finish resolving style for it.
|
|
if (matchContext.Exists()) {
|
|
matchContext->InitAncestors(nullptr);
|
|
}
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext,
|
|
GetAbsoluteContainingBlock(mDocElementContainingBlock, FIXED_POS),
|
|
nullptr,
|
|
nullptr, do_AddRef(aFrameState));
|
|
|
|
// XXXbz why, exactly?
|
|
if (!mTempFrameTreeState)
|
|
state.mPresShell->CaptureHistoryState(getter_AddRefs(mTempFrameTreeState));
|
|
|
|
// Make sure that we'll handle restyles for this document element in
|
|
// the future. We need this, because the document element might
|
|
// have stale restyle bits from a previous frame constructor for
|
|
// this document. Unlike in AddFrameConstructionItems, it's safe to
|
|
// unset all element restyle flags, since we don't have any
|
|
// siblings.
|
|
aDocElement->UnsetRestyleFlagsIfGecko();
|
|
|
|
// --------- CREATE AREA OR BOX FRAME -------
|
|
if (ServoStyleSet* set = mPresShell->StyleSet()->GetAsServo()) {
|
|
// Ensure the document element is styled at this point.
|
|
if (!aDocElement->HasServoData()) {
|
|
// NOTE(emilio): If the root has a non-null binding, we'll stop at the
|
|
// document element and won't process any children, loading the bindings
|
|
// (or failing to do so) will take care of the rest.
|
|
set->StyleNewSubtree(aDocElement);
|
|
}
|
|
}
|
|
|
|
// FIXME: Should this use ResolveStyleContext? (The calls in this
|
|
// function are the only case in nsCSSFrameConstructor where we don't
|
|
// do so for the construction of a style context for an element.)
|
|
RefPtr<nsStyleContext> styleContext =
|
|
mPresShell->StyleSet()->ResolveStyleFor(aDocElement,
|
|
nullptr,
|
|
LazyComputeBehavior::Assert);
|
|
|
|
const nsStyleDisplay* display = styleContext->StyleDisplay();
|
|
|
|
// Ensure that our XBL bindings are installed.
|
|
if (display->mBinding) {
|
|
// Get the XBL loader.
|
|
nsresult rv;
|
|
bool resolveStyle;
|
|
|
|
nsXBLService* xblService = nsXBLService::GetInstance();
|
|
if (!xblService) {
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<nsXBLBinding> binding;
|
|
rv = xblService->LoadBindings(aDocElement, display->mBinding->GetURI(),
|
|
display->mBinding->mExtraData->GetPrincipal(),
|
|
getter_AddRefs(binding), &resolveStyle);
|
|
if (NS_FAILED(rv) && rv != NS_ERROR_XBL_BLOCKED) {
|
|
// Binding will load asynchronously.
|
|
return nullptr;
|
|
}
|
|
|
|
if (binding) {
|
|
// For backwards compat, keep firing the root's constructor
|
|
// after all of its kids' constructors. So tell the binding
|
|
// manager about it right now.
|
|
mDocument->BindingManager()->AddToAttachedQueue(binding);
|
|
}
|
|
|
|
if (resolveStyle) {
|
|
// FIXME: Should this use ResolveStyleContext? (The calls in this
|
|
// function are the only case in nsCSSFrameConstructor where we
|
|
// don't do so for the construction of a style context for an
|
|
// element.)
|
|
//
|
|
// FIXME(emilio): This looks fishy. It really wants to fully re-resolve
|
|
// the style, but it wont if the element is already styled afaict... It
|
|
// seems we handle it on a subsequent restyle?
|
|
styleContext = mPresShell->StyleSet()->ResolveStyleFor(
|
|
aDocElement, nullptr, LazyComputeBehavior::Assert);
|
|
display = styleContext->StyleDisplay();
|
|
}
|
|
} else if (display->mBinding.ForceGet() && aDocElement->IsStyledByServo()) {
|
|
// See the comment in AddFrameConstructionItemsInternal for why this is
|
|
// needed.
|
|
mPresShell->StyleSet()->AsServo()->StyleNewChildren(aDocElement);
|
|
}
|
|
|
|
// --------- 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)) {
|
|
RegisterDisplayNoneStyleFor(aDocElement, styleContext);
|
|
return nullptr;
|
|
}
|
|
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(state.mTreeMatchContext);
|
|
ancestorPusher.PushAncestorAndStyleScope(aDocElement);
|
|
|
|
// Make sure to start any background image loads for the root element now.
|
|
styleContext->StartBackgroundImageLoads();
|
|
|
|
nsFrameConstructorSaveState docElementContainingBlockAbsoluteSaveState;
|
|
if (mHasRootAbsPosContainingBlock) {
|
|
// Push the absolute containing block now so we can absolutely position
|
|
// the root element
|
|
mDocElementContainingBlock->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
state.PushAbsoluteContainingBlock(mDocElementContainingBlock,
|
|
mDocElementContainingBlock,
|
|
docElementContainingBlockAbsoluteSaveState);
|
|
}
|
|
|
|
// 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. newFrame
|
|
// is the frame that will be the child of the initial containing block.
|
|
// These are usually the same frame but they can be different, in
|
|
// particular if the root frame is positioned, in which case
|
|
// contentFrame is the out-of-flow frame and newFrame is the
|
|
// placeholder.
|
|
nsContainerFrame* contentFrame;
|
|
nsIFrame* newFrame;
|
|
bool processChildren = false;
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
|
|
// Check whether we need to build a XUL box or SVG root frame
|
|
#ifdef MOZ_XUL
|
|
if (aDocElement->IsXULElement()) {
|
|
contentFrame = NS_NewDocElementBoxFrame(mPresShell, styleContext);
|
|
InitAndRestoreFrame(state, aDocElement, mDocElementContainingBlock,
|
|
contentFrame);
|
|
newFrame = contentFrame;
|
|
processChildren = true;
|
|
}
|
|
else
|
|
#endif
|
|
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.
|
|
// Use a null PendingBinding, since our binding is not in fact pending.
|
|
static const FrameConstructionData rootSVGData = FCDATA_DECL(0, nullptr);
|
|
already_AddRefed<nsStyleContext> extraRef =
|
|
RefPtr<nsStyleContext>(styleContext).forget();
|
|
AutoFrameConstructionItem item(this, &rootSVGData, aDocElement,
|
|
aDocElement->NodeInfo()->NameAtom(),
|
|
kNameSpaceID_SVG, nullptr, extraRef, true,
|
|
nullptr);
|
|
|
|
nsFrameItems frameItems;
|
|
contentFrame = static_cast<nsContainerFrame*>(
|
|
ConstructOuterSVG(state, item, mDocElementContainingBlock,
|
|
styleContext->StyleDisplay(),
|
|
frameItems));
|
|
newFrame = frameItems.FirstChild();
|
|
NS_ASSERTION(frameItems.OnlyChild(), "multiple root element frames");
|
|
} else if (display->mDisplay == StyleDisplay::Flex ||
|
|
display->mDisplay == StyleDisplay::WebkitBox) {
|
|
contentFrame = NS_NewFlexContainerFrame(mPresShell, styleContext);
|
|
InitAndRestoreFrame(state, aDocElement, mDocElementContainingBlock,
|
|
contentFrame);
|
|
newFrame = contentFrame;
|
|
processChildren = true;
|
|
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (display->IsAbsPosContainingBlock(newFrame)) {
|
|
state.PushAbsoluteContainingBlock(contentFrame, newFrame,
|
|
absoluteSaveState);
|
|
}
|
|
|
|
} else if (display->mDisplay == StyleDisplay::Grid) {
|
|
contentFrame = NS_NewGridContainerFrame(mPresShell, styleContext);
|
|
InitAndRestoreFrame(state, aDocElement, mDocElementContainingBlock,
|
|
contentFrame);
|
|
newFrame = contentFrame;
|
|
processChildren = true;
|
|
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (display->IsAbsPosContainingBlock(newFrame)) {
|
|
state.PushAbsoluteContainingBlock(contentFrame, newFrame,
|
|
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.
|
|
// Use a null PendingBinding, since our binding is not in fact pending.
|
|
static const FrameConstructionData rootTableData = FCDATA_DECL(0, nullptr);
|
|
already_AddRefed<nsStyleContext> extraRef =
|
|
RefPtr<nsStyleContext>(styleContext).forget();
|
|
AutoFrameConstructionItem item(this, &rootTableData, aDocElement,
|
|
aDocElement->NodeInfo()->NameAtom(),
|
|
kNameSpaceID_None, nullptr, extraRef, true,
|
|
nullptr);
|
|
|
|
nsFrameItems frameItems;
|
|
// if the document is a table then just populate it.
|
|
contentFrame = static_cast<nsContainerFrame*>(
|
|
ConstructTable(state, item, mDocElementContainingBlock,
|
|
styleContext->StyleDisplay(),
|
|
frameItems));
|
|
newFrame = frameItems.FirstChild();
|
|
NS_ASSERTION(frameItems.OnlyChild(), "multiple root element frames");
|
|
} else {
|
|
MOZ_ASSERT(display->mDisplay == StyleDisplay::Block ||
|
|
display->mDisplay == StyleDisplay::FlowRoot,
|
|
"Unhandled display type for root element");
|
|
contentFrame = NS_NewBlockFormattingContext(mPresShell, styleContext);
|
|
nsFrameItems frameItems;
|
|
// Use a null PendingBinding, since our binding is not in fact pending.
|
|
ConstructBlock(state, aDocElement,
|
|
state.GetGeometricParent(display,
|
|
mDocElementContainingBlock),
|
|
mDocElementContainingBlock, styleContext,
|
|
&contentFrame, frameItems,
|
|
display->IsAbsPosContainingBlock(contentFrame) ? contentFrame : nullptr,
|
|
nullptr);
|
|
newFrame = frameItems.FirstChild();
|
|
NS_ASSERTION(frameItems.OnlyChild(), "multiple root element frames");
|
|
}
|
|
|
|
MOZ_ASSERT(newFrame);
|
|
MOZ_ASSERT(contentFrame);
|
|
|
|
NS_ASSERTION(processChildren ? !mRootElementFrame :
|
|
mRootElementFrame == contentFrame,
|
|
"unexpected mRootElementFrame");
|
|
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->GetParentStyleContext(&mRootElementStyleFrame);
|
|
bool isChild = mRootElementStyleFrame &&
|
|
mRootElementStyleFrame->GetParent() == contentFrame;
|
|
if (!isChild) {
|
|
mRootElementStyleFrame = mRootElementFrame;
|
|
}
|
|
|
|
if (processChildren) {
|
|
// Still need to process the child content
|
|
nsFrameItems childItems;
|
|
|
|
NS_ASSERTION(!nsLayoutUtils::GetAsBlock(contentFrame) &&
|
|
!contentFrame->IsFrameOfType(nsIFrame::eSVG),
|
|
"Only XUL frames should reach here");
|
|
// Use a null PendingBinding, since our binding is not in fact pending.
|
|
ProcessChildren(state, aDocElement, styleContext, contentFrame, true,
|
|
childItems, false, nullptr);
|
|
|
|
// Set the initial child lists
|
|
contentFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
}
|
|
|
|
// set the primary frame
|
|
aDocElement->SetPrimaryFrame(contentFrame);
|
|
|
|
SetInitialSingleChild(mDocElementContainingBlock, newFrame);
|
|
|
|
// Create frames for anonymous contents if there is a canvas frame.
|
|
if (mDocElementContainingBlock->IsCanvasFrame()) {
|
|
ConstructAnonymousContentForCanvas(state, mDocElementContainingBlock,
|
|
aDocElement);
|
|
}
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructRootFrame()
|
|
{
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
|
|
// --------- BUILD VIEWPORT -----------
|
|
RefPtr<nsStyleContext> viewportPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::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
|
|
nsView* rootView = mPresShell->GetViewManager()->GetRootView();
|
|
viewportFrame->SetView(rootView);
|
|
|
|
viewportFrame->SyncFrameViewProperties(rootView);
|
|
nsContainerFrame::SyncWindowProperties(mPresShell->GetPresContext(), viewportFrame,
|
|
rootView, nullptr, nsContainerFrame::SET_ASYNC);
|
|
|
|
// 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)
|
|
{
|
|
NS_PRECONDITION(aDocElement, "No element?");
|
|
NS_PRECONDITION(!aDocElement->GetParent(), "Not root content?");
|
|
NS_PRECONDITION(aDocElement->GetUncomposedDoc(), "Not in a document?");
|
|
NS_PRECONDITION(aDocElement->GetUncomposedDoc()->GetRootElement() ==
|
|
aDocElement, "Not the root of the document?");
|
|
|
|
/*
|
|
how the root frame hierarchy should look
|
|
|
|
Galley presentation, non-XUL, with scrolling:
|
|
|
|
ViewportFrame [fixed-cb]
|
|
nsHTMLScrollFrame
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, nsSVGOuterSVGFrame,
|
|
nsTableWrapperFrame, nsPlaceholderFrame)
|
|
|
|
Galley presentation, XUL
|
|
|
|
ViewportFrame [fixed-cb]
|
|
nsRootBoxFrame
|
|
root element frame (nsDocElementBoxFrame)
|
|
|
|
Print presentation, non-XUL
|
|
|
|
ViewportFrame
|
|
nsSimplePageSequenceFrame
|
|
nsPageFrame
|
|
nsPageContentFrame [fixed-cb]
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, nsSVGOuterSVGFrame,
|
|
nsTableWrapperFrame, nsPlaceholderFrame)
|
|
|
|
Print-preview presentation, non-XUL
|
|
|
|
ViewportFrame
|
|
nsHTMLScrollFrame
|
|
nsSimplePageSequenceFrame
|
|
nsPageFrame
|
|
nsPageContentFrame [fixed-cb]
|
|
nsCanvasFrame [abs-cb]
|
|
root element frame (nsBlockFrame, nsSVGOuterSVGFrame,
|
|
nsTableWrapperFrame, nsPlaceholderFrame)
|
|
|
|
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 or nsRootBoxFrame)
|
|
|
|
mPageSequenceFrame is the nsSimplePageSequenceFrame, 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();
|
|
bool isPaginated = presContext->IsRootPaginatedDocument();
|
|
nsContainerFrame* viewportFrame = static_cast<nsContainerFrame*>(GetRootFrame());
|
|
nsStyleContext* viewportPseudoStyle = viewportFrame->StyleContext();
|
|
|
|
nsContainerFrame* rootFrame = nullptr;
|
|
nsAtom* rootPseudo;
|
|
|
|
if (!isPaginated) {
|
|
#ifdef MOZ_XUL
|
|
if (aDocElement->IsXULElement())
|
|
{
|
|
// pass a temporary stylecontext, the correct one will be set later
|
|
rootFrame = NS_NewRootBoxFrame(mPresShell, viewportPseudoStyle);
|
|
} else
|
|
#endif
|
|
{
|
|
// pass a temporary stylecontext, the correct one will be set later
|
|
rootFrame = NS_NewCanvasFrame(mPresShell, viewportPseudoStyle);
|
|
mHasRootAbsPosContainingBlock = true;
|
|
}
|
|
|
|
rootPseudo = nsCSSAnonBoxes::canvas;
|
|
mDocElementContainingBlock = rootFrame;
|
|
} else {
|
|
// Create a page sequence frame
|
|
rootFrame = NS_NewSimplePageSequenceFrame(mPresShell, viewportPseudoStyle);
|
|
mPageSequenceFrame = rootFrame;
|
|
rootPseudo = nsCSSAnonBoxes::pageSequence;
|
|
rootFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
}
|
|
|
|
|
|
// --------- 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?
|
|
|
|
bool isHTML = aDocElement->IsHTMLElement();
|
|
bool isXUL = false;
|
|
|
|
if (!isHTML) {
|
|
isXUL = aDocElement->IsXULElement();
|
|
}
|
|
|
|
// Never create scrollbars for XUL documents
|
|
bool isScrollable = isPaginated ? presContext->HasPaginatedScrolling() : !isXUL;
|
|
|
|
// 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 = rootFrame;
|
|
RefPtr<nsStyleContext> rootPseudoStyle;
|
|
// we must create a state because if the scrollbars are GFX it needs the
|
|
// state to build the scrollbar frames.
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell, matchContext, nullptr, nullptr, nullptr);
|
|
|
|
// Start off with the viewport as parent; we'll adjust it as needed.
|
|
nsContainerFrame* parentFrame = viewportFrame;
|
|
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
// If paginated, make sure we don't put scrollbars in
|
|
if (!isScrollable) {
|
|
rootPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(rootPseudo,
|
|
viewportPseudoStyle);
|
|
} else {
|
|
if (rootPseudo == nsCSSAnonBoxes::canvas) {
|
|
rootPseudo = nsCSSAnonBoxes::scrolledCanvas;
|
|
} else {
|
|
NS_ASSERTION(rootPseudo == nsCSSAnonBoxes::pageSequence,
|
|
"Unknown root pseudo");
|
|
rootPseudo = nsCSSAnonBoxes::scrolledPageSequence;
|
|
}
|
|
|
|
// 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<nsStyleContext> styleContext;
|
|
styleContext =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::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 = BeginBuildingScrollFrame( state,
|
|
aDocElement,
|
|
styleContext,
|
|
viewportFrame,
|
|
rootPseudo,
|
|
true,
|
|
newFrame);
|
|
parentFrame = newFrame;
|
|
}
|
|
|
|
rootFrame->SetStyleContextWithoutNotification(rootPseudoStyle);
|
|
rootFrame->Init(aDocElement, parentFrame, nullptr);
|
|
|
|
if (isScrollable) {
|
|
FinishBuildingScrollFrame(parentFrame, rootFrame);
|
|
}
|
|
|
|
if (isPaginated) {
|
|
// Create the first page
|
|
// Set the initial child lists
|
|
nsContainerFrame* canvasFrame;
|
|
nsContainerFrame* pageFrame =
|
|
ConstructPageFrame(mPresShell, rootFrame, nullptr, canvasFrame);
|
|
pageFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
SetInitialSingleChild(rootFrame, pageFrame);
|
|
|
|
// The eventual parent of the document element frame.
|
|
// XXX should this be set for every new page (in ConstructPageFrame)?
|
|
mDocElementContainingBlock = canvasFrame;
|
|
mHasRootAbsPosContainingBlock = true;
|
|
}
|
|
|
|
if (viewportFrame->GetStateBits() & NS_FRAME_FIRST_REFLOW) {
|
|
SetInitialSingleChild(viewportFrame, newFrame);
|
|
} else {
|
|
nsFrameList newFrameList(newFrame, newFrame);
|
|
viewportFrame->AppendFrames(kPrincipalList, newFrameList);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ConstructAnonymousContentForCanvas(nsFrameConstructorState& aState,
|
|
nsIFrame* aFrame,
|
|
nsIContent* aDocElement)
|
|
{
|
|
NS_ASSERTION(aFrame->IsCanvasFrame(), "aFrame should be canvas frame!");
|
|
|
|
AutoTArray<nsIAnonymousContentCreator::ContentInfo, 4> anonymousItems;
|
|
GetAnonymousContent(aDocElement, aFrame, anonymousItems);
|
|
if (anonymousItems.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
AutoFrameConstructionItemList itemsToConstruct(this);
|
|
nsContainerFrame* frameAsContainer = do_QueryFrame(aFrame);
|
|
AddFCItemsForAnonymousContent(aState, frameAsContainer, anonymousItems, itemsToConstruct);
|
|
|
|
nsFrameItems frameItems;
|
|
ConstructFramesFromItemList(aState, itemsToConstruct, frameAsContainer,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
frameItems);
|
|
frameAsContainer->AppendFrames(kPrincipalList, frameItems);
|
|
}
|
|
|
|
nsContainerFrame*
|
|
nsCSSFrameConstructor::ConstructPageFrame(nsIPresShell* aPresShell,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIFrame* aPrevPageFrame,
|
|
nsContainerFrame*& aCanvasFrame)
|
|
{
|
|
nsStyleContext* parentStyleContext = aParentFrame->StyleContext();
|
|
StyleSetHandle styleSet = aPresShell->StyleSet();
|
|
|
|
RefPtr<nsStyleContext> pagePseudoStyle;
|
|
pagePseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::page,
|
|
parentStyleContext);
|
|
|
|
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<nsStyleContext> pageContentPseudoStyle;
|
|
pageContentPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::pageContent,
|
|
pagePseudoStyle);
|
|
|
|
nsContainerFrame* pageContentFrame =
|
|
NS_NewPageContentFrame(aPresShell, pageContentPseudoStyle);
|
|
|
|
// Initialize the page content frame and force it to have a view. Also make it the
|
|
// containing block for fixed elements which are repeated on every page.
|
|
nsIFrame* prevPageContentFrame = nullptr;
|
|
if (aPrevPageFrame) {
|
|
prevPageContentFrame = aPrevPageFrame->PrincipalChildList().FirstChild();
|
|
NS_ASSERTION(prevPageContentFrame, "missing page content frame");
|
|
}
|
|
pageContentFrame->Init(nullptr, pageFrame, prevPageContentFrame);
|
|
if (!prevPageContentFrame) {
|
|
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<nsStyleContext> canvasPseudoStyle;
|
|
canvasPseudoStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::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(nsIPresShell* aPresShell,
|
|
nsIContent* aContent,
|
|
nsIFrame* aFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIFrame* aPrevInFlow,
|
|
nsFrameState aTypeBit)
|
|
{
|
|
RefPtr<nsStyleContext> placeholderStyle = aPresShell->StyleSet()->
|
|
ResolveStyleForPlaceholder();
|
|
|
|
// The placeholder frame gets a pseudo style context
|
|
nsPlaceholderFrame* placeholderFrame =
|
|
(nsPlaceholderFrame*)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)
|
|
{
|
|
NS_PRECONDITION(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);
|
|
}
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructSelectFrame(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
// Construct a frame-based listbox or combobox
|
|
dom::HTMLSelectElement* sel = dom::HTMLSelectElement::FromContent(content);
|
|
MOZ_ASSERT(sel);
|
|
if (sel->IsCombobox()) {
|
|
// Construct a frame-based combo box.
|
|
// The frame-based combo box is built out of three parts. A display area, a button and
|
|
// a dropdown list. The display area and button are created through anonymous content.
|
|
// The drop-down list's frame is created explicitly. The combobox frame shares its content
|
|
// with the drop-down list.
|
|
nsFrameState flags = NS_BLOCK_FLOAT_MGR;
|
|
nsComboboxControlFrame* comboboxFrame =
|
|
NS_NewComboboxControlFrame(mPresShell, styleContext, flags);
|
|
|
|
// Save the history state so we don't restore during construction
|
|
// since the complete tree is required before we restore.
|
|
nsILayoutHistoryState *historyState = aState.mFrameState;
|
|
aState.mFrameState = nullptr;
|
|
// Initialize the combobox frame
|
|
InitAndRestoreFrame(aState, content,
|
|
aState.GetGeometricParent(aStyleDisplay, aParentFrame),
|
|
comboboxFrame);
|
|
|
|
comboboxFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
aState.AddChild(comboboxFrame, aFrameItems, content, styleContext,
|
|
aParentFrame);
|
|
|
|
// Resolve pseudo element style for the dropdown list
|
|
RefPtr<nsStyleContext> listStyle;
|
|
listStyle = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::dropDownList,
|
|
styleContext);
|
|
|
|
// Create a listbox
|
|
nsContainerFrame* listFrame = NS_NewListControlFrame(mPresShell, listStyle);
|
|
|
|
// Notify the listbox that it is being used as a dropdown list.
|
|
nsIListControlFrame * listControlFrame = do_QueryFrame(listFrame);
|
|
if (listControlFrame) {
|
|
listControlFrame->SetComboboxFrame(comboboxFrame);
|
|
}
|
|
// Notify combobox that it should use the listbox as it's popup
|
|
comboboxFrame->SetDropDown(listFrame);
|
|
|
|
NS_ASSERTION(!listFrame->IsAbsPosContainingBlock(),
|
|
"Ended up with positioned dropdown list somehow.");
|
|
NS_ASSERTION(!listFrame->IsFloating(),
|
|
"Ended up with floating dropdown list somehow.");
|
|
|
|
// Initialize the scroll frame positioned. Note that it is NOT
|
|
// initialized as absolutely positioned.
|
|
nsContainerFrame* scrolledFrame =
|
|
NS_NewSelectsAreaFrame(mPresShell, styleContext, flags);
|
|
|
|
InitializeSelectFrame(aState, listFrame, scrolledFrame, content,
|
|
comboboxFrame, listStyle, true,
|
|
aItem.mPendingBinding, aFrameItems);
|
|
|
|
NS_ASSERTION(listFrame->GetView(), "ListFrame's view is nullptr");
|
|
|
|
// Create display and button frames from the combobox's anonymous content.
|
|
// The anonymous content is appended to existing anonymous content for this
|
|
// element (the scrollbars).
|
|
nsFrameItems childItems;
|
|
|
|
// nsComboboxControlFrame needs special frame creation behavior for its first
|
|
// piece of anonymous content, which means that we can't take the normal
|
|
// ProcessChildren path.
|
|
AutoTArray<nsIAnonymousContentCreator::ContentInfo, 2> newAnonymousItems;
|
|
DebugOnly<nsresult> rv = GetAnonymousContent(content, comboboxFrame, newAnonymousItems);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
MOZ_ASSERT(newAnonymousItems.Length() == 2);
|
|
|
|
// Manually create a frame for the special NAC.
|
|
MOZ_ASSERT(newAnonymousItems[0].mContent == comboboxFrame->GetDisplayNode());
|
|
newAnonymousItems.RemoveElementAt(0);
|
|
nsIFrame* customFrame = comboboxFrame->CreateFrameForDisplayNode();
|
|
MOZ_ASSERT(customFrame);
|
|
customFrame->AddStateBits(NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT);
|
|
childItems.AddChild(customFrame);
|
|
|
|
// The other piece of NAC can take the normal path.
|
|
AutoFrameConstructionItemList fcItems(this);
|
|
AddFCItemsForAnonymousContent(aState, comboboxFrame, newAnonymousItems,
|
|
fcItems);
|
|
ConstructFramesFromItemList(aState, fcItems, comboboxFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
childItems);
|
|
|
|
comboboxFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
|
|
// Initialize the additional popup child list which contains the
|
|
// dropdown list frame.
|
|
nsFrameItems popupItems;
|
|
popupItems.AddChild(listFrame);
|
|
comboboxFrame->SetInitialChildList(nsIFrame::kSelectPopupList,
|
|
popupItems);
|
|
|
|
aState.mFrameState = historyState;
|
|
if (aState.mFrameState) {
|
|
// Restore frame state for the entire subtree of |comboboxFrame|.
|
|
RestoreFrameState(comboboxFrame, aState.mFrameState);
|
|
}
|
|
return comboboxFrame;
|
|
}
|
|
|
|
// Listbox, not combobox
|
|
nsContainerFrame* listFrame = NS_NewListControlFrame(mPresShell, styleContext);
|
|
|
|
nsContainerFrame* scrolledFrame = NS_NewSelectsAreaFrame(
|
|
mPresShell, styleContext, NS_BLOCK_FLOAT_MGR);
|
|
|
|
// ******* this code stolen from Initialze ScrollFrame ********
|
|
// please adjust this code to use BuildScrollFrame.
|
|
|
|
InitializeSelectFrame(aState, listFrame, scrolledFrame, content,
|
|
aParentFrame, styleContext, false,
|
|
aItem.mPendingBinding, aFrameItems);
|
|
|
|
return listFrame;
|
|
}
|
|
|
|
/**
|
|
* Used to be InitializeScrollFrame but now it's only used for the select tag
|
|
* But the select tag should really be fixed to use GFX scrollbars that can
|
|
* be create with BuildScrollFrame.
|
|
*/
|
|
void
|
|
nsCSSFrameConstructor::InitializeSelectFrame(nsFrameConstructorState& aState,
|
|
nsContainerFrame* scrollFrame,
|
|
nsContainerFrame* scrolledFrame,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsStyleContext* aStyleContext,
|
|
bool aBuildCombobox,
|
|
PendingBinding* aPendingBinding,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
// Initialize it
|
|
nsContainerFrame* geometricParent =
|
|
aState.GetGeometricParent(aStyleContext->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);
|
|
|
|
if (!aBuildCombobox) {
|
|
aState.AddChild(scrollFrame, aFrameItems, aContent,
|
|
aStyleContext, aParentFrame);
|
|
}
|
|
|
|
BuildScrollFrame(aState, aContent, aStyleContext, scrolledFrame,
|
|
geometricParent, scrollFrame);
|
|
|
|
if (aState.mFrameState) {
|
|
// Restore frame state for the scroll frame
|
|
RestoreFrameStateFor(scrollFrame, aState.mFrameState);
|
|
}
|
|
|
|
// Process children
|
|
nsFrameItems childItems;
|
|
|
|
ProcessChildren(aState, aContent, aStyleContext, scrolledFrame, false,
|
|
childItems, false, aPendingBinding);
|
|
|
|
// Set the scrolled frame's initial child lists
|
|
scrolledFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructFieldSetFrame(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
nsContainerFrame* fieldsetFrame = NS_NewFieldSetFrame(mPresShell, styleContext);
|
|
|
|
// Initialize it
|
|
InitAndRestoreFrame(aState, content,
|
|
aState.GetGeometricParent(aStyleDisplay, aParentFrame),
|
|
fieldsetFrame);
|
|
|
|
fieldsetFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// Resolve style and initialize the frame
|
|
RefPtr<nsStyleContext> fieldsetContentStyle;
|
|
fieldsetContentStyle = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::fieldsetContent,
|
|
styleContext);
|
|
|
|
const nsStyleDisplay* fieldsetContentDisplay = fieldsetContentStyle->StyleDisplay();
|
|
bool isScrollable = fieldsetContentDisplay->IsScrollableOverflow();
|
|
nsContainerFrame* scrollFrame = nullptr;
|
|
if (isScrollable) {
|
|
fieldsetContentStyle =
|
|
BeginBuildingScrollFrame(aState, content, fieldsetContentStyle,
|
|
fieldsetFrame, nsCSSAnonBoxes::scrolledContent,
|
|
false, scrollFrame);
|
|
}
|
|
|
|
nsContainerFrame* absPosContainer = nullptr;
|
|
if (fieldsetFrame->IsAbsPosContainingBlock()) {
|
|
absPosContainer = fieldsetFrame;
|
|
}
|
|
|
|
// Create the inner ::-moz-fieldset-content frame.
|
|
nsContainerFrame* contentFrameTop;
|
|
nsContainerFrame* contentFrame;
|
|
auto parent = scrollFrame ? scrollFrame : fieldsetFrame;
|
|
switch (fieldsetContentDisplay->mDisplay) {
|
|
case StyleDisplay::Flex:
|
|
contentFrame = NS_NewFlexContainerFrame(mPresShell, fieldsetContentStyle);
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
break;
|
|
case StyleDisplay::Grid:
|
|
contentFrame = NS_NewGridContainerFrame(mPresShell, fieldsetContentStyle);
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
break;
|
|
default: {
|
|
MOZ_ASSERT(fieldsetContentDisplay->mDisplay == StyleDisplay::Block,
|
|
"bug in nsRuleNode::ComputeDisplayData?");
|
|
|
|
nsContainerFrame* columnSetFrame = nullptr;
|
|
RefPtr<nsStyleContext> innerSC = fieldsetContentStyle;
|
|
const nsStyleColumn* columns = fieldsetContentStyle->StyleColumn();
|
|
if (columns->mColumnCount != NS_STYLE_COLUMN_COUNT_AUTO ||
|
|
columns->mColumnWidth.GetUnit() != eStyleUnit_Auto) {
|
|
columnSetFrame =
|
|
NS_NewColumnSetFrame(mPresShell, fieldsetContentStyle,
|
|
nsFrameState(NS_FRAME_OWNS_ANON_BOXES));
|
|
InitAndRestoreFrame(aState, content, parent, columnSetFrame);
|
|
innerSC = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::columnContent,
|
|
fieldsetContentStyle);
|
|
if (absPosContainer) {
|
|
absPosContainer = columnSetFrame;
|
|
}
|
|
}
|
|
contentFrame = NS_NewBlockFormattingContext(mPresShell, innerSC);
|
|
if (columnSetFrame) {
|
|
InitAndRestoreFrame(aState, content, columnSetFrame, contentFrame);
|
|
SetInitialSingleChild(columnSetFrame, contentFrame);
|
|
contentFrameTop = columnSetFrame;
|
|
} else {
|
|
InitAndRestoreFrame(aState, content, parent, contentFrame);
|
|
contentFrameTop = contentFrame;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
aState.AddChild(fieldsetFrame, aFrameItems, content, styleContext, aParentFrame);
|
|
|
|
// Process children
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
nsFrameItems childItems;
|
|
|
|
contentFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (absPosContainer) {
|
|
aState.PushAbsoluteContainingBlock(contentFrame, absPosContainer, absoluteSaveState);
|
|
}
|
|
|
|
ProcessChildren(aState, content, styleContext, contentFrame, true,
|
|
childItems, true, aItem.mPendingBinding);
|
|
|
|
nsFrameItems fieldsetKids;
|
|
fieldsetKids.AddChild(scrollFrame ? scrollFrame : contentFrameTop);
|
|
|
|
for (nsFrameList::Enumerator e(childItems); !e.AtEnd(); e.Next()) {
|
|
nsIFrame* child = e.get();
|
|
nsContainerFrame* cif = child->GetContentInsertionFrame();
|
|
if (cif && cif->IsLegendFrame()) {
|
|
// We want the legend to be the first frame in the fieldset child list.
|
|
// That way the EventStateManager will do the right thing when tabbing
|
|
// from a selection point within the legend (bug 236071), which is
|
|
// used for implementing legend access keys (bug 81481).
|
|
// GetAdjustedParentFrame() below depends on this frame order.
|
|
childItems.RemoveFrame(child);
|
|
// Make sure to reparent the legend so it has the fieldset as the parent.
|
|
fieldsetKids.InsertFrame(fieldsetFrame, nullptr, child);
|
|
if (scrollFrame) {
|
|
StickyScrollContainer::NotifyReparentedFrameAcrossScrollFrameBoundary(
|
|
child, contentFrame);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (isScrollable) {
|
|
FinishBuildingScrollFrame(scrollFrame, contentFrameTop);
|
|
}
|
|
|
|
// Set the inner frame's initial child lists
|
|
contentFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
|
|
// Set the outer frame's initial child list
|
|
fieldsetFrame->SetInitialChildList(kPrincipalList, fieldsetKids);
|
|
|
|
fieldsetFrame->AddStateBits(NS_FRAME_MAY_HAVE_GENERATED_CONTENT);
|
|
|
|
// Our new frame returned is the outer frame, which is the fieldset frame.
|
|
return fieldsetFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructDetailsFrame(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aStyleDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
if (!aStyleDisplay->IsScrollableOverflow()) {
|
|
return ConstructNonScrollableBlockWithConstructor(aState, aItem, aParentFrame,
|
|
aStyleDisplay, aFrameItems,
|
|
NS_NewDetailsFrame);
|
|
}
|
|
|
|
// Build a scroll frame to wrap details frame if necessary.
|
|
return ConstructScrollableBlockWithConstructor(aState, aItem, aParentFrame,
|
|
aStyleDisplay, aFrameItems,
|
|
NS_NewDetailsFrame);
|
|
}
|
|
|
|
static nsIFrame*
|
|
FindAncestorWithGeneratedContentPseudo(nsIFrame* aFrame)
|
|
{
|
|
for (nsIFrame* f = aFrame->GetParent(); f; f = f->GetParent()) {
|
|
NS_ASSERTION(f->IsGeneratedContentFrame(),
|
|
"should not have exited generated content");
|
|
nsAtom* pseudo = f->StyleContext()->GetPseudo();
|
|
if (pseudo == nsCSSPseudoElements::before ||
|
|
pseudo == nsCSSPseudoElements::after)
|
|
return f;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
#define SIMPLE_FCDATA(_func) FCDATA_DECL(0, _func)
|
|
#define FULL_CTOR_FCDATA(_flags, _func) \
|
|
{ _flags | FCDATA_FUNC_IS_FULL_CTOR, { nullptr }, _func, nullptr }
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindTextData(nsIFrame* aParentFrame)
|
|
{
|
|
if (aParentFrame && IsFrameForSVG(aParentFrame)) {
|
|
nsIFrame *ancestorFrame =
|
|
nsSVGUtils::GetFirstNonAAncestorFrame(aParentFrame);
|
|
if (ancestorFrame) {
|
|
static const FrameConstructionData sSVGTextData =
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT | FCDATA_IS_SVG_TEXT,
|
|
NS_NewTextFrame);
|
|
if (nsSVGUtils::IsInSVGTextSubtree(ancestorFrame)) {
|
|
return &sSVGTextData;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sTextData =
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT, NS_NewTextFrame);
|
|
return &sTextData;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ConstructTextFrame(const FrameConstructionData* aData,
|
|
nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsStyleContext* aStyleContext,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
NS_PRECONDITION(aData, "Must have frame construction data");
|
|
|
|
nsIFrame* newFrame = (*aData->mFunc.mCreationFunc)(mPresShell, aStyleContext);
|
|
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, newFrame);
|
|
|
|
// We never need to create a view for a text frame.
|
|
|
|
if (newFrame->IsGeneratedContentFrame()) {
|
|
nsAutoPtr<nsGenConInitializer> initializer;
|
|
initializer =
|
|
static_cast<nsGenConInitializer*>(
|
|
aContent->UnsetProperty(nsGkAtoms::genConInitializerProperty));
|
|
if (initializer) {
|
|
if (initializer->mNode->InitTextFrame(initializer->mList,
|
|
FindAncestorWithGeneratedContentPseudo(newFrame), newFrame)) {
|
|
(this->*(initializer->mDirtyAll))();
|
|
}
|
|
initializer->mNode.forget();
|
|
}
|
|
}
|
|
|
|
// Add the newly constructed frame to the flow
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
if (!aState.mCreatingExtraFrames)
|
|
aContent->SetPrimaryFrame(newFrame);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDataByInt(int32_t aInt,
|
|
Element* aElement,
|
|
nsStyleContext* aStyleContext,
|
|
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, aStyleContext);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDataByTag(nsAtom* aTag,
|
|
Element* aElement,
|
|
nsStyleContext* aStyleContext,
|
|
const FrameConstructionDataByTag* aDataPtr,
|
|
uint32_t aDataLength)
|
|
{
|
|
for (const FrameConstructionDataByTag *curData = aDataPtr,
|
|
*endData = aDataPtr + aDataLength;
|
|
curData != endData;
|
|
++curData) {
|
|
if (*curData->mTag == aTag) {
|
|
const FrameConstructionData* data = &curData->mData;
|
|
if (data->mBits & FCDATA_FUNC_IS_DATA_GETTER) {
|
|
return data->mFunc.mDataGetter(aElement, aStyleContext);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
#define SUPPRESS_FCDATA() FCDATA_DECL(FCDATA_SUPPRESS_FRAME, nullptr)
|
|
#define SIMPLE_INT_CREATE(_int, _func) { _int, SIMPLE_FCDATA(_func) }
|
|
#define SIMPLE_INT_CHAIN(_int, _func) \
|
|
{ _int, FCDATA_DECL(FCDATA_FUNC_IS_DATA_GETTER, _func) }
|
|
#define COMPLEX_INT_CREATE(_int, _func) \
|
|
{ _int, FULL_CTOR_FCDATA(0, _func) }
|
|
|
|
#define SIMPLE_TAG_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, SIMPLE_FCDATA(_func) }
|
|
#define SIMPLE_TAG_CHAIN(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, FCDATA_DECL(FCDATA_FUNC_IS_DATA_GETTER, _func) }
|
|
#define COMPLEX_TAG_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, FULL_CTOR_FCDATA(0, _func) }
|
|
|
|
static bool
|
|
IsFrameForFieldSet(nsIFrame* aFrame)
|
|
{
|
|
nsAtom* pseudo = aFrame->StyleContext()->GetPseudo();
|
|
if (pseudo == nsCSSAnonBoxes::fieldsetContent ||
|
|
pseudo == nsCSSAnonBoxes::scrolledContent ||
|
|
pseudo == nsCSSAnonBoxes::columnContent) {
|
|
return IsFrameForFieldSet(aFrame->GetParent());
|
|
}
|
|
return aFrame->IsFieldSetFrame();
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindHTMLData(Element* aElement,
|
|
nsAtom* aTag,
|
|
int32_t aNameSpaceID,
|
|
nsIFrame* aParentFrame,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
// Ignore the tag if it's not HTML content and if it doesn't extend (via XBL)
|
|
// a valid HTML namespace. This check must match the one in
|
|
// ShouldHaveFirstLineStyle.
|
|
if (aNameSpaceID != kNameSpaceID_XHTML) {
|
|
return nullptr;
|
|
}
|
|
|
|
NS_ASSERTION(!aParentFrame ||
|
|
aParentFrame->StyleContext()->GetPseudo() !=
|
|
nsCSSAnonBoxes::fieldsetContent ||
|
|
aParentFrame->GetParent()->IsFieldSetFrame(),
|
|
"Unexpected parent for fieldset content anon box");
|
|
if (aTag == nsGkAtoms::legend &&
|
|
(!aParentFrame || !IsFrameForFieldSet(aParentFrame) ||
|
|
aStyleContext->StyleDisplay()->IsFloatingStyle() ||
|
|
aStyleContext->StyleDisplay()->IsAbsolutelyPositionedStyle())) {
|
|
// <legend> is only special inside fieldset, we only check the frame tree
|
|
// parent because the content tree parent may not be a <fieldset> due to
|
|
// display:contents, Shadow DOM, or XBL. For floated or absolutely
|
|
// positioned legends we want to construct by display type and
|
|
// not do special legend stuff.
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionDataByTag sHTMLData[] = {
|
|
SIMPLE_TAG_CHAIN(img, nsCSSFrameConstructor::FindImgData),
|
|
SIMPLE_TAG_CHAIN(mozgeneratedcontentimage,
|
|
nsCSSFrameConstructor::FindImgData),
|
|
{ &nsGkAtoms::br,
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT | FCDATA_IS_LINE_BREAK,
|
|
NS_NewBRFrame) },
|
|
SIMPLE_TAG_CREATE(wbr, NS_NewWBRFrame),
|
|
SIMPLE_TAG_CHAIN(input, nsCSSFrameConstructor::FindInputData),
|
|
SIMPLE_TAG_CREATE(textarea, NS_NewTextControlFrame),
|
|
COMPLEX_TAG_CREATE(select, &nsCSSFrameConstructor::ConstructSelectFrame),
|
|
SIMPLE_TAG_CHAIN(object, nsCSSFrameConstructor::FindObjectData),
|
|
SIMPLE_TAG_CHAIN(embed, nsCSSFrameConstructor::FindObjectData),
|
|
COMPLEX_TAG_CREATE(fieldset,
|
|
&nsCSSFrameConstructor::ConstructFieldSetFrame),
|
|
{ &nsGkAtoms::legend,
|
|
FCDATA_DECL(FCDATA_ALLOW_BLOCK_STYLES | FCDATA_MAY_NEED_SCROLLFRAME,
|
|
NS_NewLegendFrame) },
|
|
SIMPLE_TAG_CREATE(frameset, NS_NewHTMLFramesetFrame),
|
|
SIMPLE_TAG_CREATE(iframe, NS_NewSubDocumentFrame),
|
|
{ &nsGkAtoms::button,
|
|
FCDATA_WITH_WRAPPING_BLOCK(FCDATA_ALLOW_BLOCK_STYLES |
|
|
FCDATA_ALLOW_GRID_FLEX_COLUMNSET,
|
|
NS_NewHTMLButtonControlFrame,
|
|
nsCSSAnonBoxes::buttonContent) },
|
|
SIMPLE_TAG_CHAIN(canvas, nsCSSFrameConstructor::FindCanvasData),
|
|
SIMPLE_TAG_CREATE(video, NS_NewHTMLVideoFrame),
|
|
SIMPLE_TAG_CREATE(audio, NS_NewHTMLVideoFrame),
|
|
SIMPLE_TAG_CREATE(progress, NS_NewProgressFrame),
|
|
SIMPLE_TAG_CREATE(meter, NS_NewMeterFrame),
|
|
COMPLEX_TAG_CREATE(details, &nsCSSFrameConstructor::ConstructDetailsFrame)
|
|
};
|
|
|
|
return FindDataByTag(aTag, aElement, aStyleContext, sHTMLData,
|
|
ArrayLength(sHTMLData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindImgData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (!nsImageFrame::ShouldCreateImageFrameFor(aElement, aStyleContext)) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sImgData = SIMPLE_FCDATA(NS_NewImageFrame);
|
|
return &sImgData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindImgControlData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (!nsImageFrame::ShouldCreateImageFrameFor(aElement, aStyleContext)) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sImgControlData =
|
|
SIMPLE_FCDATA(NS_NewImageControlFrame);
|
|
return &sImgControlData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindInputData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
static const FrameConstructionDataByInt sInputData[] = {
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_CHECKBOX, NS_NewCheckboxRadioFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_RADIO, NS_NewCheckboxRadioFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_FILE, NS_NewFileControlFrame),
|
|
SIMPLE_INT_CHAIN(NS_FORM_INPUT_IMAGE,
|
|
nsCSSFrameConstructor::FindImgControlData),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_EMAIL, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_SEARCH, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_TEXT, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_TEL, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_URL, NS_NewTextControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_RANGE, NS_NewRangeFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_PASSWORD, NS_NewTextControlFrame),
|
|
{ NS_FORM_INPUT_COLOR,
|
|
FCDATA_WITH_WRAPPING_BLOCK(0, NS_NewColorControlFrame,
|
|
nsCSSAnonBoxes::buttonContent) },
|
|
// TODO: this is temporary until a frame is written: bug 635240.
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_NUMBER, NS_NewNumberControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_TIME, NS_NewDateTimeControlFrame),
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_DATE, NS_NewDateTimeControlFrame),
|
|
// TODO: this is temporary until a frame is written: bug 888320
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_MONTH, NS_NewTextControlFrame),
|
|
// TODO: this is temporary until a frame is written: bug 888320
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_WEEK, NS_NewTextControlFrame),
|
|
// TODO: this is temporary until a frame is written: bug 888320
|
|
SIMPLE_INT_CREATE(NS_FORM_INPUT_DATETIME_LOCAL, NS_NewTextControlFrame),
|
|
{ NS_FORM_INPUT_SUBMIT,
|
|
FCDATA_WITH_WRAPPING_BLOCK(0, NS_NewGfxButtonControlFrame,
|
|
nsCSSAnonBoxes::buttonContent) },
|
|
{ NS_FORM_INPUT_RESET,
|
|
FCDATA_WITH_WRAPPING_BLOCK(0, NS_NewGfxButtonControlFrame,
|
|
nsCSSAnonBoxes::buttonContent) },
|
|
{ NS_FORM_INPUT_BUTTON,
|
|
FCDATA_WITH_WRAPPING_BLOCK(0, NS_NewGfxButtonControlFrame,
|
|
nsCSSAnonBoxes::buttonContent) }
|
|
// Keeping hidden inputs out of here on purpose for so they get frames by
|
|
// display (in practice, none).
|
|
};
|
|
|
|
nsCOMPtr<nsIFormControl> control = do_QueryInterface(aElement);
|
|
NS_ASSERTION(control, "input doesn't implement nsIFormControl?");
|
|
|
|
auto controlType = control->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) NS_THEME_RADIO and NS_THEME_CHECKBOX.)
|
|
if ((controlType == NS_FORM_INPUT_CHECKBOX ||
|
|
controlType == NS_FORM_INPUT_RADIO) &&
|
|
aStyleContext->StyleDisplay()->mAppearance == NS_THEME_NONE) {
|
|
return nullptr;
|
|
}
|
|
|
|
return FindDataByInt(controlType, aElement, aStyleContext,
|
|
sInputData, ArrayLength(sInputData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindObjectData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
// GetDisplayedType isn't necessarily nsIObjectLoadingContent::TYPE_NULL for
|
|
// cases when the object is broken/suppressed/etc (e.g. a broken image), but
|
|
// we want to treat those cases as TYPE_NULL
|
|
uint32_t type;
|
|
if (aElement->State().HasAtLeastOneOfStates(NS_EVENT_STATE_BROKEN |
|
|
NS_EVENT_STATE_USERDISABLED |
|
|
NS_EVENT_STATE_SUPPRESSED)) {
|
|
type = nsIObjectLoadingContent::TYPE_NULL;
|
|
} else {
|
|
nsCOMPtr<nsIObjectLoadingContent> objContent(do_QueryInterface(aElement));
|
|
NS_ASSERTION(objContent,
|
|
"embed and object must implement "
|
|
"nsIObjectLoadingContent!");
|
|
|
|
objContent->GetDisplayedType(&type);
|
|
}
|
|
|
|
static const FrameConstructionDataByInt sObjectData[] = {
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_LOADING,
|
|
NS_NewEmptyFrame),
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_PLUGIN,
|
|
NS_NewObjectFrame),
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_IMAGE,
|
|
NS_NewImageFrame),
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_DOCUMENT,
|
|
NS_NewSubDocumentFrame),
|
|
// Fake plugin handlers load as documents
|
|
SIMPLE_INT_CREATE(nsIObjectLoadingContent::TYPE_FAKE_PLUGIN,
|
|
NS_NewSubDocumentFrame)
|
|
// Nothing for TYPE_NULL so we'll construct frames by display there
|
|
};
|
|
|
|
return FindDataByInt((int32_t)type, aElement, aStyleContext,
|
|
sObjectData, ArrayLength(sObjectData));
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindCanvasData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
// 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.
|
|
nsIDocument* doc = aElement->OwnerDoc();
|
|
if (doc->IsStaticDocument()) {
|
|
doc = doc->GetOriginalDocument();
|
|
}
|
|
if (!doc->IsScriptEnabled()) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sCanvasData =
|
|
FCDATA_WITH_WRAPPING_BLOCK(0, NS_NewHTMLCanvasFrame,
|
|
nsCSSAnonBoxes::htmlCanvasContent);
|
|
return &sCanvasData;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ConstructFrameFromItemInternal(FrameConstructionItem& aItem,
|
|
nsFrameConstructorState& aState,
|
|
nsContainerFrame* aParentFrame,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
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_MAY_NEED_SCROLLFRAME);
|
|
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
|
|
NS_ASSERTION(!(bits & FCDATA_FORCED_NON_SCROLLABLE_BLOCK) ||
|
|
((bits & FCDATA_FUNC_IS_FULL_CTOR) &&
|
|
data->mFullConstructor ==
|
|
&nsCSSFrameConstructor::ConstructNonScrollableBlock),
|
|
"Unexpected FCDATA_FORCED_NON_SCROLLABLE_BLOCK flag");
|
|
MOZ_ASSERT(!(bits & FCDATA_IS_WRAPPER_ANON_BOX) ||
|
|
(bits & FCDATA_USE_CHILD_ITEMS),
|
|
"Wrapper anon boxes should always have FCDATA_USE_CHILD_ITEMS");
|
|
|
|
// 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;
|
|
nsIContent* parent = content->GetParent();
|
|
|
|
// Push display:contents ancestors.
|
|
Maybe<AutoDisplayContentsAncestorPusher> adcp;
|
|
if (aState.mTreeMatchContext) {
|
|
adcp.emplace(*aState.mTreeMatchContext, aState.mPresContext, parent);
|
|
} else {
|
|
MOZ_ASSERT(content->IsStyledByServo());
|
|
}
|
|
|
|
// Get the parent of the content and check if it is a XBL children element.
|
|
// Push the children element as an ancestor here because it does
|
|
// not have a frame and would not otherwise be pushed as an ancestor. It is
|
|
// necessary to do so in order to correctly handle style resolution on
|
|
// descendants. (If !adcp.IsEmpty() then it was already pushed by
|
|
// AutoDisplayContentsAncestorPusher above.)
|
|
TreeMatchContext::AutoAncestorPusher
|
|
insertionPointPusher(aState.mTreeMatchContext);
|
|
if (adcp.isSome() && adcp->IsEmpty() && parent &&
|
|
nsContentUtils::IsContentInsertionPoint(parent)) {
|
|
if (aState.mTreeMatchContext->mAncestorFilter.HasFilter()) {
|
|
insertionPointPusher.PushAncestorAndStyleScope(parent);
|
|
} else {
|
|
insertionPointPusher.PushStyleScope(parent);
|
|
}
|
|
}
|
|
|
|
// Push the content as a style ancestor now, so we don't have to do
|
|
// it in our various full-constructor functions. In particular,
|
|
// since a number of full-constructor functions don't actually call
|
|
// ProcessChildren in some cases (e.g. for CSS anonymous table boxes
|
|
// or for situations where only anonymouse children are having
|
|
// frames constructed), this is the best place to bottleneck the
|
|
// pushing of the content instead of having to do it in multiple
|
|
// places.
|
|
TreeMatchContext::AutoAncestorPusher
|
|
ancestorPusher(aState.mTreeMatchContext);
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(content);
|
|
} else {
|
|
ancestorPusher.PushStyleScope(content);
|
|
}
|
|
|
|
nsIFrame* newFrame;
|
|
nsIFrame* primaryFrame;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
const nsStyleDisplay* display = styleContext->StyleDisplay();
|
|
if (bits & FCDATA_FUNC_IS_FULL_CTOR) {
|
|
newFrame =
|
|
(this->*(data->mFullConstructor))(aState, aItem, aParentFrame,
|
|
display, aFrameItems);
|
|
MOZ_ASSERT(newFrame, "Full constructor failed");
|
|
primaryFrame = newFrame;
|
|
} else {
|
|
newFrame =
|
|
(*data->mFunc.mCreationFunc)(mPresShell, styleContext);
|
|
|
|
bool allowOutOfFlow = !(bits & FCDATA_DISALLOW_OUT_OF_FLOW);
|
|
bool isPopup = aItem.mIsPopup;
|
|
NS_ASSERTION(!isPopup ||
|
|
(aState.mPopupItems.containingBlock &&
|
|
aState.mPopupItems.containingBlock->IsPopupSetFrame()),
|
|
"Should have a containing block here!");
|
|
|
|
nsContainerFrame* geometricParent =
|
|
isPopup ? aState.mPopupItems.containingBlock :
|
|
(allowOutOfFlow ? aState.GetGeometricParent(display, aParentFrame)
|
|
: aParentFrame);
|
|
|
|
// Must init frameToAddToList to null, since it's inout
|
|
nsIFrame* frameToAddToList = nullptr;
|
|
if ((bits & FCDATA_MAY_NEED_SCROLLFRAME) &&
|
|
display->IsScrollableOverflow()) {
|
|
nsContainerFrame* scrollframe = nullptr;
|
|
BuildScrollFrame(aState, content, styleContext, newFrame,
|
|
geometricParent, scrollframe);
|
|
frameToAddToList = scrollframe;
|
|
} else {
|
|
InitAndRestoreFrame(aState, content, geometricParent, newFrame);
|
|
frameToAddToList = newFrame;
|
|
}
|
|
|
|
// Use frameToAddToList as the primary frame. In the non-scrollframe case
|
|
// they're equal, but in the scrollframe case newFrame is the scrolled
|
|
// frame, while frameToAddToList is the scrollframe (and should be the
|
|
// primary frame).
|
|
primaryFrame = frameToAddToList;
|
|
|
|
// If we need to create a block formatting context to wrap our
|
|
// kids, do it now.
|
|
const nsStyleDisplay* maybeAbsoluteContainingBlockDisplay = display;
|
|
nsIFrame* maybeAbsoluteContainingBlockStyleFrame = primaryFrame;
|
|
nsIFrame* maybeAbsoluteContainingBlock = newFrame;
|
|
nsIFrame* possiblyLeafFrame = newFrame;
|
|
if (bits & FCDATA_CREATE_BLOCK_WRAPPER_FOR_ALL_KIDS) {
|
|
RefPtr<nsStyleContext> outerSC = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(*data->mAnonBoxPseudo,
|
|
styleContext);
|
|
#ifdef DEBUG
|
|
nsContainerFrame* containerFrame = do_QueryFrame(newFrame);
|
|
MOZ_ASSERT(containerFrame);
|
|
#endif
|
|
nsContainerFrame* container = static_cast<nsContainerFrame*>(newFrame);
|
|
nsContainerFrame* outerFrame;
|
|
nsContainerFrame* innerFrame;
|
|
if (bits & FCDATA_ALLOW_GRID_FLEX_COLUMNSET) {
|
|
switch (display->mDisplay) {
|
|
case StyleDisplay::Flex:
|
|
case StyleDisplay::InlineFlex:
|
|
outerFrame = NS_NewFlexContainerFrame(mPresShell, outerSC);
|
|
InitAndRestoreFrame(aState, content, container, outerFrame);
|
|
innerFrame = outerFrame;
|
|
break;
|
|
case StyleDisplay::Grid:
|
|
case StyleDisplay::InlineGrid:
|
|
outerFrame = NS_NewGridContainerFrame(mPresShell, outerSC);
|
|
InitAndRestoreFrame(aState, content, container, outerFrame);
|
|
innerFrame = outerFrame;
|
|
break;
|
|
default: {
|
|
nsContainerFrame* columnSetFrame = nullptr;
|
|
RefPtr<nsStyleContext> innerSC = outerSC;
|
|
const nsStyleColumn* columns = outerSC->StyleColumn();
|
|
if (columns->mColumnCount != NS_STYLE_COLUMN_COUNT_AUTO ||
|
|
columns->mColumnWidth.GetUnit() != eStyleUnit_Auto) {
|
|
columnSetFrame =
|
|
NS_NewColumnSetFrame(mPresShell, outerSC,
|
|
nsFrameState(NS_FRAME_OWNS_ANON_BOXES));
|
|
InitAndRestoreFrame(aState, content, container, columnSetFrame);
|
|
innerSC = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::columnContent,
|
|
outerSC);
|
|
}
|
|
innerFrame = NS_NewBlockFormattingContext(mPresShell, innerSC);
|
|
if (columnSetFrame) {
|
|
InitAndRestoreFrame(aState, content, columnSetFrame, innerFrame);
|
|
SetInitialSingleChild(columnSetFrame, innerFrame);
|
|
outerFrame = columnSetFrame;
|
|
} else {
|
|
InitAndRestoreFrame(aState, content, container, innerFrame);
|
|
outerFrame = innerFrame;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
innerFrame = NS_NewBlockFormattingContext(mPresShell, outerSC);
|
|
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.
|
|
auto outerDisplay = outerSC->StyleDisplay();
|
|
if (outerDisplay->IsAbsPosContainingBlock(outerFrame)) {
|
|
maybeAbsoluteContainingBlockDisplay = outerDisplay;
|
|
maybeAbsoluteContainingBlock = outerFrame;
|
|
maybeAbsoluteContainingBlockStyleFrame = outerFrame;
|
|
innerFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
}
|
|
|
|
// Our kids should go into the innerFrame.
|
|
newFrame = innerFrame;
|
|
}
|
|
|
|
aState.AddChild(frameToAddToList, aFrameItems, content, styleContext,
|
|
aParentFrame, allowOutOfFlow, allowOutOfFlow, isPopup);
|
|
|
|
nsContainerFrame* newFrameAsContainer = do_QueryFrame(newFrame);
|
|
if (newFrameAsContainer) {
|
|
#ifdef MOZ_XUL
|
|
// Icky XUL stuff, sadly
|
|
|
|
if (aItem.mIsRootPopupgroup) {
|
|
NS_ASSERTION(nsIRootBox::GetRootBox(mPresShell) &&
|
|
nsIRootBox::GetRootBox(mPresShell)->GetPopupSetFrame() ==
|
|
newFrame,
|
|
"Unexpected PopupSetFrame");
|
|
aState.mPopupItems.containingBlock = newFrameAsContainer;
|
|
aState.mHavePendingPopupgroup = false;
|
|
}
|
|
#endif /* MOZ_XUL */
|
|
|
|
// Process the child content if requested
|
|
nsFrameItems childItems;
|
|
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);
|
|
// This check is identical to nsStyleDisplay::IsAbsPosContainingBlock
|
|
// except without the assertion that the style display and frame match.
|
|
// When constructing scroll frames we intentionally use the style
|
|
// display for the outer, but make the inner the containing block.
|
|
if ((maybeAbsoluteContainingBlockDisplay->IsAbsolutelyPositionedStyle() ||
|
|
maybeAbsoluteContainingBlockDisplay->IsRelativelyPositionedStyle() ||
|
|
maybeAbsoluteContainingBlockDisplay->IsFixedPosContainingBlock(
|
|
maybeAbsoluteContainingBlockStyleFrame)) &&
|
|
!nsSVGUtils::IsInSVGTextSubtree(maybeAbsoluteContainingBlockStyleFrame)) {
|
|
nsContainerFrame* cf = static_cast<nsContainerFrame*>(
|
|
maybeAbsoluteContainingBlock);
|
|
aState.PushAbsoluteContainingBlock(cf, cf, absoluteSaveState);
|
|
}
|
|
}
|
|
|
|
if (!aItem.mAnonChildren.IsEmpty()) {
|
|
NS_ASSERTION(!(bits & FCDATA_USE_CHILD_ITEMS),
|
|
"We should not have both anonymous and non-anonymous "
|
|
"children in a given FrameConstructorItem");
|
|
AddFCItemsForAnonymousContent(aState, newFrameAsContainer, aItem.mAnonChildren,
|
|
aItem.mChildItems);
|
|
bits |= FCDATA_USE_CHILD_ITEMS;
|
|
}
|
|
|
|
if (bits & FCDATA_USE_CHILD_ITEMS) {
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
|
|
if (ShouldSuppressFloatingOfDescendants(newFrame)) {
|
|
aState.PushFloatContainingBlock(nullptr, floatSaveState);
|
|
} else if (newFrame->IsFloatContainingBlock()) {
|
|
aState.PushFloatContainingBlock(newFrameAsContainer, floatSaveState);
|
|
}
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems,
|
|
newFrameAsContainer,
|
|
bits & FCDATA_IS_WRAPPER_ANON_BOX,
|
|
childItems);
|
|
} else {
|
|
// Process the child frames.
|
|
ProcessChildren(aState, content, styleContext, newFrameAsContainer,
|
|
!(bits & FCDATA_DISALLOW_GENERATED_CONTENT),
|
|
childItems,
|
|
(bits & FCDATA_ALLOW_BLOCK_STYLES) != 0,
|
|
aItem.mPendingBinding, possiblyLeafFrame);
|
|
}
|
|
|
|
if (bits & FCDATA_WRAP_KIDS_IN_BLOCKS) {
|
|
nsFrameItems newItems;
|
|
nsFrameItems currentBlockItems;
|
|
nsIFrame* f;
|
|
while ((f = childItems.FirstChild()) != nullptr) {
|
|
bool wrapFrame = IsInlineFrame(f) || IsFramePartOfIBSplit(f);
|
|
if (!wrapFrame) {
|
|
FlushAccumulatedBlock(aState, content, newFrameAsContainer,
|
|
currentBlockItems, newItems);
|
|
}
|
|
|
|
childItems.RemoveFrame(f);
|
|
if (wrapFrame) {
|
|
currentBlockItems.AddChild(f);
|
|
} else {
|
|
newItems.AddChild(f);
|
|
}
|
|
}
|
|
FlushAccumulatedBlock(aState, content, newFrameAsContainer,
|
|
currentBlockItems, newItems);
|
|
|
|
if (childItems.NotEmpty()) {
|
|
// an error must have occurred, delete unprocessed frames
|
|
childItems.DestroyFrames();
|
|
}
|
|
|
|
childItems = newItems;
|
|
}
|
|
|
|
// Set the frame's initial child list
|
|
// Note that MathML depends on this being called even if
|
|
// childItems is empty!
|
|
newFrameAsContainer->SetInitialChildList(kPrincipalList, childItems);
|
|
}
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
// More icky XUL stuff
|
|
if (aItem.mNameSpaceID == kNameSpaceID_XUL &&
|
|
(aItem.mTag == nsGkAtoms::treechildren || // trees always need titletips
|
|
content->HasAttr(kNameSpaceID_None, nsGkAtoms::tooltiptext) ||
|
|
content->HasAttr(kNameSpaceID_None, nsGkAtoms::tooltip))) {
|
|
nsIRootBox* rootBox = nsIRootBox::GetRootBox(mPresShell);
|
|
if (rootBox) {
|
|
rootBox->AddTooltipSupport(content);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
NS_ASSERTION(newFrame->IsFrameOfType(nsIFrame::eLineParticipant) ==
|
|
((bits & FCDATA_IS_LINE_PARTICIPANT) != 0),
|
|
"Incorrectly set FCDATA_IS_LINE_PARTICIPANT bits");
|
|
|
|
if (aItem.mIsAnonymousContentCreatorContent) {
|
|
primaryFrame->AddStateBits(NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT);
|
|
}
|
|
|
|
// 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 ||
|
|
(primaryFrame->HasAnyStateBits(NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT |
|
|
NS_FRAME_GENERATED_CONTENT) &&
|
|
!aItem.mContent->GetPrimaryFrame())) &&
|
|
!(bits & FCDATA_SKIP_FRAMESET)) {
|
|
aItem.mContent->SetPrimaryFrame(primaryFrame);
|
|
ActiveLayerTracker::TransferActivityToFrame(aItem.mContent, primaryFrame);
|
|
}
|
|
}
|
|
|
|
static void
|
|
SetFlagsOnSubtree(nsIContent *aNode, uintptr_t aFlagsToSet)
|
|
{
|
|
#ifdef DEBUG
|
|
// Make sure that the node passed to us doesn't have any XBL children
|
|
{
|
|
FlattenedChildIterator iter(aNode);
|
|
NS_ASSERTION(!iter.XBLInvolved() || !iter.GetNextChild(),
|
|
"The node should not have any XBL children");
|
|
}
|
|
#endif
|
|
|
|
// Set the flag on the node itself
|
|
aNode->SetFlags(aFlagsToSet);
|
|
|
|
// Set the flag on all of its children recursively
|
|
for (nsIContent* child = aNode->GetFirstChild(); child;
|
|
child = child->GetNextSibling()) {
|
|
SetFlagsOnSubtree(child, aFlagsToSet);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function takes a tree of nsIAnonymousContentCreator::ContentInfo
|
|
* objects where the nsIContent nodes have just been created, and appends the
|
|
* nsIContent children in the tree to their parent. The leaf nsIContent objects
|
|
* are appended first to minimize the number of notifications that are sent
|
|
* out (i.e. by appending as many descendants as posible while their parent is
|
|
* not yet in the document tree).
|
|
*
|
|
* This function is used simply as a convenience so that implementations of
|
|
* nsIAnonymousContentCreator::CreateAnonymousContent don't all have to have
|
|
* their own code to connect the elements that they create.
|
|
*/
|
|
static void
|
|
ConnectAnonymousTreeDescendants(nsIContent* aParent,
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>& aContent)
|
|
{
|
|
uint32_t count = aContent.Length();
|
|
for (uint32_t i=0; i < count; i++) {
|
|
nsIContent* content = aContent[i].mContent;
|
|
NS_ASSERTION(content, "null anonymous content?");
|
|
|
|
ConnectAnonymousTreeDescendants(content, aContent[i].mChildren);
|
|
|
|
aParent->AppendChildTo(content, false);
|
|
}
|
|
}
|
|
|
|
static void
|
|
SetNativeAnonymousBitOnDescendants(nsIContent* aRoot)
|
|
{
|
|
for (nsIContent* curr = aRoot; curr; curr = curr->GetNextNode(aRoot)) {
|
|
curr->SetFlags(NODE_IS_NATIVE_ANONYMOUS);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
uint32_t count = aContent.Length();
|
|
for (uint32_t i=0; i < count; i++) {
|
|
// get our child's content and set its parent to our content
|
|
nsIContent* content = aContent[i].mContent;
|
|
NS_ASSERTION(content, "null anonymous content?");
|
|
|
|
ConnectAnonymousTreeDescendants(content, aContent[i].mChildren);
|
|
|
|
LayoutFrameType parentFrameType = aParentFrame->Type();
|
|
if (parentFrameType == LayoutFrameType::SVGUse) {
|
|
// least-surprise CSS binding until we do the SVG specified
|
|
// cascading rules for <svg:use> - bug 265894
|
|
content->SetFlags(NODE_IS_ANONYMOUS_ROOT);
|
|
} else {
|
|
content->SetIsNativeAnonymousRoot();
|
|
// Don't mark descendants of the custom content container
|
|
// as native anonymous. When canvas custom content is initially
|
|
// created and appended to the custom content container, in
|
|
// nsIDocument::InsertAnonymousContent, it is not considered native
|
|
// anonymous content. But if we end up reframing the root element,
|
|
// we will re-create the nsCanvasFrame, and we would end up in here,
|
|
// marking it as NAC. Existing uses of canvas custom content would
|
|
// break if it becomes NAC (since each element starts inheriting
|
|
// styles from its closest non-NAC ancestor, rather than from its
|
|
// parent).
|
|
if (!(parentFrameType == LayoutFrameType::Canvas &&
|
|
content == static_cast<nsCanvasFrame*>(aParentFrame)
|
|
->GetCustomContentContainer())) {
|
|
SetNativeAnonymousBitOnDescendants(content);
|
|
}
|
|
}
|
|
|
|
bool anonContentIsEditable = content->HasFlag(NODE_IS_EDITABLE);
|
|
|
|
// If the parent is in a shadow tree, make sure we don't
|
|
// bind with a document because shadow roots and its descendants
|
|
// are not in document.
|
|
nsIDocument* bindDocument =
|
|
aParent->HasFlag(NODE_IS_IN_SHADOW_TREE) ? nullptr : mDocument;
|
|
rv = content->BindToTree(bindDocument, aParent, aParent, true);
|
|
// If the anonymous content creator requested that the content should be
|
|
// editable, honor its request.
|
|
// We need to set the flag on the whole subtree, because existing
|
|
// children's flags have already been set as part of the BindToTree operation.
|
|
if (anonContentIsEditable) {
|
|
NS_ASSERTION(aParentFrame->IsTextInputFrame(),
|
|
"We only expect this for anonymous content under a text control frame");
|
|
SetFlagsOnSubtree(content, NODE_IS_EDITABLE);
|
|
}
|
|
if (NS_FAILED(rv)) {
|
|
content->UnbindFromTree();
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
if (ServoStyleSet* styleSet = mPresShell->StyleSet()->GetAsServo()) {
|
|
// Eagerly compute styles for the anonymous content tree.
|
|
for (auto& info : aContent) {
|
|
if (info.mContent->IsElement()) {
|
|
styleSet->StyleNewSubtree(info.mContent->AsElement());
|
|
}
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static
|
|
bool IsXULDisplayType(const nsStyleDisplay* aDisplay)
|
|
{
|
|
return (aDisplay->mDisplay == StyleDisplay::MozInlineBox ||
|
|
#ifdef MOZ_XUL
|
|
aDisplay->mDisplay == StyleDisplay::MozInlineGrid ||
|
|
aDisplay->mDisplay == StyleDisplay::MozInlineStack ||
|
|
#endif
|
|
aDisplay->mDisplay == StyleDisplay::MozBox
|
|
#ifdef MOZ_XUL
|
|
|| aDisplay->mDisplay == StyleDisplay::MozGrid ||
|
|
aDisplay->mDisplay == StyleDisplay::MozStack ||
|
|
aDisplay->mDisplay == StyleDisplay::MozGridGroup ||
|
|
aDisplay->mDisplay == StyleDisplay::MozGridLine ||
|
|
aDisplay->mDisplay == StyleDisplay::MozDeck ||
|
|
aDisplay->mDisplay == StyleDisplay::MozPopup ||
|
|
aDisplay->mDisplay == StyleDisplay::MozGroupbox
|
|
#endif
|
|
);
|
|
}
|
|
|
|
|
|
// XUL frames are not allowed to be out of flow.
|
|
#define SIMPLE_XUL_FCDATA(_func) \
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_SKIP_ABSPOS_PUSH, \
|
|
_func)
|
|
#define SCROLLABLE_XUL_FCDATA(_func) \
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_SKIP_ABSPOS_PUSH | \
|
|
FCDATA_MAY_NEED_SCROLLFRAME, _func)
|
|
// .. 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) \
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_MAY_NEED_SCROLLFRAME, _func)
|
|
|
|
#define SIMPLE_XUL_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, SIMPLE_XUL_FCDATA(_func) }
|
|
#define SCROLLABLE_XUL_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, SCROLLABLE_XUL_FCDATA(_func) }
|
|
#define SIMPLE_XUL_DISPLAY_CREATE(_display, _func) \
|
|
FCDATA_FOR_DISPLAY(_display, SIMPLE_XUL_FCDATA(_func))
|
|
#define SCROLLABLE_XUL_DISPLAY_CREATE(_display, _func) \
|
|
FCDATA_FOR_DISPLAY(_display, SCROLLABLE_XUL_FCDATA(_func))
|
|
#define SCROLLABLE_ABSPOS_CONTAINER_XUL_DISPLAY_CREATE(_display, _func) \
|
|
FCDATA_FOR_DISPLAY(_display, SCROLLABLE_ABSPOS_CONTAINER_XUL_FCDATA(_func))
|
|
|
|
static
|
|
nsIFrame* NS_NewGridBoxFrame(nsIPresShell* aPresShell,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
nsCOMPtr<nsBoxLayout> layout;
|
|
NS_NewGridLayout2(aPresShell, getter_AddRefs(layout));
|
|
return NS_NewBoxFrame(aPresShell, aStyleContext, false, layout);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULTagData(Element* aElement,
|
|
nsAtom* aTag,
|
|
int32_t aNameSpaceID,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aNameSpaceID != kNameSpaceID_XUL) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionDataByTag sXULTagData[] = {
|
|
#ifdef MOZ_XUL
|
|
SCROLLABLE_XUL_CREATE(button, NS_NewButtonBoxFrame),
|
|
SCROLLABLE_XUL_CREATE(checkbox, NS_NewButtonBoxFrame),
|
|
SCROLLABLE_XUL_CREATE(radio, NS_NewButtonBoxFrame),
|
|
SCROLLABLE_XUL_CREATE(autorepeatbutton, NS_NewAutoRepeatBoxFrame),
|
|
SCROLLABLE_XUL_CREATE(titlebar, NS_NewTitleBarFrame),
|
|
SCROLLABLE_XUL_CREATE(resizer, NS_NewResizerFrame),
|
|
SIMPLE_XUL_CREATE(image, NS_NewImageBoxFrame),
|
|
SIMPLE_XUL_CREATE(spring, NS_NewLeafBoxFrame),
|
|
SIMPLE_XUL_CREATE(spacer, NS_NewLeafBoxFrame),
|
|
SIMPLE_XUL_CREATE(treechildren, NS_NewTreeBodyFrame),
|
|
SIMPLE_XUL_CREATE(treecol, NS_NewTreeColFrame),
|
|
SIMPLE_XUL_CREATE(text, NS_NewTextBoxFrame),
|
|
SIMPLE_TAG_CHAIN(label, nsCSSFrameConstructor::FindXULLabelData),
|
|
SIMPLE_TAG_CHAIN(description, nsCSSFrameConstructor::FindXULDescriptionData),
|
|
SIMPLE_XUL_CREATE(menu, NS_NewMenuFrame),
|
|
SIMPLE_XUL_CREATE(menubutton, NS_NewMenuFrame),
|
|
SIMPLE_XUL_CREATE(menuitem, NS_NewMenuItemFrame),
|
|
#ifdef XP_MACOSX
|
|
SIMPLE_TAG_CHAIN(menubar, nsCSSFrameConstructor::FindXULMenubarData),
|
|
#else
|
|
SIMPLE_XUL_CREATE(menubar, NS_NewMenuBarFrame),
|
|
#endif /* XP_MACOSX */
|
|
SIMPLE_TAG_CHAIN(popupgroup, nsCSSFrameConstructor::FindPopupGroupData),
|
|
SIMPLE_XUL_CREATE(iframe, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(editor, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(browser, NS_NewSubDocumentFrame),
|
|
SIMPLE_XUL_CREATE(progressmeter, NS_NewProgressMeterFrame),
|
|
SIMPLE_XUL_CREATE(splitter, NS_NewSplitterFrame),
|
|
SIMPLE_TAG_CHAIN(listboxbody,
|
|
nsCSSFrameConstructor::FindXULListBoxBodyData),
|
|
SIMPLE_TAG_CHAIN(listitem, nsCSSFrameConstructor::FindXULListItemData),
|
|
#endif /* MOZ_XUL */
|
|
SIMPLE_XUL_CREATE(slider, NS_NewSliderFrame),
|
|
SIMPLE_XUL_CREATE(scrollbar, NS_NewScrollbarFrame),
|
|
SIMPLE_XUL_CREATE(scrollbarbutton, NS_NewScrollbarButtonFrame)
|
|
};
|
|
|
|
return FindDataByTag(aTag, aElement, aStyleContext, sXULTagData,
|
|
ArrayLength(sXULTagData));
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindPopupGroupData(Element* aElement,
|
|
nsStyleContext* /* unused */)
|
|
{
|
|
if (!aElement->IsRootOfNativeAnonymousSubtree()) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sPopupSetData =
|
|
SIMPLE_XUL_FCDATA(NS_NewPopupSetFrame);
|
|
return &sPopupSetData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData
|
|
nsCSSFrameConstructor::sXULTextBoxData = SIMPLE_XUL_FCDATA(NS_NewTextBoxFrame);
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULLabelData(Element* aElement,
|
|
nsStyleContext* /* unused */)
|
|
{
|
|
if (aElement->HasAttr(kNameSpaceID_None, nsGkAtoms::value)) {
|
|
return &sXULTextBoxData;
|
|
}
|
|
|
|
static const FrameConstructionData sLabelData =
|
|
SIMPLE_XUL_FCDATA(NS_NewXULLabelFrame);
|
|
return &sLabelData;
|
|
}
|
|
|
|
static nsIFrame*
|
|
NS_NewXULDescriptionFrame(nsIPresShell* aPresShell, nsStyleContext *aContext)
|
|
{
|
|
// XXXbz do we really need to set up the block formatting context root? If the
|
|
// parent is not a block we'll get it anyway, and if it is, do we want it?
|
|
return NS_NewBlockFormattingContext(aPresShell, aContext);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULDescriptionData(Element* aElement,
|
|
nsStyleContext* /* unused */)
|
|
{
|
|
if (aElement->HasAttr(kNameSpaceID_None, nsGkAtoms::value)) {
|
|
return &sXULTextBoxData;
|
|
}
|
|
|
|
static const FrameConstructionData sDescriptionData =
|
|
SIMPLE_XUL_FCDATA(NS_NewXULDescriptionFrame);
|
|
return &sDescriptionData;
|
|
}
|
|
|
|
#ifdef XP_MACOSX
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULMenubarData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
nsCOMPtr<nsIDocShell> treeItem =
|
|
aStyleContext->PresContext()->GetDocShell();
|
|
if (treeItem && nsIDocShellTreeItem::typeChrome == treeItem->ItemType()) {
|
|
nsCOMPtr<nsIDocShellTreeItem> parent;
|
|
treeItem->GetParent(getter_AddRefs(parent));
|
|
if (!parent) {
|
|
// This is the root. Suppress the menubar, since on Mac
|
|
// window menus are not attached to the window.
|
|
static const FrameConstructionData sSuppressData = SUPPRESS_FCDATA();
|
|
return &sSuppressData;
|
|
}
|
|
}
|
|
|
|
static const FrameConstructionData sMenubarData =
|
|
SIMPLE_XUL_FCDATA(NS_NewMenuBarFrame);
|
|
return &sMenubarData;
|
|
}
|
|
#endif /* XP_MACOSX */
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULListBoxBodyData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aStyleContext->StyleDisplay()->mDisplay !=
|
|
StyleDisplay::MozGridGroup) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sListBoxBodyData =
|
|
SCROLLABLE_XUL_FCDATA(NS_NewListBoxBodyFrame);
|
|
return &sListBoxBodyData;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULListItemData(Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aStyleContext->StyleDisplay()->mDisplay != StyleDisplay::MozGridLine) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sListItemData =
|
|
SCROLLABLE_XUL_FCDATA(NS_NewListItemFrame);
|
|
return &sListItemData;
|
|
}
|
|
|
|
#endif /* MOZ_XUL */
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindXULDisplayData(const nsStyleDisplay* aDisplay,
|
|
Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
static const FrameConstructionDataByDisplay sXULDisplayData[] = {
|
|
SCROLLABLE_ABSPOS_CONTAINER_XUL_DISPLAY_CREATE(StyleDisplay::MozBox,
|
|
NS_NewBoxFrame),
|
|
SCROLLABLE_ABSPOS_CONTAINER_XUL_DISPLAY_CREATE(StyleDisplay::MozInlineBox,
|
|
NS_NewBoxFrame),
|
|
#ifdef MOZ_XUL
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozGrid, NS_NewGridBoxFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozInlineGrid, NS_NewGridBoxFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozGridGroup,
|
|
NS_NewGridRowGroupFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozGridLine,
|
|
NS_NewGridRowLeafFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozStack, NS_NewStackFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozInlineStack, NS_NewStackFrame),
|
|
SIMPLE_XUL_DISPLAY_CREATE(StyleDisplay::MozDeck, NS_NewDeckFrame),
|
|
SCROLLABLE_XUL_DISPLAY_CREATE(StyleDisplay::MozGroupbox, NS_NewGroupBoxFrame),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::MozPopup,
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | FCDATA_IS_POPUP |
|
|
FCDATA_SKIP_ABSPOS_PUSH, NS_NewMenuPopupFrame))
|
|
#endif /* MOZ_XUL */
|
|
};
|
|
|
|
if (aDisplay->mDisplay < StyleDisplay::MozBox) {
|
|
return nullptr;
|
|
}
|
|
|
|
MOZ_ASSERT(aDisplay->mDisplay <= StyleDisplay::MozPopup,
|
|
"Someone added a new display value?");
|
|
|
|
const FrameConstructionDataByDisplay& data =
|
|
sXULDisplayData[size_t(aDisplay->mDisplay) - size_t(StyleDisplay::MozBox)];
|
|
MOZ_ASSERT(aDisplay->mDisplay == data.mDisplay,
|
|
"Did someone mess with the order?");
|
|
|
|
return &data.mData;
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::BeginBuildingScrollFrame(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aContentStyle,
|
|
nsContainerFrame* aParentFrame,
|
|
nsAtom* aScrolledPseudo,
|
|
bool aIsRoot,
|
|
nsContainerFrame*& aNewFrame)
|
|
{
|
|
nsContainerFrame* gfxScrollFrame = aNewFrame;
|
|
|
|
nsFrameItems anonymousItems;
|
|
|
|
RefPtr<nsStyleContext> contentStyle = aContentStyle;
|
|
|
|
if (!gfxScrollFrame) {
|
|
// Build a XULScrollFrame when the child is a box, otherwise an
|
|
// HTMLScrollFrame
|
|
// XXXbz this is the lone remaining consumer of IsXULDisplayType.
|
|
// I wonder whether we can eliminate that somehow.
|
|
const nsStyleDisplay* displayStyle = aContentStyle->StyleDisplay();
|
|
if (IsXULDisplayType(displayStyle)) {
|
|
gfxScrollFrame = NS_NewXULScrollFrame(mPresShell, contentStyle, aIsRoot,
|
|
displayStyle->mDisplay == StyleDisplay::MozStack ||
|
|
displayStyle->mDisplay == StyleDisplay::MozInlineStack);
|
|
} else {
|
|
gfxScrollFrame = NS_NewHTMLScrollFrame(mPresShell, contentStyle, aIsRoot);
|
|
}
|
|
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, gfxScrollFrame);
|
|
}
|
|
|
|
// 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, gfxScrollFrame, scrollNAC);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
if (scrollNAC.Length() > 0) {
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(aState.mTreeMatchContext);
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(aContent->AsElement());
|
|
} else {
|
|
ancestorPusher.PushStyleScope(aContent->AsElement());
|
|
}
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
AddFCItemsForAnonymousContent(aState, gfxScrollFrame, scrollNAC, items);
|
|
ConstructFramesFromItemList(aState, items, gfxScrollFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
anonymousItems);
|
|
}
|
|
|
|
aNewFrame = gfxScrollFrame;
|
|
gfxScrollFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
|
|
// we used the style that was passed in. So resolve another one.
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
RefPtr<nsStyleContext> scrolledChildStyle =
|
|
styleSet->ResolveInheritingAnonymousBoxStyle(aScrolledPseudo, contentStyle);
|
|
|
|
if (gfxScrollFrame) {
|
|
gfxScrollFrame->SetInitialChildList(kPrincipalList, anonymousItems);
|
|
}
|
|
|
|
return scrolledChildStyle.forget();
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::FinishBuildingScrollFrame(nsContainerFrame* aScrollFrame,
|
|
nsIFrame* aScrolledFrame)
|
|
{
|
|
nsFrameList scrolled(aScrolledFrame, aScrolledFrame);
|
|
aScrollFrame->AppendFrames(kPrincipalList, scrolled);
|
|
}
|
|
|
|
/**
|
|
* Called to wrap a gfx scrollframe around a frame. The hierarchy will look like this
|
|
*
|
|
* ------- for gfx scrollbars ------
|
|
*
|
|
*
|
|
* ScrollFrame
|
|
* ^
|
|
* |
|
|
* Frame (scrolled frame you passed in)
|
|
*
|
|
*
|
|
*-----------------------------------
|
|
* LEGEND:
|
|
*
|
|
* ScrollFrame: This is a frame that manages gfx cross platform frame based scrollbars.
|
|
*
|
|
* @param aContent the content node of the child to wrap.
|
|
* @param aScrolledFrame The frame of the content to wrap. This should not be
|
|
* Initialized. This method will initialize it with a scrolled pseudo
|
|
* and no nsIContent. The content will be attached to the scrollframe
|
|
* returned.
|
|
* @param aContentStyle the style context that has already been resolved for the content being passed in.
|
|
*
|
|
* @param aParentFrame The parent to attach the scroll frame to
|
|
*
|
|
* @param aNewFrame The new scrollframe or gfx scrollframe that we create. It will contain the
|
|
* scrolled frame you passed in. (returned)
|
|
* If this is not null, we'll just use it
|
|
* @param aScrolledContentStyle the style that was resolved for the scrolled frame. (returned)
|
|
*/
|
|
void
|
|
nsCSSFrameConstructor::BuildScrollFrame(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aContentStyle,
|
|
nsIFrame* aScrolledFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
nsContainerFrame*& aNewFrame)
|
|
{
|
|
RefPtr<nsStyleContext> scrolledContentStyle =
|
|
BeginBuildingScrollFrame(aState, aContent, aContentStyle, aParentFrame,
|
|
nsCSSAnonBoxes::scrolledContent,
|
|
false, aNewFrame);
|
|
|
|
aScrolledFrame->SetStyleContextWithoutNotification(scrolledContentStyle);
|
|
InitAndRestoreFrame(aState, aContent, aNewFrame, aScrolledFrame);
|
|
|
|
FinishBuildingScrollFrame(aNewFrame, aScrolledFrame);
|
|
}
|
|
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindDisplayData(const nsStyleDisplay* aDisplay,
|
|
Element* aElement,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
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() ||
|
|
aDisplay->mDisplay == StyleDisplay::Contents,
|
|
"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->UpdateViewportScrollbarStylesOverride() == aElement;
|
|
}
|
|
}
|
|
|
|
NS_ASSERTION(!propagatedScrollToViewport ||
|
|
!mPresShell->GetPresContext()->IsPaginated(),
|
|
"Shouldn't propagate scroll in paginated contexts");
|
|
|
|
if (aDisplay->IsBlockInsideStyle()) {
|
|
// 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);
|
|
bool caption = aDisplay->mDisplay == StyleDisplay::TableCaption;
|
|
bool suppressScrollFrame = false;
|
|
bool needScrollFrame = aDisplay->IsScrollableOverflow() &&
|
|
!propagatedScrollToViewport;
|
|
if (needScrollFrame) {
|
|
suppressScrollFrame = mPresShell->GetPresContext()->IsPaginated() &&
|
|
aDisplay->IsBlockOutsideStyle() &&
|
|
!aElement->IsInNativeAnonymousSubtree();
|
|
if (!suppressScrollFrame) {
|
|
static const FrameConstructionData sScrollableBlockData[2] =
|
|
{ FULL_CTOR_FCDATA(0, &nsCSSFrameConstructor::ConstructScrollableBlock),
|
|
FULL_CTOR_FCDATA(kCaptionCtorFlags,
|
|
&nsCSSFrameConstructor::ConstructScrollableBlock) };
|
|
return &sScrollableBlockData[caption];
|
|
}
|
|
|
|
// If the scrollable frame would have propagated its scrolling to the
|
|
// viewport, we still want to construct a regular block rather than a
|
|
// scrollframe so that it paginates correctly, but we don't want to set
|
|
// the bit on the block that tells it to clip at paint time.
|
|
if (mPresShell->GetPresContext()->
|
|
ElementWouldPropagateScrollbarStyles(aElement)) {
|
|
suppressScrollFrame = false;
|
|
}
|
|
}
|
|
|
|
// Handle various non-scrollable blocks.
|
|
static const FrameConstructionData sNonScrollableBlockData[2][2] = {
|
|
{ FULL_CTOR_FCDATA(0,
|
|
&nsCSSFrameConstructor::ConstructNonScrollableBlock),
|
|
FULL_CTOR_FCDATA(kCaptionCtorFlags,
|
|
&nsCSSFrameConstructor::ConstructNonScrollableBlock) },
|
|
{ FULL_CTOR_FCDATA(FCDATA_FORCED_NON_SCROLLABLE_BLOCK,
|
|
&nsCSSFrameConstructor::ConstructNonScrollableBlock),
|
|
FULL_CTOR_FCDATA(FCDATA_FORCED_NON_SCROLLABLE_BLOCK | kCaptionCtorFlags,
|
|
&nsCSSFrameConstructor::ConstructNonScrollableBlock) }
|
|
};
|
|
return &sNonScrollableBlockData[suppressScrollFrame][caption];
|
|
}
|
|
|
|
// If this is for a <body> node and we've propagated the scroll-frame to the
|
|
// viewport, we need to make sure not to add another layer of scrollbars, so
|
|
// we use a different FCData struct without FCDATA_MAY_NEED_SCROLLFRAME.
|
|
if (propagatedScrollToViewport && aDisplay->IsScrollableOverflow()) {
|
|
if (aDisplay->mDisplay == StyleDisplay::Flex ||
|
|
aDisplay->mDisplay == StyleDisplay::WebkitBox) {
|
|
static const FrameConstructionData sNonScrollableFlexData =
|
|
FCDATA_DECL(0, NS_NewFlexContainerFrame);
|
|
return &sNonScrollableFlexData;
|
|
}
|
|
if (aDisplay->mDisplay == StyleDisplay::Grid) {
|
|
static const FrameConstructionData sNonScrollableGridData =
|
|
FCDATA_DECL(0, NS_NewGridContainerFrame);
|
|
return &sNonScrollableGridData;
|
|
}
|
|
}
|
|
|
|
// NOTE: Make sure to keep this up to date with the StyleDisplay definition!
|
|
static const FrameConstructionDataByDisplay sDisplayData[] = {
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::None, UNREACHABLE_FCDATA()),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Block, UNREACHABLE_FCDATA()),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::FlowRoot, UNREACHABLE_FCDATA()),
|
|
// To keep the hash table small don't add inline frames (they're
|
|
// typically things like FONT and B), because we can quickly
|
|
// find them if we need to.
|
|
// XXXbz the "quickly" part is a bald-faced lie!
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Inline,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_INLINE | FCDATA_IS_LINE_PARTICIPANT,
|
|
&nsCSSFrameConstructor::ConstructInline)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::InlineBlock, UNREACHABLE_FCDATA()),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::ListItem, UNREACHABLE_FCDATA()),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Table,
|
|
FULL_CTOR_FCDATA(0, &nsCSSFrameConstructor::ConstructTable)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::InlineTable,
|
|
FULL_CTOR_FCDATA(0, &nsCSSFrameConstructor::ConstructTable)),
|
|
// 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 nsStyleContext::ApplyStyleFixups().
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableRowGroup,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableColumn,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeColGroup),
|
|
&nsCSSFrameConstructor::ConstructTableCol)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableColumnGroup,
|
|
FCDATA_DECL(FCDATA_IS_TABLE_PART | FCDATA_DISALLOW_OUT_OF_FLOW |
|
|
FCDATA_SKIP_ABSPOS_PUSH |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
NS_NewTableColGroupFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableHeaderGroup,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableFooterGroup,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableRow,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRowGroup),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableCell,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRow),
|
|
&nsCSSFrameConstructor::ConstructTableCell)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::TableCaption, UNREACHABLE_FCDATA()),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Flex,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewFlexContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::InlineFlex,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewFlexContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Grid,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewGridContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::InlineGrid,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewGridContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Ruby,
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT, NS_NewRubyFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::RubyBase,
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyBaseContainer),
|
|
NS_NewRubyBaseFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::RubyBaseContainer,
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby),
|
|
NS_NewRubyBaseContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::RubyText,
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyTextContainer),
|
|
NS_NewRubyTextFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::RubyTextContainer,
|
|
FCDATA_DECL(FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby),
|
|
NS_NewRubyTextContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::Contents,
|
|
FULL_CTOR_FCDATA(FCDATA_IS_CONTENTS, nullptr/*never called*/)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::WebkitBox,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewFlexContainerFrame)),
|
|
FCDATA_FOR_DISPLAY(StyleDisplay::WebkitInlineBox,
|
|
FCDATA_DECL(FCDATA_MAY_NEED_SCROLLFRAME, NS_NewFlexContainerFrame)),
|
|
};
|
|
static_assert(ArrayLength(sDisplayData) == size_t(StyleDisplay::WebkitInlineBox) + 1,
|
|
"Be sure to update sDisplayData if you touch StyleDisplay");
|
|
|
|
MOZ_ASSERT(size_t(aDisplay->mDisplay) < ArrayLength(sDisplayData),
|
|
"XUL display data should have already been handled");
|
|
|
|
// See the mDisplay fixup code in nsRuleNode::ComputeDisplayData.
|
|
MOZ_ASSERT(aDisplay->mDisplay != StyleDisplay::Contents ||
|
|
!aElement->IsRootOfNativeAnonymousSubtree(),
|
|
"display:contents on anonymous content is unsupported");
|
|
|
|
const FrameConstructionDataByDisplay& data =
|
|
sDisplayData[size_t(aDisplay->mDisplay)];
|
|
|
|
MOZ_ASSERT(data.mDisplay == aDisplay->mDisplay,
|
|
"Someone messed up the order in the display values");
|
|
|
|
return &data.mData;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructScrollableBlock(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
return ConstructScrollableBlockWithConstructor(aState, aItem, aParentFrame,
|
|
aDisplay, aFrameItems,
|
|
NS_NewBlockFormattingContext);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructScrollableBlockWithConstructor(
|
|
nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems,
|
|
BlockFrameCreationFunc aConstructor)
|
|
{
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
nsContainerFrame* newFrame = nullptr;
|
|
RefPtr<nsStyleContext> scrolledContentStyle
|
|
= BeginBuildingScrollFrame(aState, content, styleContext,
|
|
aState.GetGeometricParent(aDisplay, aParentFrame),
|
|
nsCSSAnonBoxes::scrolledContent,
|
|
false, newFrame);
|
|
|
|
// Create our block frame
|
|
// pass a temporary stylecontext, the correct one will be set later
|
|
nsContainerFrame* scrolledFrame = aConstructor(mPresShell, styleContext);
|
|
|
|
// 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, aFrameItems, content, styleContext, aParentFrame);
|
|
|
|
nsFrameItems blockItem;
|
|
ConstructBlock(aState, content, newFrame, newFrame, scrolledContentStyle,
|
|
&scrolledFrame, blockItem,
|
|
aDisplay->IsAbsPosContainingBlock(newFrame) ? newFrame : nullptr,
|
|
aItem.mPendingBinding);
|
|
|
|
MOZ_ASSERT(blockItem.OnlyChild() == scrolledFrame,
|
|
"Scrollframe's frameItems should be exactly the scrolled frame!");
|
|
FinishBuildingScrollFrame(newFrame, scrolledFrame);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructNonScrollableBlock(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
return ConstructNonScrollableBlockWithConstructor(aState, aItem, aParentFrame,
|
|
aDisplay, aFrameItems,
|
|
NS_NewBlockFrame);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructNonScrollableBlockWithConstructor(
|
|
nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems,
|
|
BlockFrameCreationFunc aConstructor)
|
|
{
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
// We want a block formatting context root in paginated contexts for
|
|
// every block that would be scrollable in a non-paginated context.
|
|
// We mark our blocks with a bit here if this condition is true, so
|
|
// we can check it later in nsFrame::ApplyPaginatedOverflowClipping.
|
|
bool clipPaginatedOverflow =
|
|
(aItem.mFCData->mBits & FCDATA_FORCED_NON_SCROLLABLE_BLOCK) != 0;
|
|
nsFrameState flags = nsFrameState(0);
|
|
if ((aDisplay->IsAbsolutelyPositionedStyle() ||
|
|
aDisplay->IsFloatingStyle() ||
|
|
StyleDisplay::InlineBlock == aDisplay->mDisplay ||
|
|
clipPaginatedOverflow) &&
|
|
!nsSVGUtils::IsInSVGTextSubtree(aParentFrame)) {
|
|
flags = NS_BLOCK_FORMATTING_CONTEXT_STATE_BITS;
|
|
if (clipPaginatedOverflow) {
|
|
flags |= NS_BLOCK_CLIP_PAGINATED_OVERFLOW;
|
|
}
|
|
}
|
|
|
|
nsContainerFrame* newFrame = aConstructor(mPresShell, styleContext);
|
|
newFrame->AddStateBits(flags);
|
|
ConstructBlock(aState, aItem.mContent,
|
|
aState.GetGeometricParent(aDisplay, aParentFrame),
|
|
aParentFrame, styleContext, &newFrame,
|
|
aFrameItems,
|
|
aDisplay->IsAbsPosContainingBlock(newFrame) ? newFrame : nullptr,
|
|
aItem.mPendingBinding);
|
|
return newFrame;
|
|
}
|
|
|
|
|
|
void
|
|
nsCSSFrameConstructor::InitAndRestoreFrame(const nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIFrame* aNewFrame,
|
|
bool aAllowCounters)
|
|
{
|
|
NS_PRECONDITION(mUpdateCount != 0,
|
|
"Should be in an update while creating frames");
|
|
|
|
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 &&
|
|
mCounterManager.AddCounterResetsAndIncrements(aNewFrame)) {
|
|
CountersDirty();
|
|
}
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::ResolveStyleContext(nsIFrame* aParentFrame,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
nsFrameConstructorState* aState)
|
|
{
|
|
MOZ_ASSERT(aContainer, "Must have parent here");
|
|
// XXX uncomment when bug 1089223 is fixed:
|
|
// MOZ_ASSERT(aContainer == aChild->GetFlattenedTreeParent());
|
|
nsStyleContext* parentStyleContext = GetDisplayContentsStyleFor(aContainer);
|
|
if (MOZ_LIKELY(!parentStyleContext)) {
|
|
aParentFrame = nsFrame::CorrectStyleParentFrame(aParentFrame, nullptr);
|
|
if (aParentFrame) {
|
|
MOZ_ASSERT(aParentFrame->GetContent() == aContainer);
|
|
// Resolve the style context based on the content object and the parent
|
|
// style context
|
|
parentStyleContext = aParentFrame->StyleContext();
|
|
} else {
|
|
// Perhaps aParentFrame is a canvasFrame and we're replicating
|
|
// fixed-pos frames.
|
|
// XXX should we create a way to tell ConstructFrame which style
|
|
// context to use, and pass it the style context for the
|
|
// previous page's fixed-pos frame?
|
|
}
|
|
}
|
|
|
|
return ResolveStyleContext(parentStyleContext, aChild, aState);
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::ResolveStyleContext(nsIFrame* aParentFrame,
|
|
nsIContent* aChild,
|
|
nsFrameConstructorState* aState)
|
|
{
|
|
return ResolveStyleContext(aParentFrame, aChild->GetFlattenedTreeParent(), aChild, aState);
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::ResolveStyleContext(const InsertionPoint& aInsertion,
|
|
nsIContent* aChild,
|
|
nsFrameConstructorState* aState)
|
|
{
|
|
return ResolveStyleContext(aInsertion.mParentFrame, aInsertion.mContainer,
|
|
aChild, aState);
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::ResolveStyleContext(nsStyleContext* aParentStyleContext,
|
|
nsIContent* aContent,
|
|
nsFrameConstructorState* aState,
|
|
Element* aOriginatingElementOrNull)
|
|
{
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
aContent->OwnerDoc()->FlushPendingLinkUpdates();
|
|
|
|
RefPtr<nsStyleContext> result;
|
|
if (aContent->IsElement()) {
|
|
auto pseudoType = aContent->AsElement()->GetPseudoElementType();
|
|
if (pseudoType == CSSPseudoElementType::NotPseudo) {
|
|
MOZ_ASSERT(!aOriginatingElementOrNull);
|
|
if (aState) {
|
|
result = styleSet->ResolveStyleFor(aContent->AsElement(),
|
|
aParentStyleContext,
|
|
LazyComputeBehavior::Assert,
|
|
aState->mTreeMatchContext);
|
|
} else {
|
|
result = styleSet->ResolveStyleFor(aContent->AsElement(),
|
|
aParentStyleContext,
|
|
LazyComputeBehavior::Assert);
|
|
}
|
|
} else {
|
|
MOZ_ASSERT(aContent->IsInNativeAnonymousSubtree());
|
|
if (!aOriginatingElementOrNull) {
|
|
// For pseudo-implementing NAC created by JS using the ChromeOnly
|
|
// document.createElement(..., { pseudo: ... }) API, we find the
|
|
// originating element by lookup the tree until we find a non-NAC
|
|
// ancestor. (These are the correct semantics for C++-generated pseudo-
|
|
// implementing NAC as well, but for those cases we already have a
|
|
// correct originating element passed in.)
|
|
MOZ_ASSERT(nsCSSPseudoElements::PseudoElementIsJSCreatedNAC(pseudoType));
|
|
aOriginatingElementOrNull =
|
|
nsContentUtils::GetClosestNonNativeAnonymousAncestor(aContent->AsElement());
|
|
}
|
|
MOZ_ASSERT(aOriginatingElementOrNull);
|
|
result = styleSet->ResolvePseudoElementStyle(aOriginatingElementOrNull,
|
|
pseudoType,
|
|
aParentStyleContext,
|
|
aContent->AsElement());
|
|
}
|
|
} else {
|
|
MOZ_ASSERT(!aOriginatingElementOrNull);
|
|
NS_ASSERTION(aContent->IsNodeOfType(nsINode::eTEXT),
|
|
"shouldn't waste time creating style contexts for "
|
|
"comments and processing instructions");
|
|
result = styleSet->ResolveStyleForText(aContent, aParentStyleContext);
|
|
}
|
|
|
|
// ServoRestyleManager does not handle transitions yet, and when it does
|
|
// it probably won't need to track reframed style contexts to start
|
|
// transitions correctly.
|
|
if (mozilla::GeckoRestyleManager* geckoRM = RestyleManager()->GetAsGecko()) {
|
|
GeckoRestyleManager::ReframingStyleContexts* rsc =
|
|
geckoRM->GetReframingStyleContexts();
|
|
if (rsc) {
|
|
GeckoStyleContext* oldStyleContext =
|
|
rsc->Get(aContent, CSSPseudoElementType::NotPseudo);
|
|
nsPresContext* presContext = mPresShell->GetPresContext();
|
|
if (oldStyleContext) {
|
|
RefPtr<GeckoStyleContext> newContext =
|
|
GeckoStyleContext::TakeRef(result.forget());
|
|
GeckoRestyleManager::TryInitiatingTransition(presContext, aContent,
|
|
oldStyleContext, &newContext);
|
|
result = newContext.forget();
|
|
} else if (aContent->IsElement()) {
|
|
presContext->TransitionManager()->
|
|
PruneCompletedTransitions(aContent->AsElement(),
|
|
CSSPseudoElementType::NotPseudo, result->AsGecko());
|
|
}
|
|
}
|
|
}
|
|
|
|
return result.forget();
|
|
}
|
|
|
|
// MathML Mod - RBS
|
|
void
|
|
nsCSSFrameConstructor::FlushAccumulatedBlock(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsFrameItems& aBlockItems,
|
|
nsFrameItems& aNewItems)
|
|
{
|
|
if (aBlockItems.IsEmpty()) {
|
|
// Nothing to do
|
|
return;
|
|
}
|
|
|
|
nsAtom* anonPseudo = nsCSSAnonBoxes::mozMathMLAnonymousBlock;
|
|
|
|
nsStyleContext* parentContext =
|
|
nsFrame::CorrectStyleParentFrame(aParentFrame,
|
|
anonPseudo)->StyleContext();
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
RefPtr<nsStyleContext> blockContext;
|
|
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, aBlockItems, false);
|
|
// We have to walk over aBlockItems before we hand it over to blockFrame.
|
|
for (nsIFrame* f : aBlockItems) {
|
|
f->SetParentIsWrapperAnonBox();
|
|
}
|
|
// abs-pos and floats are disabled in MathML children so we don't have to
|
|
// worry about messing up those.
|
|
blockFrame->SetInitialChildList(kPrincipalList, aBlockItems);
|
|
NS_ASSERTION(aBlockItems.IsEmpty(), "What happened?");
|
|
aBlockItems.Clear();
|
|
aNewItems.AddChild(blockFrame);
|
|
}
|
|
|
|
// Only <math> elements can be floated or positioned. All other MathML
|
|
// should be in-flow.
|
|
#define SIMPLE_MATHML_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, \
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_FORCE_NULL_ABSPOS_CONTAINER | \
|
|
FCDATA_WRAP_KIDS_IN_BLOCKS, _func) }
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindMathMLData(Element* aElement,
|
|
nsAtom* aTag,
|
|
int32_t aNameSpaceID,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
// Make sure that we remain confined in the MathML world
|
|
if (aNameSpaceID != kNameSpaceID_MathML)
|
|
return nullptr;
|
|
|
|
// Handle <math> specially, because it sometimes produces inlines
|
|
if (aTag == nsGkAtoms::math) {
|
|
// This needs to match the test in EnsureBlockDisplay in
|
|
// nsRuleNode.cpp. Though the behavior here for the display:table
|
|
// case is pretty weird...
|
|
if (aStyleContext->StyleDisplay()->IsBlockOutsideStyle()) {
|
|
static const FrameConstructionData sBlockMathData =
|
|
FCDATA_DECL(FCDATA_FORCE_NULL_ABSPOS_CONTAINER |
|
|
FCDATA_WRAP_KIDS_IN_BLOCKS,
|
|
NS_NewMathMLmathBlockFrame);
|
|
return &sBlockMathData;
|
|
}
|
|
|
|
static const FrameConstructionData sInlineMathData =
|
|
FCDATA_DECL(FCDATA_FORCE_NULL_ABSPOS_CONTAINER |
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_WRAP_KIDS_IN_BLOCKS,
|
|
NS_NewMathMLmathInlineFrame);
|
|
return &sInlineMathData;
|
|
}
|
|
|
|
|
|
static const 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_NewMathMLmspaceFrame),
|
|
SIMPLE_MATHML_CREATE(mprescripts_, NS_NewMathMLmspaceFrame),
|
|
SIMPLE_MATHML_CREATE(mfenced_, NS_NewMathMLmfencedFrame),
|
|
SIMPLE_MATHML_CREATE(mmultiscripts_, NS_NewMathMLmmultiscriptsFrame),
|
|
SIMPLE_MATHML_CREATE(mstyle_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(msqrt_, NS_NewMathMLmsqrtFrame),
|
|
SIMPLE_MATHML_CREATE(mroot_, NS_NewMathMLmrootFrame),
|
|
SIMPLE_MATHML_CREATE(maction_, NS_NewMathMLmactionFrame),
|
|
SIMPLE_MATHML_CREATE(mrow_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(merror_, NS_NewMathMLmrowFrame),
|
|
SIMPLE_MATHML_CREATE(menclose_, NS_NewMathMLmencloseFrame),
|
|
SIMPLE_MATHML_CREATE(semantics_, NS_NewMathMLsemanticsFrame)
|
|
};
|
|
|
|
return FindDataByTag(aTag, aElement, aStyleContext, sMathMLData,
|
|
ArrayLength(sMathMLData));
|
|
}
|
|
|
|
|
|
nsContainerFrame*
|
|
nsCSSFrameConstructor::ConstructFrameWithAnonymousChild(
|
|
nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
nsFrameItems& aFrameItems,
|
|
ContainerFrameCreationFunc aConstructor,
|
|
ContainerFrameCreationFunc aInnerConstructor,
|
|
nsICSSAnonBoxPseudo* aInnerPseudo,
|
|
bool aCandidateRootFrame)
|
|
{
|
|
nsIContent* const content = aItem.mContent;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
// Create the outer frame:
|
|
nsContainerFrame* newFrame = aConstructor(mPresShell, styleContext);
|
|
|
|
InitAndRestoreFrame(aState, content,
|
|
aCandidateRootFrame ?
|
|
aState.GetGeometricParent(styleContext->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<nsStyleContext> scForAnon;
|
|
scForAnon = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(aInnerPseudo, styleContext);
|
|
|
|
// 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, aFrameItems, content, styleContext, aParentFrame,
|
|
aCandidateRootFrame, aCandidateRootFrame);
|
|
|
|
if (!mRootElementFrame && aCandidateRootFrame) {
|
|
// The frame we're constructing will be the root element frame.
|
|
// Set mRootElementFrame before processing children.
|
|
mRootElementFrame = newFrame;
|
|
}
|
|
|
|
nsFrameItems childItems;
|
|
|
|
// Process children
|
|
NS_ASSERTION(aItem.mAnonChildren.IsEmpty(),
|
|
"nsIAnonymousContentCreator::CreateAnonymousContent should not "
|
|
"be implemented for frames for which we explicitly create an "
|
|
"anonymous child to wrap its child frames");
|
|
if (aItem.mFCData->mBits & FCDATA_USE_CHILD_ITEMS) {
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems,
|
|
innerFrame,
|
|
aItem.mFCData->mBits & FCDATA_IS_WRAPPER_ANON_BOX,
|
|
childItems);
|
|
} else {
|
|
ProcessChildren(aState, content, styleContext, innerFrame,
|
|
true, childItems, false, aItem.mPendingBinding);
|
|
}
|
|
|
|
// Set the inner wrapper frame's initial primary list
|
|
innerFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructOuterSVG(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
return ConstructFrameWithAnonymousChild(
|
|
aState, aItem, aParentFrame, aFrameItems,
|
|
NS_NewSVGOuterSVGFrame, NS_NewSVGOuterSVGAnonChildFrame,
|
|
nsCSSAnonBoxes::mozSVGOuterSVGAnonChild, true);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructMarker(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
return ConstructFrameWithAnonymousChild(
|
|
aState, aItem, aParentFrame, aFrameItems,
|
|
NS_NewSVGMarkerFrame, NS_NewSVGMarkerAnonChildFrame,
|
|
nsCSSAnonBoxes::mozSVGMarkerAnonChild, false);
|
|
}
|
|
|
|
// Only outer <svg> elements can be floated or positioned. All other SVG
|
|
// should be in-flow.
|
|
#define SIMPLE_SVG_FCDATA(_func) \
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW | \
|
|
FCDATA_SKIP_ABSPOS_PUSH | \
|
|
FCDATA_DISALLOW_GENERATED_CONTENT, _func)
|
|
#define SIMPLE_SVG_CREATE(_tag, _func) \
|
|
{ &nsGkAtoms::_tag, SIMPLE_SVG_FCDATA(_func) }
|
|
|
|
static bool
|
|
IsFilterPrimitiveChildTag(const nsAtom* aTag)
|
|
{
|
|
return aTag == nsGkAtoms::feDistantLight ||
|
|
aTag == nsGkAtoms::fePointLight ||
|
|
aTag == nsGkAtoms::feSpotLight ||
|
|
aTag == nsGkAtoms::feFuncR ||
|
|
aTag == nsGkAtoms::feFuncG ||
|
|
aTag == nsGkAtoms::feFuncB ||
|
|
aTag == nsGkAtoms::feFuncA ||
|
|
aTag == nsGkAtoms::feMergeNode;
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::FrameConstructionData*
|
|
nsCSSFrameConstructor::FindSVGData(Element* aElement,
|
|
nsAtom* aTag,
|
|
int32_t aNameSpaceID,
|
|
nsIFrame* aParentFrame,
|
|
bool aIsWithinSVGText,
|
|
bool aAllowsTextPathChild,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aNameSpaceID != kNameSpaceID_SVG) {
|
|
return nullptr;
|
|
}
|
|
|
|
static const FrameConstructionData sSuppressData = SUPPRESS_FCDATA();
|
|
static const FrameConstructionData sContainerData =
|
|
SIMPLE_SVG_FCDATA(NS_NewSVGContainerFrame);
|
|
|
|
bool parentIsSVG = aIsWithinSVGText;
|
|
nsIContent* parentContent =
|
|
aParentFrame ? aParentFrame->GetContent() : nullptr;
|
|
// XXXbz should this really be based on the XBL-resolved 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) {
|
|
int32_t parentNSID;
|
|
nsAtom* parentTag =
|
|
parentContent->OwnerDoc()->BindingManager()->
|
|
ResolveTag(parentContent, &parentNSID);
|
|
|
|
// 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 = parentNSID == kNameSpaceID_SVG &&
|
|
parentTag != nsGkAtoms::foreignObject;
|
|
}
|
|
|
|
if ((aTag != nsGkAtoms::svg && !parentIsSVG) ||
|
|
(aTag == nsGkAtoms::desc || aTag == nsGkAtoms::title ||
|
|
aTag == 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->IsNodeOfType(nsINode::eANIMATION)) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
if (aTag == nsGkAtoms::svg && !parentIsSVG) {
|
|
// We need outer <svg> elements to have an nsSVGOuterSVGFrame 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
|
|
// nsSVGOuterSVGFrame::Init.
|
|
static const FrameConstructionData sOuterSVGData =
|
|
FULL_CTOR_FCDATA(0, &nsCSSFrameConstructor::ConstructOuterSVG);
|
|
return &sOuterSVGData;
|
|
}
|
|
|
|
if (aTag == nsGkAtoms::marker) {
|
|
static const FrameConstructionData sMarkerSVGData =
|
|
FULL_CTOR_FCDATA(0, &nsCSSFrameConstructor::ConstructMarker);
|
|
return &sMarkerSVGData;
|
|
}
|
|
|
|
nsCOMPtr<SVGTests> tests(do_QueryInterface(aElement));
|
|
if (tests && !tests->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 nsSVGSwitchFrame::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 nsSVGContainerFrame (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 && (aParentFrame->IsSVGLinearGradientFrame() ||
|
|
aParentFrame->IsSVGRadialGradientFrame());
|
|
bool stop = (aTag == 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();
|
|
bool filterPrimitive = aElement->IsNodeOfType(nsINode::eFILTER);
|
|
if ((parentIsFilter && !filterPrimitive) ||
|
|
(!parentIsFilter && filterPrimitive)) {
|
|
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 && !IsFilterPrimitiveChildTag(aTag)) ||
|
|
(!parentIsFEContainerFrame && IsFilterPrimitiveChildTag(aTag))) {
|
|
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.
|
|
|
|
// We don't use ConstructInline because we want different behavior
|
|
// for generated content.
|
|
static const FrameConstructionData sTSpanData =
|
|
FCDATA_DECL(FCDATA_DISALLOW_OUT_OF_FLOW |
|
|
FCDATA_SKIP_ABSPOS_PUSH |
|
|
FCDATA_DISALLOW_GENERATED_CONTENT |
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_INLINE |
|
|
FCDATA_USE_CHILD_ITEMS,
|
|
NS_NewInlineFrame);
|
|
if (aTag == nsGkAtoms::textPath) {
|
|
if (aAllowsTextPathChild) {
|
|
return &sTSpanData;
|
|
}
|
|
} else if (aTag == nsGkAtoms::tspan ||
|
|
aTag == nsGkAtoms::a) {
|
|
return &sTSpanData;
|
|
}
|
|
return &sSuppressData;
|
|
} else if (aTag == nsGkAtoms::text) {
|
|
static const FrameConstructionData sTextData =
|
|
FCDATA_WITH_WRAPPING_BLOCK(FCDATA_DISALLOW_OUT_OF_FLOW |
|
|
FCDATA_ALLOW_BLOCK_STYLES,
|
|
NS_NewSVGTextFrame,
|
|
nsCSSAnonBoxes::mozSVGText);
|
|
return &sTextData;
|
|
} else if (aTag == nsGkAtoms::tspan ||
|
|
aTag == nsGkAtoms::textPath) {
|
|
return &sSuppressData;
|
|
}
|
|
|
|
static const 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),
|
|
SIMPLE_SVG_CREATE(generic_, NS_NewSVGGenericContainerFrame),
|
|
{ &nsGkAtoms::foreignObject,
|
|
FCDATA_WITH_WRAPPING_BLOCK(FCDATA_DISALLOW_OUT_OF_FLOW,
|
|
NS_NewSVGForeignObjectFrame,
|
|
nsCSSAnonBoxes::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(aTag, aElement, aStyleContext, sSVGData,
|
|
ArrayLength(sSVGData));
|
|
|
|
if (!data) {
|
|
data = &sContainerData;
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::AddPageBreakItem(nsIContent* aContent,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
RefPtr<nsStyleContext> pseudoStyle =
|
|
mPresShell->StyleSet()->
|
|
ResolveNonInheritingAnonymousBoxStyle(nsCSSAnonBoxes::pageBreak);
|
|
|
|
MOZ_ASSERT(pseudoStyle->StyleDisplay()->mDisplay == StyleDisplay::Block,
|
|
"Unexpected display");
|
|
|
|
static const FrameConstructionData sPageBreakData =
|
|
FCDATA_DECL(FCDATA_SKIP_FRAMESET, NS_NewPageBreakFrame);
|
|
|
|
// Lie about the tag and namespace so we don't trigger anything
|
|
// interesting during frame construction.
|
|
aItems.AppendItem(this, &sPageBreakData, aContent, nsCSSAnonBoxes::pageBreak,
|
|
kNameSpaceID_None, nullptr, pseudoStyle.forget(),
|
|
true, nullptr);
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::ShouldCreateItemsForChild(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame)
|
|
{
|
|
aContent->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME);
|
|
if (aContent->IsElement() && !aContent->IsStyledByServo()) {
|
|
// We can't just remove our pending restyle flags, since we may
|
|
// have restyle-later-siblings set on us. But we _can_ remove the
|
|
// "is possible restyle root" flags, and need to. Otherwise we can
|
|
// end up with stale such flags (e.g. if we used to have a
|
|
// display:none parent when our last restyle was posted and
|
|
// processed and now no longer do).
|
|
aContent->UnsetFlags(ELEMENT_ALL_RESTYLE_FLAGS &
|
|
~ELEMENT_PENDING_RESTYLE_FLAGS);
|
|
}
|
|
|
|
// 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) {
|
|
NS_ERROR("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->IsNodeOfType(nsINode::eCOMMENT) ||
|
|
aContent->IsNodeOfType(nsINode::ePROCESSING_INSTRUCTION)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::DoAddFrameConstructionItems(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
bool aSuppressWhiteSpaceOptimizations,
|
|
nsContainerFrame* aParentFrame,
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>* aAnonChildren,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
uint32_t flags = ITEM_ALLOW_XBL_BASE | ITEM_ALLOW_PAGE_BREAK;
|
|
if (aParentFrame) {
|
|
if (nsSVGUtils::IsInSVGTextSubtree(aParentFrame)) {
|
|
flags |= ITEM_IS_WITHIN_SVG_TEXT;
|
|
}
|
|
if (aParentFrame->IsBlockFrame() && aParentFrame->GetParent() &&
|
|
aParentFrame->GetParent()->IsSVGTextFrame()) {
|
|
flags |= ITEM_ALLOWS_TEXT_PATH_CHILD;
|
|
}
|
|
}
|
|
AddFrameConstructionItemsInternal(aState, aContent, aParentFrame,
|
|
aContent->NodeInfo()->NameAtom(),
|
|
aContent->GetNameSpaceID(),
|
|
aSuppressWhiteSpaceOptimizations,
|
|
aStyleContext,
|
|
flags, aAnonChildren,
|
|
aItems);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::AddFrameConstructionItems(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
bool aSuppressWhiteSpaceOptimizations,
|
|
const InsertionPoint& aInsertion,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
nsContainerFrame* parentFrame = aInsertion.mParentFrame;
|
|
if (!ShouldCreateItemsForChild(aState, aContent, parentFrame)) {
|
|
return;
|
|
}
|
|
RefPtr<nsStyleContext> styleContext =
|
|
ResolveStyleContext(aInsertion, aContent, &aState);
|
|
DoAddFrameConstructionItems(aState, aContent, styleContext,
|
|
aSuppressWhiteSpaceOptimizations, parentFrame,
|
|
nullptr, aItems);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::SetAsUndisplayedContent(nsFrameConstructorState& aState,
|
|
FrameConstructionItemList& aList,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
bool aIsGeneratedContent)
|
|
{
|
|
if (aStyleContext->GetPseudo()) {
|
|
if (aIsGeneratedContent) {
|
|
aContent->UnbindFromTree();
|
|
}
|
|
return;
|
|
}
|
|
NS_ASSERTION(!aIsGeneratedContent, "Should have had pseudo type");
|
|
|
|
if (aState.mCreatingExtraFrames) {
|
|
MOZ_ASSERT(GetDisplayNoneStyleFor(aContent),
|
|
"should have called RegisterDisplayNoneStyleFor earlier");
|
|
return;
|
|
}
|
|
aList.AppendUndisplayedItem(aContent, aStyleContext);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::AddFrameConstructionItemsInternal(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsAtom* aTag,
|
|
int32_t aNameSpaceID,
|
|
bool aSuppressWhiteSpaceOptimizations,
|
|
nsStyleContext* aStyleContext,
|
|
uint32_t aFlags,
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>* aAnonChildren,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
NS_PRECONDITION(aContent->IsNodeOfType(nsINode::eTEXT) ||
|
|
aContent->IsElement(),
|
|
"Shouldn't get anything else here!");
|
|
MOZ_ASSERT(aContent->IsInComposedDoc());
|
|
MOZ_ASSERT(!aContent->GetPrimaryFrame() || aState.mCreatingExtraFrames ||
|
|
aContent->NodeInfo()->NameAtom() == nsGkAtoms::area);
|
|
|
|
// The following code allows the user to specify the base tag
|
|
// of an element using XBL. XUL and HTML objects (like boxes, menus, etc.)
|
|
// can then be extended arbitrarily.
|
|
const nsStyleDisplay* display = aStyleContext->StyleDisplay();
|
|
RefPtr<nsStyleContext> styleContext(aStyleContext);
|
|
PendingBinding* pendingBinding = nullptr;
|
|
if (aFlags & ITEM_ALLOW_XBL_BASE) {
|
|
if (display->mBinding) {
|
|
// Ensure that our XBL bindings are installed.
|
|
|
|
nsXBLService* xblService = nsXBLService::GetInstance();
|
|
if (!xblService)
|
|
return;
|
|
|
|
bool resolveStyle;
|
|
|
|
nsAutoPtr<PendingBinding> newPendingBinding(new PendingBinding());
|
|
|
|
nsresult rv = xblService->LoadBindings(
|
|
aContent, display->mBinding->GetURI(),
|
|
display->mBinding->mExtraData->GetPrincipal(),
|
|
getter_AddRefs(newPendingBinding->mBinding), &resolveStyle);
|
|
if (NS_FAILED(rv) && rv != NS_ERROR_XBL_BLOCKED)
|
|
return;
|
|
|
|
if (newPendingBinding->mBinding) {
|
|
pendingBinding = newPendingBinding;
|
|
// aState takes over owning newPendingBinding
|
|
aState.AddPendingBinding(newPendingBinding.forget());
|
|
}
|
|
|
|
if (resolveStyle) {
|
|
if (styleContext->IsServo()) {
|
|
Element* element = aContent->AsElement();
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet()->AsServo();
|
|
|
|
// XXX: We should have a better way to restyle ourselves.
|
|
ServoRestyleManager::ClearServoDataFromSubtree(element);
|
|
styleSet->StyleNewSubtree(element);
|
|
|
|
// Servo's should_traverse_children() in traversal.rs skips
|
|
// styling descendants of elements with a -moz-binding the
|
|
// first time. Thus call StyleNewChildren() again.
|
|
styleSet->StyleNewChildren(element);
|
|
|
|
// Because of LazyComputeBehavior::Assert we never create a style
|
|
// context here, so it's fine to pass a null parent.
|
|
styleContext =
|
|
styleSet->ResolveStyleFor(element, nullptr,
|
|
LazyComputeBehavior::Assert);
|
|
} else {
|
|
styleContext =
|
|
ResolveStyleContext(styleContext->AsGecko()->GetParent(),
|
|
aContent, &aState);
|
|
}
|
|
|
|
display = styleContext->StyleDisplay();
|
|
aStyleContext = styleContext;
|
|
}
|
|
|
|
aTag = mDocument->BindingManager()->ResolveTag(aContent, &aNameSpaceID);
|
|
} else if (display->mBinding.ForceGet()) {
|
|
if (aContent->IsStyledByServo()) {
|
|
// Servo's should_traverse_children skips styling descendants of
|
|
// elements with a -moz-binding value. For -moz-binding URLs that can
|
|
// be resolved, we will load the binding above, which will style the
|
|
// children after they have been rearranged in the flattened tree.
|
|
// If the URL couldn't be resolved, we still need to style the children,
|
|
// so we do that here.
|
|
mPresShell->StyleSet()->AsServo()->StyleNewChildren(aContent->AsElement());
|
|
}
|
|
}
|
|
}
|
|
|
|
const bool isGeneratedContent = !!(aFlags & ITEM_IS_GENERATED_CONTENT);
|
|
|
|
// Pre-check for display "none" - if we find that, don't create
|
|
// any frame at all
|
|
if (StyleDisplay::None == display->mDisplay) {
|
|
SetAsUndisplayedContent(aState, aItems, aContent, styleContext, isGeneratedContent);
|
|
return;
|
|
}
|
|
|
|
bool isText = !aContent->IsElement();
|
|
|
|
// never create frames for non-option/optgroup kids of <select> and
|
|
// non-option kids of <optgroup> inside a <select>.
|
|
// XXXbz it's not clear how this should best work with XBL.
|
|
nsIContent *parent = aContent->GetParent();
|
|
if (parent) {
|
|
// Check tag first, since that check will usually fail
|
|
if (parent->IsAnyOfHTMLElements(nsGkAtoms::select, nsGkAtoms::optgroup) &&
|
|
// <option> is ok no matter what
|
|
!aContent->IsHTMLElement(nsGkAtoms::option) &&
|
|
// <optgroup> is OK in <select> but not in <optgroup>
|
|
(!aContent->IsHTMLElement(nsGkAtoms::optgroup) ||
|
|
!parent->IsHTMLElement(nsGkAtoms::select)) &&
|
|
// Allow native anonymous content no matter what
|
|
!aContent->IsRootOfNativeAnonymousSubtree()) {
|
|
// No frame for aContent
|
|
if (!isText) {
|
|
SetAsUndisplayedContent(aState, aItems, aContent, styleContext,
|
|
isGeneratedContent);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
// When constructing a child of a non-open <details>, create only the frame
|
|
// for the main <summary> element, and skip other elements. This only applies
|
|
// to things that are not roots of native anonymous subtrees (except for
|
|
// ::before and ::after); we always want to create "internal" anonymous
|
|
// content.
|
|
auto* details = HTMLDetailsElement::FromContentOrNull(parent);
|
|
if (details && !details->Open() &&
|
|
(!aContent->IsRootOfNativeAnonymousSubtree() ||
|
|
aContent->IsGeneratedContentContainerForBefore() ||
|
|
aContent->IsGeneratedContentContainerForAfter())) {
|
|
auto* summary = HTMLSummaryElement::FromContentOrNull(aContent);
|
|
if (!summary || !summary->IsMainSummary()) {
|
|
SetAsUndisplayedContent(aState, aItems, aContent, styleContext,
|
|
isGeneratedContent);
|
|
return;
|
|
}
|
|
}
|
|
|
|
bool isPopup = false;
|
|
// Try to find frame construction data for this content
|
|
const FrameConstructionData* data;
|
|
if (isText) {
|
|
data = FindTextData(aParentFrame);
|
|
if (!data) {
|
|
// Nothing to do here; suppressed text inside SVG
|
|
return;
|
|
}
|
|
} else {
|
|
Element* element = aContent->AsElement();
|
|
|
|
// Don't create frames for non-SVG element children of SVG elements.
|
|
if (aNameSpaceID != kNameSpaceID_SVG &&
|
|
((aParentFrame &&
|
|
IsFrameForSVG(aParentFrame) &&
|
|
!aParentFrame->IsFrameOfType(nsIFrame::eSVGForeignObject)) ||
|
|
(aFlags & ITEM_IS_WITHIN_SVG_TEXT))) {
|
|
SetAsUndisplayedContent(aState, aItems, element, styleContext,
|
|
isGeneratedContent);
|
|
return;
|
|
}
|
|
|
|
data = FindHTMLData(element, aTag, aNameSpaceID, aParentFrame,
|
|
styleContext);
|
|
if (!data) {
|
|
data = FindXULTagData(element, aTag, aNameSpaceID, styleContext);
|
|
}
|
|
if (!data) {
|
|
data = FindMathMLData(element, aTag, aNameSpaceID, styleContext);
|
|
}
|
|
if (!data) {
|
|
data = FindSVGData(element, aTag, aNameSpaceID, aParentFrame,
|
|
aFlags & ITEM_IS_WITHIN_SVG_TEXT,
|
|
aFlags & ITEM_ALLOWS_TEXT_PATH_CHILD,
|
|
styleContext);
|
|
}
|
|
|
|
// Now check for XUL display types
|
|
if (!data) {
|
|
data = FindXULDisplayData(display, element, styleContext);
|
|
}
|
|
|
|
// And general display types
|
|
if (!data) {
|
|
data = FindDisplayData(display, element, styleContext);
|
|
}
|
|
|
|
NS_ASSERTION(data, "Should have frame construction data now");
|
|
|
|
if (data->mBits & FCDATA_SUPPRESS_FRAME) {
|
|
SetAsUndisplayedContent(aState, aItems, element, styleContext, isGeneratedContent);
|
|
return;
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
if ((data->mBits & FCDATA_IS_POPUP) &&
|
|
(!aParentFrame || // Parent is inline
|
|
!aParentFrame->IsMenuFrame())) {
|
|
if (!aState.mPopupItems.containingBlock &&
|
|
!aState.mHavePendingPopupgroup) {
|
|
SetAsUndisplayedContent(aState, aItems, element, styleContext,
|
|
isGeneratedContent);
|
|
return;
|
|
}
|
|
|
|
isPopup = true;
|
|
}
|
|
#endif /* MOZ_XUL */
|
|
}
|
|
|
|
uint32_t bits = data->mBits;
|
|
|
|
// Inside colgroups, suppress everything except columns.
|
|
if (aParentFrame && aParentFrame->IsTableColGroupFrame() &&
|
|
(!(bits & FCDATA_IS_TABLE_PART) ||
|
|
display->mDisplay != StyleDisplay::TableColumn)) {
|
|
SetAsUndisplayedContent(aState, aItems, aContent, styleContext, isGeneratedContent);
|
|
return;
|
|
}
|
|
|
|
bool canHavePageBreak =
|
|
(aFlags & ITEM_ALLOW_PAGE_BREAK) && aState.mPresContext->IsPaginated() &&
|
|
!display->IsAbsolutelyPositionedStyle() &&
|
|
!(aParentFrame && aParentFrame->IsGridContainerFrame()) &&
|
|
!(bits & FCDATA_IS_TABLE_PART) && !(bits & FCDATA_IS_SVG_TEXT);
|
|
|
|
if (canHavePageBreak && display->mBreakBefore) {
|
|
AddPageBreakItem(aContent, aItems);
|
|
}
|
|
|
|
if (MOZ_UNLIKELY(bits & FCDATA_IS_CONTENTS)) {
|
|
if (!GetDisplayContentsStyleFor(aContent)) {
|
|
MOZ_ASSERT(styleContext->GetPseudo() || !isGeneratedContent,
|
|
"Should have had pseudo type");
|
|
aState.mFrameManager->RegisterDisplayContentsStyleFor(aContent,
|
|
styleContext);
|
|
} else {
|
|
aState.mFrameManager->ChangeRegisteredDisplayContentsStyleFor(aContent,
|
|
styleContext);
|
|
}
|
|
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(aState.mTreeMatchContext);
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(aContent->AsElement());
|
|
} else {
|
|
ancestorPusher.PushStyleScope(aContent->AsElement());
|
|
}
|
|
|
|
if (aParentFrame) {
|
|
aParentFrame->AddStateBits(NS_FRAME_MAY_HAVE_GENERATED_CONTENT);
|
|
}
|
|
CreateGeneratedContentItem(aState, aParentFrame, aContent, styleContext,
|
|
CSSPseudoElementType::before, aItems);
|
|
|
|
FlattenedChildIterator iter(aContent);
|
|
for (nsIContent* child = iter.GetNextChild(); child; child = iter.GetNextChild()) {
|
|
if (!ShouldCreateItemsForChild(aState, child, aParentFrame)) {
|
|
continue;
|
|
}
|
|
|
|
// Get the parent of the content and check if it is a XBL children element
|
|
// (if the content is a children element then parent != aContent because the
|
|
// FlattenedChildIterator will transitively iterate through <xbl:children>
|
|
// for default content). Push the children element as an ancestor here because
|
|
// it does not have a frame and would not otherwise be pushed as an ancestor.
|
|
nsIContent* parent = child->GetParent();
|
|
MOZ_ASSERT(parent, "Parent must be non-null because we are iterating children.");
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(aState.mTreeMatchContext);
|
|
if (parent != aContent && parent->IsElement()) {
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(parent->AsElement());
|
|
} else {
|
|
ancestorPusher.PushStyleScope(parent->AsElement());
|
|
}
|
|
}
|
|
|
|
RefPtr<nsStyleContext> childContext =
|
|
ResolveStyleContext(styleContext, child, &aState);
|
|
DoAddFrameConstructionItems(aState, child, childContext,
|
|
aSuppressWhiteSpaceOptimizations,
|
|
aParentFrame, aAnonChildren, aItems);
|
|
}
|
|
aItems.SetParentHasNoXBLChildren(!iter.XBLInvolved());
|
|
|
|
CreateGeneratedContentItem(aState, aParentFrame, aContent, styleContext,
|
|
CSSPseudoElementType::after, aItems);
|
|
if (canHavePageBreak && display->mBreakAfter) {
|
|
AddPageBreakItem(aContent, aItems);
|
|
}
|
|
return;
|
|
}
|
|
|
|
FrameConstructionItem* item = nullptr;
|
|
if (details && details->Open()) {
|
|
auto* summary = HTMLSummaryElement::FromContentOrNull(aContent);
|
|
if (summary && summary->IsMainSummary()) {
|
|
// If details is open, the main summary needs to be rendered as if it is
|
|
// the first child, so add the item to the front of the item list.
|
|
item = aItems.PrependItem(this, data, aContent, aTag, aNameSpaceID,
|
|
pendingBinding, styleContext.forget(),
|
|
aSuppressWhiteSpaceOptimizations, aAnonChildren);
|
|
}
|
|
}
|
|
|
|
if (!item) {
|
|
item = aItems.AppendItem(this, data, aContent, aTag, aNameSpaceID,
|
|
pendingBinding, styleContext.forget(),
|
|
aSuppressWhiteSpaceOptimizations, aAnonChildren);
|
|
}
|
|
item->mIsText = isText;
|
|
item->mIsGeneratedContent = isGeneratedContent;
|
|
item->mIsAnonymousContentCreatorContent =
|
|
aFlags & ITEM_IS_ANONYMOUSCONTENTCREATOR_CONTENT;
|
|
if (isGeneratedContent) {
|
|
// We need to keep this alive until the frame takes ownership.
|
|
NS_ADDREF(item->mContent);
|
|
}
|
|
item->mIsRootPopupgroup =
|
|
aNameSpaceID == kNameSpaceID_XUL && aTag == nsGkAtoms::popupgroup &&
|
|
aContent->IsRootOfNativeAnonymousSubtree();
|
|
if (item->mIsRootPopupgroup) {
|
|
aState.mHavePendingPopupgroup = true;
|
|
}
|
|
item->mIsPopup = isPopup;
|
|
item->mIsForSVGAElement = aNameSpaceID == kNameSpaceID_SVG &&
|
|
aTag == nsGkAtoms::a;
|
|
|
|
if (canHavePageBreak && display->mBreakAfter) {
|
|
AddPageBreakItem(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 & ITEM_IS_WITHIN_SVG_TEXT,
|
|
aFlags & ITEM_ALLOWS_TEXT_PATH_CHILD);
|
|
item->mHasInlineEnds = true;
|
|
item->mIsBlock = false;
|
|
} else {
|
|
// Compute a boolean isInline which is guaranteed to be false for blocks
|
|
// (but may also be false for some inlines).
|
|
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()->mDisplay == StyleDisplay::Inline)) ||
|
|
// 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 = item->mHasInlineEnds = 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. And probably remove mIsBlock
|
|
// altogether, since then it will always be equal to !mHasInlineEnds.
|
|
(!(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();
|
|
}
|
|
|
|
// Our item should be treated as a line participant if we have the relevant
|
|
// bit and are going to be in-flow. Note that this really only matters if
|
|
// our ancestor is a box or some such, so the fact that we might have an
|
|
// inline ancestor that might become a containing block is not relevant here.
|
|
if ((bits & FCDATA_IS_LINE_PARTICIPANT) &&
|
|
((bits & FCDATA_DISALLOW_OUT_OF_FLOW) ||
|
|
!aState.GetGeometricParent(display, nullptr))) {
|
|
item->mIsLineParticipant = true;
|
|
aItems.LineParticipantItemAdded();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
nsContainerFrame* adjParentFrame = aParentFrame;
|
|
FrameConstructionItem& item = aIter.item();
|
|
nsStyleContext* styleContext = item.mStyleContext;
|
|
AdjustParentFrame(&adjParentFrame, item.mFCData, styleContext);
|
|
|
|
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 XBL anonymous siblings,
|
|
// 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) &&
|
|
!styleContext->StyleText()->WhiteSpaceOrNewlineIsSignificant() &&
|
|
aIter.List()->ParentHasNoXBLChildren() &&
|
|
!(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,
|
|
adjParentFrame, styleContext,
|
|
aFrameItems);
|
|
return;
|
|
}
|
|
|
|
// Start background loads during frame construction so that we're
|
|
// guaranteed that they will be started before onload fires.
|
|
styleContext->StartBackgroundImageLoads();
|
|
|
|
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, adjParentFrame, aFrameItems);
|
|
|
|
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;
|
|
}
|
|
|
|
aState.mAdditionalStateBits = savedStateBits;
|
|
}
|
|
|
|
|
|
inline bool
|
|
IsRootBoxFrame(nsIFrame *aFrame)
|
|
{
|
|
return (aFrame->IsRootFrame());
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ReconstructDocElementHierarchy(InsertionKind aInsertionKind)
|
|
{
|
|
Element* rootElement = mDocument->GetRootElement();
|
|
if (!rootElement) {
|
|
/* nothing to do */
|
|
return;
|
|
}
|
|
RecreateFramesForContent(rootElement, aInsertionKind);
|
|
}
|
|
|
|
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->IsFrameOfType(nsIFrame::eMathML)) {
|
|
// 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() ||
|
|
(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::Scroll) {
|
|
nsIScrollableFrame* scrollFrame = do_QueryFrame(absPosCBCandidate);
|
|
absPosCBCandidate = scrollFrame->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 mHasRootAbsPosContainingBlock ? mDocElementContainingBlock : nullptr;
|
|
}
|
|
|
|
nsContainerFrame*
|
|
nsCSSFrameConstructor::GetFloatContainingBlock(nsIFrame* aFrame)
|
|
{
|
|
// Starting with aFrame, look for a frame that is a float containing block.
|
|
// IF we hit a mathml frame, bail out; we don't allow floating out of mathml
|
|
// frames, because they don't seem to be able to deal.
|
|
// The logic here needs to match the logic in ProcessChildren()
|
|
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 check whether aContainer has :after generated content.
|
|
* If so, appending to it should actually insert. The return value is the
|
|
* parent to use for newly-appended content. *aAfterFrame points to the :after
|
|
* frame before which appended content should go, if there is one.
|
|
*/
|
|
static nsContainerFrame*
|
|
AdjustAppendParentForAfterContent(nsFrameManager* aFrameManager,
|
|
nsIContent* aContainer,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIContent* aChild,
|
|
nsIFrame** aAfterFrame)
|
|
{
|
|
// If the parent frame may have an ::after pseudo or aContainer is a
|
|
// display:contents node or aContainer have display:contents children
|
|
// then we need to walk through the child frames to find the first one
|
|
// that is either a ::after frame for an ancestor of aChild or a frame
|
|
// that is for a node later in the document than aChild and return that
|
|
// in aAfterFrame.
|
|
if (nsLayoutUtils::HasPseudoStyle(aContainer, aParentFrame->StyleContext(),
|
|
CSSPseudoElementType::after,
|
|
aParentFrame->PresContext()) ||
|
|
aFrameManager->GetDisplayContentsStyleFor(aContainer) ||
|
|
aFrameManager->GetAllRegisteredDisplayContentsStylesIn(aContainer)) {
|
|
nsIFrame* afterFrame = nullptr;
|
|
nsContainerFrame* parent =
|
|
static_cast<nsContainerFrame*>(aParentFrame->LastContinuation());
|
|
bool done = false;
|
|
while (!done && parent) {
|
|
// Ensure that all normal flow children are on the principal child list.
|
|
parent->DrainSelfOverflowList();
|
|
|
|
nsIFrame* child = parent->GetChildList(nsIFrame::kPrincipalList).LastChild();
|
|
if (child && child->IsPseudoFrame(aContainer) &&
|
|
!child->IsGeneratedContentFrame()) {
|
|
// Drill down into non-generated pseudo frames of aContainer.
|
|
nsContainerFrame* childAsContainer = do_QueryFrame(child);
|
|
if (childAsContainer) {
|
|
parent = nsLayoutUtils::LastContinuationWithChild(childAsContainer);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
for (; child; child = child->GetPrevSibling()) {
|
|
nsIContent* c = child->GetContent();
|
|
if (child->IsGeneratedContentFrame()) {
|
|
nsIContent* p = c->GetParent();
|
|
if (c->NodeInfo()->NameAtom() == nsGkAtoms::mozgeneratedcontentafter) {
|
|
if (!nsContentUtils::ContentIsDescendantOf(aChild, p) &&
|
|
p != aContainer &&
|
|
nsContentUtils::PositionIsBefore(p, aChild)) {
|
|
// ::after generated content for content earlier in the doc and not
|
|
// for an ancestor. "p != aContainer" may seem redundant but it
|
|
// checks if the ::after belongs to the XBL insertion point we're
|
|
// inserting aChild into (in which case ContentIsDescendantOf is
|
|
// false even though p == aContainer).
|
|
// See layout/reftests/bugs/482592-1a.xhtml for an example of that.
|
|
done = true;
|
|
break;
|
|
}
|
|
} else if (nsContentUtils::PositionIsBefore(p, aChild)) {
|
|
// Non-::after generated content for content earlier in the doc.
|
|
done = true;
|
|
break;
|
|
}
|
|
} else if (nsContentUtils::PositionIsBefore(c, aChild)) {
|
|
// Content is before aChild.
|
|
done = true;
|
|
break;
|
|
}
|
|
afterFrame = child;
|
|
}
|
|
|
|
parent = static_cast<nsContainerFrame*>(parent->GetPrevContinuation());
|
|
}
|
|
if (afterFrame) {
|
|
*aAfterFrame = afterFrame;
|
|
return afterFrame->GetParent();
|
|
}
|
|
}
|
|
|
|
*aAfterFrame = nullptr;
|
|
|
|
if (IsFramePartOfIBSplit(aParentFrame)) {
|
|
// We might be in a situation where the last part of the {ib} split was
|
|
// empty. Since we have no ::after pseudo-element, we do in fact want to be
|
|
// appending to that last part, so advance to it if needed. Note that here
|
|
// aParentFrame is the result of a GetLastIBSplitSibling call, so must be
|
|
// either the last or next to last ib-split sibling.
|
|
nsContainerFrame* trailingInline = GetIBSplitSibling(aParentFrame);
|
|
if (trailingInline) {
|
|
aParentFrame = trailingInline;
|
|
}
|
|
|
|
// 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.
|
|
aParentFrame =
|
|
static_cast<nsContainerFrame*>(aParentFrame->LastContinuation());
|
|
}
|
|
|
|
return aParentFrame;
|
|
}
|
|
|
|
/**
|
|
* This function will get the previous sibling to use for an append operation.
|
|
* it takes a parent frame (must not be null) and its :after frame (may be
|
|
* null).
|
|
*/
|
|
static nsIFrame*
|
|
FindAppendPrevSibling(nsIFrame* aParentFrame, nsIFrame* aAfterFrame)
|
|
{
|
|
if (aAfterFrame) {
|
|
NS_ASSERTION(aAfterFrame->GetParent() == aParentFrame, "Wrong parent");
|
|
NS_ASSERTION(aAfterFrame->GetPrevSibling() ||
|
|
aParentFrame->PrincipalChildList().FirstChild() == aAfterFrame,
|
|
":after frame must be on the principal child list here");
|
|
return aAfterFrame->GetPrevSibling();
|
|
}
|
|
|
|
aParentFrame->DrainSelfOverflowList();
|
|
|
|
return aParentFrame->GetChildList(kPrincipalList).LastChild();
|
|
}
|
|
|
|
/**
|
|
* 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();
|
|
}
|
|
|
|
/**
|
|
* This function is called by ContentAppended() and ContentInserted() when
|
|
* appending flowed frames to a parent's principal child list. It handles the
|
|
* case where the parent is the trailing inline of an {ib} split.
|
|
*/
|
|
void
|
|
nsCSSFrameConstructor::AppendFramesToParent(nsFrameConstructorState& aState,
|
|
nsContainerFrame* aParentFrame,
|
|
nsFrameItems& aFrameList,
|
|
nsIFrame* aPrevSibling,
|
|
bool aIsRecursiveCall)
|
|
{
|
|
NS_PRECONDITION(!IsFramePartOfIBSplit(aParentFrame) ||
|
|
!GetIBSplitSibling(aParentFrame) ||
|
|
!GetIBSplitSibling(aParentFrame)->PrincipalChildList().FirstChild(),
|
|
"aParentFrame has a ib-split sibling with kids?");
|
|
NS_PRECONDITION(!aPrevSibling || aPrevSibling->GetParent() == aParentFrame,
|
|
"Parent and prevsibling don't match");
|
|
|
|
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()->IsInlineOutside()) {
|
|
// 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::FrameLinkEnumerator firstNonBlockEnumerator =
|
|
FindFirstNonBlock(aFrameList);
|
|
nsFrameList blockKids = aFrameList.ExtractHead(firstNonBlockEnumerator);
|
|
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::FrameLinkEnumerator firstBlockEnumerator(aFrameList);
|
|
FindFirstBlock(firstBlockEnumerator);
|
|
|
|
nsFrameList inlineKids = aFrameList.ExtractHead(firstBlockEnumerator);
|
|
if (!inlineKids.IsEmpty()) {
|
|
AppendFrames(aParentFrame, kPrincipalList, inlineKids);
|
|
}
|
|
|
|
if (!aFrameList.IsEmpty()) {
|
|
bool positioned = aParentFrame->IsRelativelyPositioned();
|
|
nsFrameItems ibSiblings;
|
|
CreateIBSiblings(aState, aParentFrame, positioned, 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,
|
|
nsIPresShell::eTreeChange,
|
|
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;
|
|
}
|
|
|
|
// Insert the frames after our aPrevSibling
|
|
InsertFrames(aParentFrame, kPrincipalList, aPrevSibling, aFrameList);
|
|
}
|
|
|
|
#define UNSET_DISPLAY static_cast<StyleDisplay>(255)
|
|
|
|
// 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,
|
|
StyleDisplay& aDisplay)
|
|
{
|
|
nsIFrame* parentFrame = aSibling->GetParent();
|
|
LayoutFrameType parentType = parentFrame->Type();
|
|
|
|
StyleDisplay siblingDisplay = aSibling->GetDisplay();
|
|
if (StyleDisplay::TableColumnGroup == siblingDisplay ||
|
|
StyleDisplay::TableColumn == siblingDisplay ||
|
|
StyleDisplay::TableCaption == siblingDisplay ||
|
|
StyleDisplay::TableHeaderGroup == siblingDisplay ||
|
|
StyleDisplay::TableRowGroup == siblingDisplay ||
|
|
StyleDisplay::TableFooterGroup == siblingDisplay ||
|
|
LayoutFrameType::Menu == parentType) {
|
|
// if we haven't already, construct a style context to find the display type of aContent
|
|
if (UNSET_DISPLAY == aDisplay) {
|
|
nsIFrame* styleParent;
|
|
aSibling->GetParentStyleContext(&styleParent);
|
|
if (!styleParent) {
|
|
styleParent = aSibling->GetParent();
|
|
}
|
|
if (!styleParent) {
|
|
NS_NOTREACHED("Shouldn't happen");
|
|
return false;
|
|
}
|
|
if (aContent->IsNodeOfType(nsINode::eCOMMENT) ||
|
|
aContent->IsNodeOfType(nsINode::ePROCESSING_INSTRUCTION)) {
|
|
// Comments and processing instructions never have frames, so we
|
|
// should not try to generate style contexts for them.
|
|
return false;
|
|
}
|
|
// XXXbz when this code is killed, the state argument to
|
|
// ResolveStyleContext can be made non-optional.
|
|
RefPtr<nsStyleContext> styleContext =
|
|
ResolveStyleContext(styleParent, aContent, nullptr);
|
|
const nsStyleDisplay* display = styleContext->StyleDisplay();
|
|
aDisplay = display->mDisplay;
|
|
}
|
|
if (LayoutFrameType::Menu == parentType) {
|
|
return
|
|
(StyleDisplay::MozPopup == aDisplay) ==
|
|
(StyleDisplay::MozPopup == siblingDisplay);
|
|
}
|
|
// 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) !=
|
|
(aDisplay == StyleDisplay::TableCaption)) {
|
|
// One's a caption and the other is not. Not valid siblings.
|
|
return false;
|
|
}
|
|
|
|
if ((siblingDisplay == StyleDisplay::TableColumnGroup ||
|
|
siblingDisplay == StyleDisplay::TableColumn) !=
|
|
(aDisplay == StyleDisplay::TableColumnGroup ||
|
|
aDisplay == 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.
|
|
}
|
|
|
|
if (IsFrameForFieldSet(parentFrame)) {
|
|
// Legends can be sibling of legends but not of other content in the fieldset
|
|
if (nsContainerFrame* cif = aSibling->GetContentInsertionFrame()) {
|
|
aSibling = cif;
|
|
}
|
|
LayoutFrameType sibType = aSibling->Type();
|
|
bool legendContent = aContent->IsHTMLElement(nsGkAtoms::legend);
|
|
|
|
if ((legendContent && (LayoutFrameType::Legend != sibType)) ||
|
|
(!legendContent && (LayoutFrameType::Legend == sibType)))
|
|
return false;
|
|
}
|
|
|
|
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,
|
|
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 getNearPseudo = [](const nsIContent* aContent) -> nsIFrame* {
|
|
return aDirection == SiblingDirection::Forward
|
|
? nsLayoutUtils::GetBeforeFrame(aContent)
|
|
: nsLayoutUtils::GetAfterFrame(aContent);
|
|
};
|
|
|
|
auto getFarPseudo = [](const nsIContent* aContent) -> nsIFrame* {
|
|
return aDirection == SiblingDirection::Forward
|
|
? nsLayoutUtils::GetAfterFrame(aContent)
|
|
: nsLayoutUtils::GetBeforeFrame(aContent);
|
|
};
|
|
|
|
while (nsIContent* sibling = nextDomSibling(aIter)) {
|
|
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 (GetDisplayContentsStyleFor(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,
|
|
mozilla::StyleDisplay& aTargetContentDisplay,
|
|
SiblingDirection aDirection)
|
|
{
|
|
if (!aSibling) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (aSibling->GetStateBits() & 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, true);
|
|
}
|
|
|
|
// 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,
|
|
StyleDisplay& aTargetContentDisplay)
|
|
{
|
|
return FindSibling<SiblingDirection::Backward>(aIter, aTargetContentDisplay);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::FindNextSibling(const FlattenedChildIterator& aIter,
|
|
StyleDisplay& aTargetContentDisplay)
|
|
{
|
|
return FindSibling<SiblingDirection::Forward>(aIter, aTargetContentDisplay);
|
|
}
|
|
|
|
template<nsCSSFrameConstructor::SiblingDirection aDirection>
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::FindSibling(const FlattenedChildIterator& aIter,
|
|
StyleDisplay& aTargetContentDisplay)
|
|
{
|
|
nsIContent* targetContent = aIter.Get();
|
|
nsIFrame* sibling =
|
|
FindSiblingInternal<aDirection>(aIter, 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 (GetDisplayContentsStyleFor(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)
|
|
{
|
|
NS_PRECONDITION(!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 = GetFieldSetBlockFrame(aParentFrame);
|
|
}
|
|
}
|
|
return newParent ? newParent : aParentFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::GetInsertionPrevSibling(InsertionPoint* aInsertion,
|
|
nsIContent* aChild,
|
|
bool* aIsAppend,
|
|
bool* aIsRangeInsertSafe,
|
|
nsIContent* aStartSkipChild,
|
|
nsIContent* aEndSkipChild)
|
|
{
|
|
NS_PRECONDITION(aInsertion->mParentFrame, "Must have parent frame to start with");
|
|
|
|
*aIsAppend = false;
|
|
|
|
// Find the frame that precedes the insertion point. Walk backwards
|
|
// from the parent frame to get the parent content, because if an
|
|
// XBL insertion point is involved, we'll need to use _that_ to find
|
|
// the preceding frame.
|
|
FlattenedChildIterator iter(aInsertion->mContainer);
|
|
bool xblCase = iter.XBLInvolved() ||
|
|
aInsertion->mParentFrame->GetContent() != aInsertion->mContainer;
|
|
if (xblCase || !aChild->IsRootOfAnonymousSubtree()) {
|
|
// The check for IsRootOfAnonymousSubtree() 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.
|
|
StyleDisplay childDisplay = UNSET_DISPLAY;
|
|
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;
|
|
if (IsFramePartOfIBSplit(aInsertion->mParentFrame)) {
|
|
// Since we're appending, we'll walk to the last anonymous frame
|
|
// that was created for the broken inline frame. But don't walk
|
|
// to the trailing inline if it's empty; stop at the block.
|
|
aInsertion->mParentFrame =
|
|
GetLastIBSplitSibling(aInsertion->mParentFrame, false);
|
|
}
|
|
// Get continuation that parents the last child.
|
|
aInsertion->mParentFrame =
|
|
nsLayoutUtils::LastContinuationWithChild(aInsertion->mParentFrame);
|
|
// Deal with fieldsets
|
|
aInsertion->mParentFrame =
|
|
::GetAdjustedParentFrame(aInsertion->mParentFrame, aChild);
|
|
prevSibling = ::FindAppendPrevSibling(aInsertion->mParentFrame, nullptr);
|
|
}
|
|
}
|
|
|
|
*aIsRangeInsertSafe = (childDisplay == UNSET_DISPLAY);
|
|
return prevSibling;
|
|
}
|
|
|
|
nsContainerFrame*
|
|
nsCSSFrameConstructor::GetContentInsertionFrameFor(nsIContent* aContent)
|
|
{
|
|
nsIFrame* frame;
|
|
while (!(frame = aContent->GetPrimaryFrame())) {
|
|
if (!GetDisplayContentsStyleFor(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(nsIPresShell* presShell, nsIContent* node);
|
|
|
|
#ifdef MOZ_XUL
|
|
|
|
static
|
|
bool
|
|
IsXULListBox(nsIContent* aContainer)
|
|
{
|
|
return (aContainer->IsXULElement(nsGkAtoms::listbox));
|
|
}
|
|
|
|
static
|
|
nsListBoxBodyFrame*
|
|
MaybeGetListBoxBodyFrame(nsIContent* aContainer, nsIContent* aChild)
|
|
{
|
|
if (!aContainer)
|
|
return nullptr;
|
|
|
|
if (IsXULListBox(aContainer) &&
|
|
aChild->IsXULElement(nsGkAtoms::listitem)) {
|
|
RefPtr<nsXULElement> xulElement = nsXULElement::FromContent(aContainer);
|
|
IgnoredErrorResult ignored;
|
|
nsCOMPtr<nsIBoxObject> boxObject = xulElement->GetBoxObject(ignored);
|
|
nsCOMPtr<nsPIListBoxObject> listBoxObject = do_QueryInterface(boxObject);
|
|
if (listBoxObject) {
|
|
return listBoxObject->GetListBoxBody(false);
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
#endif // MOZ_XUL
|
|
|
|
void
|
|
nsCSSFrameConstructor::AddTextItemIfNeeded(nsFrameConstructorState& aState,
|
|
const InsertionPoint& aInsertion,
|
|
nsIContent* aPossibleTextContent,
|
|
FrameConstructionItemList& aItems)
|
|
{
|
|
NS_PRECONDITION(aPossibleTextContent, "Must have node");
|
|
if (!aPossibleTextContent->IsNodeOfType(nsINode::eTEXT) ||
|
|
!aPossibleTextContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE)) {
|
|
// 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)
|
|
return;
|
|
}
|
|
NS_ASSERTION(!aPossibleTextContent->GetPrimaryFrame(),
|
|
"Text node has a frame and NS_CREATE_FRAME_IF_NON_WHITESPACE");
|
|
AddFrameConstructionItems(aState, aPossibleTextContent, false,
|
|
aInsertion, aItems);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ReframeTextIfNeeded(nsIContent* aParentContent,
|
|
nsIContent* aContent)
|
|
{
|
|
if (!aContent->IsNodeOfType(nsINode::eTEXT) ||
|
|
!aContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE)) {
|
|
// 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)
|
|
return;
|
|
}
|
|
NS_ASSERTION(!aContent->GetPrimaryFrame(),
|
|
"Text node has a frame and NS_CREATE_FRAME_IF_NON_WHITESPACE");
|
|
ContentInserted(aParentContent, aContent, nullptr, 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()) {
|
|
nsStyleContext* sc = GetDisplayNoneStyleFor(content);
|
|
noPrimaryFrame = !GetDisplayContentsStyleFor(content) &&
|
|
(sc && !sc->IsInDisplayNoneSubtree());
|
|
}
|
|
if (!needsFrameBitSet && content->HasFlag(NODE_NEEDS_FRAME)) {
|
|
needsFrameBitSet = true;
|
|
}
|
|
|
|
content = content->GetFlattenedTreeParent();
|
|
}
|
|
if (content && content->GetPrimaryFrame() &&
|
|
content->GetPrimaryFrame()->IsLeaf()) {
|
|
noPrimaryFrame = needsFrameBitSet = false;
|
|
}
|
|
NS_ASSERTION(!noPrimaryFrame, "Ancestors of nodes with frames to be "
|
|
"constructed lazily should have frames");
|
|
NS_ASSERTION(!needsFrameBitSet, "Ancestors of nodes with frames to be "
|
|
"constructed lazily should not have NEEDS_FRAME bit set");
|
|
}
|
|
#endif
|
|
|
|
// For inserts aChild should be valid, for appends it should be null.
|
|
// 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.
|
|
//
|
|
// But we disable lazy frame construction for shadow trees... We should fix
|
|
// that, too.
|
|
bool
|
|
nsCSSFrameConstructor::MaybeConstructLazily(Operation aOperation,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild)
|
|
{
|
|
if (mPresShell->GetPresContext()->IsChrome() || !aContainer ||
|
|
aContainer->IsInNativeAnonymousSubtree() || aContainer->IsXULElement()) {
|
|
return false;
|
|
}
|
|
|
|
if (aOperation == CONTENTINSERT) {
|
|
if (aChild->IsRootOfAnonymousSubtree() ||
|
|
(aChild->HasFlag(NODE_IS_IN_SHADOW_TREE) &&
|
|
!aChild->IsInNativeAnonymousSubtree()) ||
|
|
aChild->IsXULElement()) {
|
|
return false;
|
|
}
|
|
} else { // CONTENTAPPEND
|
|
NS_ASSERTION(aOperation == CONTENTAPPEND,
|
|
"operation should be either insert or append");
|
|
for (nsIContent* child = aChild; child; child = child->GetNextSibling()) {
|
|
NS_ASSERTION(!child->IsRootOfAnonymousSubtree(),
|
|
"Should be coming through the CONTENTAPPEND case");
|
|
if (child->IsXULElement()) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// We can construct lazily; just need to set suitable bits in the content
|
|
// tree.
|
|
nsIContent* parent = aChild->GetFlattenedTreeParent();
|
|
if (!parent) {
|
|
// Not part of the flat tree, nothing to do.
|
|
return true;
|
|
}
|
|
|
|
|
|
// 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);
|
|
}
|
|
}
|
|
|
|
// Walk up the tree setting the NODE_DESCENDANTS_NEED_FRAMES bit as we go.
|
|
// We need different handling for servo given the scoped restyle roots.
|
|
CheckBitsForLazyFrameConstruction(parent);
|
|
|
|
if (mozilla::GeckoRestyleManager* geckoRM = RestyleManager()->GetAsGecko()) {
|
|
while (parent && !parent->HasFlag(NODE_DESCENDANTS_NEED_FRAMES)) {
|
|
parent->SetFlags(NODE_DESCENDANTS_NEED_FRAMES);
|
|
parent = parent->GetFlattenedTreeParent();
|
|
}
|
|
geckoRM->PostRestyleEventForLazyConstruction();
|
|
} else {
|
|
parent->AsElement()->NoteDescendantsNeedFramesForServo();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CreateNeededFrames(
|
|
nsIContent* aContent,
|
|
TreeMatchContext& aTreeMatchContext)
|
|
{
|
|
MOZ_ASSERT(!aContent->IsStyledByServo());
|
|
NS_ASSERTION(!aContent->HasFlag(NODE_NEEDS_FRAME),
|
|
"shouldn't get here with a content node that has needs frame bit set");
|
|
NS_ASSERTION(aContent->HasFlag(NODE_DESCENDANTS_NEED_FRAMES),
|
|
"should only get here with a content node that has descendants needing frames");
|
|
MOZ_ASSERT(aTreeMatchContext.mAncestorFilter.HasFilter(),
|
|
"The whole point of having the tree match context is optimizing "
|
|
"the ancestor filter usage!");
|
|
|
|
aContent->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES);
|
|
|
|
// We could either descend first (on nodes that don't have NODE_NEEDS_FRAME
|
|
// set) or issue content notifications for our kids first. In absence of
|
|
// anything definitive either way we'll go with the latter.
|
|
|
|
// It might be better to use GetChildArray and scan it completely first and
|
|
// then issue all notifications. (We have to scan it completely first because
|
|
// constructing frames can set attributes, which can change the storage of
|
|
// child lists).
|
|
|
|
// Scan the children of aContent to see what operations (if any) we need to
|
|
// perform.
|
|
uint32_t childCount = aContent->GetChildCount();
|
|
bool inRun = false;
|
|
nsIContent* firstChildInRun = nullptr;
|
|
for (uint32_t i = 0; i < childCount; i++) {
|
|
nsIContent* child = aContent->GetChildAt(i);
|
|
if (child->HasFlag(NODE_NEEDS_FRAME)) {
|
|
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.
|
|
"NEEDS_FRAME set on a node that already has a frame?");
|
|
if (!inRun) {
|
|
inRun = true;
|
|
firstChildInRun = child;
|
|
}
|
|
} else {
|
|
if (inRun) {
|
|
inRun = false;
|
|
// generate a ContentRangeInserted for [startOfRun,i)
|
|
ContentRangeInserted(aContent, firstChildInRun, child, nullptr,
|
|
InsertionKind::Sync,
|
|
&aTreeMatchContext);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (inRun) {
|
|
ContentAppended(aContent, firstChildInRun,
|
|
InsertionKind::Sync,
|
|
&aTreeMatchContext);
|
|
}
|
|
|
|
// Now descend.
|
|
FlattenedChildIterator iter(aContent);
|
|
for (nsIContent* child = iter.GetNextChild(); child; child = iter.GetNextChild()) {
|
|
if (child->HasFlag(NODE_DESCENDANTS_NEED_FRAMES)) {
|
|
TreeMatchContext::AutoAncestorPusher insertionPointPusher(
|
|
&aTreeMatchContext);
|
|
|
|
// Handle stuff like xbl:children.
|
|
if (child->GetParent() != aContent && child->GetParent()->IsElement()) {
|
|
insertionPointPusher.PushAncestorAndStyleScope(
|
|
child->GetParent()->AsElement());
|
|
}
|
|
|
|
TreeMatchContext::AutoAncestorPusher pusher(&aTreeMatchContext);
|
|
pusher.PushAncestorAndStyleScope(child);
|
|
|
|
CreateNeededFrames(child, aTreeMatchContext);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CreateNeededFrames()
|
|
{
|
|
NS_ASSERTION(!nsContentUtils::IsSafeToRunScript(),
|
|
"Someone forgot a script blocker");
|
|
|
|
Element* rootElement = mDocument->GetRootElement();
|
|
NS_ASSERTION(!rootElement || !rootElement->HasFlag(NODE_NEEDS_FRAME),
|
|
"root element should not have frame created lazily");
|
|
if (rootElement && rootElement->HasFlag(NODE_DESCENDANTS_NEED_FRAMES)) {
|
|
BeginUpdate();
|
|
TreeMatchContext treeMatchContext(
|
|
mDocument, TreeMatchContext::ForFrameConstruction);
|
|
treeMatchContext.InitAncestors(rootElement);
|
|
CreateNeededFrames(rootElement, treeMatchContext);
|
|
EndUpdate();
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::IssueSingleInsertNofications(nsIContent* aContainer,
|
|
nsIContent* aStartChild,
|
|
nsIContent* aEndChild,
|
|
InsertionKind aInsertionKind)
|
|
{
|
|
for (nsIContent* child = aStartChild;
|
|
child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
if ((child->GetPrimaryFrame() || GetDisplayNoneStyleFor(child) ||
|
|
GetDisplayContentsStyleFor(child))
|
|
#ifdef MOZ_XUL
|
|
// Except listboxes suck, so do NOT skip anything here if
|
|
// we plan to notify a listbox.
|
|
&& !MaybeGetListBoxBodyFrame(aContainer, child)
|
|
#endif
|
|
) {
|
|
// Already have a frame or undisplayed entry for this content; a
|
|
// previous ContentRangeInserted in this loop must have reconstructed
|
|
// its insertion parent. Skip it.
|
|
continue;
|
|
}
|
|
// Call ContentRangeInserted with this node.
|
|
ContentRangeInserted(aContainer, child, child->GetNextSibling(),
|
|
mTempFrameTreeState, aInsertionKind, nullptr);
|
|
}
|
|
}
|
|
|
|
nsCSSFrameConstructor::InsertionPoint
|
|
nsCSSFrameConstructor::GetRangeInsertionPoint(nsIContent* aContainer,
|
|
nsIContent* aStartChild,
|
|
nsIContent* aEndChild,
|
|
InsertionKind aInsertionKind)
|
|
{
|
|
// See if we have an XBL 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.
|
|
InsertionPoint insertionPoint = GetInsertionPoint(aContainer, nullptr);
|
|
if (!insertionPoint.mParentFrame && !insertionPoint.mMultiple) {
|
|
return insertionPoint; // Don't build the frames.
|
|
}
|
|
|
|
if (insertionPoint.mMultiple || aStartChild->GetXBLInsertionPoint()) {
|
|
// If we have multiple insertion points or if we have an insertion point
|
|
// and the operation is not a true append or if the insertion point already
|
|
// has explicit children, then we must fall back.
|
|
if (insertionPoint.mMultiple || aEndChild ||
|
|
insertionPoint.mParentFrame->GetContent()->HasChildren()) {
|
|
// Now comes the fun part. For each inserted child, make a
|
|
// ContentInserted call as if it had just gotten inserted and
|
|
// let ContentInserted handle the mess.
|
|
IssueSingleInsertNofications(aContainer, aStartChild, aEndChild,
|
|
aInsertionKind);
|
|
insertionPoint.mParentFrame = nullptr;
|
|
}
|
|
}
|
|
|
|
return insertionPoint;
|
|
}
|
|
|
|
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();
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::StyleNewChildRange(nsIContent* aStartChild,
|
|
nsIContent* aEndChild)
|
|
{
|
|
ServoStyleSet* styleSet = mPresShell->StyleSet()->AsServo();
|
|
|
|
for (nsIContent* child = aStartChild; child != aEndChild;
|
|
child = child->GetNextSibling()) {
|
|
// Calling StyleNewChildren on one child will end up styling another child,
|
|
// if they share the same flattened tree parent. So we check HasServoData()
|
|
// to avoid a wasteful call to GetFlattenedTreeParent (on the child) and
|
|
// StyleNewChildren (on the flattened tree parent) when we detect we've
|
|
// already handled that parent. In the common case of inserting elements
|
|
// into a container that does not have an XBL binding or shadow tree with
|
|
// distributed children, this boils down to a single call to
|
|
// GetFlattenedTreeParent/StyleNewChildren, and traversing the list of
|
|
// children checking HasServoData (which is fast).
|
|
if (child->IsElement() && !child->AsElement()->HasServoData()) {
|
|
Element* parent = child->AsElement()->GetFlattenedTreeParentElement();
|
|
// NB: Parent may be null if the content is appended to a shadow root, and
|
|
// isn't assigned to any insertion point.
|
|
if (MOZ_LIKELY(parent) && parent->HasServoData()) {
|
|
styleSet->StyleNewChildren(parent);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ContentAppended(nsIContent* aContainer,
|
|
nsIContent* aFirstNewContent,
|
|
InsertionKind aInsertionKind,
|
|
TreeMatchContext* aProvidedTreeMatchContext)
|
|
{
|
|
MOZ_ASSERT(!aProvidedTreeMatchContext ||
|
|
aInsertionKind == InsertionKind::Sync);
|
|
MOZ_ASSERT(aInsertionKind == InsertionKind::Sync ||
|
|
!RestyleManager()->IsInStyleRefresh());
|
|
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
NS_PRECONDITION(mUpdateCount != 0,
|
|
"Should be in an update while creating frames");
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentAppended container=%p "
|
|
"first-child=%p lazy=%d\n",
|
|
static_cast<void*>(aContainer), aFirstNewContent,
|
|
aInsertionKind == InsertionKind::Async);
|
|
if (gReallyNoisyContentUpdates && aContainer) {
|
|
aContainer->List(stdout, 0);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
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.
|
|
NS_ASSERTION(!child->GetPrimaryFrame() ||
|
|
child->GetPrimaryFrame()->GetContent() != child,
|
|
"asked to construct a frame for a node that already has a frame");
|
|
}
|
|
#endif
|
|
|
|
#ifdef MOZ_XUL
|
|
if (aContainer) {
|
|
int32_t namespaceID;
|
|
nsAtom* tag =
|
|
mDocument->BindingManager()->ResolveTag(aContainer, &namespaceID);
|
|
|
|
// Just ignore tree tags, anyway we don't create any frames for them.
|
|
if (tag == nsGkAtoms::treechildren ||
|
|
tag == nsGkAtoms::treeitem ||
|
|
tag == nsGkAtoms::treerow)
|
|
return;
|
|
}
|
|
#endif // MOZ_XUL
|
|
|
|
// See comment in ContentRangeInserted for why this is necessary.
|
|
if (!GetContentInsertionFrameFor(aContainer) &&
|
|
!aContainer->IsActiveChildrenElement()) {
|
|
// 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 (aContainer->IsStyledByServo() &&
|
|
aInsertionKind == InsertionKind::Async) {
|
|
LazilyStyleNewChildRange(aFirstNewContent, nullptr);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (aInsertionKind == InsertionKind::Async &&
|
|
MaybeConstructLazily(CONTENTAPPEND, aContainer, aFirstNewContent)) {
|
|
if (aContainer->IsStyledByServo()) {
|
|
LazilyStyleNewChildRange(aFirstNewContent, nullptr);
|
|
}
|
|
return;
|
|
}
|
|
|
|
// We couldn't construct lazily. Make Servo eagerly traverse the new content
|
|
// if needed (when aInsertionKind == InsertionKind::Sync, we know that the
|
|
// styles are up-to-date already).
|
|
if (aInsertionKind == InsertionKind::Async && aContainer->IsStyledByServo()) {
|
|
StyleNewChildRange(aFirstNewContent, nullptr);
|
|
}
|
|
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
InsertionPoint insertion =
|
|
GetRangeInsertionPoint(aContainer, aFirstNewContent, nullptr,
|
|
aInsertionKind);
|
|
nsContainerFrame*& parentFrame = insertion.mParentFrame;
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
if (!parentFrame) {
|
|
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;
|
|
}
|
|
|
|
if (parentFrame->IsFrameOfType(nsIFrame::eMathML)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(parentFrame->GetContent(), aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
// 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.
|
|
bool parentIBSplit = IsFramePartOfIBSplit(parentFrame);
|
|
if (parentIBSplit) {
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentAppended: parentFrame=");
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
printf(" is ib-split\n");
|
|
}
|
|
#endif
|
|
|
|
// Since we're appending, we'll walk to the last anonymous frame
|
|
// that was created for the broken inline frame. But don't walk
|
|
// to the trailing inline if it's empty; stop at the block.
|
|
parentFrame = GetLastIBSplitSibling(parentFrame, false);
|
|
}
|
|
|
|
// Get continuation that parents the last child. This MUST be done
|
|
// before the AdjustAppendParentForAfterContent call.
|
|
parentFrame = nsLayoutUtils::LastContinuationWithChild(parentFrame);
|
|
|
|
// We should never get here with fieldsets or details, since they have
|
|
// multiple insertion points.
|
|
MOZ_ASSERT(!parentFrame->IsFieldSetFrame() && !parentFrame->IsDetailsFrame(),
|
|
"Parent frame should not be fieldset or details!");
|
|
|
|
// Deal with possible :after generated content on the parent
|
|
nsIFrame* parentAfterFrame;
|
|
nsContainerFrame* preAdjustedParentFrame = parentFrame;
|
|
parentFrame =
|
|
::AdjustAppendParentForAfterContent(this, insertion.mContainer, parentFrame,
|
|
aFirstNewContent, &parentAfterFrame);
|
|
|
|
// See if the containing block has :first-letter style applied.
|
|
bool haveFirstLetterStyle = false, haveFirstLineStyle = false;
|
|
nsContainerFrame* containingBlock = GetFloatContainingBlock(parentFrame);
|
|
if (containingBlock) {
|
|
haveFirstLetterStyle = HasFirstLetterStyle(containingBlock);
|
|
haveFirstLineStyle =
|
|
ShouldHaveFirstLineStyle(containingBlock->GetContent(),
|
|
containingBlock->StyleContext());
|
|
}
|
|
|
|
if (haveFirstLetterStyle) {
|
|
AutoWeakFrame wf(parentAfterFrame);
|
|
// Before we get going, remove the current letter frames
|
|
RemoveLetterFrames(mPresShell, containingBlock);
|
|
|
|
if (parentAfterFrame && !wf) {
|
|
// Ouch, parentAfterFrame was a letter frame and we just deleted it!
|
|
// Retry AdjustAppendParentForAfterContent; fortunately this is rare.
|
|
parentFrame =
|
|
::AdjustAppendParentForAfterContent(this, insertion.mContainer,
|
|
preAdjustedParentFrame,
|
|
aFirstNewContent, &parentAfterFrame);
|
|
if (parentFrame != preAdjustedParentFrame) {
|
|
containingBlock = GetFloatContainingBlock(parentFrame);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Create some new frames
|
|
//
|
|
// We use the provided tree match context, or create a new one on the fly
|
|
// otherwise.
|
|
Maybe<TreeMatchContext> matchContext;
|
|
if (!aProvidedTreeMatchContext && !aContainer->IsStyledByServo()) {
|
|
// We use GetParentElementCrossingShadowRoot to handle the case where
|
|
// aContainer is a ShadowRoot.
|
|
matchContext.emplace(mDocument, TreeMatchContext::ForFrameConstruction);
|
|
matchContext->InitAncestors(aFirstNewContent->GetParentElementCrossingShadowRoot());
|
|
}
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext.ptrOr(aProvidedTreeMatchContext),
|
|
GetAbsoluteContainingBlock(parentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(parentFrame, ABS_POS),
|
|
containingBlock);
|
|
|
|
LayoutFrameType frameType = parentFrame->Type();
|
|
|
|
FlattenedChildIterator iter(aContainer);
|
|
bool haveNoXBLChildren = (!iter.XBLInvolved() || !iter.GetNextChild());
|
|
AutoFrameConstructionItemList items(this);
|
|
if (aFirstNewContent->GetPreviousSibling() &&
|
|
GetParentType(frameType) == eTypeBlock &&
|
|
haveNoXBLChildren) {
|
|
// 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 XBL anonymous content
|
|
// (text in an XBL binding is not suppressed) or generated content
|
|
// (and bare text nodes are not generated). Native anonymous content
|
|
// generated by frames never participates in inline layout.
|
|
AddTextItemIfNeeded(state, insertion,
|
|
aFirstNewContent->GetPreviousSibling(), items);
|
|
}
|
|
for (nsIContent* child = aFirstNewContent;
|
|
child;
|
|
child = child->GetNextSibling()) {
|
|
AddFrameConstructionItems(state, child, false, insertion, items);
|
|
}
|
|
|
|
nsIFrame* prevSibling = ::FindAppendPrevSibling(parentFrame, parentAfterFrame);
|
|
|
|
// 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 (nsLayoutUtils::GetAsBlock(parentFrame) && !haveFirstLetterStyle &&
|
|
!haveFirstLineStyle && !parentIBSplit) {
|
|
items.SetLineBoundaryAtStart(!prevSibling ||
|
|
!prevSibling->IsInlineOutside() ||
|
|
prevSibling->IsBrFrame());
|
|
// :after content can't be <br> so no need to check it
|
|
items.SetLineBoundaryAtEnd(!parentAfterFrame ||
|
|
!parentAfterFrame->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.SetParentHasNoXBLChildren(haveNoXBLChildren);
|
|
|
|
nsFrameItems frameItems;
|
|
ConstructFramesFromItemList(state, items, parentFrame,
|
|
ParentIsWrapperAnonBox(parentFrame),
|
|
frameItems);
|
|
|
|
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 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.
|
|
nsFrameItems captionItems;
|
|
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(frameItems, captionItems);
|
|
}
|
|
|
|
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, frameItems);
|
|
// That moved things into line frames as needed, reparenting their
|
|
// styles. Nothing else needs to be done.
|
|
} else if (parentFrame->StyleContext()->HasPseudoElementData()) {
|
|
// parentFrame might be inside a ::first-line frame. Check whether it is,
|
|
// and if so fix up our styles.
|
|
CheckForFirstLineInsertion(parentFrame, frameItems);
|
|
CheckForFirstLineInsertion(parentFrame, captionItems);
|
|
}
|
|
|
|
// 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 (captionItems.NotEmpty()) { // append the caption to the table wrapper
|
|
NS_ASSERTION(LayoutFrameType::Table == frameType, "how did that happen?");
|
|
nsContainerFrame* outerTable = parentFrame->GetParent();
|
|
AppendFrames(outerTable, nsIFrame::kCaptionList, captionItems);
|
|
}
|
|
|
|
if (frameItems.NotEmpty()) { // append the in-flow kids
|
|
AppendFramesToParent(state, parentFrame, frameItems, prevSibling);
|
|
}
|
|
|
|
// Recover first-letter frames
|
|
if (haveFirstLetterStyle) {
|
|
RecoverLetterFrames(containingBlock);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentAppended: resulting frame model:\n");
|
|
parentFrame->List(stdout, 0);
|
|
}
|
|
#endif
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = nsIPresShell::AccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aContainer,
|
|
aFirstNewContent, nullptr);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
|
|
enum content_operation
|
|
{
|
|
CONTENT_INSERTED,
|
|
CONTENT_REMOVED
|
|
};
|
|
|
|
// Helper function to lookup the listbox body frame and send a notification
|
|
// for insertion or removal of content
|
|
static
|
|
bool NotifyListBoxBody(nsPresContext* aPresContext,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
// Only used for the removed notification
|
|
nsIContent* aOldNextSibling,
|
|
nsIFrame* aChildFrame,
|
|
content_operation aOperation)
|
|
{
|
|
nsListBoxBodyFrame* listBoxBodyFrame =
|
|
MaybeGetListBoxBodyFrame(aContainer, aChild);
|
|
if (listBoxBodyFrame) {
|
|
if (aOperation == CONTENT_REMOVED) {
|
|
// Except if we have an aChildFrame and its parent is not the right
|
|
// thing, then we don't do this. Pseudo frames are so much fun....
|
|
if (!aChildFrame || aChildFrame->GetParent() == listBoxBodyFrame) {
|
|
listBoxBodyFrame->OnContentRemoved(aPresContext, aContainer,
|
|
aChildFrame, aOldNextSibling);
|
|
return true;
|
|
}
|
|
} else {
|
|
listBoxBodyFrame->OnContentInserted(aChild);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
#endif // MOZ_XUL
|
|
|
|
void
|
|
nsCSSFrameConstructor::ContentInserted(nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
nsILayoutHistoryState* aFrameState,
|
|
InsertionKind aInsertionKind)
|
|
{
|
|
ContentRangeInserted(aContainer,
|
|
aChild,
|
|
aChild->GetNextSibling(),
|
|
aFrameState,
|
|
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* aContainer,
|
|
nsIContent* aStartChild,
|
|
nsIContent* aEndChild,
|
|
nsILayoutHistoryState* aFrameState,
|
|
InsertionKind aInsertionKind,
|
|
TreeMatchContext* aProvidedTreeMatchContext)
|
|
{
|
|
MOZ_ASSERT(!aProvidedTreeMatchContext ||
|
|
aInsertionKind == InsertionKind::Sync);
|
|
MOZ_ASSERT(aInsertionKind == InsertionKind::Sync ||
|
|
!RestyleManager()->IsInStyleRefresh());
|
|
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
NS_PRECONDITION(mUpdateCount != 0,
|
|
"Should be in an update while creating frames");
|
|
|
|
NS_PRECONDITION(aStartChild, "must always pass a child");
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentRangeInserted container=%p "
|
|
"start-child=%p end-child=%p lazy=%d\n",
|
|
static_cast<void*>(aContainer),
|
|
static_cast<void*>(aStartChild), static_cast<void*>(aEndChild),
|
|
aInsertionKind == InsertionKind::Async);
|
|
if (gReallyNoisyContentUpdates) {
|
|
if (aContainer) {
|
|
aContainer->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
|
|
|
|
auto styleNewChildRangeEagerly =
|
|
[this, aInsertionKind, aContainer, aStartChild, aEndChild]() {
|
|
// When aInsertionKind == InsertionKind::Sync, we know that the
|
|
// styles are up-to-date already.
|
|
if (aInsertionKind == InsertionKind::Async &&
|
|
aContainer->IsStyledByServo()) {
|
|
StyleNewChildRange(aStartChild, aEndChild);
|
|
}
|
|
};
|
|
|
|
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");
|
|
|
|
#ifdef MOZ_XUL
|
|
if (aContainer && IsXULListBox(aContainer)) {
|
|
// For XUL list box, we need to style the new children eagerly.
|
|
styleNewChildRangeEagerly();
|
|
if (isSingleInsert) {
|
|
if (NotifyListBoxBody(mPresShell->GetPresContext(), aContainer,
|
|
// The insert case in NotifyListBoxBody
|
|
// doesn't use "old next sibling".
|
|
aStartChild, nullptr, nullptr, CONTENT_INSERTED)) {
|
|
return;
|
|
}
|
|
} else {
|
|
// We don't handle a range insert to a listbox parent, issue single
|
|
// ContertInserted calls for each node inserted.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
IssueSingleInsertNofications(aContainer, aStartChild, aEndChild,
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
}
|
|
#endif // MOZ_XUL
|
|
|
|
// 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 (!aContainer) {
|
|
NS_ASSERTION(isSingleInsert,
|
|
"root node insertion should be a single insertion");
|
|
Element* docElement = mDocument->GetRootElement();
|
|
|
|
if (aStartChild != docElement) {
|
|
// Not the root element; just bail out
|
|
return;
|
|
}
|
|
|
|
NS_PRECONDITION(!mRootElementFrame, "root element frame already created");
|
|
|
|
// Create frames for the document element and its child elements
|
|
if (ConstructDocElementFrame(docElement, aFrameState)) {
|
|
InvalidateCanvasIfNeeded(mPresShell, aStartChild);
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentRangeInserted: resulting frame "
|
|
"model:\n");
|
|
mRootElementFrame->List(stdout, 0);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if (aFrameState) {
|
|
// Restore frame state for the root scroll frame if there is one
|
|
if (nsIFrame* rootScrollFrame = mPresShell->GetRootScrollFrame()) {
|
|
RestoreFrameStateFor(rootScrollFrame, aFrameState);
|
|
}
|
|
}
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = nsIPresShell::AccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aContainer,
|
|
aStartChild, aEndChild);
|
|
}
|
|
#endif
|
|
|
|
return;
|
|
}
|
|
|
|
nsContainerFrame* parentFrame = GetContentInsertionFrameFor(aContainer);
|
|
// The xbl:children element won't have a frame, but default content can have the children as
|
|
// a parent. While its uncommon to change the structure of the default content itself, a label,
|
|
// for example, can be reframed by having its value attribute set or removed.
|
|
if (!parentFrame &&
|
|
!(aContainer->IsActiveChildrenElement() ||
|
|
ShadowRoot::FromNode(aContainer))) {
|
|
// 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 (aContainer->IsStyledByServo() &&
|
|
aInsertionKind == InsertionKind::Async) {
|
|
LazilyStyleNewChildRange(aStartChild, aEndChild);
|
|
}
|
|
return;
|
|
}
|
|
|
|
MOZ_ASSERT_IF(ShadowRoot::FromNode(aContainer), !parentFrame);
|
|
|
|
// Otherwise, we've got parent content. Find its frame.
|
|
NS_ASSERTION(!parentFrame || parentFrame->GetContent() == aContainer ||
|
|
GetDisplayContentsStyleFor(aContainer), "New XBL code is possibly wrong!");
|
|
|
|
if (aInsertionKind == InsertionKind::Async &&
|
|
MaybeConstructLazily(CONTENTINSERT, aContainer, aStartChild)) {
|
|
if (aContainer->IsStyledByServo()) {
|
|
LazilyStyleNewChildRange(aStartChild, aEndChild);
|
|
}
|
|
return;
|
|
}
|
|
|
|
// We couldn't construct lazily. Make Servo eagerly traverse the new content
|
|
// if needed.
|
|
styleNewChildRangeEagerly();
|
|
|
|
InsertionPoint insertion;
|
|
if (isSingleInsert) {
|
|
// See if we have an XBL 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(aContainer, aStartChild);
|
|
} else {
|
|
// Get our insertion point. If we need to issue single ContentInserted's
|
|
// GetRangeInsertionPoint will take care of that for us.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
insertion = GetRangeInsertionPoint(aContainer, aStartChild, aEndChild,
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
|
|
if (!insertion.mParentFrame) {
|
|
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(aContainer, aStartChild, aEndChild,
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
nsIContent* container = insertion.mParentFrame->GetContent();
|
|
|
|
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");
|
|
if (IsFrameForFieldSet(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(),
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
// We should only get here with details when doing a single insertion because
|
|
// we treat details frame as if it has multiple insertion points.
|
|
MOZ_ASSERT(isSingleInsert || frameType != LayoutFrameType::Details);
|
|
if (frameType == LayoutFrameType::Details) {
|
|
// When inserting an element into <details>, just reframe the details frame
|
|
// and let it figure out where the element should be laid out. It might seem
|
|
// expensive to recreate the entire details frame, but it's the simplest way
|
|
// to handle the insertion.
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(insertion.mParentFrame->GetContent(),
|
|
aInsertionKind);
|
|
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;
|
|
}
|
|
|
|
if (insertion.mParentFrame->IsFrameOfType(nsIFrame::eMathML)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(insertion.mParentFrame->GetContent(),
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
Maybe<TreeMatchContext> matchContext;
|
|
if (!aProvidedTreeMatchContext && !aContainer->IsStyledByServo()) {
|
|
// We use GetParentElementCrossingShadowRoot to handle the case where
|
|
// aContainer is a ShadowRoot.
|
|
matchContext.emplace(mDocument, TreeMatchContext::ForFrameConstruction);
|
|
matchContext->InitAncestors(aStartChild->GetParentElementCrossingShadowRoot());
|
|
}
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext.ptrOr(aProvidedTreeMatchContext),
|
|
GetAbsoluteContainingBlock(insertion.mParentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(insertion.mParentFrame, ABS_POS),
|
|
GetFloatContainingBlock(insertion.mParentFrame),
|
|
do_AddRef(aFrameState));
|
|
|
|
// 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.mFloatedItems.containingBlock;
|
|
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.
|
|
StyleDisplay parentDisplay = insertion.mParentFrame->GetDisplay();
|
|
|
|
// 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 ((StyleDisplay::Block == parentDisplay) ||
|
|
(StyleDisplay::ListItem == parentDisplay) ||
|
|
(StyleDisplay::Inline == parentDisplay) ||
|
|
(StyleDisplay::InlineBlock == parentDisplay)) {
|
|
// Recover the special style flags for the containing block
|
|
if (containingBlock) {
|
|
haveFirstLetterStyle = HasFirstLetterStyle(containingBlock);
|
|
haveFirstLineStyle =
|
|
ShouldHaveFirstLineStyle(containingBlock->GetContent(),
|
|
containingBlock->StyleContext());
|
|
}
|
|
|
|
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->GetStateBits() & 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.mFloatedItems.containingBlock);
|
|
|
|
// 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.mFloatedItems.containingBlock);
|
|
|
|
// must fall back to a single ContertInserted for each child in the range
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
IssueSingleInsertNofications(aContainer, aStartChild, aEndChild,
|
|
aInsertionKind);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return;
|
|
}
|
|
|
|
container = insertion.mParentFrame->GetContent();
|
|
frameType = insertion.mParentFrame->Type();
|
|
}
|
|
}
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
ParentType parentType = GetParentType(frameType);
|
|
FlattenedChildIterator iter(aContainer);
|
|
bool haveNoXBLChildren = (!iter.XBLInvolved() || !iter.GetNextChild());
|
|
if (aStartChild->GetPreviousSibling() &&
|
|
parentType == eTypeBlock && haveNoXBLChildren) {
|
|
// 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 sitation.
|
|
AddTextItemIfNeeded(state, insertion, aStartChild->GetPreviousSibling(),
|
|
items);
|
|
}
|
|
|
|
if (isSingleInsert) {
|
|
AddFrameConstructionItems(state, aStartChild,
|
|
aStartChild->IsRootOfAnonymousSubtree(),
|
|
insertion, items);
|
|
} else {
|
|
for (nsIContent* child = aStartChild;
|
|
child != aEndChild;
|
|
child = child->GetNextSibling()){
|
|
AddFrameConstructionItems(state, child, false, insertion, items);
|
|
}
|
|
}
|
|
|
|
if (aEndChild && parentType == eTypeBlock && haveNoXBLChildren) {
|
|
// 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 sitation.
|
|
AddTextItemIfNeeded(state, 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();
|
|
|
|
// 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.
|
|
nsFrameItems frameItems, captionItems;
|
|
ConstructFramesFromItemList(state, items, insertion.mParentFrame,
|
|
ParentIsWrapperAnonBox(insertion.mParentFrame),
|
|
frameItems);
|
|
|
|
if (frameItems.NotEmpty()) {
|
|
for (nsIContent* child = aStartChild;
|
|
child != aEndChild;
|
|
child = child->GetNextSibling()){
|
|
InvalidateCanvasIfNeeded(mPresShell, child);
|
|
}
|
|
|
|
if (LayoutFrameType::Table == frameType ||
|
|
LayoutFrameType::TableWrapper == frameType) {
|
|
PullOutCaptionFrames(frameItems, captionItems);
|
|
}
|
|
}
|
|
|
|
// 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 and as it is unknown where to insert correctly we append instead
|
|
// (bug 341858).
|
|
// This can affect our prevSibling and isAppend, but should not have any
|
|
// effect on the WipeContainingBlock above, since this should only happen
|
|
// when neither parent is a ib-split frame and should not affect whitespace
|
|
// handling inside table-related frames (and in fact, can only happen when
|
|
// one of the parents is a table wrapper and one is an inner table or when the
|
|
// parent is a fieldset or fieldset content frame). So it won't affect the
|
|
// {ib} or XUL box cases in WipeContainingBlock(), and the table pseudo
|
|
// handling will only be affected by us maybe thinking we're not inserting
|
|
// at the beginning, whereas we really are. That would have made us reframe
|
|
// unnecessarily, but that's ok.
|
|
// XXXbz we should push our frame construction item code up higher, so we
|
|
// know what our items are by the time we start figuring out previous
|
|
// siblings
|
|
if (prevSibling && frameItems.NotEmpty() &&
|
|
frameItems.FirstChild()->GetParent() != prevSibling->GetParent()) {
|
|
#ifdef DEBUG
|
|
nsIFrame* frame1 = frameItems.FirstChild()->GetParent();
|
|
nsIFrame* frame2 = prevSibling->GetParent();
|
|
NS_ASSERTION(!IsFramePartOfIBSplit(frame1) &&
|
|
!IsFramePartOfIBSplit(frame2),
|
|
"Neither should be ib-split");
|
|
NS_ASSERTION((frame1->IsTableFrame() &&
|
|
frame2->IsTableWrapperFrame()) ||
|
|
(frame1->IsTableWrapperFrame() &&
|
|
frame2->IsTableFrame()) ||
|
|
frame1->IsFieldSetFrame() ||
|
|
(frame1->GetParent() &&
|
|
frame1->GetParent()->IsFieldSetFrame()),
|
|
"Unexpected frame types");
|
|
#endif
|
|
isAppend = true;
|
|
nsIFrame* appendAfterFrame;
|
|
insertion.mParentFrame =
|
|
::AdjustAppendParentForAfterContent(this, container,
|
|
frameItems.FirstChild()->GetParent(),
|
|
aStartChild, &appendAfterFrame);
|
|
prevSibling = ::FindAppendPrevSibling(insertion.mParentFrame, appendAfterFrame);
|
|
}
|
|
|
|
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, frameItems);
|
|
} else if (insertion.mParentFrame->StyleContext()->HasPseudoElementData()) {
|
|
CheckForFirstLineInsertion(insertion.mParentFrame, frameItems);
|
|
CheckForFirstLineInsertion(insertion.mParentFrame, captionItems);
|
|
}
|
|
|
|
// We might have captions; put them into the caption list of the
|
|
// table wrapper frame.
|
|
if (captionItems.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.mParentFrame, insertion.mContainer);
|
|
if (isSingleInsert) {
|
|
captionPrevSibling =
|
|
GetInsertionPrevSibling(&captionInsertion, aStartChild,
|
|
&captionIsAppend, &ignored);
|
|
} else {
|
|
nsIContent* firstCaption = captionItems.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 = nullptr;
|
|
if (GetCaptionAdjustedParent(captionInsertion.mParentFrame,
|
|
captionItems.FirstChild(),
|
|
&outerTable)) {
|
|
// 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.
|
|
NS_ASSERTION(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;
|
|
}
|
|
if (captionIsAppend) {
|
|
AppendFrames(outerTable, nsIFrame::kCaptionList, captionItems);
|
|
} else {
|
|
InsertFrames(outerTable, nsIFrame::kCaptionList,
|
|
captionPrevSibling, captionItems);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (frameItems.NotEmpty()) {
|
|
// Notify the parent frame
|
|
if (isAppend) {
|
|
AppendFramesToParent(state, insertion.mParentFrame, frameItems, prevSibling);
|
|
} else {
|
|
InsertFrames(insertion.mParentFrame, kPrincipalList, prevSibling, frameItems);
|
|
}
|
|
}
|
|
|
|
if (haveFirstLetterStyle) {
|
|
// Recover the letter frames for the containing block when
|
|
// it has first-letter style.
|
|
RecoverLetterFrames(state.mFloatedItems.containingBlock);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates && insertion.mParentFrame) {
|
|
printf("nsCSSFrameConstructor::ContentRangeInserted: resulting frame model:\n");
|
|
insertion.mParentFrame->List(stdout, 0);
|
|
}
|
|
#endif
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = nsIPresShell::AccService()) {
|
|
accService->ContentRangeInserted(mPresShell, aContainer,
|
|
aStartChild, aEndChild);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::ContentRemoved(nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
nsIContent* aOldNextSibling,
|
|
RemoveFlags aFlags)
|
|
{
|
|
MOZ_ASSERT(aChild);
|
|
AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
|
|
NS_PRECONDITION(mUpdateCount != 0,
|
|
"Should be in an update while destroying frames");
|
|
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->GetViewportScrollbarStylesOverrideElement() ||
|
|
(!aContainer && aChild->IsElement())) {
|
|
// 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->UpdateViewportScrollbarStylesOverride();
|
|
|
|
// If aChild is the root (i.e. aContainer is null), 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 (aContainer && 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",
|
|
static_cast<void*>(aContainer),
|
|
static_cast<void*>(aChild),
|
|
static_cast<void*>(aOldNextSibling));
|
|
if (gReallyNoisyContentUpdates) {
|
|
aContainer->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.
|
|
UnregisterDisplayNoneStyleFor(aChild, aContainer);
|
|
}
|
|
MOZ_ASSERT(!childFrame || !GetDisplayContentsStyleFor(aChild),
|
|
"display:contents nodes shouldn't have a frame");
|
|
if (!childFrame && GetDisplayContentsStyleFor(aChild)) {
|
|
if (HasGeneratedContent(aChild)) {
|
|
nsIContent* ancestor = aChild->GetFlattenedTreeParent();
|
|
MOZ_ASSERT(ancestor, "display: contents on the root?");
|
|
while (!ancestor->GetPrimaryFrame()) {
|
|
ancestor = ancestor->GetFlattenedTreeParent();
|
|
MOZ_ASSERT(ancestor, "we can't have a display: contents subtree root!");
|
|
}
|
|
|
|
// XXXmats Can we recreate frames only for the ::after/::before content?
|
|
// XXX Perhaps even only those that belong to the aChild sub-tree?
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(ancestor, InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
|
|
FlattenedChildIterator iter(aChild);
|
|
for (nsIContent* c = iter.GetNextChild(); c; c = iter.GetNextChild()) {
|
|
if (c->GetPrimaryFrame() || GetDisplayContentsStyleFor(c)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
bool didReconstruct =
|
|
ContentRemoved(aChild, c, nullptr, REMOVE_FOR_RECONSTRUCTION);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
if (didReconstruct) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
UnregisterDisplayContentsStyleFor(aChild, aContainer);
|
|
return false;
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
if (NotifyListBoxBody(presContext, aContainer, aChild, aOldNextSibling,
|
|
childFrame, CONTENT_REMOVED)) {
|
|
return false;
|
|
}
|
|
#endif // MOZ_XUL
|
|
|
|
// 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 had an XBL binding or display:none, even though the
|
|
// frames above it got created). We do the adjustment after the childFrame
|
|
// check above, because we do want to clear any undisplayed content we might
|
|
// have for the root. Detecting removal of a root is a little exciting; in
|
|
// particular, having a null aContainer 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.
|
|
bool isRoot = false;
|
|
if (!aContainer) {
|
|
nsIFrame* viewport = GetRootFrame();
|
|
if (viewport) {
|
|
nsIFrame* firstChild = viewport->PrincipalChildList().FirstChild();
|
|
if (firstChild && firstChild->GetContent() == aChild) {
|
|
isRoot = true;
|
|
childFrame = firstChild;
|
|
NS_ASSERTION(!childFrame->GetNextSibling(), "How did that happen?");
|
|
}
|
|
}
|
|
}
|
|
|
|
if (childFrame) {
|
|
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->IsFrameOfType(nsIFrame::eMathML)) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(parentFrame->GetContent(), InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
|
|
// Undo XUL wrapping if it's no longer needed.
|
|
// (If we're in the XUL block-wrapping situation, parentFrame is the
|
|
// wrapper frame.)
|
|
nsIFrame* grandparentFrame = parentFrame->GetParent();
|
|
if (grandparentFrame && grandparentFrame->IsXULBoxFrame() &&
|
|
(grandparentFrame->GetStateBits() & NS_STATE_BOX_WRAPS_KIDS_IN_BLOCK) &&
|
|
// check if this frame is the only one needing wrapping
|
|
aChild == AnyKidsNeedBlockParent(parentFrame->PrincipalChildList().FirstChild()) &&
|
|
!AnyKidsNeedBlockParent(childFrame->GetNextSibling())) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
RecreateFramesForContent(grandparentFrame->GetContent(),
|
|
InsertionKind::Async);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
return true;
|
|
}
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (nsAccessibilityService* accService = nsIPresShell::AccService()) {
|
|
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->GetStateBits() & 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=");
|
|
nsFrame::ListTag(stdout, containingBlock);
|
|
printf(" parentFrame=");
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
printf(" childFrame=");
|
|
nsFrame::ListTag(stdout, childFrame);
|
|
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.
|
|
UnregisterDisplayNoneStyleFor(aChild, aContainer);
|
|
return false;
|
|
}
|
|
parentFrame = childFrame->GetParent();
|
|
parentType = parentFrame->Type();
|
|
|
|
#ifdef NOISY_FIRST_LETTER
|
|
printf(" ==> revised parentFrame=");
|
|
nsFrame::ListTag(stdout, parentFrame);
|
|
printf(" childFrame=");
|
|
nsFrame::ListTag(stdout, childFrame);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ContentRemoved: childFrame=");
|
|
nsFrame::ListTag(stdout, childFrame);
|
|
putchar('\n');
|
|
parentFrame->List(stdout, 0);
|
|
}
|
|
#endif
|
|
|
|
|
|
// Notify the parent frame that it should delete the frame
|
|
if (childFrame->GetStateBits() & NS_FRAME_OUT_OF_FLOW) {
|
|
childFrame = childFrame->GetPlaceholderFrame();
|
|
NS_ASSERTION(childFrame, "Missing placeholder frame for out of flow.");
|
|
parentFrame = childFrame->GetParent();
|
|
}
|
|
RemoveFrame(nsLayoutUtils::GetChildListNameFor(childFrame), childFrame);
|
|
|
|
if (isRoot) {
|
|
mRootElementFrame = nullptr;
|
|
mRootElementStyleFrame = nullptr;
|
|
mDocElementContainingBlock = nullptr;
|
|
mPageSequenceFrame = nullptr;
|
|
mHasRootAbsPosContainingBlock = false;
|
|
}
|
|
|
|
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. We don't need
|
|
// to do this if the table parent type of our parent type is not
|
|
// eTypeBlock, though, because in that case the whitespace isn't
|
|
// being suppressed due to us anyway.
|
|
if (aContainer && !aChild->IsRootOfAnonymousSubtree() &&
|
|
aFlags == REMOVE_CONTENT &&
|
|
GetParentType(parentType) == eTypeBlock) {
|
|
// 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.
|
|
//
|
|
// FIXME(emilio): This should probably use the lazy frame construction
|
|
// bits if possible instead of reframing it in place.
|
|
if (aOldNextSibling) {
|
|
nsIContent* prevSibling = aOldNextSibling->GetPreviousSibling();
|
|
if (prevSibling && prevSibling->GetPreviousSibling()) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
ReframeTextIfNeeded(aContainer, 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 && aOldNextSibling->GetNextSibling() &&
|
|
aOldNextSibling->GetPreviousSibling()) {
|
|
LAYOUT_PHASE_TEMP_EXIT();
|
|
ReframeTextIfNeeded(aContainer, aOldNextSibling);
|
|
LAYOUT_PHASE_TEMP_REENTER();
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gReallyNoisyContentUpdates && parentFrame) {
|
|
printf("nsCSSFrameConstructor::ContentRemoved: resulting frame model:\n");
|
|
parentFrame->List(stdout, 0);
|
|
}
|
|
#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(nsIPresShell* presShell, nsIContent* node)
|
|
{
|
|
NS_PRECONDITION(presShell->GetRootFrame(), "What happened here?");
|
|
NS_PRECONDITION(presShell->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 = node->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 (!node->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 = presShell->GetRootFrame();
|
|
rootFrame->InvalidateFrameSubtree();
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::EnsureFrameForTextNodeIsCreatedAfterFlush(
|
|
nsGenericDOMDataNode* 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, nsRestyleHint(0), nsChangeHint_ReconstructFrame);
|
|
return true;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CharacterDataChanged(nsIContent* aContent,
|
|
CharacterDataChangeInfo* aInfo)
|
|
{
|
|
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
|
|
|
|
// 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::BeginUpdate() {
|
|
NS_ASSERTION(!nsContentUtils::IsSafeToRunScript(),
|
|
"Someone forgot a script blocker");
|
|
|
|
nsRootPresContext* rootPresContext =
|
|
mPresShell->GetPresContext()->GetRootPresContext();
|
|
if (rootPresContext) {
|
|
rootPresContext->IncrementDOMGeneration();
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
++mUpdateCount;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::EndUpdate()
|
|
{
|
|
#ifdef DEBUG
|
|
NS_ASSERTION(mUpdateCount, "Negative mUpdateCount!");
|
|
--mUpdateCount;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::RecalcQuotesAndCounters()
|
|
{
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
|
|
if (mQuotesDirty) {
|
|
mQuotesDirty = false;
|
|
mQuoteList.RecalcAll();
|
|
}
|
|
|
|
if (mCountersDirty) {
|
|
mCountersDirty = false;
|
|
mCounterManager.RecalcAll();
|
|
}
|
|
|
|
NS_ASSERTION(!mQuotesDirty, "Quotes updates will be lost");
|
|
NS_ASSERTION(!mCountersDirty, "Counter updates will be lost");
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::NotifyCounterStylesAreDirty()
|
|
{
|
|
NS_PRECONDITION(mUpdateCount != 0, "Should be in an update");
|
|
mCounterManager.SetAllDirty();
|
|
CountersDirty();
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::WillDestroyFrameTree()
|
|
{
|
|
#if defined(DEBUG_dbaron_off)
|
|
mCounterManager.Dump();
|
|
#endif
|
|
|
|
mIsDestroyingFrameTree = true;
|
|
|
|
// Prevent frame tree destruction from being O(N^2)
|
|
mQuoteList.Clear();
|
|
mCounterManager.Clear();
|
|
|
|
// Remove our presshell as a style flush observer. But leave
|
|
// RestyleManager::mObservingRefreshDriver true so we don't readd to
|
|
// it even if someone tries to post restyle events on us from this
|
|
// point on for some reason.
|
|
mPresShell->GetPresContext()->RefreshDriver()->
|
|
RemoveStyleFlushObserver(mPresShell);
|
|
|
|
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.
|
|
void
|
|
nsCSSFrameConstructor::GetAlternateTextFor(nsIContent* aContent,
|
|
nsAtom* aTag,
|
|
nsAString& aAltText)
|
|
{
|
|
// The "alt" attribute specifies alternate text that is rendered
|
|
// when the image can not be displayed.
|
|
if (aContent->GetAttr(kNameSpaceID_None, nsGkAtoms::alt, aAltText)) {
|
|
return;
|
|
}
|
|
|
|
if (nsGkAtoms::input == aTag) {
|
|
// If there's no "alt" attribute, and aContent is an input element, then use
|
|
// the value of the "value" attribute
|
|
if (aContent->GetAttr(kNameSpaceID_None, nsGkAtoms::value, aAltText)) {
|
|
return;
|
|
}
|
|
|
|
// If there's no "value" attribute either, then use the localized string for
|
|
// "Submit" as the alternate text.
|
|
nsContentUtils::GetLocalizedString(nsContentUtils::eFORMS_PROPERTIES,
|
|
"Submit", aAltText);
|
|
}
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::CreateContinuingOuterTableFrame(nsIPresShell* aPresShell,
|
|
nsPresContext* aPresContext,
|
|
nsIFrame* aFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
nsTableWrapperFrame* newFrame = NS_NewTableWrapperFrame(aPresShell, aStyleContext);
|
|
|
|
newFrame->Init(aContent, aParentFrame, aFrame);
|
|
|
|
// Create a continuing inner table frame, and if there's a caption then
|
|
// replicate the caption
|
|
nsFrameItems newChildFrames;
|
|
|
|
nsIFrame* childFrame = aFrame->PrincipalChildList().FirstChild();
|
|
if (childFrame) {
|
|
nsIFrame* continuingTableFrame =
|
|
CreateContinuingFrame(aPresContext, childFrame, newFrame);
|
|
newChildFrames.AddChild(continuingTableFrame);
|
|
|
|
NS_ASSERTION(!childFrame->GetNextSibling(),"there can be only one inner table frame");
|
|
}
|
|
|
|
// Set the table wrapper's initial child list
|
|
newFrame->SetInitialChildList(kPrincipalList, newChildFrames);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::CreateContinuingTableFrame(nsIPresShell* aPresShell,
|
|
nsIFrame* aFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
nsTableFrame* newFrame = NS_NewTableFrame(aPresShell, aStyleContext);
|
|
|
|
newFrame->Init(aContent, aParentFrame, aFrame);
|
|
|
|
// Replicate any header/footer frames
|
|
nsFrameItems 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;
|
|
nsFrameItems childItems;
|
|
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext,
|
|
GetAbsoluteContainingBlock(newFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(newFrame, ABS_POS),
|
|
nullptr);
|
|
state.mCreatingExtraFrames = true;
|
|
|
|
nsStyleContext* const headerFooterStyleContext = rowGroupFrame->StyleContext();
|
|
headerFooterFrame = static_cast<nsTableRowGroupFrame*>
|
|
(NS_NewTableRowGroupFrame(aPresShell, headerFooterStyleContext));
|
|
|
|
nsIContent* headerFooter = rowGroupFrame->GetContent();
|
|
headerFooterFrame->Init(headerFooter, newFrame, nullptr);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState;
|
|
MakeTablePartAbsoluteContainingBlockIfNeeded(state,
|
|
headerFooterStyleContext->StyleDisplay(),
|
|
absoluteSaveState,
|
|
headerFooterFrame);
|
|
|
|
ProcessChildren(state, headerFooter, rowGroupFrame->StyleContext(),
|
|
headerFooterFrame, true, childItems, false,
|
|
nullptr);
|
|
NS_ASSERTION(state.mFloatedItems.IsEmpty(), "unexpected floated element");
|
|
headerFooterFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
headerFooterFrame->SetRepeatable(true);
|
|
|
|
// Table specific initialization
|
|
headerFooterFrame->InitRepeatedFrame(rowGroupFrame);
|
|
|
|
// XXX Deal with absolute and fixed frames...
|
|
childFrames.AddChild(headerFooterFrame);
|
|
}
|
|
}
|
|
|
|
// Set the table frame's initial child list
|
|
newFrame->SetInitialChildList(kPrincipalList, childFrames);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::CreateContinuingFrame(nsPresContext* aPresContext,
|
|
nsIFrame* aFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
bool aIsFluid)
|
|
{
|
|
nsIPresShell* shell = aPresContext->PresShell();
|
|
nsStyleContext* styleContext = aFrame->StyleContext();
|
|
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();
|
|
|
|
NS_ASSERTION(aFrame->GetSplittableType() != NS_FRAME_NOT_SPLITTABLE,
|
|
"why CreateContinuingFrame for a non-splittable frame?");
|
|
|
|
if (LayoutFrameType::Text == frameType) {
|
|
newFrame = NS_NewContinuingTextFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Inline == frameType) {
|
|
newFrame = NS_NewInlineFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Block == frameType) {
|
|
MOZ_ASSERT(!aFrame->IsTableCaption(),
|
|
"no support for fragmenting table captions yet");
|
|
newFrame = NS_NewBlockFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
#ifdef MOZ_XUL
|
|
} else if (LayoutFrameType::XULLabel == frameType) {
|
|
newFrame = NS_NewXULLabelFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
#endif
|
|
} else if (LayoutFrameType::ColumnSet == frameType) {
|
|
MOZ_ASSERT(!aFrame->IsTableCaption(),
|
|
"no support for fragmenting table captions yet");
|
|
newFrame = NS_NewColumnSetFrame(shell, styleContext, nsFrameState(0));
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Page == frameType) {
|
|
nsContainerFrame* canvasFrame;
|
|
newFrame = ConstructPageFrame(shell, aParentFrame, aFrame, canvasFrame);
|
|
} else if (LayoutFrameType::TableWrapper == frameType) {
|
|
newFrame =
|
|
CreateContinuingOuterTableFrame(shell, aPresContext, aFrame, aParentFrame,
|
|
content, styleContext);
|
|
|
|
} else if (LayoutFrameType::Table == frameType) {
|
|
newFrame =
|
|
CreateContinuingTableFrame(shell, aFrame, aParentFrame,
|
|
content, styleContext);
|
|
|
|
} else if (LayoutFrameType::TableRowGroup == frameType) {
|
|
newFrame = NS_NewTableRowGroupFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
if (newFrame->GetStateBits() & NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN) {
|
|
nsTableFrame::RegisterPositionedTablePart(newFrame);
|
|
}
|
|
} else if (LayoutFrameType::TableRow == frameType) {
|
|
nsTableRowFrame* rowFrame = NS_NewTableRowFrame(shell, styleContext);
|
|
|
|
rowFrame->Init(content, aParentFrame, aFrame);
|
|
if (rowFrame->GetStateBits() & NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN) {
|
|
nsTableFrame::RegisterPositionedTablePart(rowFrame);
|
|
}
|
|
|
|
// Create a continuing frame for each table cell frame
|
|
nsFrameItems newChildList;
|
|
nsIFrame* cellFrame = aFrame->PrincipalChildList().FirstChild();
|
|
while (cellFrame) {
|
|
// See if it's a table cell frame
|
|
if (IS_TABLE_CELL(cellFrame->Type())) {
|
|
nsIFrame* continuingCellFrame =
|
|
CreateContinuingFrame(aPresContext, cellFrame, rowFrame);
|
|
newChildList.AddChild(continuingCellFrame);
|
|
}
|
|
cellFrame = cellFrame->GetNextSibling();
|
|
}
|
|
|
|
rowFrame->SetInitialChildList(kPrincipalList, newChildList);
|
|
newFrame = rowFrame;
|
|
|
|
} else if (IS_TABLE_CELL(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(shell, styleContext, tableFrame);
|
|
|
|
cellFrame->Init(content, aParentFrame, aFrame);
|
|
if (cellFrame->GetStateBits() & NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN) {
|
|
nsTableFrame::RegisterPositionedTablePart(cellFrame);
|
|
}
|
|
|
|
// Create a continuing area frame
|
|
nsIFrame* blockFrame = aFrame->PrincipalChildList().FirstChild();
|
|
nsIFrame* continuingBlockFrame =
|
|
CreateContinuingFrame(aPresContext, blockFrame,
|
|
static_cast<nsContainerFrame*>(cellFrame));
|
|
|
|
SetInitialSingleChild(cellFrame, continuingBlockFrame);
|
|
newFrame = cellFrame;
|
|
} else if (LayoutFrameType::Line == frameType) {
|
|
newFrame = NS_NewFirstLineFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Letter == frameType) {
|
|
newFrame = NS_NewFirstLetterFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Image == frameType) {
|
|
newFrame = NS_NewImageFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::ImageControl == frameType) {
|
|
newFrame = NS_NewImageControlFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Placeholder == frameType) {
|
|
// create a continuing out of flow frame
|
|
nsIFrame* oofFrame = nsPlaceholderFrame::GetRealFrameForPlaceholder(aFrame);
|
|
nsIFrame* oofContFrame =
|
|
CreateContinuingFrame(aPresContext, oofFrame, aParentFrame);
|
|
newFrame =
|
|
CreatePlaceholderFrameFor(shell, content, oofContFrame,
|
|
aParentFrame, aFrame,
|
|
aFrame->GetStateBits() & PLACEHOLDER_TYPE_MASK);
|
|
} else if (LayoutFrameType::FieldSet == frameType) {
|
|
nsContainerFrame* fieldset = NS_NewFieldSetFrame(shell, styleContext);
|
|
|
|
fieldset->Init(content, aParentFrame, aFrame);
|
|
|
|
// Create a continuing area frame
|
|
// XXXbz we really shouldn't have to do this by hand!
|
|
nsContainerFrame* blockFrame = GetFieldSetBlockFrame(aFrame);
|
|
if (blockFrame) {
|
|
nsIFrame* continuingBlockFrame =
|
|
CreateContinuingFrame(aPresContext, blockFrame, fieldset);
|
|
// Set the fieldset's initial child list
|
|
SetInitialSingleChild(fieldset, continuingBlockFrame);
|
|
} else {
|
|
MOZ_ASSERT(aFrame->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER,
|
|
"FieldSet block may only be null for overflow containers");
|
|
}
|
|
newFrame = fieldset;
|
|
} else if (LayoutFrameType::Legend == frameType) {
|
|
newFrame = NS_NewLegendFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::FlexContainer == frameType) {
|
|
newFrame = NS_NewFlexContainerFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::GridContainer == frameType) {
|
|
newFrame = NS_NewGridContainerFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Ruby == frameType) {
|
|
newFrame = NS_NewRubyFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::RubyBaseContainer == frameType) {
|
|
newFrame = NS_NewRubyBaseContainerFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::RubyTextContainer == frameType) {
|
|
newFrame = NS_NewRubyTextContainerFrame(shell, styleContext);
|
|
newFrame->Init(content, aParentFrame, aFrame);
|
|
} else if (LayoutFrameType::Details == frameType) {
|
|
newFrame = NS_NewDetailsFrame(shell, styleContext);
|
|
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);
|
|
}
|
|
|
|
// A continuation of generated content is also generated content
|
|
if (aFrame->GetStateBits() & NS_FRAME_GENERATED_CONTENT) {
|
|
newFrame->AddStateBits(NS_FRAME_GENERATED_CONTENT);
|
|
}
|
|
|
|
// A continuation of nsIAnonymousContentCreator content is also
|
|
// nsIAnonymousContentCreator created content
|
|
if (aFrame->GetStateBits() & NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT) {
|
|
newFrame->AddStateBits(NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT);
|
|
}
|
|
|
|
// A continuation of an out-of-flow is also an out-of-flow
|
|
if (aFrame->GetStateBits() & NS_FRAME_OUT_OF_FLOW) {
|
|
newFrame->AddStateBits(NS_FRAME_OUT_OF_FLOW);
|
|
}
|
|
|
|
if (nextInFlow) {
|
|
nextInFlow->SetPrevInFlow(newFrame);
|
|
newFrame->SetNextInFlow(nextInFlow);
|
|
} else if (nextContinuation) {
|
|
nextContinuation->SetPrevContinuation(newFrame);
|
|
newFrame->SetNextContinuation(nextContinuation);
|
|
}
|
|
|
|
MOZ_ASSERT(!newFrame->GetNextSibling(), "unexpected sibling");
|
|
return newFrame;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
nsFrameItems fixedPlaceholders;
|
|
nsIFrame* firstFixed = prevPageContentFrame->GetChildList(nsIFrame::kFixedList).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.
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext,
|
|
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();
|
|
nsStyleContext* styleContext =
|
|
nsLayoutUtils::GetStyleFrame(content)->StyleContext();
|
|
AutoFrameConstructionItemList items(this);
|
|
AddFrameConstructionItemsInternal(state, content, canvasFrame,
|
|
content->NodeInfo()->NameAtom(),
|
|
content->GetNameSpaceID(),
|
|
true,
|
|
styleContext,
|
|
ITEM_ALLOW_XBL_BASE |
|
|
ITEM_ALLOW_PAGE_BREAK,
|
|
nullptr, 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(kPrincipalList, fixedPlaceholders);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCSSFrameConstructor::InsertionPoint
|
|
nsCSSFrameConstructor::GetInsertionPoint(nsIContent* aContainer,
|
|
nsIContent* aChild)
|
|
{
|
|
nsBindingManager* bindingManager = mDocument->BindingManager();
|
|
|
|
nsIContent* insertionElement;
|
|
if (aChild) {
|
|
// We've got an explicit insertion child. Check to see if it's
|
|
// anonymous.
|
|
if (aChild->GetBindingParent() == aContainer) {
|
|
// This child content is anonymous. Don't use the insertion
|
|
// point, since that's only for the explicit kids.
|
|
return InsertionPoint(GetContentInsertionFrameFor(aContainer), aContainer);
|
|
}
|
|
|
|
if (nsContentUtils::HasDistributedChildren(aContainer) ||
|
|
ShadowRoot::FromNode(aContainer)) {
|
|
// The container distributes nodes or is a shadow root, use the frame of
|
|
// the flattened tree parent.
|
|
//
|
|
// It may be the case that the node is distributed but not matched to any
|
|
// insertion points, so there is no flattened parent.
|
|
nsIContent* flattenedParent = aChild->GetFlattenedTreeParent();
|
|
if (flattenedParent) {
|
|
return InsertionPoint(GetContentInsertionFrameFor(flattenedParent),
|
|
flattenedParent);
|
|
}
|
|
return InsertionPoint();
|
|
}
|
|
|
|
insertionElement = bindingManager->FindNestedInsertionPoint(aContainer, aChild);
|
|
} else {
|
|
if (nsContentUtils::HasDistributedChildren(aContainer)) {
|
|
// The container distributes nodes to shadow DOM insertion points.
|
|
// Return with aMultiple set to true to induce callers to insert children
|
|
// individually into the node's flattened tree parent.
|
|
return InsertionPoint(nullptr, nullptr, true);
|
|
}
|
|
|
|
bool multiple;
|
|
insertionElement = bindingManager->FindNestedSingleInsertionPoint(aContainer, &multiple);
|
|
if (multiple) {
|
|
return InsertionPoint(nullptr, nullptr, true);
|
|
}
|
|
}
|
|
|
|
if (!insertionElement) {
|
|
// The FindNested{,Single}InsertionPoint methods return null in the case
|
|
// that there is a binding with anonymous content but no insertion point.
|
|
// In that case the element doesn't belong in the flattened tree, and we
|
|
// don't want to render it.
|
|
return InsertionPoint();
|
|
}
|
|
|
|
InsertionPoint insertion(GetContentInsertionFrameFor(insertionElement),
|
|
insertionElement);
|
|
|
|
// Fieldset frames have multiple normal flow child frame lists so handle it
|
|
// the same as if it had multiple content insertion points.
|
|
if (insertion.mParentFrame && insertion.mParentFrame->IsFieldSetFrame()) {
|
|
insertion.mMultiple = true;
|
|
}
|
|
|
|
// A details frame moves the first summary frame to be its first child, so we
|
|
// treat it as if it has multiple content insertion points.
|
|
if (insertion.mParentFrame && insertion.mParentFrame->IsDetailsFrame()) {
|
|
insertion.mMultiple = true;
|
|
}
|
|
|
|
return insertion;
|
|
}
|
|
|
|
// 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
|
|
DefinitelyEqualURIsAndPrincipal(mozilla::css::URLValue* aURI1,
|
|
mozilla::css::URLValue* aURI2)
|
|
{
|
|
return aURI1 == aURI2 ||
|
|
(aURI1 && aURI2 && aURI1->DefinitelyEqualURIsAndPrincipal(*aURI2));
|
|
}
|
|
|
|
nsStyleContext*
|
|
nsCSSFrameConstructor::MaybeRecreateFramesForElement(Element* aElement)
|
|
{
|
|
RefPtr<nsStyleContext> oldContext = GetDisplayNoneStyleFor(aElement);
|
|
StyleDisplay oldDisplay = StyleDisplay::None;
|
|
if (!oldContext) {
|
|
oldContext = GetDisplayContentsStyleFor(aElement);
|
|
if (!oldContext) {
|
|
return nullptr;
|
|
}
|
|
oldDisplay = StyleDisplay::Contents;
|
|
}
|
|
|
|
// The parent has a frame, so try resolving a new context.
|
|
RefPtr<nsStyleContext> newContext = mPresShell->StyleSet()->
|
|
ResolveStyleFor(aElement, oldContext->AsGecko()->GetParent(),
|
|
LazyComputeBehavior::Assert);
|
|
|
|
if (oldDisplay == StyleDisplay::None) {
|
|
ChangeRegisteredDisplayNoneStyleFor(aElement, newContext);
|
|
} else {
|
|
ChangeRegisteredDisplayContentsStyleFor(aElement, newContext);
|
|
}
|
|
|
|
const nsStyleDisplay* disp = newContext->StyleDisplay();
|
|
if (oldDisplay == disp->mDisplay) {
|
|
// We can skip trying to recreate frames here, but only if our style
|
|
// context does not have a binding URI that differs from our old one.
|
|
// Otherwise, we should try to recreate, because we may want to apply the
|
|
// new binding
|
|
if (!disp->mBinding) {
|
|
return newContext;
|
|
}
|
|
const nsStyleDisplay* oldDisp = oldContext->PeekStyleDisplay();
|
|
if (oldDisp &&
|
|
DefinitelyEqualURIsAndPrincipal(disp->mBinding, oldDisp->mBinding)) {
|
|
return newContext;
|
|
}
|
|
}
|
|
|
|
RecreateFramesForContent(aElement, InsertionKind::Sync);
|
|
return nullptr;
|
|
}
|
|
|
|
static bool
|
|
IsWhitespaceFrame(nsIFrame* aFrame)
|
|
{
|
|
MOZ_ASSERT(aFrame, "invalid argument");
|
|
return aFrame->IsTextFrame() && aFrame->GetContent()->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
static nsIFrame*
|
|
FindFirstNonWhitespaceChild(nsIFrame* aParentFrame)
|
|
{
|
|
nsIFrame* f = aParentFrame->PrincipalChildList().FirstChild();
|
|
while (f && IsWhitespaceFrame(f)) {
|
|
f = f->GetNextSibling();
|
|
}
|
|
return f;
|
|
}
|
|
|
|
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)
|
|
{
|
|
NS_PRECONDITION(aFrame, "Must have a frame");
|
|
NS_PRECONDITION(aFrame->GetParent(), "Frame shouldn't be root");
|
|
NS_PRECONDITION(aFrame == aFrame->FirstContinuation(),
|
|
"aFrame not the result of GetPrimaryFrame()?");
|
|
|
|
if (IsFramePartOfIBSplit(aFrame)) {
|
|
// The removal functions can't handle removal of an {ib} split directly; we
|
|
// need to rebuild the containing block.
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval: "
|
|
"frame=");
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
printf(" is ib-split\n");
|
|
}
|
|
#endif
|
|
|
|
ReframeContainingBlock(aFrame);
|
|
return true;
|
|
}
|
|
|
|
nsContainerFrame* insertionFrame = aFrame->GetContentInsertionFrame();
|
|
if (insertionFrame && insertionFrame->IsLegendFrame() &&
|
|
aFrame->GetParent()->IsFieldSetFrame()) {
|
|
RecreateFramesForContent(aFrame->GetParent()->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
nsIFrame* inFlowFrame =
|
|
(aFrame->GetStateBits() & 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 (parent && parent->IsDetailsFrame()) {
|
|
HTMLSummaryElement* summary =
|
|
HTMLSummaryElement::FromContent(aFrame->GetContent());
|
|
DetailsFrame* detailsFrame = static_cast<DetailsFrame*>(parent);
|
|
|
|
// Unlike adding summary element cases, we need to check children of the
|
|
// parent details frame since at this moment the summary element has been
|
|
// already removed from the parent details element's child list.
|
|
if (summary && detailsFrame->HasMainSummaryFrame(aFrame)) {
|
|
// When removing a summary, we should reframe the parent details frame to
|
|
// ensure that another summary is used or the default summary is
|
|
// generated.
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Now check for possibly needing to reconstruct due to a pseudo parent
|
|
// For the case of ruby pseudo parent, effectively, only pseudo rb/rt frame
|
|
// need to be checked here, since all other types of parent will be catched
|
|
// by "Check ruby containers" section below.
|
|
if (IsTableOrRubyPseudo(parent)) {
|
|
if (FindFirstNonWhitespaceChild(parent) == inFlowFrame ||
|
|
!FindNextNonWhitespaceSibling(inFlowFrame->LastContinuation()) ||
|
|
// If it is a whitespace, and is the only child of the parent, the
|
|
// pseudo parent was created for the space, and should now be removed.
|
|
(IsWhitespaceFrame(aFrame) &&
|
|
parent->PrincipalChildList().OnlyChild()) ||
|
|
// If we're a table-column-group, then the OnlyChild check above is
|
|
// not going to catch cases when we're the first child.
|
|
(inFlowFrame->IsTableColGroupFrame() &&
|
|
parent->GetChildList(nsIFrame::kColGroupList).FirstChild() == inFlowFrame) ||
|
|
// Similar if we're a table-caption.
|
|
(inFlowFrame->IsTableCaption() &&
|
|
parent->GetChildList(nsIFrame::kCaptionList).FirstChild() == inFlowFrame)) {
|
|
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(!IsTableOrRubyPseudo(inFlowFrame), "Shouldn't happen here");
|
|
// 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 && IsTableOrRubyPseudo(nextSibling)) {
|
|
nsIFrame* prevSibling = FindPreviousNonWhitespaceSibling(inFlowFrame);
|
|
if (prevSibling && IsTableOrRubyPseudo(prevSibling)) {
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval: "
|
|
"frame=");
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
printf(" has a table pseudo next sibling of different type and a "
|
|
"table pseudo prevsibling\n");
|
|
}
|
|
#endif
|
|
// 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.
|
|
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.
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Might need to reconstruct things if the removed frame's nextSibling is an
|
|
// anonymous flex item. The removed frame might've been what divided two
|
|
// runs of inline content into two anonymous flex items, which would now
|
|
// need to be merged.
|
|
// NOTE: It's fine that we've advanced nextSibling past whitespace (up above);
|
|
// we're only interested in anonymous flex items here, and those can never
|
|
// be adjacent to whitespace, since they absorb contiguous runs of inline
|
|
// non-replaced content (including whitespace).
|
|
if (nextSibling && IsAnonymousFlexOrGridItem(nextSibling)) {
|
|
AssertAnonymousFlexOrGridItemParent(nextSibling, parent);
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval: "
|
|
"frame=");
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
printf(" has an anonymous flex item as its next sibling\n");
|
|
}
|
|
#endif // DEBUG
|
|
// Recreate frames for the flex container (the removed frame's parent)
|
|
RecreateFramesForContent(parent->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Might need to reconstruct things if the removed frame's nextSibling is
|
|
// null and its parent is an anonymous flex item. (This might be the last
|
|
// remaining child of that anonymous flex item, which can then go away.)
|
|
if (!nextSibling && IsAnonymousFlexOrGridItem(parent)) {
|
|
AssertAnonymousFlexOrGridItemParent(parent, parent->GetParent());
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval: "
|
|
"frame=");
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
printf(" has an anonymous flex item as its parent\n");
|
|
}
|
|
#endif // DEBUG
|
|
// Recreate frames for the flex container (the removed frame's grandparent)
|
|
RecreateFramesForContent(parent->GetParent()->GetContent(),
|
|
InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
#ifdef MOZ_XUL
|
|
if (aFrame->IsPopupSetFrame()) {
|
|
nsIRootBox* rootBox = nsIRootBox::GetRootBox(mPresShell);
|
|
if (rootBox && rootBox->GetPopupSetFrame() == aFrame) {
|
|
ReconstructDocElementHierarchy(InsertionKind::Async);
|
|
return true;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// 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()))) {
|
|
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;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::MaybeRecreateContainerForFrameRemoval: "
|
|
"frame=");
|
|
nsFrame::ListTag(stdout, parent);
|
|
printf(" is ib-split\n");
|
|
}
|
|
#endif
|
|
|
|
ReframeContainingBlock(parent);
|
|
return true;
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
// Is the frame ib-split? If so, we need to reframe the containing
|
|
// block *here*, rather than trying to remove and re-insert the
|
|
// content (which would otherwise result in *two* nested reframe
|
|
// containing block from ContentRemoved() and ContentInserted(),
|
|
// below!). We'd really like to optimize away one of those
|
|
// containing block reframes, hence the code here.
|
|
|
|
nsIFrame* frame = aContent->GetPrimaryFrame();
|
|
if (frame && frame->IsFrameOfType(nsIFrame::eMathML)) {
|
|
// Reframe the topmost MathML element to prevent exponential blowup
|
|
// (see bug 397518)
|
|
while (true) {
|
|
nsIContent* parentContent = aContent->GetParent();
|
|
nsIFrame* parentContentFrame = parentContent->GetPrimaryFrame();
|
|
if (!parentContentFrame || !parentContentFrame->IsFrameOfType(nsIFrame::eMathML))
|
|
break;
|
|
aContent = parentContent;
|
|
frame = parentContentFrame;
|
|
}
|
|
}
|
|
|
|
if (frame) {
|
|
nsIFrame* nonGeneratedAncestor = nsLayoutUtils::GetNonGeneratedAncestor(frame);
|
|
if (nonGeneratedAncestor->GetContent() != aContent) {
|
|
return RecreateFramesForContent(nonGeneratedAncestor->GetContent(),
|
|
InsertionKind::Async);
|
|
}
|
|
|
|
if (frame->GetStateBits() & NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT) {
|
|
// Recreate the frames for the entire nsIAnonymousContentCreator tree
|
|
// since |frame| or one of its descendants may need an nsStyleContext
|
|
// that associates it to a CSS pseudo-element, and only the
|
|
// nsIAnonymousContentCreator that created this content knows how to make
|
|
// that happen.
|
|
nsIAnonymousContentCreator* acc = nullptr;
|
|
nsIFrame* ancestor = nsLayoutUtils::GetParentOrPlaceholderFor(frame);
|
|
while (!(acc = do_QueryFrame(ancestor))) {
|
|
ancestor = nsLayoutUtils::GetParentOrPlaceholderFor(ancestor);
|
|
}
|
|
NS_ASSERTION(acc, "Where is the nsIAnonymousContentCreator? We may fail "
|
|
"to recreate its content correctly");
|
|
// nsSVGUseFrame is special, and we know this is unnecessary for it.
|
|
if (!ancestor->IsSVGUseFrame()) {
|
|
NS_ASSERTION(aContent->IsInNativeAnonymousSubtree(),
|
|
"Why is NS_FRAME_ANONYMOUSCONTENTCREATOR_CONTENT set?");
|
|
return RecreateFramesForContent(ancestor->GetContent(),
|
|
InsertionKind::Async);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
nsINode* containerNode = aContent->GetParentNode();
|
|
// XXXbz how can containerNode be null here?
|
|
if (containerNode) {
|
|
// Before removing the frames associated with the content object,
|
|
// ask them to save their state onto a temporary state object.
|
|
CaptureStateForFramesOf(aContent, mTempFrameTreeState);
|
|
|
|
// Need the nsIContent parent, which might be null here, since we need to
|
|
// pass it to ContentInserted and ContentRemoved.
|
|
nsIContent* container = aContent->GetParent();
|
|
|
|
// Remove the frames associated with the content object.
|
|
nsIContent* nextSibling = aContent->IsRootOfAnonymousSubtree() ?
|
|
nullptr : aContent->GetNextSibling();
|
|
bool didReconstruct =
|
|
ContentRemoved(container, 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(),
|
|
nsRestyleHint(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(container, aContent, aContent->GetNextSibling(),
|
|
mTempFrameTreeState,
|
|
aInsertionKind,
|
|
nullptr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::DestroyFramesFor(Element* aElement)
|
|
{
|
|
MOZ_ASSERT(aElement && aElement->GetParentNode());
|
|
|
|
nsIContent* nextSibling =
|
|
aElement->IsRootOfAnonymousSubtree() ? nullptr : aElement->GetNextSibling();
|
|
|
|
CaptureStateForFramesOf(aElement, mTempFrameTreeState);
|
|
return ContentRemoved(aElement->GetParent(),
|
|
aElement,
|
|
nextSibling,
|
|
REMOVE_FOR_RECONSTRUCTION);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// Block frame construction code
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::GetFirstLetterStyle(nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aContent) {
|
|
return mPresShell->StyleSet()->
|
|
ResolvePseudoElementStyle(aContent->AsElement(),
|
|
CSSPseudoElementType::firstLetter,
|
|
aStyleContext,
|
|
nullptr);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext>
|
|
nsCSSFrameConstructor::GetFirstLineStyle(nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
if (aContent) {
|
|
return mPresShell->StyleSet()->
|
|
ResolvePseudoElementStyle(aContent->AsElement(),
|
|
CSSPseudoElementType::firstLine,
|
|
aStyleContext,
|
|
nullptr);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// Predicate to see if a given content (block element) has
|
|
// first-letter style applied to it.
|
|
bool
|
|
nsCSSFrameConstructor::ShouldHaveFirstLetterStyle(nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
return nsLayoutUtils::HasPseudoStyle(aContent, aStyleContext,
|
|
CSSPseudoElementType::firstLetter,
|
|
mPresShell->GetPresContext());
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::HasFirstLetterStyle(nsIFrame* aBlockFrame)
|
|
{
|
|
NS_PRECONDITION(aBlockFrame, "Need a frame");
|
|
NS_ASSERTION(nsLayoutUtils::GetAsBlock(aBlockFrame),
|
|
"Not a block frame?");
|
|
|
|
return (aBlockFrame->GetStateBits() & NS_BLOCK_HAS_FIRST_LETTER_STYLE) != 0;
|
|
}
|
|
|
|
bool
|
|
nsCSSFrameConstructor::ShouldHaveFirstLineStyle(nsIContent* aContent,
|
|
nsStyleContext* aStyleContext)
|
|
{
|
|
bool hasFirstLine =
|
|
nsLayoutUtils::HasPseudoStyle(aContent, aStyleContext,
|
|
CSSPseudoElementType::firstLine,
|
|
mPresShell->GetPresContext());
|
|
if (hasFirstLine) {
|
|
// But disable for fieldsets
|
|
int32_t namespaceID;
|
|
nsAtom* tag = mDocument->BindingManager()->ResolveTag(aContent,
|
|
&namespaceID);
|
|
// This check must match the one in FindHTMLData.
|
|
hasFirstLine = tag != nsGkAtoms::fieldset ||
|
|
namespaceID != kNameSpaceID_XHTML;
|
|
}
|
|
|
|
return hasFirstLine;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ShouldHaveSpecialBlockStyle(nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
bool* aHaveFirstLetterStyle,
|
|
bool* aHaveFirstLineStyle)
|
|
{
|
|
*aHaveFirstLetterStyle =
|
|
ShouldHaveFirstLetterStyle(aContent, aStyleContext);
|
|
*aHaveFirstLineStyle =
|
|
ShouldHaveFirstLineStyle(aContent, aStyleContext);
|
|
}
|
|
|
|
/* static */
|
|
const nsCSSFrameConstructor::PseudoParentData
|
|
nsCSSFrameConstructor::sPseudoParentData[eParentTypeCount] = {
|
|
{ // Cell
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRow),
|
|
&nsCSSFrameConstructor::ConstructTableCell),
|
|
&nsCSSAnonBoxes::tableCell
|
|
},
|
|
{ // Row
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRowGroup),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup),
|
|
&nsCSSAnonBoxes::tableRow
|
|
},
|
|
{ // Row group
|
|
FULL_CTOR_FCDATA(FCDATA_IS_TABLE_PART | FCDATA_SKIP_FRAMESET |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
&nsCSSFrameConstructor::ConstructTableRowOrRowGroup),
|
|
&nsCSSAnonBoxes::tableRowGroup
|
|
},
|
|
{ // Column group
|
|
FCDATA_DECL(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 style contexts.
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
|
|
NS_NewTableColGroupFrame),
|
|
&nsCSSAnonBoxes::tableColGroup
|
|
},
|
|
{ // Table
|
|
FULL_CTOR_FCDATA(FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX,
|
|
&nsCSSFrameConstructor::ConstructTable),
|
|
&nsCSSAnonBoxes::table
|
|
},
|
|
{ // Ruby
|
|
FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_SKIP_FRAMESET,
|
|
NS_NewRubyFrame),
|
|
&nsCSSAnonBoxes::ruby
|
|
},
|
|
{ // Ruby Base
|
|
FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyBaseContainer) |
|
|
FCDATA_SKIP_FRAMESET,
|
|
NS_NewRubyBaseFrame),
|
|
&nsCSSAnonBoxes::rubyBase
|
|
},
|
|
{ // Ruby Base Container
|
|
FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby) |
|
|
FCDATA_SKIP_FRAMESET,
|
|
NS_NewRubyBaseContainerFrame),
|
|
&nsCSSAnonBoxes::rubyBaseContainer
|
|
},
|
|
{ // Ruby Text
|
|
FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_LINE_PARTICIPANT |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyTextContainer) |
|
|
FCDATA_SKIP_FRAMESET,
|
|
NS_NewRubyTextFrame),
|
|
&nsCSSAnonBoxes::rubyText
|
|
},
|
|
{ // Ruby Text Container
|
|
FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX |
|
|
FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeRuby) |
|
|
FCDATA_SKIP_FRAMESET,
|
|
NS_NewRubyTextContainerFrame),
|
|
&nsCSSAnonBoxes::rubyTextContainer
|
|
}
|
|
};
|
|
|
|
void
|
|
nsCSSFrameConstructor::CreateNeededAnonFlexOrGridItems(
|
|
nsFrameConstructorState& aState,
|
|
FrameConstructionItemList& aItems,
|
|
nsIFrame* aParentFrame)
|
|
{
|
|
if (aItems.IsEmpty()) {
|
|
return;
|
|
}
|
|
const LayoutFrameType parentType = aParentFrame->Type();
|
|
if (parentType != LayoutFrameType::FlexContainer &&
|
|
parentType != LayoutFrameType::GridContainer) {
|
|
return;
|
|
}
|
|
|
|
const bool isWebkitBox = IsFlexContainerForLegacyBox(aParentFrame);
|
|
FCItemIterator iter(aItems);
|
|
do {
|
|
// Advance iter past children that don't want to be wrapped
|
|
if (iter.SkipItemsThatDontNeedAnonFlexOrGridItem(aState, isWebkitBox)) {
|
|
// 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, isWebkitBox);
|
|
|
|
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, isWebkitBox),
|
|
"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, isWebkitBox);
|
|
|
|
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|.
|
|
nsAtom* pseudoType = (aParentFrame->IsFlexContainerFrame())
|
|
? nsCSSAnonBoxes::anonymousFlexItem
|
|
: nsCSSAnonBoxes::anonymousGridItem;
|
|
nsStyleContext* parentStyle = aParentFrame->StyleContext();
|
|
nsIContent* parentContent = aParentFrame->GetContent();
|
|
already_AddRefed<nsStyleContext> wrapperStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(pseudoType,
|
|
parentStyle);
|
|
|
|
static const FrameConstructionData sBlockFormattingContextFCData =
|
|
FCDATA_DECL(FCDATA_SKIP_FRAMESET |
|
|
FCDATA_USE_CHILD_ITEMS |
|
|
FCDATA_IS_WRAPPER_ANON_BOX,
|
|
NS_NewBlockFormattingContext);
|
|
|
|
FrameConstructionItem* newItem =
|
|
new (this) FrameConstructionItem(&sBlockFormattingContextFCData,
|
|
// Use the content of our parent frame
|
|
parentContent,
|
|
// Lie about the tag; it doesn't matter anyway
|
|
pseudoType,
|
|
iter.item().mNameSpaceID,
|
|
// no pending binding
|
|
nullptr,
|
|
wrapperStyle,
|
|
true, nullptr);
|
|
|
|
newItem->mIsAllInline = newItem->mHasInlineEnds =
|
|
newItem->mStyleContext->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.SetParentHasNoXBLChildren(
|
|
aItems.ParentHasNoXBLChildren());
|
|
|
|
// 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(nsStyleDisplay::IsRubyDisplayType(aPrevDisplay) &&
|
|
nsStyleDisplay::IsRubyDisplayType(aNextDisplay));
|
|
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().mStyleContext->StyleDisplay()->mDisplay,
|
|
aEndIter.item().mStyleContext->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,
|
|
nsStyleContext* 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,
|
|
nsStyleContext* 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)) {
|
|
// 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();
|
|
nsStyleContext* parentStyle = aParentFrame->StyleContext();
|
|
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;
|
|
}
|
|
}
|
|
|
|
nsStyleContext* parentStyle = aParentFrame->StyleContext();
|
|
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,
|
|
nsStyleContext* aParentStyle,
|
|
ParentType aWrapperType,
|
|
FCItemIterator& aIter,
|
|
const FCItemIterator& aEndIter)
|
|
{
|
|
const PseudoParentData& pseudoData = sPseudoParentData[aWrapperType];
|
|
nsAtom* pseudoType = *pseudoData.mPseudoType;
|
|
StyleDisplay parentDisplay = aParentStyle->StyleDisplay()->mDisplay;
|
|
|
|
if (pseudoType == nsCSSAnonBoxes::table &&
|
|
(parentDisplay == StyleDisplay::Inline ||
|
|
parentDisplay == StyleDisplay::RubyBase ||
|
|
parentDisplay == StyleDisplay::RubyText)) {
|
|
pseudoType = nsCSSAnonBoxes::inlineTable;
|
|
}
|
|
|
|
already_AddRefed<nsStyleContext> wrapperStyle;
|
|
if (pseudoData.mFCData.mBits & FCDATA_IS_WRAPPER_ANON_BOX) {
|
|
wrapperStyle =
|
|
mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(pseudoType,
|
|
aParentStyle);
|
|
} else {
|
|
wrapperStyle =
|
|
mPresShell->StyleSet()->ResolveNonInheritingAnonymousBoxStyle(pseudoType);
|
|
}
|
|
|
|
FrameConstructionItem* newItem =
|
|
new (this) FrameConstructionItem(&pseudoData.mFCData,
|
|
// Use the content of our parent frame
|
|
aParentContent,
|
|
// Lie about the tag; it doesn't matter anyway
|
|
pseudoType,
|
|
// The namespace does matter, however; it needs
|
|
// to match that of our first child item to
|
|
// match the old behavior
|
|
aIter.item().mNameSpaceID,
|
|
// no pending binding
|
|
nullptr,
|
|
wrapperStyle,
|
|
true, nullptr);
|
|
|
|
const nsStyleDisplay* disp = newItem->mStyleContext->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 = newItem->mHasInlineEnds =
|
|
disp->IsInlineOutsideStyle();
|
|
|
|
bool isRuby = disp->IsRubyDisplayType();
|
|
// All types of ruby frames need a block frame to provide line layout,
|
|
// hence they are always line participant.
|
|
newItem->mIsLineParticipant = isRuby;
|
|
|
|
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.SetParentHasNoXBLChildren(
|
|
aIter.List()->ParentHasNoXBLChildren());
|
|
|
|
// 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().mStyleContext->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];
|
|
already_AddRefed<nsStyleContext> pseudoStyle = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(*pseudoData.mPseudoType,
|
|
aParentFrame->StyleContext());
|
|
FrameConstructionItem* newItem =
|
|
new (this) FrameConstructionItem(&pseudoData.mFCData,
|
|
// Use the content of the parent frame
|
|
aParentFrame->GetContent(),
|
|
// Tag type
|
|
*pseudoData.mPseudoType,
|
|
// Use the namespace of the rtc frame
|
|
iter.item().mNameSpaceID,
|
|
// no pending binding
|
|
nullptr,
|
|
pseudoStyle,
|
|
true, nullptr);
|
|
newItem->mIsAllInline = true;
|
|
newItem->mChildItems.SetParentHasNoXBLChildren(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)
|
|
{
|
|
LayoutFrameType containerType = aContainerFrame->Type();
|
|
MOZ_ASSERT(containerType == LayoutFrameType::FlexContainer ||
|
|
containerType == LayoutFrameType::GridContainer);
|
|
|
|
// Any line-participant frames (e.g. text) definitely want to be wrapped in
|
|
// an anonymous flex/grid item.
|
|
if (aFrame->IsFrameOfType(nsIFrame::eLineParticipant)) {
|
|
return true;
|
|
}
|
|
|
|
// If the container is a -webkit-box/-webkit-inline-box, then placeholders
|
|
// also need to be wrapped, for compatibility.
|
|
if (containerType == LayoutFrameType::FlexContainer &&
|
|
aContainerFrame->HasAnyStateBits(NS_STATE_FLEX_IS_LEGACY_WEBKIT_BOX) &&
|
|
aFrame->IsPlaceholderFrame()) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
VerifyGridFlexContainerChildren(nsIFrame* aParentFrame,
|
|
const nsFrameList& aChildren)
|
|
{
|
|
#ifdef DEBUG
|
|
auto parentType = aParentFrame->Type();
|
|
if (parentType != LayoutFrameType::FlexContainer &&
|
|
parentType != LayoutFrameType::GridContainer) {
|
|
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 (IsAnonymousFlexOrGridItem(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
|
|
}
|
|
|
|
inline void
|
|
nsCSSFrameConstructor::ConstructFramesFromItemList(nsFrameConstructorState& aState,
|
|
FrameConstructionItemList& aItems,
|
|
nsContainerFrame* aParentFrame,
|
|
bool aParentIsWrapperAnonBox,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
// 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);
|
|
|
|
CreateNeededPseudoContainers(aState, aItems, aParentFrame);
|
|
CreateNeededAnonFlexOrGridItems(aState, aItems, aParentFrame);
|
|
CreateNeededPseudoInternalRubyBoxes(aState, aItems, aParentFrame);
|
|
CreateNeededPseudoSiblings(aState, aItems, aParentFrame);
|
|
|
|
aItems.SetTriedConstructingFrames();
|
|
for (FCItemIterator iter(aItems); !iter.IsDone(); iter.Next()) {
|
|
NS_ASSERTION(iter.item().DesiredParentType() == GetParentType(aParentFrame),
|
|
"Needed pseudos didn't get created; expect bad things");
|
|
ConstructFramesFromItem(aState, iter, aParentFrame, aFrameItems);
|
|
}
|
|
|
|
VerifyGridFlexContainerChildren(aParentFrame, aFrameItems);
|
|
NS_ASSERTION(!aState.mHavePendingPopupgroup,
|
|
"Should have proccessed it by now");
|
|
|
|
if (aParentIsWrapperAnonBox) {
|
|
for (nsIFrame* f : aFrameItems) {
|
|
f->SetParentIsWrapperAnonBox();
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::AddFCItemsForAnonymousContent(
|
|
nsFrameConstructorState& aState,
|
|
nsContainerFrame* aFrame,
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>& aAnonymousItems,
|
|
FrameConstructionItemList& aItemsToConstruct,
|
|
uint32_t aExtraFlags)
|
|
{
|
|
for (uint32_t i = 0; i < aAnonymousItems.Length(); ++i) {
|
|
nsIContent* content = aAnonymousItems[i].mContent;
|
|
// Gecko-styled nodes should have no pending restyle flags.
|
|
MOZ_ASSERT(content->IsStyledByServo() ||
|
|
!content->IsElement() ||
|
|
!(content->GetFlags() & ELEMENT_ALL_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->IsNodeOfType(nsINode::eCOMMENT) &&
|
|
!content->IsNodeOfType(nsINode::ePROCESSING_INSTRUCTION),
|
|
"Why is someone creating garbage anonymous content");
|
|
|
|
RefPtr<nsStyleContext> styleContext;
|
|
Maybe<TreeMatchContext::AutoParentDisplayBasedStyleFixupSkipper>
|
|
parentDisplayBasedStyleFixupSkipper;
|
|
MOZ_ASSERT(aState.mTreeMatchContext || content->IsStyledByServo());
|
|
if (aState.mTreeMatchContext) {
|
|
parentDisplayBasedStyleFixupSkipper.emplace(*aState.mTreeMatchContext);
|
|
}
|
|
|
|
// Make sure we eagerly performed the servo cascade when the anonymous
|
|
// nodes were created.
|
|
MOZ_ASSERT(!content->IsStyledByServo() || !content->IsElement() ||
|
|
content->AsElement()->HasServoData());
|
|
|
|
// Determine whether this NAC is pseudo-implementing.
|
|
nsAtom* pseudo = nullptr;
|
|
if (content->IsElement()) {
|
|
auto pseudoType = content->AsElement()->GetPseudoElementType();
|
|
if (pseudoType != CSSPseudoElementType::NotPseudo) {
|
|
pseudo = nsCSSPseudoElements::GetPseudoAtom(pseudoType);
|
|
}
|
|
}
|
|
|
|
// Determine the appropriate parent style for this NAC, and if the NAC
|
|
// implements a pseudo-element, the appropriate originating element
|
|
// (that is to say, the element to the left of the ::pseudo-element in
|
|
// the selector). This is all rather tricky, and merits some discussion.
|
|
//
|
|
// First, it's important to note that author stylesheets generally do not
|
|
// apply to elements in native-anonymous subtrees. The exceptions to
|
|
// this are web-exposed pseudo-elements, where authors can style the
|
|
// pseudo-implementing NAC if the originating element is not itself in a NAC
|
|
// subtree.
|
|
//
|
|
// For this reason, it's very important that we avoid using a style parent
|
|
// that is inside a NAC subtree together with an originating element that
|
|
// is not inside a NAC subtree, since that would allow authors to
|
|
// explicitly inherit styles from internal elements, potentially making
|
|
// the NAC hierarchy observable. To ensure this, and generally simplify
|
|
// things, we always set the originating element to the style parent.
|
|
//
|
|
// As a consequence of the above, all web-exposed pseudo-elements (which,
|
|
// by definition, must have a content-accessible originating element) must
|
|
// also inherit style from that same content-accessible element. To avoid
|
|
// unintuitive behavior differences between NAC elements that do and don't
|
|
// correspond to web-exposed pseudo-elements, we follow this protocol for
|
|
// all NAC, pseudo-implementing or not.
|
|
//
|
|
// However, things get tricky with the <video> element, where we have a
|
|
// bunch of XBL-generated anonymous content descending from a native-
|
|
// anonymous XULElement. The XBL elements inherit style from their
|
|
// flattened tree parent, because that's how XBL works. But then we need
|
|
// to figure out what to do when one of those anonymous XBL elements
|
|
// (like an <input> element) generates its own (possibly pseudo-element-
|
|
// implementing) NAC.
|
|
//
|
|
// In this case, we inherit style from the XBL-generated NAC-creating
|
|
// element, rather than the <video> element. There are a number of good
|
|
// reasons for this. First, inheriting from the great-grandparent while
|
|
// the parent inherits from the grandparent would be bizarre at best.
|
|
// Second, exposing pseudo-elements from elements within our particular
|
|
// XBL implementation would allow content styles to (un)intentionally
|
|
// alter the video controls, which would be very bad. Third, our UA
|
|
// stylesheets have selectors like:
|
|
//
|
|
// input[type=range][orient=horizontal]::-moz-range-track
|
|
//
|
|
// and we need to make sure that the originating element is the <input>,
|
|
// not the <video>, because that's where the |orient| attribute lives.
|
|
//
|
|
// The upshot of all of this is that, to find the style parent (and
|
|
// originating element, if applicable), we walk up our parent chain to the
|
|
// first element that is not itself NAC (distinct from whether it happens
|
|
// to be in a NAC subtree).
|
|
//
|
|
// The one exception to all of this is scrollbar content, which we parent
|
|
// directly to the scrollframe. This is because the special-snowflake
|
|
// construction of scroll frames doesn't result in the placeholder frame
|
|
// being constructed until later, which means that GetInFlowParent() doesn't
|
|
// work right in the case of out-of-flow scrollframes.
|
|
//
|
|
// To implement all this, we need to pass the correct parent style context
|
|
// here because SetPrimaryFrame() may not have been called on the content
|
|
// yet and thus ResolveStyleContext can't find it otherwise.
|
|
//
|
|
// We don't need to worry about display:contents here, because such
|
|
// elements don't get a frame and thus can't generate NAC. But we do need
|
|
// to worry about anonymous boxes, which CorrectStyleParentFrame handles
|
|
// for us.
|
|
nsIFrame* inheritFrame = aFrame;
|
|
if (!content->IsNativeScrollbarContent()) {
|
|
while (inheritFrame->GetContent()->IsNativeAnonymous()) {
|
|
inheritFrame = inheritFrame->GetInFlowParent();
|
|
}
|
|
}
|
|
|
|
nsIFrame* styleParentFrame =
|
|
nsFrame::CorrectStyleParentFrame(inheritFrame, pseudo);
|
|
// The only way we can not have a style parent now is if inheritFrame is the
|
|
// canvas frame and we're the NAC parent for all the things added via
|
|
// nsIDocument::InsertAnonymousContent.
|
|
MOZ_ASSERT(styleParentFrame || inheritFrame->IsCanvasFrame());
|
|
// And that anonymous div has no pseudo.
|
|
MOZ_ASSERT(styleParentFrame || !pseudo);
|
|
|
|
Element* originating =
|
|
pseudo ? styleParentFrame->GetContent()->AsElement() : nullptr;
|
|
nsStyleContext* parentStyle =
|
|
styleParentFrame ? styleParentFrame->StyleContext() : nullptr;
|
|
styleContext =
|
|
ResolveStyleContext(parentStyle, content, &aState, originating);
|
|
|
|
nsTArray<nsIAnonymousContentCreator::ContentInfo>* anonChildren = nullptr;
|
|
if (!aAnonymousItems[i].mChildren.IsEmpty()) {
|
|
anonChildren = &aAnonymousItems[i].mChildren;
|
|
}
|
|
|
|
uint32_t flags = ITEM_ALLOW_XBL_BASE | ITEM_ALLOW_PAGE_BREAK |
|
|
ITEM_IS_ANONYMOUSCONTENTCREATOR_CONTENT | aExtraFlags;
|
|
|
|
AddFrameConstructionItemsInternal(aState, content, aFrame,
|
|
content->NodeInfo()->NameAtom(),
|
|
content->GetNameSpaceID(),
|
|
true, styleContext, flags,
|
|
anonChildren, aItemsToConstruct);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ProcessChildren(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsStyleContext* aStyleContext,
|
|
nsContainerFrame* aFrame,
|
|
const bool aCanHaveGeneratedContent,
|
|
nsFrameItems& aFrameItems,
|
|
const bool aAllowBlockStyles,
|
|
PendingBinding* aPendingBinding,
|
|
nsIFrame* aPossiblyLeafFrame)
|
|
{
|
|
NS_PRECONDITION(aFrame, "Must have parent frame here");
|
|
NS_PRECONDITION(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...
|
|
|
|
bool haveFirstLetterStyle = false, haveFirstLineStyle = false;
|
|
if (aAllowBlockStyles) {
|
|
ShouldHaveSpecialBlockStyle(aContent, aStyleContext, &haveFirstLetterStyle,
|
|
&haveFirstLineStyle);
|
|
}
|
|
|
|
const bool isFlexOrGridContainer = ::IsFlexOrGridContainer(aFrame);
|
|
// The logic here needs to match the logic in GetFloatContainingBlock()
|
|
// (Since we already have isFlexOrGridContainer, we check that eagerly instead
|
|
// of letting ShouldSuppressFloatingOfDescendants look it up redundantly.)
|
|
nsFrameConstructorSaveState floatSaveState;
|
|
if (isFlexOrGridContainer ||
|
|
ShouldSuppressFloatingOfDescendants(aFrame)) {
|
|
aState.PushFloatContainingBlock(nullptr, floatSaveState);
|
|
} else if (aFrame->IsFloatContainingBlock()) {
|
|
aState.PushFloatContainingBlock(aFrame, floatSaveState);
|
|
}
|
|
|
|
nsFrameConstructorState::PendingBindingAutoPusher pusher(aState,
|
|
aPendingBinding);
|
|
|
|
AutoFrameConstructionItemList itemsToConstruct(this);
|
|
|
|
// If we have first-letter or first-line style then frames can get
|
|
// moved around so don't set these flags.
|
|
if (aAllowBlockStyles && !haveFirstLetterStyle && !haveFirstLineStyle) {
|
|
itemsToConstruct.SetLineBoundaryAtStart(true);
|
|
itemsToConstruct.SetLineBoundaryAtEnd(true);
|
|
}
|
|
|
|
// Create any anonymous frames we need here. This must happen before the
|
|
// non-anonymous children are processed to ensure that popups are never
|
|
// constructed before the popupset.
|
|
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->IsRootOfAnonymousSubtree(),
|
|
"Content should know it's an anonymous subtree");
|
|
}
|
|
#endif
|
|
AddFCItemsForAnonymousContent(aState, aFrame, anonymousItems,
|
|
itemsToConstruct);
|
|
|
|
if (!aPossiblyLeafFrame->IsLeaf()) {
|
|
// :before/:after content should have the same style context parent
|
|
// as normal kids.
|
|
// Note that we don't use this style context 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.
|
|
nsStyleContext* styleContext;
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
aFrame->AddStateBits(NS_FRAME_MAY_HAVE_GENERATED_CONTENT);
|
|
styleContext =
|
|
nsFrame::CorrectStyleParentFrame(aFrame, nullptr)->StyleContext();
|
|
// Probe for generated content before
|
|
CreateGeneratedContentItem(aState, aFrame, aContent, styleContext,
|
|
CSSPseudoElementType::before,
|
|
itemsToConstruct);
|
|
}
|
|
|
|
const bool addChildItems = MOZ_LIKELY(mCurrentDepth < kMaxDepth);
|
|
if (!addChildItems) {
|
|
NS_WARNING("ProcessChildren max depth exceeded");
|
|
}
|
|
|
|
// Don't blockify 'display' in ApplyStyleFixups unless aFrame really is
|
|
// a flex/grid container frame, not just has display:flex/grid.
|
|
Maybe<TreeMatchContext::AutoParentDisplayBasedStyleFixupSkipper>
|
|
parentDisplayBasedStyleFixupSkipper;
|
|
MOZ_ASSERT(aState.mTreeMatchContext || aContent->IsStyledByServo());
|
|
if (!isFlexOrGridContainer && aState.mTreeMatchContext) {
|
|
parentDisplayBasedStyleFixupSkipper.emplace(*aState.mTreeMatchContext);
|
|
}
|
|
|
|
InsertionPoint insertion(aFrame, nullptr);
|
|
FlattenedChildIterator iter(aContent);
|
|
for (nsIContent* child = iter.GetNextChild(); child; child = iter.GetNextChild()) {
|
|
// Get the parent of the content and check if it is a XBL children element
|
|
// (if the content is a children element then parent != aContent because the
|
|
// FlattenedChildIterator will transitively iterate through <xbl:children>
|
|
// for default content). Push the children element as an ancestor here because
|
|
// it does not have a frame and would not otherwise be pushed as an ancestor.
|
|
insertion.mContainer = aContent;
|
|
nsIContent* parent = child->GetParent();
|
|
MOZ_ASSERT(parent, "Parent must be non-null because we are iterating children.");
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(aState.mTreeMatchContext);
|
|
if (parent != aContent && parent->IsElement()) {
|
|
insertion.mContainer = child->GetFlattenedTreeParent();
|
|
MOZ_ASSERT(insertion.mContainer == GetInsertionPoint(parent, child).mContainer);
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(parent->AsElement());
|
|
} else {
|
|
ancestorPusher.PushStyleScope(parent->AsElement());
|
|
}
|
|
}
|
|
|
|
// Frame construction item construction should not post
|
|
// restyles, so removing restyle flags here is safe.
|
|
child->UnsetRestyleFlagsIfGecko();
|
|
if (addChildItems) {
|
|
AddFrameConstructionItems(aState, child, iter.XBLInvolved(), insertion,
|
|
itemsToConstruct);
|
|
} else {
|
|
ClearLazyBits(child, child->GetNextSibling());
|
|
}
|
|
}
|
|
itemsToConstruct.SetParentHasNoXBLChildren(!iter.XBLInvolved());
|
|
|
|
if (aCanHaveGeneratedContent) {
|
|
// Probe for generated content after
|
|
CreateGeneratedContentItem(aState, aFrame, aContent, styleContext,
|
|
CSSPseudoElementType::after,
|
|
itemsToConstruct);
|
|
}
|
|
} else {
|
|
ClearLazyBits(aContent->GetFirstChild(), nullptr);
|
|
}
|
|
|
|
ConstructFramesFromItemList(aState, itemsToConstruct, aFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
aFrameItems);
|
|
|
|
NS_ASSERTION(!aAllowBlockStyles || !aFrame->IsXULBoxFrame(),
|
|
"can't be both block and box");
|
|
|
|
if (haveFirstLetterStyle) {
|
|
WrapFramesInFirstLetterFrame(aFrame, aFrameItems);
|
|
}
|
|
if (haveFirstLineStyle) {
|
|
WrapFramesInFirstLineFrame(aState, aContent, aFrame, nullptr,
|
|
aFrameItems);
|
|
}
|
|
|
|
// We might end up with first-line frames that change
|
|
// AnyKidsNeedBlockParent() without changing itemsToConstruct, but that
|
|
// should never happen for cases whan aFrame->IsXULBoxFrame().
|
|
NS_ASSERTION(!haveFirstLineStyle || !aFrame->IsXULBoxFrame(),
|
|
"Shouldn't have first-line style if we're a box");
|
|
NS_ASSERTION(!aFrame->IsXULBoxFrame() ||
|
|
itemsToConstruct.AnyItemsNeedBlockParent() ==
|
|
(AnyKidsNeedBlockParent(aFrameItems.FirstChild()) != nullptr),
|
|
"Something went awry in our block parent calculations");
|
|
|
|
if (aFrame->IsXULBoxFrame() && itemsToConstruct.AnyItemsNeedBlockParent()) {
|
|
// XXXbz we could do this on the FrameConstructionItemList level,
|
|
// no? And if we cared we could look through the item list
|
|
// instead of groveling through the framelist here..
|
|
nsStyleContext *frameStyleContext = aFrame->StyleContext();
|
|
// Report a warning for non-GC frames, for chrome:
|
|
if (!aFrame->IsGeneratedContentFrame() &&
|
|
mPresShell->GetPresContext()->IsChrome()) {
|
|
nsIContent *badKid = AnyKidsNeedBlockParent(aFrameItems.FirstChild());
|
|
nsDependentAtomString parentTag(aContent->NodeInfo()->NameAtom()),
|
|
kidTag(badKid->NodeInfo()->NameAtom());
|
|
const char16_t* params[] = { parentTag.get(), kidTag.get() };
|
|
const nsStyleDisplay *display = frameStyleContext->StyleDisplay();
|
|
const char *message =
|
|
(display->mDisplay == StyleDisplay::MozInlineBox)
|
|
? "NeededToWrapXULInlineBox" : "NeededToWrapXUL";
|
|
nsContentUtils::ReportToConsole(nsIScriptError::warningFlag,
|
|
NS_LITERAL_CSTRING("Layout: FrameConstructor"),
|
|
mDocument,
|
|
nsContentUtils::eXUL_PROPERTIES,
|
|
message,
|
|
params, ArrayLength(params));
|
|
}
|
|
|
|
RefPtr<nsStyleContext> blockSC = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::mozXULAnonymousBlock,
|
|
frameStyleContext);
|
|
nsBlockFrame* blockFrame = NS_NewBlockFrame(mPresShell, blockSC);
|
|
// We might, in theory, want to set NS_BLOCK_FLOAT_MGR and
|
|
// NS_BLOCK_MARGIN_ROOT, but I think it's a bad idea given that
|
|
// a real block placed here wouldn't get those set on it.
|
|
|
|
InitAndRestoreFrame(aState, aContent, aFrame, blockFrame, false);
|
|
|
|
NS_ASSERTION(!blockFrame->HasView(), "need to do view reparenting");
|
|
ReparentFrames(this, blockFrame, aFrameItems, false);
|
|
|
|
blockFrame->SetInitialChildList(kPrincipalList, aFrameItems);
|
|
NS_ASSERTION(aFrameItems.IsEmpty(), "How did that happen?");
|
|
aFrameItems.Clear();
|
|
aFrameItems.AddChild(blockFrame);
|
|
|
|
aFrame->AddStateBits(NS_STATE_BOX_WRAPS_KIDS_IN_BLOCK);
|
|
MOZ_ASSERT(!aFrame->IsLeaf(),
|
|
"Why do we have an nsLeafBoxFrame here?");
|
|
aFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// 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,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
// Find the part of aFrameItems that we want to put in the first-line
|
|
nsFrameList::FrameLinkEnumerator link(aFrameItems);
|
|
while (!link.AtEnd() && link.NextFrame()->IsInlineOutside()) {
|
|
link.Next();
|
|
}
|
|
|
|
nsFrameList firstLineChildren = aFrameItems.ExtractHead(link);
|
|
|
|
if (firstLineChildren.IsEmpty()) {
|
|
// Nothing is supposed to go into the first-line; nothing to do
|
|
return;
|
|
}
|
|
|
|
if (!aLineFrame) {
|
|
// Create line frame
|
|
nsStyleContext* parentStyle =
|
|
nsFrame::CorrectStyleParentFrame(aBlockFrame,
|
|
nsCSSPseudoElements::firstLine)->
|
|
StyleContext();
|
|
RefPtr<nsStyleContext> 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 aFrameItems.
|
|
aFrameItems.InsertFrame(nullptr, nullptr, aLineFrame);
|
|
|
|
NS_ASSERTION(aLineFrame->StyleContext() == firstLineStyle,
|
|
"Bogus style context 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->GetStateBits() & NS_FRAME_FIRST_REFLOW)) {
|
|
aLineFrame->SetInitialChildList(kPrincipalList, firstLineChildren);
|
|
} else {
|
|
AppendFrames(aLineFrame, kPrincipalList, 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,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
// 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, aFrameItems);
|
|
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, aFrameItems);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CheckForFirstLineInsertion(nsIFrame* aParentFrame,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
MOZ_ASSERT(aParentFrame->StyleContext()->HasPseudoElementData(),
|
|
"Why were we called?");
|
|
|
|
if (aFrameItems.IsEmpty()) {
|
|
// Happens often enough, with the caption stuff. No need to do the ancestor
|
|
// walk here.
|
|
return;
|
|
}
|
|
|
|
class RestyleManager* restyleManager = RestyleManager();
|
|
if (!restyleManager->IsServo()) {
|
|
// Gecko's style resolution is frame-based, so already has the right styles
|
|
// even in the ::first-line case.
|
|
return;
|
|
}
|
|
|
|
// 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->StyleContext()->HasPseudoElementData()) {
|
|
// We know we won't find a ::first-line now.
|
|
return;
|
|
}
|
|
|
|
if (!ancestor->IsLineFrame()) {
|
|
ancestor = ancestor->GetParent();
|
|
continue;
|
|
}
|
|
|
|
if (!ancestor->StyleContext()->IsPseudoElement()) {
|
|
// This is a continuation lineframe, not the first line; no need to do
|
|
// anything to the styles.
|
|
return;
|
|
}
|
|
|
|
// Fix up the styles of aFrameItems for ::first-line.
|
|
for (nsIFrame* f : aFrameItems) {
|
|
restyleManager->ReparentStyleContext(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;
|
|
|
|
int32_t i, n = aFragment->GetLength();
|
|
for (i = 0; i < n; i++) {
|
|
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(nsIContent* aContent)
|
|
{
|
|
NS_PRECONDITION(aContent, "null ptr");
|
|
|
|
bool result = false;
|
|
if (aContent) {
|
|
const nsTextFragment* frag = aContent->GetText();
|
|
if (frag) {
|
|
int32_t flc = FirstLetterCount(frag);
|
|
int32_t tl = frag->GetLength();
|
|
if (flc < tl) {
|
|
result = true;
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static bool IsFirstLetterContent(nsIContent* aContent)
|
|
{
|
|
return aContent->TextLength() &&
|
|
!aContent->TextIsOnlyWhitespace();
|
|
}
|
|
|
|
/**
|
|
* Create a letter frame, only make it a floating frame.
|
|
*/
|
|
nsFirstLetterFrame*
|
|
nsCSSFrameConstructor::CreateFloatingLetterFrame(
|
|
nsFrameConstructorState& aState,
|
|
nsIContent* aTextContent,
|
|
nsIFrame* aTextFrame,
|
|
nsContainerFrame* aParentFrame,
|
|
nsStyleContext* aParentStyleContext,
|
|
nsStyleContext* aStyleContext,
|
|
nsFrameItems& aResult)
|
|
{
|
|
MOZ_ASSERT(aParentStyleContext);
|
|
|
|
nsFirstLetterFrame* letterFrame =
|
|
NS_NewFirstLetterFrame(mPresShell, aStyleContext);
|
|
// 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). So use its parent for the
|
|
// first-letter.
|
|
nsIContent* letterContent = aTextContent->GetParent();
|
|
nsContainerFrame* containingBlock = aState.GetGeometricParent(
|
|
aStyleContext->StyleDisplay(), aParentFrame);
|
|
InitAndRestoreFrame(aState, letterContent, containingBlock, letterFrame);
|
|
|
|
// Init the text frame to refer to the letter frame. Make sure we
|
|
// get a proper style context 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).
|
|
StyleSetHandle styleSet = mPresShell->StyleSet();
|
|
RefPtr<nsStyleContext> textSC = styleSet->
|
|
ResolveStyleForText(aTextContent, aStyleContext);
|
|
aTextFrame->SetStyleContextWithoutNotification(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(aState.mPresContext, aTextFrame, aParentFrame);
|
|
RefPtr<nsStyleContext> newSC = styleSet->
|
|
ResolveStyleForText(aTextContent, aParentStyleContext);
|
|
nextTextFrame->SetStyleContext(newSC);
|
|
}
|
|
|
|
NS_ASSERTION(aResult.IsEmpty(), "aResult should be an empty nsFrameItems!");
|
|
// 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.
|
|
nsFrameList::FrameLinkEnumerator link(aState.mFloatedItems);
|
|
while (!link.AtEnd() && link.NextFrame()->GetParent() != containingBlock) {
|
|
link.Next();
|
|
}
|
|
|
|
aState.AddChild(letterFrame, aResult, letterContent, aStyleContext,
|
|
aParentFrame, false, true, false, true,
|
|
link.PrevFrame());
|
|
|
|
if (nextTextFrame) {
|
|
aResult.AddChild(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,
|
|
nsIContent* aTextContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsFrameItems& aResult)
|
|
{
|
|
NS_PRECONDITION(aTextContent->IsNodeOfType(nsINode::eTEXT),
|
|
"aTextContent isn't text");
|
|
NS_ASSERTION(nsLayoutUtils::GetAsBlock(aBlockFrame),
|
|
"Not a block frame?");
|
|
|
|
// Get style context for the first-letter-frame. Keep this in sync with
|
|
// nsBlockFrame::UpdatePseudoElementStyles.
|
|
nsIFrame* parentFrame =
|
|
nsFrame::CorrectStyleParentFrame(aParentFrame,
|
|
nsCSSPseudoElements::firstLetter);
|
|
|
|
nsStyleContext* parentStyleContext = parentFrame->StyleContext();
|
|
|
|
// Use content from containing block so that we can actually
|
|
// find a matching style rule.
|
|
nsIContent* blockContent = aBlockFrame->GetContent();
|
|
|
|
// Create first-letter style rule
|
|
RefPtr<nsStyleContext> sc =
|
|
GetFirstLetterStyle(blockContent, parentStyleContext);
|
|
|
|
if (sc) {
|
|
if (sc->IsServo() && parentFrame->IsLineFrame()) {
|
|
ServoStyleContext* parentStyleIgnoringFirstLine =
|
|
aBlockFrame->StyleContext()->AsServo();
|
|
sc =
|
|
mPresShell->StyleSet()->AsServo()->ReparentStyleContext(
|
|
sc->AsServo(),
|
|
parentStyleContext->AsServo(),
|
|
parentStyleIgnoringFirstLine,
|
|
parentStyleContext->AsServo(),
|
|
blockContent->AsElement());
|
|
}
|
|
|
|
RefPtr<nsStyleContext> 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");
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext,
|
|
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() &&
|
|
!nsSVGUtils::IsInSVGTextSubtree(aParentFrame)) {
|
|
// Make a floating first-letter frame
|
|
letterFrame = CreateFloatingLetterFrame(state, aTextContent, textFrame,
|
|
aParentFrame, parentStyleContext,
|
|
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). So use its parent for the first-letter.
|
|
nsIContent* letterContent = aTextContent->GetParent();
|
|
letterFrame->Init(letterContent, aParentFrame, nullptr);
|
|
|
|
InitAndRestoreFrame(state, aTextContent, letterFrame, textFrame);
|
|
|
|
SetInitialSingleChild(letterFrame, textFrame);
|
|
aResult.Clear();
|
|
aResult.AddChild(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());
|
|
parent->SetHasFirstLetterChild();
|
|
aBlockFrame->SetProperty(nsContainerFrame::FirstLetterProperty(),
|
|
letterFrame);
|
|
aTextContent->SetPrimaryFrame(textFrame);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
|
nsContainerFrame* aBlockFrame,
|
|
nsFrameItems& aBlockFrames)
|
|
{
|
|
aBlockFrame->AddStateBits(NS_BLOCK_HAS_FIRST_LETTER_STYLE);
|
|
|
|
nsContainerFrame* parentFrame = nullptr;
|
|
nsIFrame* textFrame = nullptr;
|
|
nsIFrame* prevFrame = nullptr;
|
|
nsFrameItems letterFrames;
|
|
bool stopLooking = false;
|
|
WrapFramesInFirstLetterFrame(aBlockFrame, aBlockFrame, aBlockFrame,
|
|
aBlockFrames.FirstChild(),
|
|
&parentFrame, &textFrame, &prevFrame,
|
|
letterFrames, &stopLooking);
|
|
if (parentFrame) {
|
|
if (parentFrame == aBlockFrame) {
|
|
// Take textFrame out of the block's frame list and substitute the
|
|
// letter frame(s) instead.
|
|
aBlockFrames.DestroyFrame(textFrame);
|
|
aBlockFrames.InsertFrames(nullptr, prevFrame, letterFrames);
|
|
}
|
|
else {
|
|
// Take the old textFrame out of the inline parent's child list
|
|
RemoveFrame(kPrincipalList, textFrame);
|
|
|
|
// Insert in the letter frame(s)
|
|
parentFrame->InsertFrames(kPrincipalList, prevFrame, letterFrames);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::WrapFramesInFirstLetterFrame(
|
|
nsContainerFrame* aBlockFrame,
|
|
nsContainerFrame* aBlockContinuation,
|
|
nsContainerFrame* aParentFrame,
|
|
nsIFrame* aParentFrameList,
|
|
nsContainerFrame** aModifiedParent,
|
|
nsIFrame** aTextFrame,
|
|
nsIFrame** aPrevFrame,
|
|
nsFrameItems& aLetterFrames,
|
|
bool* aStopLooking)
|
|
{
|
|
nsIFrame* prevFrame = nullptr;
|
|
nsIFrame* frame = aParentFrameList;
|
|
|
|
while (frame) {
|
|
nsIFrame* nextFrame = frame->GetNextSibling();
|
|
|
|
LayoutFrameType frameType = frame->Type();
|
|
if (LayoutFrameType::Text == frameType) {
|
|
// Wrap up first-letter content in a letter frame
|
|
nsIContent* textContent = frame->GetContent();
|
|
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, nsIFrame::ChildListID aListID)
|
|
{
|
|
nsFrameList list = aFrame->GetChildList(aListID);
|
|
for (nsFrameList::Enumerator e(list); !e.AtEnd(); e.Next()) {
|
|
if (e.get()->IsLetterFrame()) {
|
|
return e.get();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::RemoveFloatingFirstLetterFrames(
|
|
nsIPresShell* aPresShell,
|
|
nsIFrame* aBlockFrame)
|
|
{
|
|
// Look for the first letter frame on the kFloatList, then kPushedFloatsList.
|
|
nsIFrame* floatFrame =
|
|
::FindFirstLetterFrame(aBlockFrame, nsIFrame::kFloatList);
|
|
if (!floatFrame) {
|
|
floatFrame =
|
|
::FindFirstLetterFrame(aBlockFrame, nsIFrame::kPushedFloatsList);
|
|
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;
|
|
}
|
|
static_cast<nsContainerFrame*>(parentFrame->FirstContinuation())->
|
|
ClearHasFirstLetterChild();
|
|
|
|
// Create a new text frame with the right style context that maps
|
|
// all of the content that was previously part of the letter frame
|
|
// (and probably continued elsewhere).
|
|
nsStyleContext* parentSC = parentFrame->StyleContext();
|
|
nsIContent* textContent = textFrame->GetContent();
|
|
if (!textContent) {
|
|
return;
|
|
}
|
|
RefPtr<nsStyleContext> 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();
|
|
while (frameToDelete != textFrame) {
|
|
nsIFrame* nextFrameToDelete = frameToDelete->GetPrevContinuation();
|
|
RemoveFrame(kPrincipalList, 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(kPrincipalList, 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
|
|
nsFrameList textList(newTextFrame, newTextFrame);
|
|
InsertFrames(parentFrame, kPrincipalList, prevSibling, textList);
|
|
|
|
if (offsetsNeedFixing) {
|
|
prevSibling->RemoveStateBits(TEXT_OFFSETS_NEED_FIXING);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::RemoveFirstLetterFrames(nsIPresShell* aPresShell,
|
|
nsContainerFrame* aFrame,
|
|
nsContainerFrame* aBlockFrame,
|
|
bool* aStopLooking)
|
|
{
|
|
nsIFrame* prevSibling = nullptr;
|
|
nsIFrame* kid = aFrame->PrincipalChildList().FirstChild();
|
|
|
|
while (kid) {
|
|
if (kid->IsLetterFrame()) {
|
|
// Bingo. Found it. First steal away the text frame.
|
|
static_cast<nsContainerFrame*>(aFrame->FirstContinuation())->
|
|
ClearHasFirstLetterChild();
|
|
nsIFrame* textFrame = kid->PrincipalChildList().FirstChild();
|
|
if (!textFrame) {
|
|
break;
|
|
}
|
|
|
|
// Create a new textframe
|
|
nsStyleContext* parentSC = aFrame->StyleContext();
|
|
if (!parentSC) {
|
|
break;
|
|
}
|
|
nsIContent* textContent = textFrame->GetContent();
|
|
if (!textContent) {
|
|
break;
|
|
}
|
|
RefPtr<nsStyleContext> newSC = aPresShell->StyleSet()->
|
|
ResolveStyleForText(textContent, parentSC);
|
|
textFrame = NS_NewTextFrame(aPresShell, newSC);
|
|
textFrame->Init(textContent, aFrame, nullptr);
|
|
|
|
// Next rip out the kid and replace it with the text frame
|
|
RemoveFrame(kPrincipalList, 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
|
|
nsFrameList textList(textFrame, textFrame);
|
|
InsertFrames(aFrame, kPrincipalList, prevSibling, textList);
|
|
|
|
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;
|
|
}
|
|
else 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(nsIPresShell* 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;
|
|
nsFrameItems 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) {
|
|
// Take the old textFrame out of the parent's child list
|
|
RemoveFrame(kPrincipalList, textFrame);
|
|
|
|
// Insert in the letter frame(s)
|
|
parentFrame->InsertFrames(kPrincipalList, prevFrame, letterFrames);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// listbox Widget Routines
|
|
|
|
void
|
|
nsCSSFrameConstructor::CreateListBoxContent(nsContainerFrame* aParentFrame,
|
|
nsIFrame* aPrevFrame,
|
|
nsIContent* aChild,
|
|
nsIFrame** aNewFrame,
|
|
bool aIsAppend)
|
|
{
|
|
#ifdef MOZ_XUL
|
|
// Construct a new frame
|
|
if (nullptr != aParentFrame) {
|
|
nsFrameItems frameItems;
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell,
|
|
matchContext,
|
|
GetAbsoluteContainingBlock(aParentFrame, FIXED_POS),
|
|
GetAbsoluteContainingBlock(aParentFrame, ABS_POS),
|
|
GetFloatContainingBlock(aParentFrame),
|
|
do_AddRef(mTempFrameTreeState));
|
|
|
|
// If we ever initialize the ancestor filter on |state|, make sure
|
|
// to push the right parent!
|
|
|
|
RefPtr<nsStyleContext> styleContext;
|
|
styleContext = ResolveStyleContext(aParentFrame, aChild, &state);
|
|
|
|
// Pre-check for display "none" - only if we find that, do we create
|
|
// any frame at all
|
|
const nsStyleDisplay* display = styleContext->StyleDisplay();
|
|
|
|
if (StyleDisplay::None == display->mDisplay) {
|
|
*aNewFrame = nullptr;
|
|
return;
|
|
}
|
|
|
|
BeginUpdate();
|
|
|
|
AutoFrameConstructionItemList items(this);
|
|
AddFrameConstructionItemsInternal(state, aChild, aParentFrame,
|
|
aChild->NodeInfo()->NameAtom(),
|
|
aChild->GetNameSpaceID(),
|
|
true, styleContext,
|
|
ITEM_ALLOW_XBL_BASE, nullptr, items);
|
|
ConstructFramesFromItemList(state, items, aParentFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
frameItems);
|
|
|
|
nsIFrame* newFrame = frameItems.FirstChild();
|
|
*aNewFrame = newFrame;
|
|
|
|
if (newFrame) {
|
|
// Notify the parent frame
|
|
if (aIsAppend)
|
|
((nsListBoxBodyFrame*)aParentFrame)->ListBoxAppendFrames(frameItems);
|
|
else
|
|
((nsListBoxBodyFrame*)aParentFrame)->ListBoxInsertFrames(aPrevFrame, frameItems);
|
|
}
|
|
|
|
EndUpdate();
|
|
|
|
#ifdef ACCESSIBILITY
|
|
if (newFrame) {
|
|
nsAccessibilityService* accService = nsIPresShell::AccService();
|
|
if (accService) {
|
|
accService->ContentRangeInserted(mPresShell, aChild->GetParent(),
|
|
aChild, aChild->GetNextSibling());
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
#endif
|
|
}
|
|
|
|
//----------------------------------------
|
|
|
|
void
|
|
nsCSSFrameConstructor::ConstructBlock(nsFrameConstructorState& aState,
|
|
nsIContent* aContent,
|
|
nsContainerFrame* aParentFrame,
|
|
nsContainerFrame* aContentParentFrame,
|
|
nsStyleContext* aStyleContext,
|
|
nsContainerFrame** aNewFrame,
|
|
nsFrameItems& aFrameItems,
|
|
nsIFrame* aPositionedFrameForAbsPosContainer,
|
|
PendingBinding* aPendingBinding)
|
|
{
|
|
// Create column wrapper if necessary
|
|
nsContainerFrame* blockFrame = *aNewFrame;
|
|
NS_ASSERTION((blockFrame->IsBlockFrame() || blockFrame->IsDetailsFrame()),
|
|
"not a block frame nor a details frame?");
|
|
nsContainerFrame* parent = aParentFrame;
|
|
RefPtr<nsStyleContext> blockStyle = aStyleContext;
|
|
const nsStyleColumn* columns = aStyleContext->StyleColumn();
|
|
|
|
if (columns->mColumnCount != NS_STYLE_COLUMN_COUNT_AUTO
|
|
|| columns->mColumnWidth.GetUnit() != eStyleUnit_Auto) {
|
|
nsContainerFrame* columnSetFrame =
|
|
NS_NewColumnSetFrame(mPresShell, aStyleContext,
|
|
nsFrameState(NS_FRAME_OWNS_ANON_BOXES));
|
|
|
|
InitAndRestoreFrame(aState, aContent, aParentFrame, columnSetFrame);
|
|
blockStyle = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::columnContent,
|
|
aStyleContext);
|
|
parent = columnSetFrame;
|
|
*aNewFrame = columnSetFrame;
|
|
if (aPositionedFrameForAbsPosContainer == blockFrame) {
|
|
aPositionedFrameForAbsPosContainer = columnSetFrame;
|
|
}
|
|
|
|
SetInitialSingleChild(columnSetFrame, blockFrame);
|
|
}
|
|
|
|
blockFrame->SetStyleContextWithoutNotification(blockStyle);
|
|
InitAndRestoreFrame(aState, aContent, parent, blockFrame);
|
|
|
|
aState.AddChild(*aNewFrame, aFrameItems, aContent, aStyleContext,
|
|
aContentParentFrame ? aContentParentFrame :
|
|
aParentFrame);
|
|
if (!mRootElementFrame) {
|
|
// The frame we're constructing will be the root element frame.
|
|
// Set mRootElementFrame before processing children.
|
|
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) {
|
|
// NS_ASSERTION(aRelPos, "should have made area frame for this");
|
|
aState.PushAbsoluteContainingBlock(*aNewFrame, aPositionedFrameForAbsPosContainer, absoluteSaveState);
|
|
}
|
|
|
|
// Process the child content
|
|
nsFrameItems childItems;
|
|
ProcessChildren(aState, aContent, aStyleContext, blockFrame, true,
|
|
childItems, true, aPendingBinding);
|
|
|
|
// Set the frame's initial child list
|
|
blockFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
}
|
|
|
|
nsIFrame*
|
|
nsCSSFrameConstructor::ConstructInline(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aItem,
|
|
nsContainerFrame* aParentFrame,
|
|
const nsStyleDisplay* aDisplay,
|
|
nsFrameItems& aFrameItems)
|
|
{
|
|
// 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 context for the
|
|
// runs of inlines, and put all these frames, in order, into aFrameItems. 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.
|
|
//
|
|
// 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;
|
|
nsStyleContext* const styleContext = aItem.mStyleContext;
|
|
|
|
bool positioned =
|
|
StyleDisplay::Inline == aDisplay->mDisplay &&
|
|
aDisplay->IsRelativelyPositionedStyle() &&
|
|
!nsSVGUtils::IsInSVGTextSubtree(aParentFrame);
|
|
|
|
nsInlineFrame* newFrame = NS_NewInlineFrame(mPresShell, styleContext);
|
|
|
|
// Initialize the frame
|
|
InitAndRestoreFrame(aState, content, aParentFrame, newFrame);
|
|
|
|
// Inline frames can always have generated content
|
|
newFrame->AddStateBits(NS_FRAME_MAY_HAVE_GENERATED_CONTENT);
|
|
|
|
nsFrameConstructorSaveState absoluteSaveState; // definition cannot be inside next block
|
|
// because the object's destructor is significant
|
|
// this is part of the fix for bug 42372
|
|
|
|
newFrame->AddStateBits(NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (positioned) {
|
|
// Relatively positioned frames becomes a container for child
|
|
// frames that are positioned
|
|
aState.PushAbsoluteContainingBlock(newFrame, newFrame, absoluteSaveState);
|
|
}
|
|
|
|
// Process the child content
|
|
nsFrameItems childItems;
|
|
ConstructFramesFromItemList(aState, aItem.mChildItems, newFrame,
|
|
/* aParentIsWrapperAnonBox = */ false,
|
|
childItems);
|
|
|
|
nsFrameList::FrameLinkEnumerator firstBlockEnumerator(childItems);
|
|
if (!aItem.mIsAllInline) {
|
|
FindFirstBlock(firstBlockEnumerator);
|
|
}
|
|
|
|
if (aItem.mIsAllInline || firstBlockEnumerator.AtEnd()) {
|
|
// 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(kPrincipalList, childItems);
|
|
aState.AddChild(newFrame, aFrameItems, content, styleContext, 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 = childItems.ExtractHead(firstBlockEnumerator);
|
|
newFrame->SetInitialChildList(kPrincipalList, firstInlineKids);
|
|
|
|
aFrameItems.AddChild(newFrame);
|
|
|
|
newFrame->AddStateBits(NS_FRAME_OWNS_ANON_BOXES);
|
|
CreateIBSiblings(aState, newFrame, positioned, childItems, aFrameItems);
|
|
|
|
return newFrame;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CreateIBSiblings(nsFrameConstructorState& aState,
|
|
nsContainerFrame* aInitialInline,
|
|
bool aIsPositioned,
|
|
nsFrameItems& aChildItems,
|
|
nsFrameItems& aSiblings)
|
|
{
|
|
nsIContent* content = aInitialInline->GetContent();
|
|
nsStyleContext* styleContext = aInitialInline->StyleContext();
|
|
nsContainerFrame* parentFrame = aInitialInline->GetParent();
|
|
|
|
// Resolve the right style context 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<nsStyleContext> blockSC = mPresShell->StyleSet()->
|
|
ResolveInheritingAnonymousBoxStyle(nsCSSAnonBoxes::mozBlockInsideInlineWrapper,
|
|
styleContext);
|
|
|
|
nsContainerFrame* lastNewInline =
|
|
static_cast<nsContainerFrame*>(aInitialInline->FirstContinuation());
|
|
do {
|
|
// On entry to this loop aChildItems is not empty and the first frame in it
|
|
// is block-level.
|
|
NS_PRECONDITION(aChildItems.NotEmpty(), "Should have child items");
|
|
NS_PRECONDITION(!aChildItems.FirstChild()->IsInlineOutside(),
|
|
"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);
|
|
|
|
// Find the first non-block child which defines the end of our block kids
|
|
// and the start of our next inline's kids
|
|
nsFrameList::FrameLinkEnumerator firstNonBlock =
|
|
FindFirstNonBlock(aChildItems);
|
|
nsFrameList blockKids = aChildItems.ExtractHead(firstNonBlock);
|
|
|
|
MoveChildrenTo(aInitialInline, blockFrame, blockKids);
|
|
|
|
SetFrameIsIBSplit(lastNewInline, blockFrame);
|
|
aSiblings.AddChild(blockFrame);
|
|
|
|
// Now grab the initial inlines in aChildItems and put them into an inline
|
|
// frame.
|
|
nsInlineFrame* inlineFrame = NS_NewInlineFrame(mPresShell, styleContext);
|
|
InitAndRestoreFrame(aState, content, parentFrame, inlineFrame, false);
|
|
inlineFrame->AddStateBits(NS_FRAME_MAY_HAVE_GENERATED_CONTENT |
|
|
NS_FRAME_CAN_HAVE_ABSPOS_CHILDREN);
|
|
if (aIsPositioned) {
|
|
inlineFrame->MarkAsAbsoluteContainingBlock();
|
|
}
|
|
|
|
if (aChildItems.NotEmpty()) {
|
|
nsFrameList::FrameLinkEnumerator firstBlock(aChildItems);
|
|
FindFirstBlock(firstBlock);
|
|
nsFrameList inlineKids = aChildItems.ExtractHead(firstBlock);
|
|
|
|
MoveChildrenTo(aInitialInline, inlineFrame, inlineKids);
|
|
}
|
|
|
|
SetFrameIsIBSplit(blockFrame, inlineFrame);
|
|
aSiblings.AddChild(inlineFrame);
|
|
lastNewInline = inlineFrame;
|
|
} while (aChildItems.NotEmpty());
|
|
|
|
SetFrameIsIBSplit(lastNewInline, nullptr);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::BuildInlineChildItems(nsFrameConstructorState& aState,
|
|
FrameConstructionItem& aParentItem,
|
|
bool aItemIsWithinSVGText,
|
|
bool aItemAllowsTextPathChild)
|
|
{
|
|
// XXXbz should we preallocate aParentItem.mChildItems to some sane
|
|
// length? Maybe even to parentContent->GetChildCount()?
|
|
nsFrameConstructorState::PendingBindingAutoPusher
|
|
pusher(aState, aParentItem.mPendingBinding);
|
|
|
|
nsStyleContext* const parentStyleContext = aParentItem.mStyleContext;
|
|
nsIContent* const parentContent = aParentItem.mContent;
|
|
|
|
TreeMatchContext::AutoAncestorPusher ancestorPusher(aState.mTreeMatchContext);
|
|
if (aState.HasAncestorFilter()) {
|
|
ancestorPusher.PushAncestorAndStyleScope(parentContent->AsElement());
|
|
} else {
|
|
ancestorPusher.PushStyleScope(parentContent->AsElement());
|
|
}
|
|
|
|
if (!aItemIsWithinSVGText) {
|
|
// Probe for generated content before
|
|
CreateGeneratedContentItem(aState, nullptr, parentContent, parentStyleContext,
|
|
CSSPseudoElementType::before,
|
|
aParentItem.mChildItems);
|
|
}
|
|
|
|
uint32_t flags = ITEM_ALLOW_XBL_BASE | ITEM_ALLOW_PAGE_BREAK;
|
|
if (aItemIsWithinSVGText) {
|
|
flags |= ITEM_IS_WITHIN_SVG_TEXT;
|
|
}
|
|
if (aItemAllowsTextPathChild && aParentItem.mIsForSVGAElement) {
|
|
flags |= ITEM_ALLOWS_TEXT_PATH_CHILD;
|
|
}
|
|
|
|
if (!aParentItem.mAnonChildren.IsEmpty()) {
|
|
// Use the anon-children list instead of the content tree child list so
|
|
// that we use any special style context that should be associated with
|
|
// the children, and so that we won't try to construct grandchildren frame
|
|
// constructor items before the frame is available for their parent.
|
|
AddFCItemsForAnonymousContent(aState, nullptr, aParentItem.mAnonChildren,
|
|
aParentItem.mChildItems, flags);
|
|
} else {
|
|
// Use the content tree child list:
|
|
FlattenedChildIterator iter(parentContent);
|
|
for (nsIContent* content = iter.GetNextChild(); content; content = iter.GetNextChild()) {
|
|
// Get the parent of the content and check if it is a XBL children element
|
|
// (if the content is a children element then contentParent != parentContent because the
|
|
// FlattenedChildIterator will transitively iterate through <xbl:children>
|
|
// for default content). Push the children element as an ancestor here because
|
|
// it does not have a frame and would not otherwise be pushed as an ancestor.
|
|
nsIContent* contentParent = content->GetParent();
|
|
MOZ_ASSERT(contentParent, "Parent must be non-null because we are iterating children.");
|
|
TreeMatchContext::AutoAncestorPusher insertionPointPusher(aState.mTreeMatchContext);
|
|
if (contentParent != parentContent && contentParent->IsElement()) {
|
|
if (aState.HasAncestorFilter()) {
|
|
insertionPointPusher.PushAncestorAndStyleScope(contentParent->AsElement());
|
|
} else {
|
|
insertionPointPusher.PushStyleScope(contentParent->AsElement());
|
|
}
|
|
}
|
|
|
|
// Manually check for comments/PIs, since we don't have a frame to pass to
|
|
// AddFrameConstructionItems. We know our parent is a non-replaced inline,
|
|
// so there is no need to do the NeedFrameFor check.
|
|
content->UnsetFlags(NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME);
|
|
if (content->IsNodeOfType(nsINode::eCOMMENT) ||
|
|
content->IsNodeOfType(nsINode::ePROCESSING_INSTRUCTION)) {
|
|
continue;
|
|
}
|
|
|
|
// See comment explaining why we need to remove the "is possible
|
|
// restyle root" flags in AddFrameConstructionItems. But note
|
|
// that we can remove all restyle flags, just like in
|
|
// ProcessChildren and for the same reason.
|
|
content->UnsetRestyleFlagsIfGecko();
|
|
|
|
RefPtr<nsStyleContext> childContext =
|
|
ResolveStyleContext(parentStyleContext, content, &aState);
|
|
|
|
AddFrameConstructionItemsInternal(aState, content, nullptr,
|
|
content->NodeInfo()->NameAtom(),
|
|
content->GetNameSpaceID(),
|
|
iter.XBLInvolved(), childContext,
|
|
flags, nullptr,
|
|
aParentItem.mChildItems);
|
|
}
|
|
}
|
|
|
|
if (!aItemIsWithinSVGText) {
|
|
// Probe for generated content after
|
|
CreateGeneratedContentItem(aState, nullptr, parentContent, parentStyleContext,
|
|
CSSPseudoElementType::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)
|
|
{
|
|
NS_PRECONDITION(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::WipeContainingBlock(nsFrameConstructorState& aState,
|
|
nsIFrame* aContainingBlock,
|
|
nsIFrame* aFrame,
|
|
FrameConstructionItemList& aItems,
|
|
bool aIsAppend,
|
|
nsIFrame* aPrevSibling)
|
|
{
|
|
if (aItems.IsEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
// Before we go and append the frames, we must check for several
|
|
// special situations.
|
|
|
|
// Situation #1 is a XUL frame that contains frames that are required
|
|
// to be wrapped in blocks.
|
|
if (aFrame->IsXULBoxFrame() &&
|
|
!(aFrame->GetStateBits() & NS_STATE_BOX_WRAPS_KIDS_IN_BLOCK) &&
|
|
aItems.AnyItemsNeedBlockParent()) {
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
nsIFrame* nextSibling = ::GetInsertNextSibling(aFrame, aPrevSibling);
|
|
|
|
// Situation #2 is a flex or grid container frame into which we're inserting
|
|
// new inline non-replaced children, adjacent to an existing anonymous
|
|
// flex or grid item.
|
|
LayoutFrameType frameType = aFrame->Type();
|
|
if (frameType == LayoutFrameType::FlexContainer ||
|
|
frameType == LayoutFrameType::GridContainer) {
|
|
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 isWebkitBox = IsFlexContainerForLegacyBox(aFrame);
|
|
if (aPrevSibling && IsAnonymousFlexOrGridItem(aPrevSibling) &&
|
|
iter.item().NeedsAnonFlexOrGridItem(aState, isWebkitBox)) {
|
|
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 && IsAnonymousFlexOrGridItem(nextSibling)) {
|
|
// Jump to the last entry in the list
|
|
iter.SetToEnd();
|
|
iter.Prev();
|
|
if (iter.item().NeedsAnonFlexOrGridItem(aState, isWebkitBox)) {
|
|
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 (IsAnonymousFlexOrGridItem(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 isWebkitBox = IsFlexContainerForLegacyBox(containerFrame);
|
|
if (!iter.SkipItemsThatNeedAnonFlexOrGridItem(aState, isWebkitBox)) {
|
|
// We hit something that _doesn't_ need an anonymous flex item!
|
|
// Rebuild the flex container to bust it out.
|
|
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 ruby-related frame that's getting new children.
|
|
// The situation for ruby is complex, especially when interacting with
|
|
// spaces. It containes 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.
|
|
RecreateFramesForContent(aFrame->GetContent(), InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
// Situation #5 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->GetChildList(kPrincipalList).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 (!aItems.AllWantParentType(parentType)) {
|
|
// Reframing aFrame->GetContent() is good enough, since the content of
|
|
// table pseudo-frames is the ancestor content.
|
|
RecreateFramesForContent(aFrame->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 (0);
|
|
|
|
// 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 nsStyleContext::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->StyleContext()->GetPseudo()) {
|
|
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();
|
|
#ifdef DEBUG
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::WipeContainingBlock: blockContent=%p\n",
|
|
static_cast<void*>(blockContent));
|
|
}
|
|
#endif
|
|
RecreateFramesForContent(blockContent, InsertionKind::Async);
|
|
return true;
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::ReframeContainingBlock(nsIFrame* aFrame)
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
// ReframeContainingBlock is a NASTY routine, it causes terrible performance problems
|
|
// so I want to see when it is happening! Unfortunately, it is happening way to often because
|
|
// so much content on the web causes block-in-inline frame situations and we handle them
|
|
// very poorly
|
|
if (gNoisyContentUpdates) {
|
|
printf("nsCSSFrameConstructor::ReframeContainingBlock frame=%p\n",
|
|
static_cast<void*>(aFrame));
|
|
}
|
|
#endif
|
|
|
|
// 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", static_cast<void*>(blockContent));
|
|
}
|
|
#endif
|
|
RecreateFramesForContent(blockContent->AsElement(), InsertionKind::Async);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// If we get here, we're screwed!
|
|
RecreateFramesForContent(mPresShell->GetDocument()->GetRootElement(),
|
|
InsertionKind::Async);
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::GenerateChildFrames(nsContainerFrame* aFrame)
|
|
{
|
|
{
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
BeginUpdate();
|
|
|
|
nsFrameItems childItems;
|
|
TreeMatchContextHolder matchContext(mDocument);
|
|
nsFrameConstructorState state(mPresShell, matchContext, nullptr, nullptr, nullptr);
|
|
// We don't have a parent frame with a pending binding constructor here,
|
|
// so no need to worry about ordering of the kids' constructors with it.
|
|
// Pass null for the PendingBinding.
|
|
ProcessChildren(state, aFrame->GetContent(), aFrame->StyleContext(),
|
|
aFrame, false, childItems, false,
|
|
nullptr);
|
|
|
|
aFrame->SetInitialChildList(kPrincipalList, childItems);
|
|
|
|
EndUpdate();
|
|
}
|
|
|
|
#ifdef ACCESSIBILITY
|
|
nsAccessibilityService* accService = nsIPresShell::AccService();
|
|
if (accService) {
|
|
nsIContent* container = aFrame->GetContent();
|
|
nsIContent* child = container->GetFirstChild();
|
|
if (child) {
|
|
accService->ContentRangeInserted(mPresShell, container, child, nullptr);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// call XBL constructors after the frames are created
|
|
mPresShell->GetDocument()->BindingManager()->ProcessAttachedQueue();
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////
|
|
// nsCSSFrameConstructor::FrameConstructionItem methods //
|
|
//////////////////////////////////////////////////////////
|
|
bool
|
|
nsCSSFrameConstructor::
|
|
FrameConstructionItem::IsWhitespace(nsFrameConstructorState& aState) const
|
|
{
|
|
NS_PRECONDITION(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)
|
|
{
|
|
NS_PRECONDITION(aDelta == 1 || aDelta == -1, "Unexpected delta");
|
|
mItemCount += aDelta;
|
|
if (aItem->mIsAllInline) {
|
|
mInlineCount += aDelta;
|
|
}
|
|
if (aItem->mIsBlock) {
|
|
mBlockCount += aDelta;
|
|
}
|
|
if (aItem->mIsLineParticipant) {
|
|
mLineParticipantCount += aDelta;
|
|
}
|
|
mDesiredParentCounts[aItem->DesiredParentType()] += aDelta;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// nsCSSFrameConstructor::FrameConstructionItemList::Iterator methods //
|
|
////////////////////////////////////////////////////////////////////////
|
|
inline bool
|
|
nsCSSFrameConstructor::FrameConstructionItemList::
|
|
Iterator::SkipItemsWantingParentType(ParentType aParentType)
|
|
{
|
|
NS_PRECONDITION(!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)
|
|
{
|
|
NS_PRECONDITION(!IsDone(), "Shouldn't be done yet");
|
|
while (item().DesiredParentType() != aParentType) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Note: we implement -webkit-box & -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 aIsWebkitBox)
|
|
{
|
|
if (mFCData->mBits & FCDATA_IS_LINE_PARTICIPANT) {
|
|
// This will be an inline non-replaced box.
|
|
return true;
|
|
}
|
|
|
|
if (aIsWebkitBox) {
|
|
if (mStyleContext->StyleDisplay()->IsInlineOutsideStyle()) {
|
|
// In a -webkit-box, all inline-level content gets wrapped in anon item.
|
|
return true;
|
|
}
|
|
if (!(mFCData->mBits & FCDATA_DISALLOW_OUT_OF_FLOW) &&
|
|
aState.GetGeometricParent(mStyleContext->StyleDisplay(), nullptr)) {
|
|
// We're abspos or fixedpos, which means we'll spawn a placeholder which
|
|
// (because our container is a -webkit-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 aIsWebkitBox)
|
|
{
|
|
NS_PRECONDITION(!IsDone(), "Shouldn't be done yet");
|
|
while (item().NeedsAnonFlexOrGridItem(aState, aIsWebkitBox)) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool
|
|
nsCSSFrameConstructor::FrameConstructionItemList::
|
|
Iterator::SkipItemsThatDontNeedAnonFlexOrGridItem(
|
|
const nsFrameConstructorState& aState,
|
|
bool aIsWebkitBox)
|
|
{
|
|
NS_PRECONDITION(!IsDone(), "Shouldn't be done yet");
|
|
while (!(item().NeedsAnonFlexOrGridItem(aState, aIsWebkitBox))) {
|
|
Next();
|
|
if (IsDone()) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
inline bool
|
|
nsCSSFrameConstructor::FrameConstructionItemList::
|
|
Iterator::SkipItemsNotWantingRubyParent()
|
|
{
|
|
NS_PRECONDITION(!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)
|
|
{
|
|
NS_PRECONDITION(!IsDone(), "Shouldn't be done yet");
|
|
NS_PRECONDITION(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");
|
|
NS_PRECONDITION(!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");
|
|
NS_PRECONDITION(&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() ||
|
|
!aTargetList.mUndisplayedItems.IsEmpty()) {
|
|
do {
|
|
AppendItemToList(aTargetList);
|
|
} while (*this != aEnd);
|
|
return;
|
|
}
|
|
|
|
// Move our entire list of items into the empty target list.
|
|
aTargetList.mItems = Move(mList.mItems);
|
|
|
|
// Copy over the various counters
|
|
aTargetList.mInlineCount = mList.mInlineCount;
|
|
aTargetList.mBlockCount = mList.mBlockCount;
|
|
aTargetList.mLineParticipantCount = mList.mLineParticipantCount;
|
|
aTargetList.mItemCount = mList.mItemCount;
|
|
memcpy(aTargetList.mDesiredParentCounts, mList.mDesiredParentCounts,
|
|
sizeof(aTargetList.mDesiredParentCounts));
|
|
|
|
// Swap out undisplayed item arrays, before we nuke the array on our end
|
|
aTargetList.mUndisplayedItems.SwapElements(mList.mUndisplayedItems);
|
|
|
|
// 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)
|
|
{
|
|
NS_PRECONDITION(&mList == &aEnd.mList, "End iterator for some other list?");
|
|
NS_PRECONDITION(*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()
|
|
{
|
|
NS_PRECONDITION(mUpdateCount != 0, "Instant quote updates are bad news");
|
|
mQuotesDirty = true;
|
|
mPresShell->SetNeedLayoutFlush();
|
|
}
|
|
|
|
void
|
|
nsCSSFrameConstructor::CountersDirty()
|
|
{
|
|
NS_PRECONDITION(mUpdateCount != 0, "Instant counter updates are bad news");
|
|
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;
|
|
}
|
|
}
|