/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ /* rendering object for CSS "display: grid | inline-grid" */ #ifndef nsGridContainerFrame_h___ #define nsGridContainerFrame_h___ #include "mozilla/CSSOrderAwareFrameIterator.h" #include "mozilla/IntrinsicISizesCache.h" #include "mozilla/MathAlgorithms.h" #include "mozilla/Maybe.h" #include "mozilla/HashTable.h" #include "nsAtomHashKeys.h" #include "nsContainerFrame.h" #include "nsILineIterator.h" namespace mozilla { class PresShell; namespace dom { class Grid; } } // namespace mozilla /** * Factory function. * @return a newly allocated nsGridContainerFrame (infallible) */ nsContainerFrame* NS_NewGridContainerFrame(mozilla::PresShell* aPresShell, mozilla::ComputedStyle* aStyle); namespace mozilla { /** * The number of implicit / explicit tracks and their sizes. */ struct ComputedGridTrackInfo { ComputedGridTrackInfo( uint32_t aNumLeadingImplicitTracks, uint32_t aNumExplicitTracks, uint32_t aStartFragmentTrack, uint32_t aEndFragmentTrack, nsTArray&& aPositions, nsTArray&& aSizes, nsTArray&& aStates, nsTArray&& aRemovedRepeatTracks, uint32_t aRepeatFirstTrack, nsTArray>&& aResolvedLineNames, bool aIsSubgrid, bool aIsMasonry) : mNumLeadingImplicitTracks(aNumLeadingImplicitTracks), mNumExplicitTracks(aNumExplicitTracks), mStartFragmentTrack(aStartFragmentTrack), mEndFragmentTrack(aEndFragmentTrack), mPositions(std::move(aPositions)), mSizes(std::move(aSizes)), mStates(std::move(aStates)), mRemovedRepeatTracks(std::move(aRemovedRepeatTracks)), mResolvedLineNames(std::move(aResolvedLineNames)), mRepeatFirstTrack(aRepeatFirstTrack), mIsSubgrid(aIsSubgrid), mIsMasonry(aIsMasonry) {} uint32_t mNumLeadingImplicitTracks; uint32_t mNumExplicitTracks; uint32_t mStartFragmentTrack; uint32_t mEndFragmentTrack; nsTArray mPositions; nsTArray mSizes; nsTArray mStates; // Indicates if a track has been collapsed. This will be populated for each // track in the repeat(auto-fit) and repeat(auto-fill), even if there are no // collapsed tracks. nsTArray mRemovedRepeatTracks; // Contains lists of all line name lists, including the name lists inside // repeats. When a repeat(auto) track exists, the internal track names will // appear once each in this array. nsTArray> mResolvedLineNames; uint32_t mRepeatFirstTrack; bool mIsSubgrid; bool mIsMasonry; }; struct ComputedGridLineInfo { explicit ComputedGridLineInfo( nsTArray>>&& aNames, const nsTArray>& aNamesBefore, const nsTArray>& aNamesAfter, nsTArray>&& aNamesFollowingRepeat) : mNames(std::move(aNames)), mNamesBefore(aNamesBefore.Clone()), mNamesAfter(aNamesAfter.Clone()), mNamesFollowingRepeat(std::move(aNamesFollowingRepeat)) {} nsTArray>> mNames; nsTArray> mNamesBefore; nsTArray> mNamesAfter; nsTArray> mNamesFollowingRepeat; }; } // namespace mozilla class nsGridContainerFrame final : public nsContainerFrame, public nsILineIterator { public: NS_DECL_FRAMEARENA_HELPERS(nsGridContainerFrame) NS_DECL_QUERYFRAME using ComputedGridTrackInfo = mozilla::ComputedGridTrackInfo; using ComputedGridLineInfo = mozilla::ComputedGridLineInfo; using LogicalAxis = mozilla::LogicalAxis; using BaselineSharingGroup = mozilla::BaselineSharingGroup; using NamedArea = mozilla::StyleNamedArea; template using PerBaseline = mozilla::EnumeratedArray; template using PerLogicalAxis = mozilla::EnumeratedArray; // nsIFrame overrides void Reflow(nsPresContext* aPresContext, ReflowOutput& aDesiredSize, const ReflowInput& aReflowInput, nsReflowStatus& aStatus) override; void Init(nsIContent* aContent, nsContainerFrame* aParent, nsIFrame* aPrevInFlow) override; void DidSetComputedStyle(ComputedStyle* aOldStyle) override; nscoord IntrinsicISize(const mozilla::IntrinsicSizeInput& aInput, mozilla::IntrinsicISizeType aType) override; void MarkIntrinsicISizesDirty() override; void BuildDisplayList(nsDisplayListBuilder* aBuilder, const nsDisplayListSet& aLists) override; Maybe GetNaturalBaselineBOffset( mozilla::WritingMode aWM, BaselineSharingGroup aBaselineGroup, BaselineExportContext) const override { if (StyleDisplay()->IsContainLayout() || HasAnyStateBits(NS_STATE_GRID_SYNTHESIZE_BASELINE)) { return Nothing{}; } return mozilla::Some(GetBBaseline(aBaselineGroup)); } #ifdef DEBUG_FRAME_DUMP nsresult GetFrameName(nsAString& aResult) const override; void ExtraContainerFrameInfo(nsACString& aTo) const override; #endif // nsContainerFrame overrides bool DrainSelfOverflowList() override; void AppendFrames(ChildListID aListID, nsFrameList&& aFrameList) override; void InsertFrames(ChildListID aListID, nsIFrame* aPrevFrame, const nsLineList::iterator* aPrevFrameLine, nsFrameList&& aFrameList) override; void RemoveFrame(DestroyContext&, ChildListID, nsIFrame*) override; #ifdef DEBUG void SetInitialChildList(ChildListID aListID, nsFrameList&& aChildList) override; #endif /** * Return the containing block for aChild which MUST be an abs.pos. child * of a grid container and that container must have been reflowed. */ static const nsRect& GridItemCB(nsIFrame* aChild); NS_DECLARE_FRAME_PROPERTY_DELETABLE(GridItemContainingBlockRect, nsRect) /** * These properties are created by a call to * nsGridContainerFrame::GetGridFrameWithComputedInfo, typically from * Element::GetGridFragments. */ NS_DECLARE_FRAME_PROPERTY_DELETABLE(GridColTrackInfo, ComputedGridTrackInfo) const ComputedGridTrackInfo* GetComputedTemplateColumns() { const ComputedGridTrackInfo* info = GetProperty(GridColTrackInfo()); MOZ_ASSERT(info, "Property generation wasn't requested."); return info; } NS_DECLARE_FRAME_PROPERTY_DELETABLE(GridRowTrackInfo, ComputedGridTrackInfo) const ComputedGridTrackInfo* GetComputedTemplateRows() { const ComputedGridTrackInfo* info = GetProperty(GridRowTrackInfo()); MOZ_ASSERT(info, "Property generation wasn't requested."); return info; } NS_DECLARE_FRAME_PROPERTY_DELETABLE(GridColumnLineInfo, ComputedGridLineInfo) const ComputedGridLineInfo* GetComputedTemplateColumnLines() { const ComputedGridLineInfo* info = GetProperty(GridColumnLineInfo()); MOZ_ASSERT(info, "Property generation wasn't requested."); return info; } NS_DECLARE_FRAME_PROPERTY_DELETABLE(GridRowLineInfo, ComputedGridLineInfo) const ComputedGridLineInfo* GetComputedTemplateRowLines() { const ComputedGridLineInfo* info = GetProperty(GridRowLineInfo()); MOZ_ASSERT(info, "Property generation wasn't requested."); return info; } /** * This property is set by the creation of a dom::Grid object, and cleared * during GC unlink. Since the Grid object manages the lifecycle, the property * itself is set without a destructor. The property is also cleared whenever * new grid computed info is generated during reflow, ensuring that we aren't * holding a stale dom::Grid object. */ NS_DECLARE_FRAME_PROPERTY_WITHOUT_DTOR(GridFragmentInfo, mozilla::dom::Grid) mozilla::dom::Grid* GetGridFragmentInfo() { return GetProperty(GridFragmentInfo()); } using ImplicitNamedAreas = mozilla::HashMap; NS_DECLARE_FRAME_PROPERTY_DELETABLE(ImplicitNamedAreasProperty, ImplicitNamedAreas) ImplicitNamedAreas* GetImplicitNamedAreas() const { return GetProperty(ImplicitNamedAreasProperty()); } using ExplicitNamedAreas = mozilla::StyleOwnedSlice; NS_DECLARE_FRAME_PROPERTY_DELETABLE(ExplicitNamedAreasProperty, ExplicitNamedAreas) ExplicitNamedAreas* GetExplicitNamedAreas() const { return GetProperty(ExplicitNamedAreasProperty()); } using nsContainerFrame::IsMasonry; /** Return true if this frame has masonry layout in any axis. */ bool IsMasonry() const { return HasAnyStateBits(NS_STATE_GRID_IS_ROW_MASONRY | NS_STATE_GRID_IS_COL_MASONRY); } /** Return true if this frame is subgridded in its aAxis. */ bool IsSubgrid(LogicalAxis aAxis) const { return HasAnyStateBits(aAxis == mozilla::LogicalAxis::Block ? NS_STATE_GRID_IS_ROW_SUBGRID : NS_STATE_GRID_IS_COL_SUBGRID); } bool IsColSubgrid() const { return IsSubgrid(mozilla::LogicalAxis::Inline); } bool IsRowSubgrid() const { return IsSubgrid(mozilla::LogicalAxis::Block); } /** Return true if this frame is subgridded in any axis. */ bool IsSubgrid() const { return HasAnyStateBits(NS_STATE_GRID_IS_ROW_SUBGRID | NS_STATE_GRID_IS_COL_SUBGRID); } /** Return true if this frame has an item that is subgridded in our aAxis. */ bool HasSubgridItems(LogicalAxis aAxis) const { return HasAnyStateBits(aAxis == mozilla::LogicalAxis::Block ? NS_STATE_GRID_HAS_ROW_SUBGRID_ITEM : NS_STATE_GRID_HAS_COL_SUBGRID_ITEM); } /** Return true if this frame has any subgrid items. */ bool HasSubgridItems() const { return HasAnyStateBits(NS_STATE_GRID_HAS_ROW_SUBGRID_ITEM | NS_STATE_GRID_HAS_COL_SUBGRID_ITEM); } /** * Return true if the grid item aChild should stretch in its aAxis (i.e. aAxis * is in the aChild's writing-mode). * * Note: this method does *not* consider the grid item's aspect-ratio and * natural size in the axis when the self-alignment value is 'normal' per * https://drafts.csswg.org/css-grid/#grid-item-sizing */ bool GridItemShouldStretch(const nsIFrame* aChild, LogicalAxis aAxis) const; /** * Returns true if aFrame forms an independent formatting context and hence * should be inhibited from being a subgrid (i.e. if the used value of * 'grid-template-{rows,columns}:subgrid' should be 'none'). * https://drafts.csswg.org/css-grid-2/#subgrid-listing * * (Note this only makes sense to call if aFrame is itself either a grid * container frame or a wrapper frame for a grid container frame, e.g. a * scroll container frame for a scrollable grid. Having said that, this is * technically safe to call on any non-null frame.) */ static bool ShouldInhibitSubgridDueToIFC(const nsIFrame* aFrame); /** * Return a container grid frame for the supplied frame, if available. * @return nullptr if aFrame has no grid container. */ static nsGridContainerFrame* GetGridContainerFrame(nsIFrame* aFrame); /** * Return a container grid frame, and ensure it has computed grid info * @return nullptr if aFrame has no grid container, or frame was destroyed * @note this might destroy layout/style data since it may flush layout */ MOZ_CAN_RUN_SCRIPT_BOUNDARY static nsGridContainerFrame* GetGridFrameWithComputedInfo(nsIFrame* aFrame); struct Subgrid; struct UsedTrackSizes; struct TrackSize; struct GridItemInfo; struct GridReflowInput; struct FindItemInGridOrderResult { // The first(last) item in (reverse) grid order. const GridItemInfo* mItem; // Does the above item span the first(last) track? bool mIsInEdgeTrack; }; /** Return our parent grid container; |this| MUST be a subgrid. */ nsGridContainerFrame* ParentGridContainerForSubgrid() const; // https://drafts.csswg.org/css-sizing/#constraints enum class SizingConstraint { MinContent, // sizing under min-content constraint MaxContent, // sizing under max-content constraint NoConstraint // no constraint, used during Reflow }; protected: typedef mozilla::LogicalPoint LogicalPoint; typedef mozilla::LogicalRect LogicalRect; typedef mozilla::LogicalSize LogicalSize; typedef mozilla::WritingMode WritingMode; struct Grid; struct GridArea; class LineNameMap; struct LineRange; struct SharedGridData; struct SubgridFallbackTrackSizingFunctions; struct TrackSizingFunctions; struct Tracks; struct TranslatedLineRange; friend nsContainerFrame* NS_NewGridContainerFrame( mozilla::PresShell* aPresShell, ComputedStyle* aStyle); explicit nsGridContainerFrame(ComputedStyle* aStyle, nsPresContext* aPresContext) : nsContainerFrame(aStyle, aPresContext, kClassID) { for (auto& perAxisBaseline : mBaseline) { for (auto& baseline : perAxisBaseline) { baseline = NS_INTRINSIC_ISIZE_UNKNOWN; } } } /** * XXX temporary - move the ImplicitNamedAreas stuff to the style system. * The implicit area names that come from x-start .. x-end lines in * grid-template-columns / grid-template-rows are stored in this frame * property when needed, as a ImplicitNamedAreas* value. */ void InitImplicitNamedAreas(const nsStylePosition* aStyle); using LineNameList = const mozilla::StyleOwnedSlice; void AddImplicitNamedAreas(mozilla::Span); using StyleLineNameListValue = const mozilla::StyleGenericLineNameListValue; void AddImplicitNamedAreas(mozilla::Span); /** * Reflow and place our children. * @return the consumed size of all of this grid container's continuations * so far including this frame */ nscoord ReflowChildren(GridReflowInput& aState, const LogicalRect& aContentArea, const nsSize& aContainerSize, ReflowOutput& aDesiredSize, nsReflowStatus& aStatus); /** * Helper to implement IntrinsicISize(). */ nscoord ComputeIntrinsicISize(const mozilla::IntrinsicSizeInput& aInput, mozilla::IntrinsicISizeType aType); nscoord GetBBaseline(BaselineSharingGroup aBaselineGroup) const { return mBaseline[mozilla::LogicalAxis::Block][aBaselineGroup]; } nscoord GetIBaseline(BaselineSharingGroup aBaselineGroup) const { return mBaseline[mozilla::LogicalAxis::Inline][aBaselineGroup]; } /** * Calculate this grid container's baselines. * @param aBaselineSet which baseline(s) to derive from a baseline-group or * items; a baseline not included is synthesized from the border-box instead. * @param aFragmentStartTrack is the first track in this fragment in the same * axis as aMajor. Pass zero if that's not the axis we're fragmenting in. * @param aFirstExcludedTrack should be the first track in the next fragment * or one beyond the final track in the last fragment, in aMajor's axis. * Pass the number of tracks if that's not the axis we're fragmenting in. */ enum BaselineSet : uint32_t { eNone = 0x0, eFirst = 0x1, eLast = 0x2, eBoth = eFirst | eLast, }; void CalculateBaselines(BaselineSet aBaselineSet, mozilla::CSSOrderAwareFrameIterator* aIter, const nsTArray* aGridItems, const Tracks& aTracks, uint32_t aFragmentStartTrack, uint32_t aFirstExcludedTrack, WritingMode aWM, const nsSize& aCBPhysicalSize, nscoord aCBBorderPaddingStart, nscoord aCBBorderPaddingStartEnd, nscoord aCBSize); /** * Synthesize a Grid container baseline for aGroup. */ nscoord SynthesizeBaseline(const FindItemInGridOrderResult& aGridOrderItem, LogicalAxis aAxis, BaselineSharingGroup aGroup, const nsSize& aCBPhysicalSize, nscoord aCBSize, WritingMode aCBWM); /** * Find the first item in Grid Order in this fragment. * https://drafts.csswg.org/css-grid/#grid-order * @param aFragmentStartTrack is the first track in this fragment in the same * axis as aMajor. Pass zero if that's not the axis we're fragmenting in. */ static FindItemInGridOrderResult FindFirstItemInGridOrder( mozilla::CSSOrderAwareFrameIterator& aIter, const nsTArray& aGridItems, LineRange GridArea::*aMajor, LineRange GridArea::*aMinor, uint32_t aFragmentStartTrack); /** * Find the last item in Grid Order in this fragment. * @param aFragmentStartTrack is the first track in this fragment in the same * axis as aMajor. Pass zero if that's not the axis we're fragmenting in. * @param aFirstExcludedTrack should be the first track in the next fragment * or one beyond the final track in the last fragment, in aMajor's axis. * Pass the number of tracks if that's not the axis we're fragmenting in. */ static FindItemInGridOrderResult FindLastItemInGridOrder( mozilla::ReverseCSSOrderAwareFrameIterator& aIter, const nsTArray& aGridItems, LineRange GridArea::*aMajor, LineRange GridArea::*aMinor, uint32_t aFragmentStartTrack, uint32_t aFirstExcludedTrack); /** * Update our NS_STATE_GRID_IS_COL/ROW_SUBGRID bits and related subgrid state * on our entire continuation chain based on the current style. * This is needed because grid-template-columns/rows style changes only * trigger a reflow so we need to update this dynamically. */ void UpdateSubgridFrameState(); /** * Return the NS_STATE_GRID_IS_COL/ROW_SUBGRID and * NS_STATE_GRID_IS_ROW/COL_MASONRY bits we ought to have. */ nsFrameState ComputeSelfSubgridMasonryBits() const; private: // Helpers for ReflowChildren struct Fragmentainer { /** * The distance from the first grid container fragment's block-axis content * edge to the fragmentainer end. */ nscoord mToFragmentainerEnd; /** * True if the current fragment is at the start of the fragmentainer. */ bool mIsTopOfPage; /** * Is there a Class C break opportunity at the start content edge? */ bool mCanBreakAtStart; /** * Is there a Class C break opportunity at the end content edge? */ bool mCanBreakAtEnd; /** * Is the grid container's block-size unconstrained? */ bool mIsAutoBSize; }; mozilla::Maybe GetNearestFragmentainer( const GridReflowInput& aState) const; // @return the consumed size of all continuations so far including this frame nscoord ReflowInFragmentainer(GridReflowInput& aState, const LogicalRect& aContentArea, ReflowOutput& aDesiredSize, nsReflowStatus& aStatus, Fragmentainer& aFragmentainer, const nsSize& aContainerSize); // Helper for ReflowInFragmentainer // @return the consumed size of all continuations so far including this frame nscoord ReflowRowsInFragmentainer( GridReflowInput& aState, const LogicalRect& aContentArea, ReflowOutput& aDesiredSize, nsReflowStatus& aStatus, Fragmentainer& aFragmentainer, const nsSize& aContainerSize, const nsTArray& aItems, uint32_t aStartRow, uint32_t aEndRow, nscoord aBSize, nscoord aAvailableSize); // Helper for ReflowChildren / ReflowInFragmentainer void ReflowInFlowChild(nsIFrame* aChild, const GridItemInfo* aGridItemInfo, nsSize aContainerSize, const mozilla::Maybe& aStretchBSize, const Fragmentainer* aFragmentainer, const GridReflowInput& aState, const LogicalRect& aContentArea, ReflowOutput& aDesiredSize, nsReflowStatus& aStatus); /** * Places and reflows items when we have masonry layout. * It handles unconstrained reflow and also fragmentation when the row axis * is the masonry axis. ReflowInFragmentainer handles the case when we're * fragmenting and our row axis is a grid axis and it handles masonry layout * in the column axis in that case. * @return the intrinsic size in the masonry axis */ nscoord MasonryLayout(GridReflowInput& aState, const LogicalRect& aContentArea, SizingConstraint aConstraint, ReflowOutput& aDesiredSize, nsReflowStatus& aStatus, Fragmentainer* aFragmentainer, const nsSize& aContainerSize); // Return the stored UsedTrackSizes, if any. UsedTrackSizes* GetUsedTrackSizes() const; // Store the given TrackSizes in aAxis on a UsedTrackSizes frame property. void StoreUsedTrackSizes(LogicalAxis aAxis, const nsTArray& aSizes); // The internal implementation for AddImplicitNamedAreas(). void AddImplicitNamedAreasInternal(LineNameList& aNameList, ImplicitNamedAreas*& aAreas); mozilla::IntrinsicISizesCache mCachedIntrinsicSizes; // Our baselines, one per BaselineSharingGroup per axis. PerLogicalAxis> mBaseline; public: // A cached result for a grid item's block-axis measuring reflow. This // cache prevents us from doing exponential reflows in cases of deeply // nested grid frames. // // We store the cached value in the grid item's frame property table. // // We cache the following as a "key" // - The size of the grid area in the item's inline axis // - The item's block axis baseline padding // ...and we cache the following as the "value", // - The item's border-box BSize class CachedBAxisMeasurement { public: NS_DECLARE_FRAME_PROPERTY_SMALL_VALUE(Prop, CachedBAxisMeasurement) CachedBAxisMeasurement(const nsIFrame* aFrame, const LogicalSize& aCBSize, const nscoord aBSize) : mKey(aFrame, aCBSize), mBSize(aBSize) {} CachedBAxisMeasurement() = default; bool IsValidFor(const nsIFrame* aFrame, const LogicalSize& aCBSize) const { if (aFrame->IsSubtreeDirty()) { return false; } const mozilla::Maybe maybeKey = Key::TryHash(aFrame, aCBSize); return maybeKey.isSome() && mKey == *maybeKey; } static bool CanCacheMeasurement(const nsIFrame* aFrame, const LogicalSize& aCBSize) { return Key::CanHash(aFrame, aCBSize); } nscoord BSize() const { return mBSize; } void Update(const nsIFrame* aFrame, const LogicalSize& aCBSize, const nscoord aBSize) { mKey.UpdateHash(aFrame, aCBSize); mBSize = aBSize; } private: class Key { // mHashKey is generated by combining these 2 variables together // 1. The containing block size in the item's inline axis used // for measuring reflow // 2. The item's baseline padding property uint32_t mHashKey; explicit Key(uint32_t aHashKey) : mHashKey(aHashKey) {} public: Key() = default; Key(const nsIFrame* aFrame, const LogicalSize& aCBSize) { UpdateHash(aFrame, aCBSize); } void UpdateHash(const nsIFrame* aFrame, const LogicalSize& aCBSize) { const mozilla::Maybe maybeKey = TryHash(aFrame, aCBSize); MOZ_ASSERT(maybeKey.isSome()); mHashKey = maybeKey->mHashKey; } static mozilla::Maybe TryHash(const nsIFrame* aFrame, const LogicalSize& aCBSize) { const nscoord gridAreaISize = aCBSize.ISize(aFrame->GetWritingMode()); const nscoord bBaselinePaddingProperty = abs(aFrame->GetProperty(nsIFrame::BBaselinePadProperty())); const uint_fast8_t bitsNeededForISize = mozilla::FloorLog2(gridAreaISize) + 1; const uint_fast8_t bitsNeededForBBaselinePadding = mozilla::FloorLog2(bBaselinePaddingProperty) + 1; if (bitsNeededForISize + bitsNeededForBBaselinePadding > 32) { return mozilla::Nothing(); } const uint32_t hashKey = (gridAreaISize << (32 - bitsNeededForISize)) | bBaselinePaddingProperty; return mozilla::Some(Key(hashKey)); } static bool CanHash(const nsIFrame* aFrame, const LogicalSize& aCBSize) { return TryHash(aFrame, aCBSize).isSome(); } bool operator==(const Key& aOther) const { return mHashKey == aOther.mHashKey; } }; Key mKey; nscoord mBSize; }; bool CanProvideLineIterator() const final { return true; } nsILineIterator* GetLineIterator() final { return this; } int32_t GetNumLines() const final; bool IsLineIteratorFlowRTL() final; mozilla::Result GetLine(int32_t aLineNumber) final; int32_t FindLineContaining(nsIFrame* aFrame, int32_t aStartLine = 0) final; NS_IMETHOD FindFrameAt(int32_t aLineNumber, nsPoint aPos, nsIFrame** aFrameFound, bool* aPosIsBeforeFirstFrame, bool* aPosIsAfterLastFrame) final; NS_IMETHOD CheckLineOrder(int32_t aLine, bool* aIsReordered, nsIFrame** aFirstVisual, nsIFrame** aLastVisual) final; }; #endif /* nsGridContainerFrame_h___ */