6e6803fd9850d3ba4b392ececc1d7fb2cee60010
[WebKit-https.git] / Source / WebCore / rendering / RenderTableRow.cpp
1 /**
2  * Copyright (C) 1997 Martin Jones (mjones@kde.org)
3  *           (C) 1997 Torben Weis (weis@kde.org)
4  *           (C) 1998 Waldo Bastian (bastian@kde.org)
5  *           (C) 1999 Lars Knoll (knoll@kde.org)
6  *           (C) 1999 Antti Koivisto (koivisto@kde.org)
7  * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Library General Public
11  * License as published by the Free Software Foundation; either
12  * version 2 of the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * Library General Public License for more details.
18  *
19  * You should have received a copy of the GNU Library General Public License
20  * along with this library; see the file COPYING.LIB.  If not, write to
21  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
22  * Boston, MA 02110-1301, USA.
23  */
24
25 #include "config.h"
26 #include "RenderTableRow.h"
27
28 #include "Document.h"
29 #include "HTMLNames.h"
30 #include "HitTestResult.h"
31 #include "PaintInfo.h"
32 #include "RenderTableCell.h"
33 #include "RenderView.h"
34 #include "StyleInheritedData.h"
35 #include <wtf/StackStats.h>
36
37 namespace WebCore {
38
39 using namespace HTMLNames;
40
41 RenderTableRow::RenderTableRow(Element& element, RenderStyle&& style)
42     : RenderBox(element, WTFMove(style), 0)
43     , m_rowIndex(unsetRowIndex)
44 {
45     setInline(false);
46 }
47
48 RenderTableRow::RenderTableRow(Document& document, RenderStyle&& style)
49     : RenderBox(document, WTFMove(style), 0)
50     , m_rowIndex(unsetRowIndex)
51 {
52     setInline(false);
53 }
54
55 void RenderTableRow::willBeRemovedFromTree()
56 {
57     RenderBox::willBeRemovedFromTree();
58
59     section()->setNeedsCellRecalc();
60 }
61
62 static bool borderWidthChanged(const RenderStyle* oldStyle, const RenderStyle* newStyle)
63 {
64     return oldStyle->borderLeftWidth() != newStyle->borderLeftWidth()
65         || oldStyle->borderTopWidth() != newStyle->borderTopWidth()
66         || oldStyle->borderRightWidth() != newStyle->borderRightWidth()
67         || oldStyle->borderBottomWidth() != newStyle->borderBottomWidth();
68 }
69
70 void RenderTableRow::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
71 {
72     ASSERT(style().display() == TABLE_ROW);
73
74     RenderBox::styleDidChange(diff, oldStyle);
75     propagateStyleToAnonymousChildren(PropagateToAllChildren);
76
77     if (section() && oldStyle && style().logicalHeight() != oldStyle->logicalHeight())
78         section()->rowLogicalHeightChanged(rowIndex());
79
80     // If border was changed, notify table.
81     if (RenderTable* table = this->table()) {
82         if (oldStyle && oldStyle->border() != style().border())
83             table->invalidateCollapsedBorders();
84
85         if (oldStyle && diff == StyleDifferenceLayout && needsLayout() && table->collapseBorders() && borderWidthChanged(oldStyle, &style())) {
86             // If the border width changes on a row, we need to make sure the cells in the row know to lay out again.
87             // This only happens when borders are collapsed, since they end up affecting the border sides of the cell
88             // itself.
89             for (RenderTableCell* cell = firstCell(); cell; cell = cell->nextCell())
90                 cell->setChildNeedsLayout(MarkOnlyThis);
91         }
92     }
93 }
94
95 const BorderValue& RenderTableRow::borderAdjoiningStartCell(const RenderTableCell& cell) const
96 {
97     ASSERT_UNUSED(cell, cell.isFirstOrLastCellInRow());
98     // FIXME: https://webkit.org/b/79272 - Add support for mixed directionality at the cell level.
99     return style().borderStart();
100 }
101
102 const BorderValue& RenderTableRow::borderAdjoiningEndCell(const RenderTableCell& cell) const
103 {
104     ASSERT_UNUSED(cell, cell.isFirstOrLastCellInRow());
105     // FIXME: https://webkit.org/b/79272 - Add support for mixed directionality at the cell level.
106     return style().borderEnd();
107 }
108
109 void RenderTableRow::addChild(RenderObject* child, RenderObject* beforeChild)
110 {
111     if (!is<RenderTableCell>(*child)) {
112         RenderObject* last = beforeChild;
113         if (!last)
114             last = lastCell();
115         if (last && last->isAnonymous() && is<RenderTableCell>(*last) && !last->isBeforeOrAfterContent()) {
116             RenderTableCell& cell = downcast<RenderTableCell>(*last);
117             if (beforeChild == &cell)
118                 beforeChild = cell.firstChild();
119             cell.addChild(child, beforeChild);
120             return;
121         }
122
123         if (beforeChild && !beforeChild->isAnonymous() && beforeChild->parent() == this) {
124             RenderObject* cell = beforeChild->previousSibling();
125             if (is<RenderTableCell>(cell) && cell->isAnonymous()) {
126                 downcast<RenderTableCell>(*cell).addChild(child);
127                 return;
128             }
129         }
130
131         // If beforeChild is inside an anonymous cell, insert into the cell.
132         if (last && !is<RenderTableCell>(*last) && last->parent() && last->parent()->isAnonymous() && !last->parent()->isBeforeOrAfterContent()) {
133             last->parent()->addChild(child, beforeChild);
134             return;
135         }
136
137         RenderTableCell* cell = RenderTableCell::createAnonymousWithParentRenderer(this);
138         addChild(cell, beforeChild);
139         cell->addChild(child);
140         return;
141     } 
142
143     if (beforeChild && beforeChild->parent() != this)
144         beforeChild = splitAnonymousBoxesAroundChild(beforeChild);    
145
146     RenderTableCell& cell = downcast<RenderTableCell>(*child);
147
148     // Generated content can result in us having a null section so make sure to null check our parent.
149     if (RenderTableSection* section = this->section())
150         section->addCell(&cell, this);
151
152     ASSERT(!beforeChild || is<RenderTableCell>(*beforeChild));
153     RenderBox::addChild(&cell, beforeChild);
154
155     if (beforeChild || nextRow())
156         section()->setNeedsCellRecalc();
157     if (RenderTable* table = this->table())
158         table->invalidateCollapsedBorders();
159 }
160
161 void RenderTableRow::layout()
162 {
163     StackStats::LayoutCheckPoint layoutCheckPoint;
164     ASSERT(needsLayout());
165
166     // Table rows do not add translation.
167     LayoutStateMaintainer statePusher(view(), *this, LayoutSize(), hasTransform() || hasReflection() || style().isFlippedBlocksWritingMode());
168
169     bool paginated = view().layoutState()->isPaginated();
170                 
171     for (RenderTableCell* cell = firstCell(); cell; cell = cell->nextCell()) {
172         if (!cell->needsLayout() && paginated && (view().layoutState()->pageLogicalHeightChanged() || (view().layoutState()->pageLogicalHeight() && view().layoutState()->pageLogicalOffset(cell, cell->logicalTop()) != cell->pageLogicalOffset())))
173             cell->setChildNeedsLayout(MarkOnlyThis);
174
175         if (cell->needsLayout()) {
176             cell->computeAndSetBlockDirectionMargins(*table());
177             cell->layout();
178         }
179     }
180
181     clearOverflow();
182     addVisualEffectOverflow();
183     // We only ever need to repaint if our cells didn't, which menas that they didn't need
184     // layout, so we know that our bounds didn't change. This code is just making up for
185     // the fact that we did not repaint in setStyle() because we had a layout hint.
186     // We cannot call repaint() because our clippedOverflowRectForRepaint() is taken from the
187     // parent table, and being mid-layout, that is invalid. Instead, we repaint our cells.
188     if (selfNeedsLayout() && checkForRepaintDuringLayout()) {
189         for (RenderTableCell* cell = firstCell(); cell; cell = cell->nextCell())
190             cell->repaint();
191     }
192
193     statePusher.pop();
194     // RenderTableSection::layoutRows will set our logical height and width later, so it calls updateLayerTransform().
195     clearNeedsLayout();
196 }
197
198 LayoutRect RenderTableRow::clippedOverflowRectForRepaint(const RenderLayerModelObject* repaintContainer) const
199 {
200     ASSERT(parent());
201     // Rows and cells are in the same coordinate space. We need to both compute our overflow rect (which
202     // will accommodate a row outline and any visual effects on the row itself), but we also need to add in
203     // the repaint rects of cells.
204     LayoutRect result = RenderBox::clippedOverflowRectForRepaint(repaintContainer);
205     for (RenderTableCell* cell = firstCell(); cell; cell = cell->nextCell()) {
206         // Even if a cell is a repaint container, it's the row that paints the background behind it.
207         // So we don't care if a cell is a repaintContainer here.
208         result.uniteIfNonZero(cell->clippedOverflowRectForRepaint(repaintContainer));
209     }
210     return result;
211 }
212
213 // Hit Testing
214 bool RenderTableRow::nodeAtPoint(const HitTestRequest& request, HitTestResult& result, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset, HitTestAction action)
215 {
216     // Table rows cannot ever be hit tested.  Effectively they do not exist.
217     // Just forward to our children always.
218     for (RenderTableCell* cell = lastCell(); cell; cell = cell->previousCell()) {
219         // FIXME: We have to skip over inline flows, since they can show up inside table rows
220         // at the moment (a demoted inline <form> for example). If we ever implement a
221         // table-specific hit-test method (which we should do for performance reasons anyway),
222         // then we can remove this check.
223         if (!cell->hasSelfPaintingLayer()) {
224             LayoutPoint cellPoint = flipForWritingModeForChild(cell, accumulatedOffset);
225             if (cell->nodeAtPoint(request, result, locationInContainer, cellPoint, action)) {
226                 updateHitTestResult(result, locationInContainer.point() - toLayoutSize(cellPoint));
227                 return true;
228             }
229         }
230     }
231
232     return false;
233 }
234
235 void RenderTableRow::paintOutlineForRowIfNeeded(PaintInfo& paintInfo, const LayoutPoint& paintOffset)
236 {
237     LayoutPoint adjustedPaintOffset = paintOffset + location();
238     PaintPhase paintPhase = paintInfo.phase;
239     if ((paintPhase == PaintPhaseOutline || paintPhase == PaintPhaseSelfOutline) && style().visibility() == VISIBLE)
240         paintOutline(paintInfo, LayoutRect(adjustedPaintOffset, size()));
241 }
242
243 void RenderTableRow::paint(PaintInfo& paintInfo, const LayoutPoint& paintOffset)
244 {
245     ASSERT(hasSelfPaintingLayer());
246
247     paintOutlineForRowIfNeeded(paintInfo, paintOffset);
248     for (RenderTableCell* cell = firstCell(); cell; cell = cell->nextCell()) {
249         // Paint the row background behind the cell.
250         if (paintInfo.phase == PaintPhaseBlockBackground || paintInfo.phase == PaintPhaseChildBlockBackground)
251             cell->paintBackgroundsBehindCell(paintInfo, paintOffset, this);
252         if (!cell->hasSelfPaintingLayer())
253             cell->paint(paintInfo, paintOffset);
254     }
255 }
256
257 void RenderTableRow::imageChanged(WrappedImagePtr, const IntRect*)
258 {
259     // FIXME: Examine cells and repaint only the rect the image paints in.
260     repaint();
261 }
262
263 RenderTableRow* RenderTableRow::createAnonymousWithParentRenderer(const RenderObject* parent)
264 {
265     auto newRow = new RenderTableRow(parent->document(), RenderStyle::createAnonymousStyleWithDisplay(parent->style(), TABLE_ROW));
266     newRow->initializeStyle();
267     return newRow;
268 }
269
270 } // namespace WebCore