Index: Source/core/rendering/RenderGrid.cpp |
diff --git a/Source/core/rendering/RenderGrid.cpp b/Source/core/rendering/RenderGrid.cpp |
index 11474d6b9153b483983e0f05f4bbff8f4b23fdc0..21517c2f89fd339f26bb4f84385dbbe402ff8493 100644 |
--- a/Source/core/rendering/RenderGrid.cpp |
+++ b/Source/core/rendering/RenderGrid.cpp |
@@ -224,6 +224,10 @@ void RenderGrid::addChild(RenderObject* newChild, RenderObject* beforeChild) |
return; |
} |
+ // Positioned items that shouldn't take up space or otherwise participate in the layout of the grid. |
+ if (newChild->isOutOfFlowPositioned()) |
+ return; |
+ |
// If the new child has been inserted inside an existent anonymous block, we can simply ignore it as the anonymous |
// block is an already known grid item. |
if (newChild->parent() != this) |
@@ -252,19 +256,30 @@ void RenderGrid::addChild(RenderObject* newChild, RenderObject* beforeChild) |
void RenderGrid::addChildToIndexesMap(RenderBox& child) |
{ |
ASSERT(!m_gridItemsIndexesMap.contains(&child)); |
- RenderBox* sibling = child.nextSiblingBox(); |
+ RenderBox* sibling; |
+ for (sibling = child.nextSiblingBox(); sibling; sibling = sibling->nextSiblingBox()) { |
+ if (!sibling->isOutOfFlowPositioned()) |
+ break; |
+ } |
bool lastSibling = !sibling; |
- if (lastSibling) |
- sibling = child.previousSiblingBox(); |
+ if (lastSibling) { |
+ for (sibling = child.previousSiblingBox(); sibling; sibling = sibling->previousSiblingBox()) { |
Julien - ping for review
2014/11/19 18:46:43
That would probably be nice to move into a helper
Manuel Rego
2014/11/19 23:26:23
Yes I agree, it makes the code easier to understan
|
+ if (!sibling->isOutOfFlowPositioned()) |
+ break; |
+ } |
+ } |
size_t index = 0; |
if (sibling) |
index = lastSibling ? m_gridItemsIndexesMap.get(sibling) + 1 : m_gridItemsIndexesMap.get(sibling); |
if (sibling && !lastSibling) { |
- for (; sibling; sibling = sibling->nextSiblingBox()) |
+ for (; sibling; sibling = sibling->nextSiblingBox()) { |
+ if (sibling->isOutOfFlowPositioned()) |
+ continue; |
m_gridItemsIndexesMap.set(sibling, m_gridItemsIndexesMap.get(sibling) + 1); |
+ } |
} |
m_gridItemsIndexesMap.set(&child, index); |
@@ -286,6 +301,9 @@ void RenderGrid::removeChild(RenderObject* child) |
return; |
} |
+ if (child->isOutOfFlowPositioned()) |
+ return; |
+ |
const RenderBox* childBox = toRenderBox(child); |
GridCoordinate coordinate = m_gridItemCoordinate.take(childBox); |
@@ -898,6 +916,10 @@ void RenderGrid::placeItemsOnGrid() |
Vector<RenderBox*> autoMajorAxisAutoGridItems; |
Vector<RenderBox*> specifiedMajorAxisAutoGridItems; |
for (RenderBox* child = m_orderIterator.first(); child; child = m_orderIterator.next()) { |
+ if (child->isOutOfFlowPositioned()) { |
+ continue; |
+ } |
+ |
// FIXME: We never re-resolve positions if the grid is grown during auto-placement which may lead auto / <integer> |
// positions to not match the author's intent. The specification is unclear on what should be done in this case. |
OwnPtr<GridSpan> rowPositions = GridResolvedPosition::resolveGridPositionsFromStyle(*style(), *child, ForRows); |
@@ -940,6 +962,10 @@ void RenderGrid::populateExplicitGridAndOrderIterator() |
ASSERT(m_gridItemsIndexesMap.isEmpty()); |
size_t childIndex = 0; |
for (RenderBox* child = firstChildBox(); child; child = child->nextSiblingBox()) { |
+ if (child->isOutOfFlowPositioned()) { |
+ continue; |
+ } |
+ |
populator.collectChild(child); |
m_gridItemsIndexesMap.set(child, childIndex++); |
@@ -1098,10 +1124,9 @@ void RenderGrid::layoutGridItems() |
for (RenderBox* child = firstChildBox(); child; child = child->nextSiblingBox()) { |
if (child->isOutOfFlowPositioned()) { |
- // FIXME: Absolute positioned grid items should have a special |
- // behavior as described in the spec (crbug.com/273898): |
- // http://www.w3.org/TR/css-grid-1/#abspos-items |
child->containingBlock()->insertPositionedObject(child); |
+ adjustPositionedGridItem(child); |
+ continue; |
} |
// Because the grid area cannot be styled, we don't need to adjust |
@@ -1145,6 +1170,86 @@ void RenderGrid::layoutGridItems() |
setLogicalHeight(logicalHeight() + borderAndPaddingLogicalHeight()); |
} |
+void RenderGrid::adjustPositionedGridItem(RenderBox* child) |
+{ |
+ RenderLayer* childLayer = child->layer(); |
+ childLayer->setStaticInlinePosition(borderAndPaddingStart()); |
+ childLayer->setStaticBlockPosition(borderAndPaddingBefore()); |
+} |
+ |
+void RenderGrid::layoutPositionedObjects(bool relayoutChildren, PositionedLayoutBehavior info) |
+{ |
+ TrackedRendererListHashSet* positionedDescendants = positionedObjects(); |
+ if (!positionedDescendants) |
+ return; |
+ |
+ bool containerHasHorizontalWritingMode = isHorizontalWritingMode(); |
+ TrackedRendererListHashSet::iterator end = positionedDescendants->end(); |
+ for (TrackedRendererListHashSet::iterator it = positionedDescendants->begin(); it != end; ++it) { |
+ RenderBox* child = *it; |
+ |
+ bool hasOrthogonalWritingMode = child->isHorizontalWritingMode() != containerHasHorizontalWritingMode; |
+ if (hasOrthogonalWritingMode) { |
+ // FIXME: Properly support orthogonal writing mode. |
+ continue; |
+ } |
+ |
+ LayoutUnit columnBreadth = clientLogicalWidth(); |
+ LayoutUnit columnOffset = offsetForPositionedChild(*child, ForColumns, columnBreadth); |
+ LayoutUnit rowBreadth = clientLogicalHeight(); |
+ LayoutUnit rowOffset = offsetForPositionedChild(*child, ForRows, rowBreadth); |
+ |
+ RenderLayer* childLayer = child->layer(); |
+ childLayer->setStaticInlinePosition(childLayer->staticInlinePosition() + (containerHasHorizontalWritingMode ? columnOffset : rowOffset)); |
+ childLayer->setStaticBlockPosition(childLayer->staticBlockPosition() + (containerHasHorizontalWritingMode ? rowOffset : columnOffset)); |
+ |
+ child->setOverrideContainingBlockContentLogicalWidth(columnBreadth); |
+ child->setOverrideContainingBlockContentLogicalHeight(rowBreadth); |
+ } |
+ |
+ RenderBlock::layoutPositionedObjects(relayoutChildren, info); |
+} |
+ |
+LayoutUnit RenderGrid::offsetForPositionedChild(const RenderBox& child, GridTrackSizingDirection direction, LayoutUnit& breadth) |
Julien - ping for review
2014/11/19 18:46:43
The API is weird: you wouldn't assume that it touc
Manuel Rego
2014/11/19 23:26:23
I added "andBreadth" but kept the return value, as
|
+{ |
+ ASSERT(child.isHorizontalWritingMode() == isHorizontalWritingMode()); |
+ |
+ LayoutUnit offset = LayoutUnit(0); |
Julien - ping for review
2014/11/19 18:46:43
You don't use offset until below so we should move
Manuel Rego
2014/11/19 23:26:23
Done.
|
+ |
+ OwnPtr<GridSpan> positions = GridResolvedPosition::resolveGridPositionsFromStyle(*style(), child, direction); |
+ if (!positions) |
+ return offset; |
Julien - ping for review
2014/11/19 18:46:43
And explicitly return LayoutUnit(0) here (I like e
Manuel Rego
2014/11/19 23:26:24
Done.
|
+ |
+ if ((direction == ForColumns && !child.style()->gridColumnStart().isAuto()) || (direction == ForRows && !child.style()->gridRowStart().isAuto())) { |
Julien - ping for review
2014/11/19 18:46:43
Unfortunately this doesn't handle correctly spanni
Manuel Rego
2014/11/19 23:26:24
I'm not sure if I'm getting what you mean.
For ex
Julien - ping for review
2014/11/24 21:44:40
This is the case I am thinking of:
grid-column: s
Manuel Rego
2014/12/01 11:10:36
I'm not sure about this one, so I've asked in www-
|
+ breadth -= (direction == ForColumns) ? paddingStart() : paddingBefore(); |
+ |
+ GridResolvedPosition firstPosition = GridResolvedPosition(0); |
+ if (positions->resolvedInitialPosition != firstPosition) { |
+ for (GridResolvedPosition position = firstPosition; position < positions->resolvedInitialPosition; ++position) { |
+ LayoutUnit trackBreadth = (direction == ForColumns) ? m_columnPositions[position.toInt() + 1] - m_columnPositions[position.toInt()] : m_rowPositions[position.toInt() + 1] - m_rowPositions[position.toInt()]; |
+ breadth -= trackBreadth; |
+ offset += trackBreadth; |
+ } |
+ } |
+ } else if ((direction == ForColumns && !child.style()->gridColumnEnd().isAuto()) || (direction == ForRows && !child.style()->gridRowEnd().isAuto())) { |
Julien - ping for review
2014/11/19 18:46:43
Not sure why this is inside an 'else'. Nothing in
Manuel Rego
2014/11/19 23:26:23
Let me explain with an example. Imagine a grid wit
Manuel Rego
2014/11/21 10:11:00
I've removed the "else" as it was not needed at al
|
+ offset -= (direction == ForColumns) ? paddingStart() : paddingBefore(); |
+ } |
+ |
+ if ((direction == ForColumns && !child.style()->gridColumnEnd().isAuto()) || (direction == ForRows && !child.style()->gridRowEnd().isAuto())) { |
+ breadth -= (direction == ForColumns) ? paddingEnd() : paddingAfter(); |
+ |
+ GridResolvedPosition lastPosition = GridResolvedPosition(direction == ForColumns ? gridColumnCount() : gridRowCount()); |
+ if (positions->resolvedFinalPosition.next() != lastPosition) { |
+ for (GridResolvedPosition position = positions->resolvedFinalPosition.next(); position < lastPosition; ++position) { |
+ LayoutUnit trackBreadth = (direction == ForColumns) ? m_columnPositions[position.toInt() + 1] - m_columnPositions[position.toInt()] : m_rowPositions[position.toInt() + 1] - m_rowPositions[position.toInt()]; |
+ breadth -= trackBreadth; |
+ } |
+ } |
+ } |
+ |
+ return offset; |
+} |
+ |
GridCoordinate RenderGrid::cachedGridCoordinate(const RenderBox& gridItem) const |
{ |
ASSERT(m_gridItemCoordinate.contains(&gridItem)); |