forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			13341 lines
		
	
	
	
		
			530 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			13341 lines
		
	
	
	
		
			530 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| // vim:cindent:ts=2:et:sw=2:
 | |
| /* 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 "nsIAtom.h"
 | |
| #include "nsIFrameInlines.h"
 | |
| #include "nsGkAtoms.h"
 | |
| #include "nsPresContext.h"
 | |
| #include "nsIDocument.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 nsIAtom* 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 nsIAtom* 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)
 | |
| {
 | |
|   aFrame->SetParent(aNewParentFrame);
 | |
|   aRestyleManager->ReparentStyleContext(aFrame);
 | |
| }
 | |
| 
 | |
| static void
 | |
| ReparentFrames(nsCSSFrameConstructor* aFrameConstructor,
 | |
|                nsContainerFrame* aNewParentFrame,
 | |
|                const nsFrameList& aFrameList)
 | |
| {
 | |
|   RestyleManager* restyleManager = aFrameConstructor->RestyleManager();
 | |
|   for (nsFrameList::Enumerator e(aFrameList); !e.AtEnd(); e.Next()) {
 | |
|     ReparentFrame(restyleManager, aNewParentFrame, e.get());
 | |
|   }
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| //
 | |
| // 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;
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| 
 | |
| // 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) {
 | |
|         return &mTopLayerFixedItems;
 | |
|       }
 | |
|       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, PLACEHOLDER_FOR_TOPLAYER);
 | |
|   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);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NS_POSTCONDITION(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)
 | |
|   , 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);
 | |
| }
 | |
| 
 | |
| 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: {
 | |
|       nsCOMPtr<nsIAtom> 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();
 | |
|         }
 | |
| 
 | |
|         nsXPIDLString 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;
 | |
|   nsIAtom* 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.
 | |
|   nsIAtom* 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.
 | |
|   bool isServo = pseudoStyleContext->IsServo();
 | |
|   bool hasServoAnimations = false;
 | |
|   if (isServo) {
 | |
|     ServoComputedValues* servoStyle = pseudoStyleContext->ComputedValues();
 | |
|     hasServoAnimations = Servo_ComputedValues_SpecifiesAnimationsOrTransitions(servoStyle);
 | |
|     if (!hasServoAnimations) {
 | |
|       Servo_SetExplicitStyle(container, servoStyle);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // stylo: 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) {
 | |
|       nsStyleContext* oldStyleContext = rsc->Get(container, aPseudoElement);
 | |
|       if (oldStyleContext) {
 | |
|         GeckoRestyleManager::TryInitiatingTransition(aState.mPresContext,
 | |
|                                                      container,
 | |
|                                                      oldStyleContext,
 | |
|                                                      &pseudoStyleContext);
 | |
|       } else {
 | |
|         aState.mPresContext->TransitionManager()->
 | |
|           PruneCompletedTransitions(aParentContent->AsElement(),
 | |
|                                     aPseudoElement, pseudoStyleContext);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   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 (isServo) {
 | |
|     if (hasServoAnimations) {
 | |
|       // If animations are involved, we avoid the SetExplicitStyle optimization
 | |
|       // above.
 | |
|       mPresShell->StyleSet()->AsServo()->StyleNewSubtree(container);
 | |
|     } else 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)
 | |
| {
 | |
|   nsIAtom* 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, 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,
 | |
|                                 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,
 | |
|                                 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 -------
 | |
|   // 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;
 | |
|   styleContext = mPresShell->StyleSet()->ResolveStyleFor(aDocElement,
 | |
|                                                          nullptr,
 | |
|                                                          LazyComputeBehavior::Allow);
 | |
| 
 | |
|   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)
 | |
|       return nullptr; // Binding will load asynchronously.
 | |
| 
 | |
|     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.)
 | |
|       styleContext = mPresShell->StyleSet()->ResolveStyleFor(aDocElement,
 | |
|                                                              nullptr,
 | |
|                                                              LazyComputeBehavior::Allow);
 | |
|       display = styleContext->StyleDisplay();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // We delay traversing the entire document until here, since we per above we
 | |
|   // may invalidate the root style when we load doc stylesheets.
 | |
|   if (ServoStyleSet* set = mPresShell->StyleSet()->GetAsServo()) {
 | |
|     set->StyleDocument(TraversalRestyleBehavior::Normal);
 | |
|   }
 | |
| 
 | |
|   // --------- 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)) {
 | |
|     SetUndisplayedContent(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();
 | |
|     FrameConstructionItem item(&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();
 | |
|     FrameConstructionItem item(&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;
 | |
|   nsIAtom* 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;
 | |
|   }
 | |
| 
 | |
|   FrameConstructionItemList itemsToConstruct;
 | |
|   nsContainerFrame* frameAsContainer = do_QueryFrame(aFrame);
 | |
|   AddFCItemsForAnonymousContent(aState, frameAsContainer, anonymousItems, itemsToConstruct);
 | |
| 
 | |
|   nsFrameItems frameItems;
 | |
|   ConstructFramesFromItemList(aState, itemsToConstruct, frameAsContainer, 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.
 | |
|     FrameConstructionItemList fcItems;
 | |
|     AddFCItemsForAnonymousContent(aState, comboboxFrame, newAnonymousItems,
 | |
|                                   fcItems);
 | |
|     ConstructFramesFromItemList(aState, fcItems, comboboxFrame, 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");
 | |
|     nsIAtom* 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(nsIAtom* 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)
 | |
| {
 | |
|   nsIAtom* 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,
 | |
|                                     nsIAtom* 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(applet, 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_NewGfxCheckboxControlFrame),
 | |
|     SIMPLE_INT_CREATE(NS_FORM_INPUT_RADIO, NS_NewGfxRadioControlFrame),
 | |
|     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),
 | |
| #if defined(MOZ_WIDGET_ANDROID)
 | |
|     // On Android, date/time input appears as a normal text box.
 | |
|     SIMPLE_INT_CREATE(NS_FORM_INPUT_TIME, NS_NewTextControlFrame),
 | |
|     SIMPLE_INT_CREATE(NS_FORM_INPUT_DATE, NS_NewTextControlFrame),
 | |
| #else
 | |
|     SIMPLE_INT_CREATE(NS_FORM_INPUT_TIME, NS_NewDateTimeControlFrame),
 | |
|     SIMPLE_INT_CREATE(NS_FORM_INPUT_DATE, NS_NewDateTimeControlFrame),
 | |
| #endif
 | |
|     // 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();
 | |
| 
 | |
|   // Note that Android widgets don't have theming support and thus
 | |
|   // appearance:none is the same as any other appearance value.
 | |
|   // So this chunk doesn't apply there:
 | |
| #if !defined(MOZ_WIDGET_ANDROID)
 | |
|   // 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;
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   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,
 | |
|                  "applet, 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");
 | |
| 
 | |
|   // 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,
 | |
|                                     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
 | |
|   uint32_t count;
 | |
|   nsIContent * const *children = aNode->GetChildArray(&count);
 | |
| 
 | |
|   for (uint32_t index = 0; index < count; ++index) {
 | |
|     SetFlagsOnSubtree(children[index], 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);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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,
 | |
|                                       nsIAtom* 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,
 | |
|                                                 nsIAtom*                 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());
 | |
|     }
 | |
| 
 | |
|     FrameConstructionItemList items;
 | |
|     AddFCItemsForAnonymousContent(aState, gfxScrollFrame, scrollNAC, items);
 | |
|     ConstructFramesFromItemList(aState, items, gfxScrollFrame, 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) {
 | |
|       nsStyleContext* oldStyleContext =
 | |
|         rsc->Get(aContent, CSSPseudoElementType::NotPseudo);
 | |
|       nsPresContext* presContext = mPresShell->GetPresContext();
 | |
|       if (oldStyleContext) {
 | |
|         GeckoRestyleManager::TryInitiatingTransition(presContext, aContent,
 | |
|                                                      oldStyleContext, &result);
 | |
|       } else if (aContent->IsElement()) {
 | |
|         presContext->TransitionManager()->
 | |
|           PruneCompletedTransitions(aContent->AsElement(),
 | |
|             CSSPseudoElementType::NotPseudo, result);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   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;
 | |
|   }
 | |
| 
 | |
|   nsIAtom* 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);
 | |
|   blockFrame->AddStateBits(NS_BLOCK_FORMATTING_CONTEXT_STATE_BITS);
 | |
| 
 | |
|   InitAndRestoreFrame(aState, aContent, aParentFrame, blockFrame);
 | |
|   ReparentFrames(this, blockFrame, aBlockItems);
 | |
|   // 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,
 | |
|                                       nsIAtom* 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, 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 nsIAtom* 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,
 | |
|                                    nsIAtom* 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;
 | |
|     nsIAtom* 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(&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(GetUndisplayedContent(aContent),
 | |
|                "should have called SetUndisplayedContent earlier");
 | |
|     return;
 | |
|   }
 | |
|   aList.AppendUndisplayedItem(aContent, aStyleContext);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::AddFrameConstructionItemsInternal(nsFrameConstructorState& aState,
 | |
|                                                          nsIContent* aContent,
 | |
|                                                          nsContainerFrame* aParentFrame,
 | |
|                                                          nsIAtom* 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->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 (aContent->IsStyledByServo()) {
 | |
|           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);
 | |
| 
 | |
|           styleContext =
 | |
|             styleSet->ResolveStyleFor(element, nullptr, LazyComputeBehavior::Allow);
 | |
|         } else {
 | |
|           styleContext =
 | |
|             ResolveStyleContext(styleContext->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());
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   bool isGeneratedContent = ((aFlags & ITEM_IS_GENERATED_CONTENT) != 0);
 | |
| 
 | |
|   // 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->SetDisplayContents(aContent, styleContext);
 | |
|     } else {
 | |
|       aState.mFrameManager->ChangeDisplayContents(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(data, aContent, aTag, aNameSpaceID,
 | |
|                                 pendingBinding, styleContext.forget(),
 | |
|                                 aSuppressWhiteSpaceOptimizations, aAnonChildren);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!item) {
 | |
|     item = aItems.AppendItem(data, aContent, aTag, aNameSpaceID,
 | |
|                              pendingBinding, styleContext.forget(),
 | |
|                              aSuppressWhiteSpaceOptimizations, aAnonChildren);
 | |
|   }
 | |
|   item->mIsText = isText;
 | |
|   item->mIsGeneratedContent = isGeneratedContent;
 | |
|   item->mIsAnonymousContentCreatorContent =
 | |
|     aFlags & ITEM_IS_ANONYMOUSCONTENTCREATOR_CONTENT;
 | |
|   if (isGeneratedContent) {
 | |
|     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();
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void
 | |
| AddGenConPseudoToFrame(nsIFrame* aOwnerFrame, nsIContent* aContent)
 | |
| {
 | |
|   NS_ASSERTION(nsLayoutUtils::IsFirstContinuationOrIBSplitSibling(aOwnerFrame),
 | |
|                "property should only be set on first continuation/ib-sibling");
 | |
| 
 | |
|   // FIXME(emilio): Remove this property, and use the frame of the generated
 | |
|   // content itself to tear the content down? It should be quite simpler.
 | |
| 
 | |
|   nsIFrame::ContentArray* value =
 | |
|     aOwnerFrame->GetProperty(nsIFrame::GenConProperty());
 | |
|   if (!value) {
 | |
|     value = new nsIFrame::ContentArray;
 | |
|     aOwnerFrame->AddProperty(nsIFrame::GenConProperty(), value);
 | |
|   }
 | |
|   value->AppendElement(aContent);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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;
 | |
| 
 | |
|     // Note that we're not necessarily setting this property on the primary
 | |
|     // frame for the content for which this is generated content.  We might be
 | |
|     // setting it on a table pseudo-frame inserted under that instead.  That's
 | |
|     // OK, though; we just need to do the property set so that the content will
 | |
|     // get cleaned up when the frame is destroyed.
 | |
|     ::AddGenConPseudoToFrame(aParentFrame, item.mContent);
 | |
| 
 | |
|     // 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;
 | |
|   }
 | |
| 
 | |
|   // XXXbz maybe just inline ConstructFrameFromItemInternal here or something?
 | |
|   ConstructFrameFromItemInternal(item, aState, adjParentFrame, aFrameItems);
 | |
| 
 | |
|   aState.mAdditionalStateBits = savedStateBits;
 | |
| }
 | |
| 
 | |
| 
 | |
| inline bool
 | |
| IsRootBoxFrame(nsIFrame *aFrame)
 | |
| {
 | |
|   return (aFrame->IsRootFrame());
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::ReconstructDocElementHierarchy()
 | |
| {
 | |
|   Element* rootElement = mDocument->GetRootElement();
 | |
|   if (!rootElement) {
 | |
|     /* nothing to do */
 | |
|     return;
 | |
|   }
 | |
|   RecreateFramesForContent(rootElement, false, REMOVE_FOR_RECONSTRUCTION,
 | |
|                            nullptr);
 | |
| }
 | |
| 
 | |
| 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 has any pseudo-elements or aContainer is a
 | |
|   // display:contents node 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 (aParentFrame->GetProperty(nsIFrame::GenConProperty()) ||
 | |
|       nsLayoutUtils::HasPseudoStyle(aContainer, aParentFrame->StyleContext(),
 | |
|                                     CSSPseudoElementType::after,
 | |
|                                     aParentFrame->PresContext()) ||
 | |
|       aFrameManager->GetDisplayContentsStyleFor(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;
 | |
| }
 | |
| 
 | |
| nsIFrame*
 | |
| nsCSSFrameConstructor::FindFrameForContentSibling(nsIContent* aContent,
 | |
|                                                   nsIContent* aTargetContent,
 | |
|                                                   StyleDisplay& aTargetContentDisplay,
 | |
|                                                   nsContainerFrame* aParentFrame,
 | |
|                                                   bool aPrevSibling)
 | |
| {
 | |
|   nsIFrame* sibling = aContent->GetPrimaryFrame();
 | |
|   if (!sibling && GetDisplayContentsStyleFor(aContent)) {
 | |
|     // A display:contents node - check if it has a ::before / ::after frame...
 | |
|     sibling = aPrevSibling ? nsLayoutUtils::GetAfterFrame(aContent)
 | |
|                            : nsLayoutUtils::GetBeforeFrame(aContent);
 | |
|     if (!sibling) {
 | |
|       // ... then recurse into children ...
 | |
|       const bool forward = !aPrevSibling;
 | |
|       FlattenedChildIterator iter(aContent, forward);
 | |
|       sibling = aPrevSibling ?
 | |
|         FindPreviousSibling(iter, aTargetContent, aTargetContentDisplay, aParentFrame) :
 | |
|         FindNextSibling(iter, aTargetContent, aTargetContentDisplay, aParentFrame);
 | |
|     }
 | |
|     if (!sibling) {
 | |
|       // ... then ::after / ::before on the opposite end.
 | |
|       sibling = aPrevSibling ? nsLayoutUtils::GetAfterFrame(aContent)
 | |
|                              : nsLayoutUtils::GetBeforeFrame(aContent);
 | |
|     }
 | |
|     if (!sibling) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   } else if (!sibling || sibling->GetContent() != aContent) {
 | |
|     // XXX the GetContent() != aContent check is needed due to bug 135040.
 | |
|     // Remove it once that's fixed.
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   // If the frame is out-of-flow, GetPrimaryFrame() will have returned the
 | |
|   // out-of-flow frame; we want the placeholder.
 | |
|   if (sibling->GetStateBits() & NS_FRAME_OUT_OF_FLOW) {
 | |
|     nsIFrame* placeholderFrame = sibling->GetPlaceholderFrame();
 | |
|     NS_ASSERTION(placeholderFrame, "no placeholder for out-of-flow frame");
 | |
|     sibling = placeholderFrame;
 | |
|   }
 | |
| 
 | |
|   // The frame we have now should never be a continuation
 | |
|   NS_ASSERTION(!sibling->GetPrevContinuation(), "How did that happen?");
 | |
| 
 | |
|   if (aPrevSibling) {
 | |
|     // 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(sibling)) {
 | |
|       sibling = GetLastIBSplitSibling(sibling, true);
 | |
|     }
 | |
| 
 | |
|     // The frame may have a continuation. If so, we want the last
 | |
|     // non-overflow-container continuation as our previous sibling.
 | |
|     sibling = sibling->GetTailContinuation();
 | |
|   }
 | |
| 
 | |
|   if (aTargetContent &&
 | |
|       !IsValidSibling(sibling, aTargetContent, aTargetContentDisplay)) {
 | |
|     sibling = nullptr;
 | |
|   }
 | |
| 
 | |
|   return sibling;
 | |
| }
 | |
| 
 | |
| nsIFrame*
 | |
| nsCSSFrameConstructor::FindPreviousSibling(FlattenedChildIterator aIter,
 | |
|                                            nsIContent* aTargetContent,
 | |
|                                            StyleDisplay& aTargetContentDisplay,
 | |
|                                            nsContainerFrame* aParentFrame)
 | |
| {
 | |
|   // Note: not all content objects are associated with a frame (e.g., if it's
 | |
|   // `display: none') so keep looking until we find a previous frame.
 | |
|   while (nsIContent* sibling = aIter.GetPreviousChild()) {
 | |
|     MOZ_ASSERT(sibling != aTargetContent);
 | |
|     nsIFrame* prevSibling =
 | |
|       FindFrameForContentSibling(sibling, aTargetContent, aTargetContentDisplay,
 | |
|                                  aParentFrame, true);
 | |
|     if (prevSibling) {
 | |
|       // Found a previous sibling, we're done!
 | |
|       return prevSibling;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| nsIFrame*
 | |
| nsCSSFrameConstructor::FindNextSibling(FlattenedChildIterator aIter,
 | |
|                                        nsIContent* aTargetContent,
 | |
|                                        StyleDisplay& aTargetContentDisplay,
 | |
|                                        nsContainerFrame* aParentFrame)
 | |
| {
 | |
|   while (nsIContent* sibling = aIter.GetNextChild()) {
 | |
|     MOZ_ASSERT(sibling != aTargetContent);
 | |
|     nsIFrame* nextSibling =
 | |
|       FindFrameForContentSibling(sibling, aTargetContent, aTargetContentDisplay,
 | |
|                                  aParentFrame, false);
 | |
| 
 | |
|     if (nextSibling) {
 | |
|       // We found a next sibling, we're done!
 | |
|       return nextSibling;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   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, iter.Get(), childDisplay, aInsertion->mParentFrame);
 | |
| 
 | |
|   // 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
 | |
|     if (aEndSkipChild) {
 | |
|       iter.Seek(aEndSkipChild);
 | |
|       iter.GetPreviousChild();
 | |
|     }
 | |
|     nsIFrame* nextSibling =
 | |
|       FindNextSibling(iter, iter.Get(), childDisplay, aInsertion->mParentFrame);
 | |
|     if (GetDisplayContentsStyleFor(aInsertion->mContainer)) {
 | |
|       if (!nextSibling) {
 | |
|         // Our siblings (if any) does not have a frame to guide us.
 | |
|         // The frame for aChild should be inserted whereever a frame for
 | |
|         // the container would be inserted.  This is needed when inserting
 | |
|         // into nested display:contents nodes.
 | |
|         nsIContent* child = aInsertion->mContainer;
 | |
|         nsIContent* parent = child->GetParent();
 | |
|         aInsertion->mParentFrame =
 | |
|           ::GetAdjustedParentFrame(aInsertion->mParentFrame, parent);
 | |
|         InsertionPoint fakeInsertion(aInsertion->mParentFrame, parent);
 | |
|         nsIFrame* result = GetInsertionPrevSibling(&fakeInsertion, child, aIsAppend,
 | |
|                                                    aIsRangeInsertSafe, nullptr, nullptr);
 | |
|         MOZ_ASSERT(aInsertion->mParentFrame->GetContent() ==
 | |
|                    fakeInsertion.mParentFrame->GetContent());
 | |
|         // fakeInsertion.mParentFrame may now be a continuation of the frame
 | |
|         // we started with in the ctor above.
 | |
|         aInsertion->mParentFrame = fakeInsertion.mParentFrame;
 | |
|         return result;
 | |
|       }
 | |
| 
 | |
|       prevSibling = nextSibling->GetPrevSibling();
 | |
|     }
 | |
| 
 | |
|     if (nextSibling) {
 | |
|       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.  This MUST be done
 | |
|       // before the AdjustAppendParentForAfterContent call.
 | |
|       aInsertion->mParentFrame =
 | |
|         nsLayoutUtils::LastContinuationWithChild(aInsertion->mParentFrame);
 | |
|       // Deal with fieldsets
 | |
|       aInsertion->mParentFrame =
 | |
|         ::GetAdjustedParentFrame(aInsertion->mParentFrame, aChild);
 | |
|       nsIFrame* appendAfterFrame;
 | |
|       aInsertion->mParentFrame =
 | |
|         ::AdjustAppendParentForAfterContent(this, aInsertion->mContainer,
 | |
|                                             aInsertion->mParentFrame,
 | |
|                                             aChild, &appendAfterFrame);
 | |
|       prevSibling = ::FindAppendPrevSibling(aInsertion->mParentFrame, appendAfterFrame);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   *aIsRangeInsertSafe = (childDisplay == UNSET_DISPLAY);
 | |
|   return prevSibling;
 | |
| }
 | |
| 
 | |
| nsContainerFrame*
 | |
| nsCSSFrameConstructor::GetContentInsertionFrameFor(nsIContent* aContent)
 | |
| {
 | |
|   // Get the primary frame associated with the content
 | |
|   nsIFrame* frame = aContent->GetPrimaryFrame();
 | |
| 
 | |
|   if (!frame) {
 | |
|     if (GetDisplayContentsStyleFor(aContent)) {
 | |
|       nsIContent* parent = aContent->GetParent();
 | |
|       if (parent && parent == aContent->GetContainingShadow()) {
 | |
|         parent = parent->GetBindingParent();
 | |
|       }
 | |
|       frame = parent ? GetContentInsertionFrameFor(parent) : nullptr;
 | |
|     }
 | |
|     if (!frame) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   } else {
 | |
|     // 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, false);
 | |
| }
 | |
| 
 | |
| // For inserts aChild should be valid, for appends it should be null.
 | |
| // Returns true if this operation can be lazy, false if not.
 | |
| bool
 | |
| nsCSSFrameConstructor::MaybeConstructLazily(Operation aOperation,
 | |
|                                             nsIContent* aContainer,
 | |
|                                             nsIContent* aChild)
 | |
| {
 | |
|   // XXXmats no lazy frames for display:contents direct descendants yet
 | |
|   // (bug 979782).
 | |
|   if (mPresShell->GetPresContext()->IsChrome() || !aContainer ||
 | |
|       aContainer->IsInNativeAnonymousSubtree() || aContainer->IsXULElement() ||
 | |
|       GetDisplayContentsStyleFor(aContainer)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (aOperation == CONTENTINSERT) {
 | |
|     if (aChild->IsRootOfAnonymousSubtree() ||
 | |
|         (aChild->HasFlag(NODE_IS_IN_SHADOW_TREE) &&
 | |
|          !aChild->IsInNativeAnonymousSubtree()) ||
 | |
|         aChild->IsEditable() || 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() || child->IsEditable()) {
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // We can construct lazily; just need to set suitable bits in the content
 | |
|   // tree.
 | |
| 
 | |
|   // Walk up the tree setting the NODE_DESCENDANTS_NEED_FRAMES bit as we go.
 | |
|   nsIContent* content = aContainer;
 | |
| 
 | |
| #ifdef DEBUG
 | |
|   // 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.
 | |
|   //
 | |
|   // Also, it's fine if one of the nodes without primary frame is a display:
 | |
|   // contents node except if it's the direct ancestor of the children we're
 | |
|   // recreating frames for.
 | |
|   bool noPrimaryFrame = false;
 | |
|   bool needsFrameBitSet = false;
 | |
| #endif
 | |
|   while (content &&
 | |
|          !content->HasFlag(NODE_DESCENDANTS_NEED_FRAMES)) {
 | |
| #ifdef DEBUG
 | |
|     if (content->GetPrimaryFrame() && content->GetPrimaryFrame()->IsLeaf()) {
 | |
|       noPrimaryFrame = needsFrameBitSet = false;
 | |
|     }
 | |
|     if (!noPrimaryFrame && !content->GetPrimaryFrame() &&
 | |
|         !GetDisplayContentsStyleFor(content)) {
 | |
|       noPrimaryFrame = true;
 | |
|     }
 | |
|     if (!needsFrameBitSet && content->HasFlag(NODE_NEEDS_FRAME)) {
 | |
|       needsFrameBitSet = true;
 | |
|     }
 | |
| #endif
 | |
|     content->SetFlags(NODE_DESCENDANTS_NEED_FRAMES);
 | |
|     content = content->GetFlattenedTreeParent();
 | |
|   }
 | |
| #ifdef DEBUG
 | |
|   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
 | |
| 
 | |
|   // 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);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (mozilla::GeckoRestyleManager* geckoRM = RestyleManager()->GetAsGecko()) {
 | |
|     geckoRM->PostRestyleEventForLazyConstruction();
 | |
|   }
 | |
| 
 | |
|   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,
 | |
|                              false, // aAllowLazyConstruction
 | |
|                              true,  // aForReconstruction
 | |
|                              &aTreeMatchContext);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (inRun) {
 | |
|     ContentAppended(aContent, firstChildInRun,
 | |
|                     false, // aAllowLazyConstruction
 | |
|                     true,  // aForReconstruction
 | |
|                     &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,
 | |
|                                                     bool aAllowLazyConstruction,
 | |
|                                                     bool aForReconstruction)
 | |
| {
 | |
|   for (nsIContent* child = aStartChild;
 | |
|        child != aEndChild;
 | |
|        child = child->GetNextSibling()) {
 | |
|     if ((child->GetPrimaryFrame() || GetUndisplayedContent(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, aAllowLazyConstruction,
 | |
|                          aForReconstruction, nullptr);
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsCSSFrameConstructor::InsertionPoint
 | |
| nsCSSFrameConstructor::GetRangeInsertionPoint(nsIContent* aContainer,
 | |
|                                               nsIContent* aStartChild,
 | |
|                                               nsIContent* aEndChild,
 | |
|                                               bool aAllowLazyConstruction,
 | |
|                                               bool aForReconstruction)
 | |
| {
 | |
|   // 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.
 | |
|   }
 | |
| 
 | |
|   bool hasInsertion = false;
 | |
|   if (!insertionPoint.mMultiple) {
 | |
|     // XXXbz XBL2/sXBL issue
 | |
|     nsIDocument* document = aStartChild->GetComposedDoc();
 | |
|     // XXXbz how would |document| be null here?
 | |
|     if (document && aStartChild->GetXBLInsertionParent()) {
 | |
|       hasInsertion = true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (insertionPoint.mMultiple || hasInsertion) {
 | |
|     // We have an insertion point.  There are some additional tests we need to do
 | |
|     // in order to ensure that an append is a safe operation.
 | |
|     uint32_t childCount = 0;
 | |
| 
 | |
|     if (!insertionPoint.mMultiple) {
 | |
|       // We may need to make multiple ContentInserted calls instead.  A
 | |
|       // reasonable heuristic to employ (in order to maintain good performance)
 | |
|       // is to find out if the insertion point's content node contains any
 | |
|       // explicit children.  If it does not, then it is highly likely that
 | |
|       // an append is occurring.  (Note it is not definite, and there are insane
 | |
|       // cases we will not deal with by employing this heuristic, but it beats
 | |
|       // always falling back to multiple ContentInserted calls).
 | |
|       //
 | |
|       // In the multiple insertion point case, we know we're going to need to do
 | |
|       // multiple ContentInserted calls anyway.
 | |
|       // XXXndeakin This test doesn't work in the new world. Or rather, it works, but
 | |
|       // it's slow
 | |
|       childCount = insertionPoint.mParentFrame->GetContent()->GetChildCount();
 | |
|     }
 | |
| 
 | |
|     // 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 != nullptr || childCount > 0) {
 | |
|       // 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,
 | |
|                                    aAllowLazyConstruction, aForReconstruction);
 | |
|       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(), false,
 | |
|                                  REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|         return true;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::LazilyStyleNewChildRange(nsIContent* aStartChild,
 | |
|                                                 nsIContent* aEndChild)
 | |
| {
 | |
|   for (nsIContent* child = aStartChild; child != aEndChild;
 | |
|        child = child->GetNextSibling()) {
 | |
|     nsINode* parent = child->GetFlattenedTreeParent();
 | |
|     if (MOZ_LIKELY(parent) && parent->IsElement()) {
 | |
|       parent->AsElement()->NoteDirtyDescendantsForServo();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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::StyleChildRangeForReconstruct(nsIContent* aStartChild,
 | |
|                                                      nsIContent* aEndChild)
 | |
| {
 | |
|   ServoStyleSet* styleSet = mPresShell->StyleSet()->AsServo();
 | |
| 
 | |
|   // We take a parallelism hit here, since we don't have a great API to pass
 | |
|   // a range of elements to style to Servo.
 | |
|   for (nsIContent* child = aStartChild; child != aEndChild;
 | |
|        child = child->GetNextSibling()) {
 | |
|     if (child->IsElement()) {
 | |
|       styleSet->StyleSubtreeForReconstruct(child->AsElement());
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::ContentAppended(nsIContent* aContainer,
 | |
|                                        nsIContent* aFirstNewContent,
 | |
|                                        bool aAllowLazyConstruction,
 | |
|                                        bool aForReconstruction,
 | |
|                                        TreeMatchContext* aProvidedTreeMatchContext)
 | |
| {
 | |
|   MOZ_ASSERT(!aProvidedTreeMatchContext || !aAllowLazyConstruction);
 | |
|   MOZ_ASSERT(!aAllowLazyConstruction || !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,
 | |
|            aAllowLazyConstruction);
 | |
|     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;
 | |
|     nsIAtom* 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
 | |
| 
 | |
|   // The frame constructor uses this codepath both for bonafide newly-added
 | |
|   // content and for RestyleManager-driven frame construction (RECONSTRUCT_FRAME
 | |
|   // and lazy frame construction). If we're using the Servo style system, we
 | |
|   // want to ensure that styles get resolved in the first case, whereas for the
 | |
|   // second case they should have already been resolved if needed.
 | |
|   bool isNewlyAddedContentForServo = aContainer->IsStyledByServo() &&
 | |
|                                      !aForReconstruction;
 | |
| 
 | |
|   bool isNewShadowTreeContent =
 | |
|     aContainer && aContainer->HasFlag(NODE_IS_IN_SHADOW_TREE) &&
 | |
|     !aContainer->IsInNativeAnonymousSubtree() &&
 | |
|     !aFirstNewContent->IsInNativeAnonymousSubtree();
 | |
| 
 | |
|   if (!isNewShadowTreeContent) {
 | |
|     // 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.
 | |
|       if (isNewlyAddedContentForServo) {
 | |
|         LazilyStyleNewChildRange(aFirstNewContent, nullptr);
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (aAllowLazyConstruction &&
 | |
|         MaybeConstructLazily(CONTENTAPPEND, aContainer, aFirstNewContent)) {
 | |
|       if (isNewlyAddedContentForServo) {
 | |
|         LazilyStyleNewChildRange(aFirstNewContent, nullptr);
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // We couldn't construct lazily. Make Servo eagerly traverse the new content.
 | |
|   if (aContainer->IsStyledByServo()) {
 | |
|     if (aForReconstruction) {
 | |
|       StyleChildRangeForReconstruct(aFirstNewContent, nullptr);
 | |
|     } else {
 | |
|       StyleNewChildRange(aFirstNewContent, nullptr);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (isNewShadowTreeContent) {
 | |
|     // Recreate frames if content is appended into a ShadowRoot
 | |
|     // because children of ShadowRoot are rendered in place of children
 | |
|     // of the host.
 | |
|     //XXXsmaug This is super unefficient!
 | |
|     nsIContent* bindingParent = aContainer->GetBindingParent();
 | |
|     LAYOUT_PHASE_TEMP_EXIT();
 | |
|     RecreateFramesForContent(bindingParent, false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     LAYOUT_PHASE_TEMP_REENTER();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   LAYOUT_PHASE_TEMP_EXIT();
 | |
|   InsertionPoint insertion =
 | |
|     GetRangeInsertionPoint(aContainer, aFirstNewContent, nullptr,
 | |
|                            aAllowLazyConstruction, aForReconstruction);
 | |
|   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(), false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     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;
 | |
|   parentFrame =
 | |
|     ::AdjustAppendParentForAfterContent(this, insertion.mContainer, parentFrame,
 | |
|                                         aFirstNewContent, &parentAfterFrame);
 | |
| 
 | |
|   // 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()) {
 | |
|     matchContext.emplace(mDocument, TreeMatchContext::ForFrameConstruction);
 | |
|     matchContext->InitAncestors(aContainer->AsElement());
 | |
|   }
 | |
|   nsFrameConstructorState state(mPresShell,
 | |
|                                 matchContext.ptrOr(aProvidedTreeMatchContext),
 | |
|                                 GetAbsoluteContainingBlock(parentFrame, FIXED_POS),
 | |
|                                 GetAbsoluteContainingBlock(parentFrame, ABS_POS),
 | |
|                                 GetFloatContainingBlock(parentFrame));
 | |
| 
 | |
|   // See if the containing block has :first-letter style applied.
 | |
|   bool haveFirstLetterStyle = false, haveFirstLineStyle = false;
 | |
|   nsContainerFrame* containingBlock = state.mFloatedItems.containingBlock;
 | |
|   if (containingBlock) {
 | |
|     haveFirstLetterStyle = HasFirstLetterStyle(containingBlock);
 | |
|     haveFirstLineStyle =
 | |
|       ShouldHaveFirstLineStyle(containingBlock->GetContent(),
 | |
|                                containingBlock->StyleContext());
 | |
|   }
 | |
| 
 | |
|   if (haveFirstLetterStyle) {
 | |
|     // Before we get going, remove the current letter frames
 | |
|     RemoveLetterFrames(state.mPresShell, containingBlock);
 | |
|   }
 | |
| 
 | |
|   LayoutFrameType frameType = parentFrame->Type();
 | |
| 
 | |
|   FlattenedChildIterator iter(aContainer);
 | |
|   bool haveNoXBLChildren = (!iter.XBLInvolved() || !iter.GetNextChild());
 | |
|   FrameConstructionItemList items;
 | |
|   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, 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);
 | |
|   }
 | |
| 
 | |
|   // 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
 | |
|   nsAccessibilityService* accService = nsIPresShell::AccService();
 | |
|   if (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,
 | |
|                                        bool                   aAllowLazyConstruction)
 | |
| {
 | |
|   ContentRangeInserted(aContainer,
 | |
|                        aChild,
 | |
|                        aChild->GetNextSibling(),
 | |
|                        aFrameState,
 | |
|                        aAllowLazyConstruction);
 | |
| }
 | |
| 
 | |
| // 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,
 | |
|                                             bool aAllowLazyConstruction,
 | |
|                                             bool aForReconstruction,
 | |
|                                             TreeMatchContext* aProvidedTreeMatchContext)
 | |
| {
 | |
|   MOZ_ASSERT(!aProvidedTreeMatchContext || !aAllowLazyConstruction);
 | |
|   MOZ_ASSERT(!aAllowLazyConstruction || !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");
 | |
| 
 | |
|   // XXXldb Do we need to re-resolve style to handle the CSS2 + combinator and
 | |
|   // the :empty pseudo-class?
 | |
| #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),
 | |
|            aAllowLazyConstruction);
 | |
|     if (gReallyNoisyContentUpdates) {
 | |
|       if (aContainer) {
 | |
|         aContainer->List(stdout,0);
 | |
|       } else {
 | |
|         aStartChild->List(stdout, 0);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| #endif
 | |
| 
 | |
| #ifdef DEBUG
 | |
|   for (nsIContent* child = aStartChild;
 | |
|        child != aEndChild;
 | |
|        child = child->GetNextSibling()) {
 | |
|     // XXX the GetContent() != child check is needed due to bug 135040.
 | |
|     // Remove it once that's fixed.
 | |
|     NS_ASSERTION(!child->GetPrimaryFrame() ||
 | |
|                  child->GetPrimaryFrame()->GetContent() != child,
 | |
|                  "asked to construct a frame for a node that already has a frame");
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   bool isSingleInsert = (aStartChild->GetNextSibling() == aEndChild);
 | |
|   NS_ASSERTION(isSingleInsert || !aAllowLazyConstruction,
 | |
|                "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)) {
 | |
|     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,
 | |
|                                    aAllowLazyConstruction, aForReconstruction);
 | |
|       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(nullptr == mRootElementFrame,
 | |
|                     "root element frame already created");
 | |
| 
 | |
|     // Create frames for the document element and its child elements
 | |
|     nsIFrame* docElementFrame =
 | |
|       ConstructDocElementFrame(docElement, aFrameState);
 | |
| 
 | |
|     if (docElementFrame) {
 | |
|       InvalidateCanvasIfNeeded(mPresShell, aStartChild);
 | |
| #ifdef DEBUG
 | |
|       if (gReallyNoisyContentUpdates) {
 | |
|         printf("nsCSSFrameConstructor::ContentRangeInserted: resulting frame "
 | |
|                "model:\n");
 | |
|         docElementFrame->List(stdout, 0);
 | |
|       }
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     if (aFrameState) {
 | |
|       // Restore frame state for the root scroll frame if there is one
 | |
|       nsIFrame* rootScrollFrame = mPresShell->GetRootScrollFrame();
 | |
|       if (rootScrollFrame) {
 | |
|         RestoreFrameStateFor(rootScrollFrame, aFrameState);
 | |
|       }
 | |
|     }
 | |
| 
 | |
| #ifdef ACCESSIBILITY
 | |
|     nsAccessibilityService* accService = nsIPresShell::AccService();
 | |
|     if (accService) {
 | |
|       accService->ContentRangeInserted(mPresShell, aContainer,
 | |
|                                        aStartChild, aEndChild);
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // The frame constructor uses this codepath both for bonafide newly-added
 | |
|   // content and for RestyleManager-driven frame construction (RECONSTRUCT_FRAME
 | |
|   // and lazy frame construction). If we're using the Servo style system, we
 | |
|   // want to ensure that styles get resolved in the first case, whereas for the
 | |
|   // second case they should have already been resolved if needed.
 | |
|   bool isNewlyAddedContentForServo = aContainer->IsStyledByServo() &&
 | |
|                                      !aForReconstruction;
 | |
| 
 | |
|   bool isNewShadowTreeContent =
 | |
|     aContainer->HasFlag(NODE_IS_IN_SHADOW_TREE) &&
 | |
|     !aContainer->IsInNativeAnonymousSubtree() &&
 | |
|     (!aStartChild || !aStartChild->IsInNativeAnonymousSubtree()) &&
 | |
|     (!aEndChild || !aEndChild->IsInNativeAnonymousSubtree());
 | |
| 
 | |
|   if (!isNewShadowTreeContent) {
 | |
|     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()) {
 | |
|       // 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.
 | |
|       if (isNewlyAddedContentForServo) {
 | |
|         LazilyStyleNewChildRange(aStartChild, aEndChild);
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Otherwise, we've got parent content. Find its frame.
 | |
|     NS_ASSERTION(!parentFrame || parentFrame->GetContent() == aContainer ||
 | |
|                  GetDisplayContentsStyleFor(aContainer), "New XBL code is possibly wrong!");
 | |
| 
 | |
|     if (aAllowLazyConstruction &&
 | |
|         MaybeConstructLazily(CONTENTINSERT, aContainer, aStartChild)) {
 | |
|       if (isNewlyAddedContentForServo) {
 | |
|         LazilyStyleNewChildRange(aStartChild, aEndChild);
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // We couldn't construct lazily. Make Servo eagerly traverse the new content.
 | |
|   if (aContainer->IsStyledByServo()) {
 | |
|     if (aForReconstruction) {
 | |
|       StyleChildRangeForReconstruct(aStartChild, aEndChild);
 | |
|     } else {
 | |
|       StyleNewChildRange(aStartChild, aEndChild);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (isNewShadowTreeContent) {
 | |
|     // Recreate frames if content is inserted into a ShadowRoot
 | |
|     // because children of ShadowRoot are rendered in place of
 | |
|     // the children of the host.
 | |
|     //XXXsmaug This is super unefficient!
 | |
|     nsIContent* bindingParent = aContainer->GetBindingParent();
 | |
|     LAYOUT_PHASE_TEMP_EXIT();
 | |
|     RecreateFramesForContent(bindingParent, false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     LAYOUT_PHASE_TEMP_REENTER();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   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,
 | |
|                                        aAllowLazyConstruction,
 | |
|                                        aForReconstruction);
 | |
|     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,
 | |
|                                  aAllowLazyConstruction, aForReconstruction);
 | |
|     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(), false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     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(), false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     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(), false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     LAYOUT_PHASE_TEMP_REENTER();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Maybe<TreeMatchContext> matchContext;
 | |
|   if (!aProvidedTreeMatchContext && !aContainer->IsStyledByServo()) {
 | |
|     matchContext.emplace(mDocument, TreeMatchContext::ForFrameConstruction);
 | |
|     matchContext->InitAncestors(aContainer ? aContainer->AsElement() : nullptr);
 | |
|   }
 | |
|   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,
 | |
|                                      aAllowLazyConstruction, aForReconstruction);
 | |
|         LAYOUT_PHASE_TEMP_REENTER();
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       container = insertion.mParentFrame->GetContent();
 | |
|       frameType = insertion.mParentFrame->Type();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!prevSibling) {
 | |
|     // We're inserting the new frames as the first child. See if the
 | |
|     // parent has a :before pseudo-element
 | |
|     nsIFrame* firstChild = insertion.mParentFrame->PrincipalChildList().FirstChild();
 | |
| 
 | |
|     if (firstChild &&
 | |
|         nsLayoutUtils::IsGeneratedContentFor(container, firstChild,
 | |
|                                              nsCSSPseudoElements::before)) {
 | |
|       // Insert the new frames after the last continuation of the :before
 | |
|       prevSibling = firstChild->GetTailContinuation();
 | |
|       insertion.mParentFrame = prevSibling->GetParent()->GetContentInsertionFrame();
 | |
|       // Don't change isAppend here; we'll can call AppendFrames as needed, and
 | |
|       // the change to our prevSibling doesn't affect that.
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   FrameConstructionItemList items;
 | |
|   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, 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) {
 | |
|     // It's possible that the new frame goes into a first-line
 | |
|     // frame. Look at it and see...
 | |
|     if (isAppend) {
 | |
|       // Use append logic when appending
 | |
|       AppendFirstLineFrames(state, containingBlock->GetContent(),
 | |
|                             containingBlock, frameItems);
 | |
|     }
 | |
|     else {
 | |
|       // Use more complicated insert logic when inserting
 | |
|       // XXXbz this method is a no-op, so it's easy for the args being passed
 | |
|       // here to make no sense without anyone noticing...  If it ever stops
 | |
|       // being a no-op, vet them carefully!
 | |
|       InsertFirstLineFrames(state, container, containingBlock, &insertion.mParentFrame,
 | |
|                             prevSibling, frameItems);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // 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
 | |
|   nsAccessibilityService* accService = nsIPresShell::AccService();
 | |
|   if (accService) {
 | |
|     accService->ContentRangeInserted(mPresShell, aContainer,
 | |
|                                      aStartChild, aEndChild);
 | |
|   }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::ContentRemoved(nsIContent*  aContainer,
 | |
|                                       nsIContent*  aChild,
 | |
|                                       nsIContent*  aOldNextSibling,
 | |
|                                       RemoveFlags  aFlags,
 | |
|                                       bool*        aDidReconstruct,
 | |
|                                       nsIContent** aDestroyedFramesFor)
 | |
| {
 | |
|   AUTO_LAYOUT_PHASE_ENTRY_POINT(mPresShell->GetPresContext(), FrameC);
 | |
|   NS_PRECONDITION(mUpdateCount != 0,
 | |
|                   "Should be in an update while destroying frames");
 | |
| 
 | |
|   *aDidReconstruct = false;
 | |
|   if (aDestroyedFramesFor) {
 | |
|     *aDestroyedFramesFor = aChild;
 | |
|   }
 | |
| 
 | |
|   // We're destroying our frame(s). This normally happens either when the content
 | |
|   // is being removed from the DOM (in which case we'll drop all Servo data in
 | |
|   // UnbindFromTree), or when we're recreating frames (usually in response to
 | |
|   // having retrieved a ReconstructFrame change hint after restyling). In both of
 | |
|   // those cases, there are no pending restyles we need to worry about.
 | |
|   //
 | |
|   // However, there is also the (rare) DestroyFramesFor path, in which we tear
 | |
|   // down (and usually recreate) the frames for a subtree. In this case, leaving
 | |
|   // the style data on the elements is problematic for our invariants, because
 | |
|   // there might be pending restyles in the subtree. If we simply leave them as-is,
 | |
|   // the subsequent traversal when recreating frames will generate a bunch of bogus
 | |
|   // change hints to update frames that no longer exist.
 | |
|   //
 | |
|   // So the two obvious options are to (1) process all pending restyles and take all
 | |
|   // the change hints before destroying the frames, or (2) drop all the style data.
 | |
|   // We chose the latter, since that matches the performance characteristics of the
 | |
|   // old Gecko style system.
 | |
|   //
 | |
|   // That said, it's almost certainly possible to optimize this if it turns out to be
 | |
|   // hot. It's just not a priority at the moment.
 | |
|   if (aFlags == REMOVE_DESTROY_FRAMES && aChild->IsElement() && aChild->IsStyledByServo()) {
 | |
|     ServoRestyleManager::ClearServoDataFromSubtree(aChild->AsElement());
 | |
|   }
 | |
| 
 | |
|   nsPresContext* presContext = mPresShell->GetPresContext();
 | |
|   MOZ_ASSERT(presContext, "Our presShell should have a valid presContext");
 | |
| 
 | |
|   if (aChild->IsHTMLElement(nsGkAtoms::body) ||
 | |
|       (!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.)
 | |
|     presContext->UpdateViewportScrollbarStylesOverride();
 | |
|   }
 | |
| 
 | |
|   // XXXldb Do we need to re-resolve style to handle the CSS2 + combinator and
 | |
|   // the :empty pseudo-class?
 | |
| 
 | |
| #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.
 | |
|     ClearUndisplayedContentIn(aChild, aContainer);
 | |
|   }
 | |
|   MOZ_ASSERT(!childFrame || !GetDisplayContentsStyleFor(aChild),
 | |
|              "display:contents nodes shouldn't have a frame");
 | |
|   if (!childFrame && GetDisplayContentsStyleFor(aChild)) {
 | |
|     nsIContent* ancestor = aContainer;
 | |
|     MOZ_ASSERT(ancestor, "display: contents on the root?");
 | |
|     while (!ancestor->GetPrimaryFrame()) {
 | |
|       // FIXME(emilio): Should this use the flattened tree parent instead?
 | |
|       ancestor = ancestor->GetParent();
 | |
|       MOZ_ASSERT(ancestor, "we can't have a display: contents subtree root!");
 | |
|     }
 | |
| 
 | |
|     nsIFrame* ancestorFrame = ancestor->GetPrimaryFrame();
 | |
|     if (ancestorFrame->GetProperty(nsIFrame::GenConProperty())) {
 | |
|       *aDidReconstruct = true;
 | |
|       LAYOUT_PHASE_TEMP_EXIT();
 | |
| 
 | |
|       // XXXmats Can we recreate frames only for the ::after/::before content?
 | |
|       // XXX Perhaps even only those that belong to the aChild sub-tree?
 | |
|       RecreateFramesForContent(ancestor, false, aFlags, aDestroyedFramesFor);
 | |
|       LAYOUT_PHASE_TEMP_REENTER();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     FlattenedChildIterator iter(aChild);
 | |
|     for (nsIContent* c = iter.GetNextChild(); c; c = iter.GetNextChild()) {
 | |
|       if (c->GetPrimaryFrame() || GetDisplayContentsStyleFor(c)) {
 | |
|         LAYOUT_PHASE_TEMP_EXIT();
 | |
|         ContentRemoved(aChild, c, nullptr, aFlags, aDidReconstruct, aDestroyedFramesFor);
 | |
|         LAYOUT_PHASE_TEMP_REENTER();
 | |
|         if (aFlags != REMOVE_DESTROY_FRAMES && *aDidReconstruct) {
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     ClearDisplayContentsIn(aChild, aContainer);
 | |
|   }
 | |
| 
 | |
| #ifdef MOZ_XUL
 | |
|   if (NotifyListBoxBody(presContext, aContainer, aChild, aOldNextSibling,
 | |
|                         childFrame, CONTENT_REMOVED)) {
 | |
|     if (aFlags == REMOVE_DESTROY_FRAMES) {
 | |
|       CaptureStateForFramesOf(aChild, mTempFrameTreeState);
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
| #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 (aContainer && aContainer->HasFlag(NODE_IS_IN_SHADOW_TREE) &&
 | |
|       !aContainer->IsInNativeAnonymousSubtree() &&
 | |
|       !aChild->IsInNativeAnonymousSubtree()) {
 | |
|     // Recreate frames if content is removed from a ShadowRoot
 | |
|     // because it may contain an insertion point which can change
 | |
|     // how the host is rendered.
 | |
|     //XXXsmaug This is super unefficient!
 | |
|     nsIContent* bindingParent = aContainer->GetBindingParent();
 | |
|     *aDidReconstruct = true;
 | |
|     LAYOUT_PHASE_TEMP_EXIT();
 | |
|     RecreateFramesForContent(bindingParent, false, aFlags, aDestroyedFramesFor);
 | |
|     LAYOUT_PHASE_TEMP_REENTER();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (aFlags == REMOVE_DESTROY_FRAMES) {
 | |
|     CaptureStateForFramesOf(aChild, mTempFrameTreeState);
 | |
|   }
 | |
| 
 | |
|   if (childFrame) {
 | |
|     InvalidateCanvasIfNeeded(mPresShell, aChild);
 | |
| 
 | |
|     // See whether we need to remove more than just childFrame
 | |
|     LAYOUT_PHASE_TEMP_EXIT();
 | |
|     nsIContent* container;
 | |
|     if (MaybeRecreateContainerForFrameRemoval(childFrame, aFlags, &container)) {
 | |
|       LAYOUT_PHASE_TEMP_REENTER();
 | |
|       MOZ_ASSERT(container);
 | |
|       *aDidReconstruct = true;
 | |
|       if (aDestroyedFramesFor) {
 | |
|         *aDestroyedFramesFor = container;
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
|     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.
 | |
|       *aDidReconstruct = true;
 | |
|       LAYOUT_PHASE_TEMP_EXIT();
 | |
|       RecreateFramesForContent(parentFrame->GetContent(), false,
 | |
|                                aFlags, aDestroyedFramesFor);
 | |
|       LAYOUT_PHASE_TEMP_REENTER();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // 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)) {
 | |
|       *aDidReconstruct = true;
 | |
|       LAYOUT_PHASE_TEMP_EXIT();
 | |
|       RecreateFramesForContent(possibleMathMLAncestor->GetContent(),
 | |
|                                false, aFlags, aDestroyedFramesFor);
 | |
|       LAYOUT_PHASE_TEMP_REENTER();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // 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())) {
 | |
|       *aDidReconstruct = true;
 | |
|       LAYOUT_PHASE_TEMP_EXIT();
 | |
|       RecreateFramesForContent(grandparentFrame->GetContent(), true,
 | |
|                                aFlags, aDestroyedFramesFor);
 | |
|       LAYOUT_PHASE_TEMP_REENTER();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
| #ifdef ACCESSIBILITY
 | |
|     nsAccessibilityService* accService = nsIPresShell::AccService();
 | |
|     if (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.
 | |
|         ClearUndisplayedContentIn(aChild, aContainer);
 | |
|         return;
 | |
|       }
 | |
|       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.
 | |
|       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
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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();
 | |
| }
 | |
| 
 | |
| nsIFrame*
 | |
| nsCSSFrameConstructor::EnsureFrameForTextNode(nsGenericDOMDataNode* aContent)
 | |
| {
 | |
|   if (aContent->HasFlag(NS_CREATE_FRAME_IF_NON_WHITESPACE) &&
 | |
|       !mAlwaysCreateFramesForIgnorableWhitespace) {
 | |
|     // 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;
 | |
|     nsAutoScriptBlocker blocker;
 | |
|     BeginUpdate();
 | |
|     ReconstructDocElementHierarchy();
 | |
|     EndUpdate();
 | |
|   }
 | |
|   return aContent->GetPrimaryFrame();
 | |
| }
 | |
| 
 | |
| 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, false,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     LAYOUT_PHASE_TEMP_REENTER();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Find the child frame
 | |
|   nsIFrame* frame = aContent->GetPrimaryFrame();
 | |
| 
 | |
|   // Notify the first frame that maps the content. It will generate a reflow
 | |
|   // command
 | |
| 
 | |
|   // 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 (nullptr != frame) {
 | |
| #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!");
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     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,
 | |
|                                                 nsIAtom*       aTag,  // content object's tag
 | |
|                                                 nsXPIDLString& aAltText)
 | |
| {
 | |
|   // The "alt" attribute specifies alternate text that is rendered
 | |
|   // when the image can not be displayed
 | |
| 
 | |
|   // 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::alt, aAltText) &&
 | |
|       nsGkAtoms::input == aTag) {
 | |
|     // If there's no "value" attribute either, then use the localized string
 | |
|     // for "Submit" as the alternate text.
 | |
|     if (!aContent->GetAttr(kNameSpaceID_None, nsGkAtoms::value, aAltText)) {
 | |
|       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);
 | |
|   }
 | |
| 
 | |
|   NS_POSTCONDITION(!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();
 | |
|       FrameConstructionItemList items;
 | |
|       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, 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)) {
 | |
|       // The container distributes nodes, 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 = GetUndisplayedContent(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->GetParent(),
 | |
|                     LazyComputeBehavior::Assert);
 | |
| 
 | |
|   if (oldDisplay == StyleDisplay::None) {
 | |
|     ChangeUndisplayedContent(aElement, newContext);
 | |
|   } else {
 | |
|     ChangeDisplayContents(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, false, REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|   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,
 | |
|                                                              RemoveFlags aFlags,
 | |
|                                                              nsIContent** aDestroyedFramesFor)
 | |
| {
 | |
|   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()?");
 | |
| 
 | |
|   *aDestroyedFramesFor = nullptr;
 | |
| 
 | |
|   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, aFlags, aDestroyedFramesFor);
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   nsContainerFrame* insertionFrame = aFrame->GetContentInsertionFrame();
 | |
|   if (insertionFrame && insertionFrame->IsLegendFrame() &&
 | |
|       aFrame->GetParent()->IsFieldSetFrame()) {
 | |
|     // When we remove the legend for a fieldset, we should reframe
 | |
|     // the fieldset to ensure another legend is used, if there is one
 | |
|     RecreateFramesForContent(aFrame->GetParent()->GetContent(), false,
 | |
|                              aFlags, aDestroyedFramesFor);
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   if (insertionFrame && aFrame->GetParent()->IsDetailsFrame()) {
 | |
|     HTMLSummaryElement* summary =
 | |
|       HTMLSummaryElement::FromContent(insertionFrame->GetContent());
 | |
| 
 | |
|     if (summary && summary->IsMainSummary()) {
 | |
|       // 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(aFrame->GetParent()->GetContent(),
 | |
|                                false, REMOVE_FOR_RECONSTRUCTION,
 | |
|                                aDestroyedFramesFor);
 | |
|       return true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Now check for possibly needing to reconstruct due to a pseudo parent
 | |
|   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();
 | |
|   // 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)) {
 | |
|       // We're the first or last frame in the pseudo.  Need to reframe.
 | |
|       // Good enough to recreate frames for |parent|'s content
 | |
|       RecreateFramesForContent(parent->GetContent(), true, aFlags,
 | |
|                                aDestroyedFramesFor);
 | |
|       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(), true, aFlags,
 | |
|                                aDestroyedFramesFor);
 | |
|       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(), true, aFlags,
 | |
|                              aDestroyedFramesFor);
 | |
|     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(), true, aFlags,
 | |
|                              aDestroyedFramesFor);
 | |
|     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(), true,
 | |
|                              aFlags, aDestroyedFramesFor);
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
| #ifdef MOZ_XUL
 | |
|   if (aFrame->IsPopupSetFrame()) {
 | |
|     nsIRootBox* rootBox = nsIRootBox::GetRootBox(mPresShell);
 | |
|     if (rootBox && rootBox->GetPopupSetFrame() == aFrame) {
 | |
|       ReconstructDocElementHierarchy();
 | |
|       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(), true, aFlags,
 | |
|                              aDestroyedFramesFor);
 | |
|     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, aFlags, aDestroyedFramesFor);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::RecreateFramesForContent(nsIContent*  aContent,
 | |
|                                                 bool         aAsyncInsert,
 | |
|                                                 RemoveFlags  aFlags,
 | |
|                                                 nsIContent** aDestroyedFramesFor)
 | |
| {
 | |
|   NS_PRECONDITION(!aAsyncInsert || aContent->IsElement(),
 | |
|                   "Can only insert elements async");
 | |
|   // 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(), aAsyncInsert,
 | |
|                                       aFlags, aDestroyedFramesFor);
 | |
|     }
 | |
| 
 | |
|     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(), aAsyncInsert,
 | |
|                                         aFlags, aDestroyedFramesFor);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     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, aAsyncInsert, aFlags,
 | |
|                                       aDestroyedFramesFor);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   nsIContent* container;
 | |
|   if (frame && MaybeRecreateContainerForFrameRemoval(frame, aFlags,
 | |
|                                                      &container)) {
 | |
|     MOZ_ASSERT(container);
 | |
|     if (aDestroyedFramesFor) {
 | |
|       *aDestroyedFramesFor = container;
 | |
|     }
 | |
|     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.
 | |
|     nsCOMPtr<nsIContent> container = aContent->GetParent();
 | |
| 
 | |
|     // Remove the frames associated with the content object.
 | |
|     bool didReconstruct;
 | |
|     nsIContent* nextSibling = aContent->IsRootOfAnonymousSubtree() ?
 | |
|       nullptr : aContent->GetNextSibling();
 | |
|     const bool reconstruct = aFlags != REMOVE_DESTROY_FRAMES;
 | |
|     RemoveFlags flags = reconstruct ? REMOVE_FOR_RECONSTRUCTION : aFlags;
 | |
|     ContentRemoved(container, aContent, nextSibling, flags,
 | |
|                    &didReconstruct, aDestroyedFramesFor);
 | |
|     if (reconstruct && !didReconstruct) {
 | |
|       // 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.
 | |
|       if (aAsyncInsert) {
 | |
|         // XXXmats doesn't frame state need to be restored in this case too?
 | |
|         RestyleManager()->PostRestyleEvent(
 | |
|           aContent->AsElement(), nsRestyleHint(0), nsChangeHint_ReconstructFrame);
 | |
|       } else {
 | |
|         ContentRangeInserted(container, aContent, aContent->GetNextSibling(),
 | |
|                              mTempFrameTreeState,
 | |
|                              false, // aAllowLazyConstruction
 | |
|                              true,  // aForReconstruction
 | |
|                              nullptr);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::DestroyFramesFor(nsIContent*  aContent,
 | |
|                                         nsIContent** aDestroyedFramesFor)
 | |
| {
 | |
|   MOZ_ASSERT(aContent && aContent->GetParentNode());
 | |
| 
 | |
|   bool didReconstruct;
 | |
|   nsIContent* nextSibling =
 | |
|     aContent->IsRootOfAnonymousSubtree() ? nullptr : aContent->GetNextSibling();
 | |
|   ContentRemoved(aContent->GetParent(), aContent, nextSibling,
 | |
|                  REMOVE_DESTROY_FRAMES, &didReconstruct, aDestroyedFramesFor);
 | |
| }
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| // 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;
 | |
|     nsIAtom* 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_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_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_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 |
 | |
|                 FCDATA_DESIRED_PARENT_TYPE_TO_BITS(eTypeTable),
 | |
|                 NS_NewTableColGroupFrame),
 | |
|     &nsCSSAnonBoxes::tableColGroup
 | |
|   },
 | |
|   { // Table
 | |
|     FULL_CTOR_FCDATA(FCDATA_SKIP_FRAMESET | FCDATA_USE_CHILD_ITEMS,
 | |
|                      &nsCSSFrameConstructor::ConstructTable),
 | |
|     &nsCSSAnonBoxes::table
 | |
|   },
 | |
|   { // Ruby
 | |
|     FCDATA_DECL(FCDATA_IS_LINE_PARTICIPANT |
 | |
|                 FCDATA_USE_CHILD_ITEMS |
 | |
|                 FCDATA_SKIP_FRAMESET,
 | |
|                 NS_NewRubyFrame),
 | |
|     &nsCSSAnonBoxes::ruby
 | |
|   },
 | |
|   { // Ruby Base
 | |
|     FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
 | |
|                 FCDATA_IS_LINE_PARTICIPANT |
 | |
|                 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_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_DESIRED_PARENT_TYPE_TO_BITS(eTypeRubyTextContainer) |
 | |
|                 FCDATA_SKIP_FRAMESET,
 | |
|                 NS_NewRubyTextFrame),
 | |
|     &nsCSSAnonBoxes::rubyText
 | |
|   },
 | |
|   { // Ruby Text Container
 | |
|     FCDATA_DECL(FCDATA_USE_CHILD_ITEMS |
 | |
|                 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(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|.
 | |
|     nsIAtom* 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,
 | |
|                   NS_NewBlockFormattingContext);
 | |
| 
 | |
|     FrameConstructionItem* newItem =
 | |
|       new 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(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(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(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(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(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];
 | |
|   nsIAtom* 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 =
 | |
|     mPresShell->StyleSet()->ResolveInheritingAnonymousBoxStyle(pseudoType,
 | |
|                                                                aParentStyle);
 | |
|   FrameConstructionItem* newItem =
 | |
|     new 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(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 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,
 | |
|                                                    nsFrameItems& aFrameItems)
 | |
| {
 | |
|   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");
 | |
| }
 | |
| 
 | |
| 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.
 | |
|     nsIAtom* 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);
 | |
| 
 | |
|   FrameConstructionItemList itemsToConstruct;
 | |
| 
 | |
|   // 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, 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);
 | |
| 
 | |
|     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);
 | |
|   }
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| 
 | |
| // 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);
 | |
|   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);
 | |
| }
 | |
| 
 | |
| // Special routine to handle inserting a new frame into 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::InsertFirstLineFrames(
 | |
|   nsFrameConstructorState& aState,
 | |
|   nsIContent*              aContent,
 | |
|   nsIFrame*                aBlockFrame,
 | |
|   nsContainerFrame**       aParentFrame,
 | |
|   nsIFrame*                aPrevSibling,
 | |
|   nsFrameItems&            aFrameItems)
 | |
| {
 | |
|   // XXXbz If you make this method actually do something, check to
 | |
|   // make sure that the caller is passing what you expect.  In
 | |
|   // particular, which content is aContent?  And audit the rest of
 | |
|   // this code too; it makes bogus assumptions and may not build.
 | |
| #if 0
 | |
|   nsIFrame* parentFrame = *aParentFrame;
 | |
|   nsIFrame* newFrame = aFrameItems.childList;
 | |
|   bool isInline = IsInlineOutside(newFrame);
 | |
| 
 | |
|   if (!aPrevSibling) {
 | |
|     // Insertion will become the first frame. Two cases: we either
 | |
|     // already have a first-line frame or we don't.
 | |
|     nsIFrame* firstBlockKid = aBlockFrame->PrincipalChildList().FirstChild();
 | |
|     if (firstBlockKid->IsLineFrame()) {
 | |
|       // We already have a first-line frame
 | |
|       nsIFrame* lineFrame = firstBlockKid;
 | |
| 
 | |
|       if (isInline) {
 | |
|         // Easy case: the new inline frame will go into the lineFrame.
 | |
|         ReparentFrame(this, lineFrame, newFrame);
 | |
|         InsertFrames(lineFrame, kPrincipalList, nullptr, newFrame);
 | |
| 
 | |
|         // Since the frame is going into the lineFrame, don't let it
 | |
|         // go into the block too.
 | |
|         aFrameItems.childList = nullptr;
 | |
|         aFrameItems.lastChild = nullptr;
 | |
|       }
 | |
|       else {
 | |
|         // Harder case: We are about to insert a block level element
 | |
|         // before the first-line frame.
 | |
|         // XXX need a method to steal away frames from the line-frame
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       // We do not have a first-line frame
 | |
|       if (isInline) {
 | |
|         // We now need a first-line frame to contain the inline frame.
 | |
|         nsIFrame* lineFrame = NS_NewFirstLineFrame(firstLineStyle);
 | |
| 
 | |
|         if (NS_SUCCEEDED(rv)) {
 | |
|           // Lookup first-line style context
 | |
|           nsStyleContext* parentStyle =
 | |
|             nsFrame::CorrectStyleParentFrame(aBlockFrame,
 | |
|                                              nsCSSPseudoElements::firstLine)->
 | |
|               StyleContext();
 | |
|           RefPtr<nsStyleContext> firstLineStyle =
 | |
|             GetFirstLineStyle(aContent, parentStyle);
 | |
| 
 | |
|           // Initialize the line frame
 | |
|           InitAndRestoreFrame(aState, aContent, aBlockFrame, lineFrame);
 | |
| 
 | |
|           // Make sure the caller inserts the lineFrame into the
 | |
|           // blocks list of children.
 | |
|           aFrameItems.childList = lineFrame;
 | |
|           aFrameItems.lastChild = lineFrame;
 | |
| 
 | |
|           // Give the inline frames to the lineFrame <b>after</b>
 | |
|           // reparenting them
 | |
|           NS_ASSERTION(lineFrame->StyleContext() == firstLineStyle,
 | |
|                        "Bogus style context on line frame");
 | |
|           ReparentFrame(aPresContext, lineFrame, newFrame);
 | |
|           lineFrame->SetInitialChildList(kPrincipalList, newFrame);
 | |
|         }
 | |
|       }
 | |
|       else {
 | |
|         // Easy case: the regular insertion logic can insert the new
 | |
|         // frame because it's a block frame.
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   else {
 | |
|     // Insertion will not be the first frame.
 | |
|     nsIFrame* prevSiblingParent = aPrevSibling->GetParent();
 | |
|     if (prevSiblingParent == aBlockFrame) {
 | |
|       // Easy case: The prev-siblings parent is the block
 | |
|       // frame. Therefore the prev-sibling is not currently in a
 | |
|       // line-frame. Therefore the new frame which is going after it,
 | |
|       // regardless of type, is not going into a line-frame.
 | |
|     }
 | |
|     else {
 | |
|       // If the prevSiblingParent is not the block-frame then it must
 | |
|       // be a line-frame (if it were a letter-frame, that logic would
 | |
|       // already have adjusted the prev-sibling to be the
 | |
|       // letter-frame).
 | |
|       if (isInline) {
 | |
|         // Easy case: the insertion can go where the caller thinks it
 | |
|         // should go (which is into prevSiblingParent).
 | |
|       }
 | |
|       else {
 | |
|         // Block elements don't end up in line-frames, therefore
 | |
|         // change the insertion point to aBlockFrame. However, there
 | |
|         // might be more inline elements following aPrevSibling that
 | |
|         // need to be pulled out of the line-frame and become children
 | |
|         // of the block.
 | |
|         nsIFrame* nextSibling = aPrevSibling->GetNextSibling();
 | |
|         nsIFrame* nextLineFrame = prevSiblingParent->GetNextInFlow();
 | |
|         if (nextSibling || nextLineFrame) {
 | |
|           // Oy. We have work to do. Create a list of the new frames
 | |
|           // that are going into the block by stripping them away from
 | |
|           // the line-frame(s).
 | |
|           if (nextSibling) {
 | |
|             nsLineFrame* lineFrame = (nsLineFrame*) prevSiblingParent;
 | |
|             nsFrameList tail = lineFrame->StealFramesAfter(aPrevSibling);
 | |
|             // XXX do something with 'tail'
 | |
|           }
 | |
| 
 | |
|           nsLineFrame* nextLineFrame = (nsLineFrame*) lineFrame;
 | |
|           for (;;) {
 | |
|             nextLineFrame = nextLineFrame->GetNextInFlow();
 | |
|             if (!nextLineFrame) {
 | |
|               break;
 | |
|             }
 | |
|             nsIFrame* kids = nextLineFrame->PrincipalChildList().FirstChild();
 | |
|           }
 | |
|         }
 | |
|         else {
 | |
|           // We got lucky: aPrevSibling was the last inline frame in
 | |
|           // the line-frame.
 | |
|           ReparentFrame(this, aBlockFrame, newFrame);
 | |
|           InsertFrames(aBlockFrame, kPrincipalList,
 | |
|                        prevSiblingParent, newFrame);
 | |
|           aFrameItems.childList = nullptr;
 | |
|           aFrameItems.lastChild = nullptr;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
| #endif
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| 
 | |
| // 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* aStyleContext,
 | |
|   nsFrameItems& aResult)
 | |
| {
 | |
|   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);
 | |
|     // Repair the continuations style context
 | |
|     nsStyleContext* parentStyleContext = aStyleContext->GetParentAllowServo();
 | |
|     if (parentStyleContext) {
 | |
|       RefPtr<nsStyleContext> newSC = styleSet->
 | |
|         ResolveStyleForText(aTextContent, parentStyleContext);
 | |
|       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.
 | |
|   nsStyleContext* parentStyleContext =
 | |
|     nsFrame::CorrectStyleParentFrame(aParentFrame,
 | |
|                                      nsCSSPseudoElements::firstLetter)->
 | |
|       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) {
 | |
|     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, 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?");
 | |
|     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;
 | |
|   }
 | |
| 
 | |
|   // 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.
 | |
|       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 parents 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.get()));
 | |
| 
 | |
|     // 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();
 | |
| 
 | |
|     FrameConstructionItemList items;
 | |
|     AddFrameConstructionItemsInternal(state, aChild, aParentFrame,
 | |
|                                       aChild->NodeInfo()->NameAtom(),
 | |
|                                       aChild->GetNameSpaceID(),
 | |
|                                       true, styleContext,
 | |
|                                       ITEM_ALLOW_XBL_BASE, nullptr, items);
 | |
|     ConstructFramesFromItemList(state, items, aParentFrame, 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, 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(), true,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     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(), true,
 | |
|                                REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|       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(), true,
 | |
|                                  REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|         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(), true,
 | |
|                                REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|       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(), true,
 | |
|                              REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|     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(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(), true,
 | |
|                                REMOVE_FOR_RECONSTRUCTION, nullptr);
 | |
|       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, true, REMOVE_FOR_RECONSTRUCTION,
 | |
|                            nullptr);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::ReframeContainingBlock(nsIFrame*    aFrame,
 | |
|                                               RemoveFlags  aFlags,
 | |
|                                               nsIContent** aDestroyedFramesFor)
 | |
| {
 | |
| 
 | |
| #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
 | |
|     nsCOMPtr<nsIContent> blockContent = containingBlock->GetContent();
 | |
|     if (blockContent) {
 | |
| #ifdef DEBUG
 | |
|       if (gNoisyContentUpdates) {
 | |
|         printf("  ==> blockContent=%p\n", static_cast<void*>(blockContent));
 | |
|       }
 | |
| #endif
 | |
|       return RecreateFramesForContent(blockContent, true, aFlags, aDestroyedFramesFor);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If we get here, we're screwed!
 | |
|   return RecreateFramesForContent(mPresShell->GetDocument()->GetRootElement(),
 | |
|                                   true, aFlags, nullptr);
 | |
| }
 | |
| 
 | |
| 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(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.~FrameConstructionItemList();
 | |
|   new (&mList) FrameConstructionItemList();
 | |
| 
 | |
|   // Point ourselves to aEnd, as advertised
 | |
|   SetToEnd();
 | |
|   NS_POSTCONDITION(*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);
 | |
| 
 | |
|   NS_POSTCONDITION(aItem->getNext() == mCurrent, "How did that happen?");
 | |
| }
 | |
| 
 | |
| void
 | |
| nsCSSFrameConstructor::FrameConstructionItemList::
 | |
| Iterator::DeleteItemsTo(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);
 | |
|     delete item;
 | |
|   } 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();
 | |
| }
 | 
