| /* |
| * Copyright (C) 2011, 2022 Apple Inc. All rights reserved. |
| * Copyright (C) 2013-2017 Igalia S.L. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #pragma once |
| |
| #include "Grid.h" |
| #include "GridTrackSizingAlgorithm.h" |
| #include "RenderBlock.h" |
| |
| namespace WebCore { |
| |
| class GridArea; |
| class GridSpan; |
| |
| struct ContentAlignmentData { |
| WTF_MAKE_NONCOPYABLE(ContentAlignmentData); WTF_MAKE_FAST_ALLOCATED; |
| public: |
| ContentAlignmentData() = default; |
| bool isValid() const { return positionOffset >= 0 && distributionOffset >= 0; } |
| |
| LayoutUnit positionOffset; |
| LayoutUnit distributionOffset; |
| }; |
| |
| enum GridAxisPosition {GridAxisStart, GridAxisEnd, GridAxisCenter}; |
| |
| class RenderGrid final : public RenderBlock { |
| WTF_MAKE_ISO_ALLOCATED(RenderGrid); |
| public: |
| RenderGrid(Element&, RenderStyle&&); |
| virtual ~RenderGrid(); |
| |
| Element& element() const { return downcast<Element>(nodeForNonAnonymous()); } |
| |
| void styleDidChange(StyleDifference, const RenderStyle* oldStyle) override; |
| void layoutBlock(bool relayoutChildren, LayoutUnit pageLogicalHeight = 0_lu) override; |
| |
| bool avoidsFloats() const override { return true; } |
| bool canDropAnonymousBlockChild() const override { return false; } |
| |
| void dirtyGrid(); |
| Vector<LayoutUnit> trackSizesForComputedStyle(GridTrackSizingDirection) const; |
| |
| const Vector<LayoutUnit>& columnPositions() const { return m_columnPositions; } |
| const Vector<LayoutUnit>& rowPositions() const { return m_rowPositions; } |
| |
| unsigned autoRepeatCountForDirection(GridTrackSizingDirection direction) const { return m_grid.autoRepeatTracks(direction); } |
| unsigned explicitGridStartForDirection(GridTrackSizingDirection direction) const { return m_grid.explicitGridStart(direction); } |
| |
| // Required by GridTrackSizingAlgorithm. Keep them under control. |
| LayoutUnit guttersSize(const Grid&, GridTrackSizingDirection, unsigned startLine, unsigned span, std::optional<LayoutUnit> availableSize) const; |
| LayoutUnit gridItemOffset(GridTrackSizingDirection) const; |
| |
| void updateGridAreaLogicalSize(RenderBox&, std::optional<LayoutUnit> width, std::optional<LayoutUnit> height) const; |
| bool isBaselineAlignmentForChild(const RenderBox&) const; |
| bool isBaselineAlignmentForChild(const RenderBox& child, GridAxis, AllowedBaseLine = BothLines) const; |
| |
| StyleSelfAlignmentData selfAlignmentForChild(GridAxis, const RenderBox&, const RenderStyle* = nullptr) const; |
| |
| StyleContentAlignmentData contentAlignment(GridTrackSizingDirection) const; |
| |
| // Computes the span relative to this RenderGrid, even if the RenderBox is a child |
| // of a descendant subgrid. |
| GridSpan gridSpanForChild(const RenderBox&, GridTrackSizingDirection) const; |
| |
| bool isSubgrid(GridTrackSizingDirection) const; |
| bool isSubgridRows() const |
| { |
| return isSubgrid(ForRows); |
| } |
| bool isSubgridColumns() const |
| { |
| return isSubgrid(ForColumns); |
| } |
| bool isSubgridInParentDirection(GridTrackSizingDirection parentDirection) const; |
| |
| // Returns true if this grid is inheriting subgridded tracks for |
| // the given direction from the specified ancestor. This handles |
| // nested subgrids, where ancestor may not be our direct parent. |
| bool isSubgridOf(GridTrackSizingDirection, const RenderGrid& ancestor); |
| |
| const Grid& currentGrid() const |
| { |
| return m_grid; |
| } |
| |
| unsigned numTracks(GridTrackSizingDirection direction) const |
| { |
| return numTracks(direction, m_grid); |
| } |
| |
| void placeItems(); |
| |
| std::optional<LayoutUnit> availableSpaceForGutters(GridTrackSizingDirection) const; |
| LayoutUnit gridGap(GridTrackSizingDirection) const; |
| LayoutUnit gridGap(GridTrackSizingDirection, std::optional<LayoutUnit> availableSize) const; |
| |
| private: |
| friend class GridTrackSizingAlgorithm; |
| |
| ItemPosition selfAlignmentNormalBehavior(const RenderBox* child = nullptr) const override |
| { |
| ASSERT(child); |
| return child->isRenderReplaced() ? ItemPosition::Start : ItemPosition::Stretch; |
| } |
| |
| ASCIILiteral renderName() const override; |
| bool isRenderGrid() const override { return true; } |
| void computeIntrinsicLogicalWidths(LayoutUnit& minLogicalWidth, LayoutUnit& maxLogicalWidth) const override; |
| |
| bool selfAlignmentChangedToStretch(GridAxis, const RenderStyle& oldStyle, const RenderStyle& newStyle, const RenderBox&) const; |
| bool selfAlignmentChangedFromStretch(GridAxis, const RenderStyle& oldStyle, const RenderStyle& newStyle, const RenderBox&) const; |
| |
| bool explicitGridDidResize(const RenderStyle&) const; |
| bool namedGridLinesDefinitionDidChange(const RenderStyle&) const; |
| bool implicitGridLinesDefinitionDidChange(const RenderStyle&) const; |
| |
| unsigned computeAutoRepeatTracksCount(GridTrackSizingDirection, std::optional<LayoutUnit> availableSize) const; |
| |
| unsigned clampAutoRepeatTracks(GridTrackSizingDirection, unsigned autoRepeatTracks) const; |
| |
| std::unique_ptr<OrderedTrackIndexSet> computeEmptyTracksForAutoRepeat(Grid&, GridTrackSizingDirection) const; |
| |
| void performGridItemsPreLayout(const GridTrackSizingAlgorithm&) const; |
| |
| void placeItemsOnGrid(GridTrackSizingAlgorithm&, std::optional<LayoutUnit> availableLogicalWidth) const; |
| void populateExplicitGridAndOrderIterator(Grid&) const; |
| std::unique_ptr<GridArea> createEmptyGridAreaAtSpecifiedPositionsOutsideGrid(Grid&, const RenderBox&, GridTrackSizingDirection, const GridSpan&) const; |
| void placeSpecifiedMajorAxisItemsOnGrid(Grid&, const Vector<RenderBox*>&) const; |
| void placeAutoMajorAxisItemsOnGrid(Grid&, const Vector<RenderBox*>&) const; |
| typedef std::pair<unsigned, unsigned> AutoPlacementCursor; |
| void placeAutoMajorAxisItemOnGrid(Grid&, RenderBox&, AutoPlacementCursor&) const; |
| GridTrackSizingDirection autoPlacementMajorAxisDirection() const; |
| GridTrackSizingDirection autoPlacementMinorAxisDirection() const; |
| |
| bool canPerformSimplifiedLayout() const final; |
| void prepareChildForPositionedLayout(RenderBox&); |
| bool hasStaticPositionForChild(const RenderBox&, GridTrackSizingDirection) const; |
| void layoutPositionedObject(RenderBox&, bool relayoutChildren, bool fixedPositionObjectsOnly) override; |
| |
| void computeTrackSizesForDefiniteSize(GridTrackSizingDirection, LayoutUnit availableSpace); |
| void computeTrackSizesForIndefiniteSize(GridTrackSizingAlgorithm&, GridTrackSizingDirection, LayoutUnit* minIntrinsicSize = nullptr, LayoutUnit* maxIntrinsicSize = nullptr) const; |
| LayoutUnit computeTrackBasedLogicalHeight() const; |
| |
| void repeatTracksSizingIfNeeded(LayoutUnit availableSpaceForColumns, LayoutUnit availableSpaceForRows); |
| |
| void updateGridAreaForAspectRatioItems(const Vector<RenderBox*>&); |
| void layoutGridItems(); |
| void populateGridPositionsForDirection(GridTrackSizingDirection); |
| |
| LayoutUnit gridAreaBreadthForOutOfFlowChild(const RenderBox&, GridTrackSizingDirection); |
| LayoutUnit logicalOffsetForOutOfFlowChild(const RenderBox&, GridTrackSizingDirection, LayoutUnit) const; |
| void gridAreaPositionForOutOfFlowChild(const RenderBox&, GridTrackSizingDirection, LayoutUnit& start, LayoutUnit& end) const; |
| void gridAreaPositionForInFlowChild(const RenderBox&, GridTrackSizingDirection, LayoutUnit& start, LayoutUnit& end) const; |
| void gridAreaPositionForChild(const RenderBox&, GridTrackSizingDirection, LayoutUnit& start, LayoutUnit& end) const; |
| |
| GridAxisPosition columnAxisPositionForChild(const RenderBox&) const; |
| GridAxisPosition rowAxisPositionForChild(const RenderBox&) const; |
| LayoutUnit columnAxisOffsetForChild(const RenderBox&) const; |
| LayoutUnit rowAxisOffsetForChild(const RenderBox&) const; |
| void computeContentPositionAndDistributionOffset(GridTrackSizingDirection, const LayoutUnit& availableFreeSpace, unsigned numberOfGridTracks); |
| void setLogicalPositionForChild(RenderBox&) const; |
| void setLogicalOffsetForChild(RenderBox&, GridTrackSizingDirection) const; |
| LayoutUnit logicalOffsetForChild(const RenderBox&, GridTrackSizingDirection) const; |
| |
| LayoutUnit gridAreaBreadthForChildIncludingAlignmentOffsets(const RenderBox&, GridTrackSizingDirection) const; |
| |
| void paintChildren(PaintInfo& forSelf, const LayoutPoint& paintOffset, PaintInfo& forChild, bool usePrintRect) override; |
| LayoutUnit availableAlignmentSpaceForChildBeforeStretching(LayoutUnit gridAreaBreadthForChild, const RenderBox&, GridTrackSizingDirection) const; |
| StyleSelfAlignmentData justifySelfForChild(const RenderBox&, StretchingMode = StretchingMode::Any, const RenderStyle* = nullptr) const; |
| StyleSelfAlignmentData alignSelfForChild(const RenderBox&, StretchingMode = StretchingMode::Any, const RenderStyle* = nullptr) const; |
| void applyStretchAlignmentToChildIfNeeded(RenderBox&); |
| void applySubgridStretchAlignmentToChildIfNeeded(RenderBox&); |
| bool hasAutoSizeInColumnAxis(const RenderBox& child) const; |
| bool hasAutoSizeInRowAxis(const RenderBox& child) const; |
| bool allowedToStretchChildAlongColumnAxis(const RenderBox& child) const { return alignSelfForChild(child).position() == ItemPosition::Stretch && hasAutoSizeInColumnAxis(child) && !hasAutoMarginsInColumnAxis(child); } |
| bool allowedToStretchChildAlongRowAxis(const RenderBox& child) const { return justifySelfForChild(child).position() == ItemPosition::Stretch && hasAutoSizeInRowAxis(child) && !hasAutoMarginsInRowAxis(child); } |
| bool hasAutoMarginsInColumnAxis(const RenderBox&) const; |
| bool hasAutoMarginsInRowAxis(const RenderBox&) const; |
| void resetAutoMarginsAndLogicalTopInColumnAxis(RenderBox& child); |
| void updateAutoMarginsInColumnAxisIfNeeded(RenderBox&); |
| void updateAutoMarginsInRowAxisIfNeeded(RenderBox&); |
| |
| LayoutUnit baselinePosition(FontBaseline, bool firstLine, LineDirectionMode, LinePositionMode = PositionOnContainingLine) const final; |
| std::optional<LayoutUnit> firstLineBaseline() const final; |
| std::optional<LayoutUnit> inlineBlockBaseline(LineDirectionMode) const final; |
| bool isInlineBaselineAlignedChild(const RenderBox&) const; |
| |
| LayoutUnit columnAxisBaselineOffsetForChild(const RenderBox&) const; |
| LayoutUnit rowAxisBaselineOffsetForChild(const RenderBox&) const; |
| |
| unsigned nonCollapsedTracks(GridTrackSizingDirection) const; |
| unsigned numTracks(GridTrackSizingDirection, const Grid&) const; |
| |
| LayoutUnit translateRTLCoordinate(LayoutUnit) const; |
| |
| bool shouldResetLogicalHeightBeforeLayout() const override { return true; } |
| bool establishesIndependentFormattingContext() const override; |
| |
| bool aspectRatioPrefersInline(const RenderBox& child, bool blockFlowIsColumnAxis); |
| |
| Vector<RenderBox*> computeAspectRatioDependentAndBaselineItems(); |
| |
| GridSpan gridSpanForOutOfFlowChild(const RenderBox&, GridTrackSizingDirection) const; |
| |
| bool computeGridPositionsForOutOfFlowChild(const RenderBox&, GridTrackSizingDirection, int&, bool&, int&, bool&) const; |
| |
| Grid m_grid; |
| |
| GridTrackSizingAlgorithm m_trackSizingAlgorithm; |
| |
| Vector<LayoutUnit> m_columnPositions; |
| Vector<LayoutUnit> m_rowPositions; |
| ContentAlignmentData m_offsetBetweenColumns; |
| ContentAlignmentData m_offsetBetweenRows; |
| |
| typedef HashMap<const RenderBox*, std::optional<size_t>> OutOfFlowPositionsMap; |
| OutOfFlowPositionsMap m_outOfFlowItemColumn; |
| OutOfFlowPositionsMap m_outOfFlowItemRow; |
| |
| std::optional<LayoutUnit> m_minContentSize; |
| std::optional<LayoutUnit> m_maxContentSize; |
| |
| bool m_hasAnyOrthogonalItem {false}; |
| bool m_hasAspectRatioBlockSizeDependentItem { false }; |
| bool m_baselineItemsCached {false}; |
| bool m_hasAnyBaselineAlignmentItem { false }; |
| }; |
| |
| } // namespace WebCore |
| |
| SPECIALIZE_TYPE_TRAITS_RENDER_OBJECT(RenderGrid, isRenderGrid()) |