Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(6910)

Unified Diff: Source/core/rendering/RenderGrid.cpp

Issue 637033003: [CSS Grid Layout] Fix positioned grid children position and size (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@master
Patch Set: Add missing orthogonal modes check Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « Source/core/rendering/RenderGrid.h ('k') | no next file » | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
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));
« no previous file with comments | « Source/core/rendering/RenderGrid.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698