• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 1999 Lars Knoll (knoll@kde.org)
3  *           (C) 1999 Antti Koivisto (koivisto@kde.org)
4  *           (C) 2007 David Smith (catfish.man@gmail.com)
5  * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
6  * Copyright (C) Research In Motion Limited 2010. All rights reserved.
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Library General Public
10  * License as published by the Free Software Foundation; either
11  * version 2 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Library General Public License for more details.
17  *
18  * You should have received a copy of the GNU Library General Public License
19  * along with this library; see the file COPYING.LIB.  If not, write to
20  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21  * Boston, MA 02110-1301, USA.
22  */
23 
24 #include "config.h"
25 #include "RenderBlock.h"
26 
27 #include "ColumnInfo.h"
28 #include "Document.h"
29 #include "Element.h"
30 #include "FloatQuad.h"
31 #include "Frame.h"
32 #include "FrameView.h"
33 #include "GraphicsContext.h"
34 #include "HTMLFormElement.h"
35 #include "HTMLNames.h"
36 #include "HitTestResult.h"
37 #include "InlineIterator.h"
38 #include "InlineTextBox.h"
39 #include "PaintInfo.h"
40 #include "RenderCombineText.h"
41 #include "RenderFlexibleBox.h"
42 #include "RenderImage.h"
43 #include "RenderInline.h"
44 #include "RenderLayer.h"
45 #include "RenderMarquee.h"
46 #include "RenderReplica.h"
47 #include "RenderTableCell.h"
48 #include "RenderTextFragment.h"
49 #include "RenderTheme.h"
50 #include "RenderView.h"
51 #include "SelectionController.h"
52 #include "Settings.h"
53 #include "TextRun.h"
54 #include "TransformState.h"
55 #include <wtf/StdLibExtras.h>
56 
57 #ifdef ANDROID_LAYOUT
58 #include "Settings.h"
59 #endif
60 
61 using namespace std;
62 using namespace WTF;
63 using namespace Unicode;
64 
65 namespace WebCore {
66 
67 using namespace HTMLNames;
68 
69 typedef WTF::HashMap<const RenderBox*, ColumnInfo*> ColumnInfoMap;
70 static ColumnInfoMap* gColumnInfoMap = 0;
71 
72 typedef WTF::HashMap<const RenderBlock*, HashSet<RenderBox*>*> PercentHeightDescendantsMap;
73 static PercentHeightDescendantsMap* gPercentHeightDescendantsMap = 0;
74 
75 typedef WTF::HashMap<const RenderBox*, HashSet<RenderBlock*>*> PercentHeightContainerMap;
76 static PercentHeightContainerMap* gPercentHeightContainerMap = 0;
77 
78 typedef WTF::HashMap<RenderBlock*, ListHashSet<RenderInline*>*> ContinuationOutlineTableMap;
79 
80 typedef WTF::HashSet<RenderBlock*> DelayedUpdateScrollInfoSet;
81 static int gDelayUpdateScrollInfo = 0;
82 static DelayedUpdateScrollInfoSet* gDelayedUpdateScrollInfoSet = 0;
83 
84 bool RenderBlock::s_canPropagateFloatIntoSibling = false;
85 
86 // Our MarginInfo state used when laying out block children.
MarginInfo(RenderBlock * block,int beforeBorderPadding,int afterBorderPadding)87 RenderBlock::MarginInfo::MarginInfo(RenderBlock* block, int beforeBorderPadding, int afterBorderPadding)
88     : m_atBeforeSideOfBlock(true)
89     , m_atAfterSideOfBlock(false)
90     , m_marginBeforeQuirk(false)
91     , m_marginAfterQuirk(false)
92     , m_determinedMarginBeforeQuirk(false)
93 {
94     // Whether or not we can collapse our own margins with our children.  We don't do this
95     // if we had any border/padding (obviously), if we're the root or HTML elements, or if
96     // we're positioned, floating, a table cell.
97     m_canCollapseWithChildren = !block->isRenderView() && !block->isRoot() && !block->isPositioned()
98         && !block->isFloating() && !block->isTableCell() && !block->hasOverflowClip() && !block->isInlineBlockOrInlineTable()
99         && !block->isWritingModeRoot();
100 
101     m_canCollapseMarginBeforeWithChildren = m_canCollapseWithChildren && (beforeBorderPadding == 0) && block->style()->marginBeforeCollapse() != MSEPARATE;
102 
103     // If any height other than auto is specified in CSS, then we don't collapse our bottom
104     // margins with our children's margins.  To do otherwise would be to risk odd visual
105     // effects when the children overflow out of the parent block and yet still collapse
106     // with it.  We also don't collapse if we have any bottom border/padding.
107     m_canCollapseMarginAfterWithChildren = m_canCollapseWithChildren && (afterBorderPadding == 0) &&
108         (block->style()->logicalHeight().isAuto() && block->style()->logicalHeight().value() == 0) && block->style()->marginAfterCollapse() != MSEPARATE;
109 
110     m_quirkContainer = block->isTableCell() || block->isBody() || block->style()->marginBeforeCollapse() == MDISCARD ||
111         block->style()->marginAfterCollapse() == MDISCARD;
112 
113     m_positiveMargin = m_canCollapseMarginBeforeWithChildren ? block->maxPositiveMarginBefore() : 0;
114     m_negativeMargin = m_canCollapseMarginBeforeWithChildren ? block->maxNegativeMarginBefore() : 0;
115 }
116 
117 // -------------------------------------------------------------------------------------------------------
118 
RenderBlock(Node * node)119 RenderBlock::RenderBlock(Node* node)
120       : RenderBox(node)
121       , m_floatingObjects(0)
122       , m_positionedObjects(0)
123       , m_rareData(0)
124       , m_lineHeight(-1)
125       , m_beingDestroyed(false)
126 {
127     setChildrenInline(true);
128 }
129 
~RenderBlock()130 RenderBlock::~RenderBlock()
131 {
132     if (m_floatingObjects)
133         deleteAllValues(m_floatingObjects->set());
134 
135     if (hasColumns())
136         delete gColumnInfoMap->take(this);
137 
138     if (gPercentHeightDescendantsMap) {
139         if (HashSet<RenderBox*>* descendantSet = gPercentHeightDescendantsMap->take(this)) {
140             HashSet<RenderBox*>::iterator end = descendantSet->end();
141             for (HashSet<RenderBox*>::iterator descendant = descendantSet->begin(); descendant != end; ++descendant) {
142                 HashSet<RenderBlock*>* containerSet = gPercentHeightContainerMap->get(*descendant);
143                 ASSERT(containerSet);
144                 if (!containerSet)
145                     continue;
146                 ASSERT(containerSet->contains(this));
147                 containerSet->remove(this);
148                 if (containerSet->isEmpty()) {
149                     gPercentHeightContainerMap->remove(*descendant);
150                     delete containerSet;
151                 }
152             }
153             delete descendantSet;
154         }
155     }
156 }
157 
destroy()158 void RenderBlock::destroy()
159 {
160     // Mark as being destroyed to avoid trouble with merges in removeChild().
161     m_beingDestroyed = true;
162 
163     // Make sure to destroy anonymous children first while they are still connected to the rest of the tree, so that they will
164     // properly dirty line boxes that they are removed from. Effects that do :before/:after only on hover could crash otherwise.
165     children()->destroyLeftoverChildren();
166 
167     // Destroy our continuation before anything other than anonymous children.
168     // The reason we don't destroy it before anonymous children is that they may
169     // have continuations of their own that are anonymous children of our continuation.
170     RenderBoxModelObject* continuation = this->continuation();
171     if (continuation) {
172         continuation->destroy();
173         setContinuation(0);
174     }
175 
176     if (!documentBeingDestroyed()) {
177         if (firstLineBox()) {
178             // We can't wait for RenderBox::destroy to clear the selection,
179             // because by then we will have nuked the line boxes.
180             // FIXME: The SelectionController should be responsible for this when it
181             // is notified of DOM mutations.
182             if (isSelectionBorder())
183                 view()->clearSelection();
184 
185             // If we are an anonymous block, then our line boxes might have children
186             // that will outlast this block. In the non-anonymous block case those
187             // children will be destroyed by the time we return from this function.
188             if (isAnonymousBlock()) {
189                 for (InlineFlowBox* box = firstLineBox(); box; box = box->nextLineBox()) {
190                     while (InlineBox* childBox = box->firstChild())
191                         childBox->remove();
192                 }
193             }
194         } else if (parent())
195             parent()->dirtyLinesFromChangedChild(this);
196     }
197 
198     m_lineBoxes.deleteLineBoxes(renderArena());
199 
200     RenderBox::destroy();
201 }
202 
styleWillChange(StyleDifference diff,const RenderStyle * newStyle)203 void RenderBlock::styleWillChange(StyleDifference diff, const RenderStyle* newStyle)
204 {
205     s_canPropagateFloatIntoSibling = style() ? !isFloatingOrPositioned() && !avoidsFloats() : false;
206 
207     setReplaced(newStyle->isDisplayInlineType());
208 
209     if (style() && parent() && diff == StyleDifferenceLayout && style()->position() != newStyle->position()) {
210         if (newStyle->position() == StaticPosition)
211             // Clear our positioned objects list. Our absolutely positioned descendants will be
212             // inserted into our containing block's positioned objects list during layout.
213             removePositionedObjects(0);
214         else if (style()->position() == StaticPosition) {
215             // Remove our absolutely positioned descendants from their current containing block.
216             // They will be inserted into our positioned objects list during layout.
217             RenderObject* cb = parent();
218             while (cb && (cb->style()->position() == StaticPosition || (cb->isInline() && !cb->isReplaced())) && !cb->isRenderView()) {
219                 if (cb->style()->position() == RelativePosition && cb->isInline() && !cb->isReplaced()) {
220                     cb = cb->containingBlock();
221                     break;
222                 }
223                 cb = cb->parent();
224             }
225 
226             if (cb->isRenderBlock())
227                 toRenderBlock(cb)->removePositionedObjects(this);
228         }
229 
230         if (containsFloats() && !isFloating() && !isPositioned() && (newStyle->position() == AbsolutePosition || newStyle->position() == FixedPosition))
231             markAllDescendantsWithFloatsForLayout();
232     }
233 
234     RenderBox::styleWillChange(diff, newStyle);
235 }
236 
styleDidChange(StyleDifference diff,const RenderStyle * oldStyle)237 void RenderBlock::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
238 {
239     RenderBox::styleDidChange(diff, oldStyle);
240 
241     if (!isAnonymousBlock()) {
242         // Ensure that all of our continuation blocks pick up the new style.
243         for (RenderBlock* currCont = blockElementContinuation(); currCont; currCont = currCont->blockElementContinuation()) {
244             RenderBoxModelObject* nextCont = currCont->continuation();
245             currCont->setContinuation(0);
246             currCont->setStyle(style());
247             currCont->setContinuation(nextCont);
248         }
249     }
250 
251     // FIXME: We could save this call when the change only affected non-inherited properties
252     for (RenderObject* child = firstChild(); child; child = child->nextSibling()) {
253         if (child->isAnonymousBlock()) {
254             RefPtr<RenderStyle> newStyle = RenderStyle::createAnonymousStyle(style());
255             if (style()->specifiesColumns()) {
256                 if (child->style()->specifiesColumns())
257                     newStyle->inheritColumnPropertiesFrom(style());
258                 if (child->style()->columnSpan())
259                     newStyle->setColumnSpan(true);
260             }
261             newStyle->setDisplay(BLOCK);
262             child->setStyle(newStyle.release());
263         }
264     }
265 
266     m_lineHeight = -1;
267 
268     // Update pseudos for :before and :after now.
269     if (!isAnonymous() && document()->usesBeforeAfterRules() && canHaveChildren()) {
270         updateBeforeAfterContent(BEFORE);
271         updateBeforeAfterContent(AFTER);
272     }
273 
274     // After our style changed, if we lose our ability to propagate floats into next sibling
275     // blocks, then we need to find the top most parent containing that overhanging float and
276     // then mark its descendants with floats for layout and clear all floats from its next
277     // sibling blocks that exist in our floating objects list. See bug 56299 and 62875.
278     bool canPropagateFloatIntoSibling = !isFloatingOrPositioned() && !avoidsFloats();
279     if (diff == StyleDifferenceLayout && s_canPropagateFloatIntoSibling && !canPropagateFloatIntoSibling && hasOverhangingFloats()) {
280         RenderBlock* parentBlock = this;
281         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
282         FloatingObjectSetIterator end = floatingObjectSet.end();
283 
284         for (RenderObject* curr = parent(); curr && !curr->isRenderView(); curr = curr->parent()) {
285             if (curr->isRenderBlock()) {
286                 RenderBlock* currBlock = toRenderBlock(curr);
287 
288                 if (currBlock->hasOverhangingFloats()) {
289                     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
290                         RenderBox* renderer = (*it)->renderer();
291                         if (currBlock->hasOverhangingFloat(renderer)) {
292                             parentBlock = currBlock;
293                             break;
294                         }
295                     }
296                 }
297             }
298         }
299 
300         parentBlock->markAllDescendantsWithFloatsForLayout();
301         parentBlock->markSiblingsWithFloatsForLayout();
302     }
303 }
304 
updateBeforeAfterContent(PseudoId pseudoId)305 void RenderBlock::updateBeforeAfterContent(PseudoId pseudoId)
306 {
307     // If this is an anonymous wrapper, then the parent applies its own pseudo-element style to it.
308     if (parent() && parent()->createsAnonymousWrapper())
309         return;
310     return children()->updateBeforeAfterContent(this, pseudoId);
311 }
312 
continuationBefore(RenderObject * beforeChild)313 RenderBlock* RenderBlock::continuationBefore(RenderObject* beforeChild)
314 {
315     if (beforeChild && beforeChild->parent() == this)
316         return this;
317 
318     RenderBlock* curr = toRenderBlock(continuation());
319     RenderBlock* nextToLast = this;
320     RenderBlock* last = this;
321     while (curr) {
322         if (beforeChild && beforeChild->parent() == curr) {
323             if (curr->firstChild() == beforeChild)
324                 return last;
325             return curr;
326         }
327 
328         nextToLast = last;
329         last = curr;
330         curr = toRenderBlock(curr->continuation());
331     }
332 
333     if (!beforeChild && !last->firstChild())
334         return nextToLast;
335     return last;
336 }
337 
addChildToContinuation(RenderObject * newChild,RenderObject * beforeChild)338 void RenderBlock::addChildToContinuation(RenderObject* newChild, RenderObject* beforeChild)
339 {
340     RenderBlock* flow = continuationBefore(beforeChild);
341     ASSERT(!beforeChild || beforeChild->parent()->isAnonymousColumnSpanBlock() || beforeChild->parent()->isRenderBlock());
342     RenderBoxModelObject* beforeChildParent = 0;
343     if (beforeChild)
344         beforeChildParent = toRenderBoxModelObject(beforeChild->parent());
345     else {
346         RenderBoxModelObject* cont = flow->continuation();
347         if (cont)
348             beforeChildParent = cont;
349         else
350             beforeChildParent = flow;
351     }
352 
353     if (newChild->isFloatingOrPositioned())
354         return beforeChildParent->addChildIgnoringContinuation(newChild, beforeChild);
355 
356     // A continuation always consists of two potential candidates: a block or an anonymous
357     // column span box holding column span children.
358     bool childIsNormal = newChild->isInline() || !newChild->style()->columnSpan();
359     bool bcpIsNormal = beforeChildParent->isInline() || !beforeChildParent->style()->columnSpan();
360     bool flowIsNormal = flow->isInline() || !flow->style()->columnSpan();
361 
362     if (flow == beforeChildParent)
363         return flow->addChildIgnoringContinuation(newChild, beforeChild);
364 
365     // The goal here is to match up if we can, so that we can coalesce and create the
366     // minimal # of continuations needed for the inline.
367     if (childIsNormal == bcpIsNormal)
368         return beforeChildParent->addChildIgnoringContinuation(newChild, beforeChild);
369     if (flowIsNormal == childIsNormal)
370         return flow->addChildIgnoringContinuation(newChild, 0); // Just treat like an append.
371     return beforeChildParent->addChildIgnoringContinuation(newChild, beforeChild);
372 }
373 
374 
addChildToAnonymousColumnBlocks(RenderObject * newChild,RenderObject * beforeChild)375 void RenderBlock::addChildToAnonymousColumnBlocks(RenderObject* newChild, RenderObject* beforeChild)
376 {
377     ASSERT(!continuation()); // We don't yet support column spans that aren't immediate children of the multi-column block.
378 
379     // The goal is to locate a suitable box in which to place our child.
380     RenderBlock* beforeChildParent = toRenderBlock(beforeChild && beforeChild->parent()->isRenderBlock() ? beforeChild->parent() : lastChild());
381 
382     // If the new child is floating or positioned it can just go in that block.
383     if (newChild->isFloatingOrPositioned())
384         return beforeChildParent->addChildIgnoringAnonymousColumnBlocks(newChild, beforeChild);
385 
386     // See if the child can be placed in the box.
387     bool newChildHasColumnSpan = newChild->style()->columnSpan() && !newChild->isInline();
388     bool beforeChildParentHoldsColumnSpans = beforeChildParent->isAnonymousColumnSpanBlock();
389 
390     if (newChildHasColumnSpan == beforeChildParentHoldsColumnSpans)
391         return beforeChildParent->addChildIgnoringAnonymousColumnBlocks(newChild, beforeChild);
392 
393     if (!beforeChild) {
394         // Create a new block of the correct type.
395         RenderBlock* newBox = newChildHasColumnSpan ? createAnonymousColumnSpanBlock() : createAnonymousColumnsBlock();
396         children()->appendChildNode(this, newBox);
397         newBox->addChildIgnoringAnonymousColumnBlocks(newChild, 0);
398         return;
399     }
400 
401     RenderObject* immediateChild = beforeChild;
402     bool isPreviousBlockViable = true;
403     while (immediateChild->parent() != this) {
404         if (isPreviousBlockViable)
405             isPreviousBlockViable = !immediateChild->previousSibling();
406         immediateChild = immediateChild->parent();
407     }
408     if (isPreviousBlockViable && immediateChild->previousSibling())
409         return toRenderBlock(immediateChild->previousSibling())->addChildIgnoringAnonymousColumnBlocks(newChild, 0); // Treat like an append.
410 
411     // Split our anonymous blocks.
412     RenderObject* newBeforeChild = splitAnonymousBlocksAroundChild(beforeChild);
413 
414     // Create a new anonymous box of the appropriate type.
415     RenderBlock* newBox = newChildHasColumnSpan ? createAnonymousColumnSpanBlock() : createAnonymousColumnsBlock();
416     children()->insertChildNode(this, newBox, newBeforeChild);
417     newBox->addChildIgnoringAnonymousColumnBlocks(newChild, 0);
418     return;
419 }
420 
containingColumnsBlock(bool allowAnonymousColumnBlock)421 RenderBlock* RenderBlock::containingColumnsBlock(bool allowAnonymousColumnBlock)
422 {
423     RenderBlock* firstChildIgnoringAnonymousWrappers = 0;
424     for (RenderObject* curr = this; curr; curr = curr->parent()) {
425         if (!curr->isRenderBlock() || curr->isFloatingOrPositioned() || curr->isTableCell() || curr->isRoot() || curr->isRenderView() || curr->hasOverflowClip()
426             || curr->isInlineBlockOrInlineTable())
427             return 0;
428 
429         RenderBlock* currBlock = toRenderBlock(curr);
430         if (!currBlock->createsAnonymousWrapper())
431             firstChildIgnoringAnonymousWrappers = currBlock;
432 
433         if (currBlock->style()->specifiesColumns() && (allowAnonymousColumnBlock || !currBlock->isAnonymousColumnsBlock()))
434             return firstChildIgnoringAnonymousWrappers;
435 
436         if (currBlock->isAnonymousColumnSpanBlock())
437             return 0;
438     }
439     return 0;
440 }
441 
clone() const442 RenderBlock* RenderBlock::clone() const
443 {
444     RenderBlock* cloneBlock;
445     if (isAnonymousBlock())
446         cloneBlock = createAnonymousBlock();
447     else {
448         cloneBlock = new (renderArena()) RenderBlock(node());
449         cloneBlock->setStyle(style());
450         if (!childrenInline() && cloneBlock->firstChild() && cloneBlock->firstChild()->isInline())
451             cloneBlock->makeChildrenNonInline();
452     }
453     cloneBlock->setChildrenInline(childrenInline());
454     return cloneBlock;
455 }
456 
splitBlocks(RenderBlock * fromBlock,RenderBlock * toBlock,RenderBlock * middleBlock,RenderObject * beforeChild,RenderBoxModelObject * oldCont)457 void RenderBlock::splitBlocks(RenderBlock* fromBlock, RenderBlock* toBlock,
458                               RenderBlock* middleBlock,
459                               RenderObject* beforeChild, RenderBoxModelObject* oldCont)
460 {
461     // Create a clone of this inline.
462     RenderBlock* cloneBlock = clone();
463     if (!isAnonymousBlock())
464         cloneBlock->setContinuation(oldCont);
465 
466     // Now take all of the children from beforeChild to the end and remove
467     // them from |this| and place them in the clone.
468     if (!beforeChild && isAfterContent(lastChild()))
469         beforeChild = lastChild();
470     moveChildrenTo(cloneBlock, beforeChild, 0);
471 
472     // Hook |clone| up as the continuation of the middle block.
473     if (!cloneBlock->isAnonymousBlock())
474         middleBlock->setContinuation(cloneBlock);
475 
476     // We have been reparented and are now under the fromBlock.  We need
477     // to walk up our block parent chain until we hit the containing anonymous columns block.
478     // Once we hit the anonymous columns block we're done.
479     RenderBoxModelObject* curr = toRenderBoxModelObject(parent());
480     RenderBoxModelObject* currChild = this;
481 
482     while (curr && curr != fromBlock) {
483         ASSERT(curr->isRenderBlock());
484 
485         RenderBlock* blockCurr = toRenderBlock(curr);
486 
487         // Create a new clone.
488         RenderBlock* cloneChild = cloneBlock;
489         cloneBlock = blockCurr->clone();
490 
491         // Insert our child clone as the first child.
492         cloneBlock->children()->appendChildNode(cloneBlock, cloneChild);
493 
494         // Hook the clone up as a continuation of |curr|.  Note we do encounter
495         // anonymous blocks possibly as we walk up the block chain.  When we split an
496         // anonymous block, there's no need to do any continuation hookup, since we haven't
497         // actually split a real element.
498         if (!blockCurr->isAnonymousBlock()) {
499             oldCont = blockCurr->continuation();
500             blockCurr->setContinuation(cloneBlock);
501             cloneBlock->setContinuation(oldCont);
502         }
503 
504         // Someone may have indirectly caused a <q> to split.  When this happens, the :after content
505         // has to move into the inline continuation.  Call updateBeforeAfterContent to ensure that the inline's :after
506         // content gets properly destroyed.
507         if (document()->usesBeforeAfterRules())
508             blockCurr->children()->updateBeforeAfterContent(blockCurr, AFTER);
509 
510         // Now we need to take all of the children starting from the first child
511         // *after* currChild and append them all to the clone.
512         RenderObject* afterContent = isAfterContent(cloneBlock->lastChild()) ? cloneBlock->lastChild() : 0;
513         blockCurr->moveChildrenTo(cloneBlock, currChild->nextSibling(), 0, afterContent);
514 
515         // Keep walking up the chain.
516         currChild = curr;
517         curr = toRenderBoxModelObject(curr->parent());
518     }
519 
520     // Now we are at the columns block level. We need to put the clone into the toBlock.
521     toBlock->children()->appendChildNode(toBlock, cloneBlock);
522 
523     // Now take all the children after currChild and remove them from the fromBlock
524     // and put them in the toBlock.
525     fromBlock->moveChildrenTo(toBlock, currChild->nextSibling(), 0);
526 }
527 
splitFlow(RenderObject * beforeChild,RenderBlock * newBlockBox,RenderObject * newChild,RenderBoxModelObject * oldCont)528 void RenderBlock::splitFlow(RenderObject* beforeChild, RenderBlock* newBlockBox,
529                             RenderObject* newChild, RenderBoxModelObject* oldCont)
530 {
531     RenderBlock* pre = 0;
532     RenderBlock* block = containingColumnsBlock();
533 
534     // Delete our line boxes before we do the inline split into continuations.
535     block->deleteLineBoxTree();
536 
537     bool madeNewBeforeBlock = false;
538     if (block->isAnonymousColumnsBlock()) {
539         // We can reuse this block and make it the preBlock of the next continuation.
540         pre = block;
541         pre->removePositionedObjects(0);
542         block = toRenderBlock(block->parent());
543     } else {
544         // No anonymous block available for use.  Make one.
545         pre = block->createAnonymousColumnsBlock();
546         pre->setChildrenInline(false);
547         madeNewBeforeBlock = true;
548     }
549 
550     RenderBlock* post = block->createAnonymousColumnsBlock();
551     post->setChildrenInline(false);
552 
553     RenderObject* boxFirst = madeNewBeforeBlock ? block->firstChild() : pre->nextSibling();
554     if (madeNewBeforeBlock)
555         block->children()->insertChildNode(block, pre, boxFirst);
556     block->children()->insertChildNode(block, newBlockBox, boxFirst);
557     block->children()->insertChildNode(block, post, boxFirst);
558     block->setChildrenInline(false);
559 
560     if (madeNewBeforeBlock)
561         block->moveChildrenTo(pre, boxFirst, 0);
562 
563     splitBlocks(pre, post, newBlockBox, beforeChild, oldCont);
564 
565     // We already know the newBlockBox isn't going to contain inline kids, so avoid wasting
566     // time in makeChildrenNonInline by just setting this explicitly up front.
567     newBlockBox->setChildrenInline(false);
568 
569     // We delayed adding the newChild until now so that the |newBlockBox| would be fully
570     // connected, thus allowing newChild access to a renderArena should it need
571     // to wrap itself in additional boxes (e.g., table construction).
572     newBlockBox->addChild(newChild);
573 
574     // Always just do a full layout in order to ensure that line boxes (especially wrappers for images)
575     // get deleted properly.  Because objects moves from the pre block into the post block, we want to
576     // make new line boxes instead of leaving the old line boxes around.
577     pre->setNeedsLayoutAndPrefWidthsRecalc();
578     block->setNeedsLayoutAndPrefWidthsRecalc();
579     post->setNeedsLayoutAndPrefWidthsRecalc();
580 }
581 
splitAnonymousBlocksAroundChild(RenderObject * beforeChild)582 RenderObject* RenderBlock::splitAnonymousBlocksAroundChild(RenderObject* beforeChild)
583 {
584     while (beforeChild->parent() != this) {
585         RenderBlock* blockToSplit = toRenderBlock(beforeChild->parent());
586         if (blockToSplit->firstChild() != beforeChild) {
587             // We have to split the parentBlock into two blocks.
588             RenderBlock* post = createAnonymousBlockWithSameTypeAs(blockToSplit);
589             post->setChildrenInline(blockToSplit->childrenInline());
590             RenderBlock* parentBlock = toRenderBlock(blockToSplit->parent());
591             parentBlock->children()->insertChildNode(parentBlock, post, blockToSplit->nextSibling());
592             blockToSplit->moveChildrenTo(post, beforeChild, 0, blockToSplit->hasLayer());
593             post->setNeedsLayoutAndPrefWidthsRecalc();
594             blockToSplit->setNeedsLayoutAndPrefWidthsRecalc();
595             beforeChild = post;
596         } else
597             beforeChild = blockToSplit;
598     }
599     return beforeChild;
600 }
601 
makeChildrenAnonymousColumnBlocks(RenderObject * beforeChild,RenderBlock * newBlockBox,RenderObject * newChild)602 void RenderBlock::makeChildrenAnonymousColumnBlocks(RenderObject* beforeChild, RenderBlock* newBlockBox, RenderObject* newChild)
603 {
604     RenderBlock* pre = 0;
605     RenderBlock* post = 0;
606     RenderBlock* block = this; // Eventually block will not just be |this|, but will also be a block nested inside |this|.  Assign to a variable
607                                // so that we don't have to patch all of the rest of the code later on.
608 
609     // Delete the block's line boxes before we do the split.
610     block->deleteLineBoxTree();
611 
612     if (beforeChild && beforeChild->parent() != this)
613         beforeChild = splitAnonymousBlocksAroundChild(beforeChild);
614 
615     if (beforeChild != firstChild()) {
616         pre = block->createAnonymousColumnsBlock();
617         pre->setChildrenInline(block->childrenInline());
618     }
619 
620     if (beforeChild) {
621         post = block->createAnonymousColumnsBlock();
622         post->setChildrenInline(block->childrenInline());
623     }
624 
625     RenderObject* boxFirst = block->firstChild();
626     if (pre)
627         block->children()->insertChildNode(block, pre, boxFirst);
628     block->children()->insertChildNode(block, newBlockBox, boxFirst);
629     if (post)
630         block->children()->insertChildNode(block, post, boxFirst);
631     block->setChildrenInline(false);
632 
633     // The pre/post blocks always have layers, so we know to always do a full insert/remove (so we pass true as the last argument).
634     block->moveChildrenTo(pre, boxFirst, beforeChild, true);
635     block->moveChildrenTo(post, beforeChild, 0, true);
636 
637     // We already know the newBlockBox isn't going to contain inline kids, so avoid wasting
638     // time in makeChildrenNonInline by just setting this explicitly up front.
639     newBlockBox->setChildrenInline(false);
640 
641     // We delayed adding the newChild until now so that the |newBlockBox| would be fully
642     // connected, thus allowing newChild access to a renderArena should it need
643     // to wrap itself in additional boxes (e.g., table construction).
644     newBlockBox->addChild(newChild);
645 
646     // Always just do a full layout in order to ensure that line boxes (especially wrappers for images)
647     // get deleted properly.  Because objects moved from the pre block into the post block, we want to
648     // make new line boxes instead of leaving the old line boxes around.
649     if (pre)
650         pre->setNeedsLayoutAndPrefWidthsRecalc();
651     block->setNeedsLayoutAndPrefWidthsRecalc();
652     if (post)
653         post->setNeedsLayoutAndPrefWidthsRecalc();
654 }
655 
columnsBlockForSpanningElement(RenderObject * newChild)656 RenderBlock* RenderBlock::columnsBlockForSpanningElement(RenderObject* newChild)
657 {
658     // FIXME: This function is the gateway for the addition of column-span support.  It will
659     // be added to in three stages:
660     // (1) Immediate children of a multi-column block can span.
661     // (2) Nested block-level children with only block-level ancestors between them and the multi-column block can span.
662     // (3) Nested children with block or inline ancestors between them and the multi-column block can span (this is when we
663     // cross the streams and have to cope with both types of continuations mixed together).
664     // This function currently supports (1) and (2).
665     RenderBlock* columnsBlockAncestor = 0;
666     if (!newChild->isText() && newChild->style()->columnSpan() && !newChild->isBeforeOrAfterContent()
667         && !newChild->isFloatingOrPositioned() && !newChild->isInline() && !isAnonymousColumnSpanBlock()) {
668         columnsBlockAncestor = containingColumnsBlock(false);
669         if (columnsBlockAncestor) {
670             // Make sure that none of the parent ancestors have a continuation.
671             // If yes, we do not want split the block into continuations.
672             RenderObject* curr = this;
673             while (curr && curr != columnsBlockAncestor) {
674                 if (curr->isRenderBlock() && toRenderBlock(curr)->continuation()) {
675                     columnsBlockAncestor = 0;
676                     break;
677                 }
678                 curr = curr->parent();
679             }
680         }
681     }
682     return columnsBlockAncestor;
683 }
684 
addChildIgnoringAnonymousColumnBlocks(RenderObject * newChild,RenderObject * beforeChild)685 void RenderBlock::addChildIgnoringAnonymousColumnBlocks(RenderObject* newChild, RenderObject* beforeChild)
686 {
687     // Make sure we don't append things after :after-generated content if we have it.
688     if (!beforeChild) {
689         RenderObject* lastRenderer = lastChild();
690         if (isAfterContent(lastRenderer))
691             beforeChild = lastRenderer;
692         else if (lastRenderer && lastRenderer->isAnonymousBlock() && isAfterContent(lastRenderer->lastChild()))
693             beforeChild = lastRenderer->lastChild();
694     }
695 
696     // If the requested beforeChild is not one of our children, then this is because
697     // there is an anonymous container within this object that contains the beforeChild.
698     if (beforeChild && beforeChild->parent() != this) {
699         RenderObject* anonymousChild = beforeChild->parent();
700         ASSERT(anonymousChild);
701 
702         while (anonymousChild->parent() != this)
703             anonymousChild = anonymousChild->parent();
704 
705         ASSERT(anonymousChild->isAnonymous());
706 
707         if (anonymousChild->isAnonymousBlock()) {
708             // Insert the child into the anonymous block box instead of here.
709             if (newChild->isInline() || beforeChild->parent()->firstChild() != beforeChild)
710                 beforeChild->parent()->addChild(newChild, beforeChild);
711             else
712                 addChild(newChild, beforeChild->parent());
713             return;
714         }
715 
716         ASSERT(anonymousChild->isTable());
717         if ((newChild->isTableCol() && newChild->style()->display() == TABLE_COLUMN_GROUP)
718                 || (newChild->isRenderBlock() && newChild->style()->display() == TABLE_CAPTION)
719                 || newChild->isTableSection()
720                 || newChild->isTableRow()
721                 || newChild->isTableCell()) {
722             // Insert into the anonymous table.
723             anonymousChild->addChild(newChild, beforeChild);
724             return;
725         }
726 
727         // Go on to insert before the anonymous table.
728         beforeChild = anonymousChild;
729     }
730 
731     // Check for a spanning element in columns.
732     RenderBlock* columnsBlockAncestor = columnsBlockForSpanningElement(newChild);
733     if (columnsBlockAncestor) {
734         // We are placing a column-span element inside a block.
735         RenderBlock* newBox = createAnonymousColumnSpanBlock();
736 
737         if (columnsBlockAncestor != this) {
738             // We are nested inside a multi-column element and are being split by the span.  We have to break up
739             // our block into continuations.
740             RenderBoxModelObject* oldContinuation = continuation();
741             setContinuation(newBox);
742 
743             // Someone may have put a <p> inside a <q>, causing a split.  When this happens, the :after content
744             // has to move into the inline continuation.  Call updateBeforeAfterContent to ensure that our :after
745             // content gets properly destroyed.
746             bool isLastChild = (beforeChild == lastChild());
747             if (document()->usesBeforeAfterRules())
748                 children()->updateBeforeAfterContent(this, AFTER);
749             if (isLastChild && beforeChild != lastChild())
750                 beforeChild = 0; // We destroyed the last child, so now we need to update our insertion
751                                  // point to be 0.  It's just a straight append now.
752 
753             splitFlow(beforeChild, newBox, newChild, oldContinuation);
754             return;
755         }
756 
757         // We have to perform a split of this block's children.  This involves creating an anonymous block box to hold
758         // the column-spanning |newChild|.  We take all of the children from before |newChild| and put them into
759         // one anonymous columns block, and all of the children after |newChild| go into another anonymous block.
760         makeChildrenAnonymousColumnBlocks(beforeChild, newBox, newChild);
761         return;
762     }
763 
764     bool madeBoxesNonInline = false;
765 
766     // A block has to either have all of its children inline, or all of its children as blocks.
767     // So, if our children are currently inline and a block child has to be inserted, we move all our
768     // inline children into anonymous block boxes.
769     if (childrenInline() && !newChild->isInline() && !newChild->isFloatingOrPositioned()) {
770         // This is a block with inline content. Wrap the inline content in anonymous blocks.
771         makeChildrenNonInline(beforeChild);
772         madeBoxesNonInline = true;
773 
774         if (beforeChild && beforeChild->parent() != this) {
775             beforeChild = beforeChild->parent();
776             ASSERT(beforeChild->isAnonymousBlock());
777             ASSERT(beforeChild->parent() == this);
778         }
779     } else if (!childrenInline() && (newChild->isFloatingOrPositioned() || newChild->isInline())) {
780         // If we're inserting an inline child but all of our children are blocks, then we have to make sure
781         // it is put into an anomyous block box. We try to use an existing anonymous box if possible, otherwise
782         // a new one is created and inserted into our list of children in the appropriate position.
783         RenderObject* afterChild = beforeChild ? beforeChild->previousSibling() : lastChild();
784 
785         if (afterChild && afterChild->isAnonymousBlock()) {
786             afterChild->addChild(newChild);
787             return;
788         }
789 
790         if (newChild->isInline()) {
791             // No suitable existing anonymous box - create a new one.
792             RenderBlock* newBox = createAnonymousBlock();
793             RenderBox::addChild(newBox, beforeChild);
794             newBox->addChild(newChild);
795             return;
796         }
797     }
798 
799     RenderBox::addChild(newChild, beforeChild);
800 
801     if (madeBoxesNonInline && parent() && isAnonymousBlock() && parent()->isRenderBlock())
802         toRenderBlock(parent())->removeLeftoverAnonymousBlock(this);
803     // this object may be dead here
804 }
805 
addChild(RenderObject * newChild,RenderObject * beforeChild)806 void RenderBlock::addChild(RenderObject* newChild, RenderObject* beforeChild)
807 {
808     if (continuation() && !isAnonymousBlock())
809         return addChildToContinuation(newChild, beforeChild);
810     return addChildIgnoringContinuation(newChild, beforeChild);
811 }
812 
addChildIgnoringContinuation(RenderObject * newChild,RenderObject * beforeChild)813 void RenderBlock::addChildIgnoringContinuation(RenderObject* newChild, RenderObject* beforeChild)
814 {
815     if (!isAnonymousBlock() && firstChild() && (firstChild()->isAnonymousColumnsBlock() || firstChild()->isAnonymousColumnSpanBlock()))
816         return addChildToAnonymousColumnBlocks(newChild, beforeChild);
817     return addChildIgnoringAnonymousColumnBlocks(newChild, beforeChild);
818 }
819 
getInlineRun(RenderObject * start,RenderObject * boundary,RenderObject * & inlineRunStart,RenderObject * & inlineRunEnd)820 static void getInlineRun(RenderObject* start, RenderObject* boundary,
821                          RenderObject*& inlineRunStart,
822                          RenderObject*& inlineRunEnd)
823 {
824     // Beginning at |start| we find the largest contiguous run of inlines that
825     // we can.  We denote the run with start and end points, |inlineRunStart|
826     // and |inlineRunEnd|.  Note that these two values may be the same if
827     // we encounter only one inline.
828     //
829     // We skip any non-inlines we encounter as long as we haven't found any
830     // inlines yet.
831     //
832     // |boundary| indicates a non-inclusive boundary point.  Regardless of whether |boundary|
833     // is inline or not, we will not include it in a run with inlines before it.  It's as though we encountered
834     // a non-inline.
835 
836     // Start by skipping as many non-inlines as we can.
837     RenderObject * curr = start;
838     bool sawInline;
839     do {
840         while (curr && !(curr->isInline() || curr->isFloatingOrPositioned()))
841             curr = curr->nextSibling();
842 
843         inlineRunStart = inlineRunEnd = curr;
844 
845         if (!curr)
846             return; // No more inline children to be found.
847 
848         sawInline = curr->isInline();
849 
850         curr = curr->nextSibling();
851         while (curr && (curr->isInline() || curr->isFloatingOrPositioned()) && (curr != boundary)) {
852             inlineRunEnd = curr;
853             if (curr->isInline())
854                 sawInline = true;
855             curr = curr->nextSibling();
856         }
857     } while (!sawInline);
858 }
859 
deleteLineBoxTree()860 void RenderBlock::deleteLineBoxTree()
861 {
862     m_lineBoxes.deleteLineBoxTree(renderArena());
863 }
864 
createRootInlineBox()865 RootInlineBox* RenderBlock::createRootInlineBox()
866 {
867     return new (renderArena()) RootInlineBox(this);
868 }
869 
createAndAppendRootInlineBox()870 RootInlineBox* RenderBlock::createAndAppendRootInlineBox()
871 {
872     RootInlineBox* rootBox = createRootInlineBox();
873     m_lineBoxes.appendLineBox(rootBox);
874     return rootBox;
875 }
876 
moveChildTo(RenderBlock * to,RenderObject * child,RenderObject * beforeChild,bool fullRemoveInsert)877 void RenderBlock::moveChildTo(RenderBlock* to, RenderObject* child, RenderObject* beforeChild, bool fullRemoveInsert)
878 {
879     ASSERT(this == child->parent());
880     ASSERT(!beforeChild || to == beforeChild->parent());
881     to->children()->insertChildNode(to, children()->removeChildNode(this, child, fullRemoveInsert), beforeChild, fullRemoveInsert);
882 }
883 
moveChildrenTo(RenderBlock * to,RenderObject * startChild,RenderObject * endChild,RenderObject * beforeChild,bool fullRemoveInsert)884 void RenderBlock::moveChildrenTo(RenderBlock* to, RenderObject* startChild, RenderObject* endChild, RenderObject* beforeChild, bool fullRemoveInsert)
885 {
886     ASSERT(!beforeChild || to == beforeChild->parent());
887     RenderObject* nextChild = startChild;
888     while (nextChild && nextChild != endChild) {
889         RenderObject* child = nextChild;
890         nextChild = child->nextSibling();
891         to->children()->insertChildNode(to, children()->removeChildNode(this, child, fullRemoveInsert), beforeChild, fullRemoveInsert);
892         if (child == endChild)
893             return;
894     }
895 }
896 
makeChildrenNonInline(RenderObject * insertionPoint)897 void RenderBlock::makeChildrenNonInline(RenderObject *insertionPoint)
898 {
899     // makeChildrenNonInline takes a block whose children are *all* inline and it
900     // makes sure that inline children are coalesced under anonymous
901     // blocks.  If |insertionPoint| is defined, then it represents the insertion point for
902     // the new block child that is causing us to have to wrap all the inlines.  This
903     // means that we cannot coalesce inlines before |insertionPoint| with inlines following
904     // |insertionPoint|, because the new child is going to be inserted in between the inlines,
905     // splitting them.
906     ASSERT(isInlineBlockOrInlineTable() || !isInline());
907     ASSERT(!insertionPoint || insertionPoint->parent() == this);
908 
909     setChildrenInline(false);
910 
911     RenderObject *child = firstChild();
912     if (!child)
913         return;
914 
915     deleteLineBoxTree();
916 
917     while (child) {
918         RenderObject *inlineRunStart, *inlineRunEnd;
919         getInlineRun(child, insertionPoint, inlineRunStart, inlineRunEnd);
920 
921         if (!inlineRunStart)
922             break;
923 
924         child = inlineRunEnd->nextSibling();
925 
926         RenderBlock* block = createAnonymousBlock();
927         children()->insertChildNode(this, block, inlineRunStart);
928         moveChildrenTo(block, inlineRunStart, child);
929     }
930 
931 #ifndef NDEBUG
932     for (RenderObject *c = firstChild(); c; c = c->nextSibling())
933         ASSERT(!c->isInline());
934 #endif
935 
936     repaint();
937 }
938 
removeLeftoverAnonymousBlock(RenderBlock * child)939 void RenderBlock::removeLeftoverAnonymousBlock(RenderBlock* child)
940 {
941     ASSERT(child->isAnonymousBlock());
942     ASSERT(!child->childrenInline());
943 
944     if (child->continuation() || (child->firstChild() && (child->isAnonymousColumnSpanBlock() || child->isAnonymousColumnsBlock())))
945         return;
946 
947     RenderObject* firstAnChild = child->m_children.firstChild();
948     RenderObject* lastAnChild = child->m_children.lastChild();
949     if (firstAnChild) {
950         RenderObject* o = firstAnChild;
951         while (o) {
952             o->setParent(this);
953             o = o->nextSibling();
954         }
955         firstAnChild->setPreviousSibling(child->previousSibling());
956         lastAnChild->setNextSibling(child->nextSibling());
957         if (child->previousSibling())
958             child->previousSibling()->setNextSibling(firstAnChild);
959         if (child->nextSibling())
960             child->nextSibling()->setPreviousSibling(lastAnChild);
961 
962         if (child == m_children.firstChild())
963             m_children.setFirstChild(firstAnChild);
964         if (child == m_children.lastChild())
965             m_children.setLastChild(lastAnChild);
966     } else {
967         if (child == m_children.firstChild())
968             m_children.setFirstChild(child->nextSibling());
969         if (child == m_children.lastChild())
970             m_children.setLastChild(child->previousSibling());
971 
972         if (child->previousSibling())
973             child->previousSibling()->setNextSibling(child->nextSibling());
974         if (child->nextSibling())
975             child->nextSibling()->setPreviousSibling(child->previousSibling());
976     }
977     child->setParent(0);
978     child->setPreviousSibling(0);
979     child->setNextSibling(0);
980 
981     child->children()->setFirstChild(0);
982     child->m_next = 0;
983 
984     child->destroy();
985 }
986 
canMergeContiguousAnonymousBlocks(RenderObject * oldChild,RenderObject * prev,RenderObject * next)987 static bool canMergeContiguousAnonymousBlocks(RenderObject* oldChild, RenderObject* prev, RenderObject* next)
988 {
989     if (oldChild->documentBeingDestroyed() || oldChild->isInline() || oldChild->virtualContinuation())
990         return false;
991 
992     if (oldChild->parent() && oldChild->parent()->isDetails())
993         return false;
994 
995     if ((prev && (!prev->isAnonymousBlock() || toRenderBlock(prev)->continuation() || toRenderBlock(prev)->beingDestroyed()))
996         || (next && (!next->isAnonymousBlock() || toRenderBlock(next)->continuation() || toRenderBlock(next)->beingDestroyed())))
997         return false;
998 
999     // FIXME: This check isn't required when inline run-ins can't be split into continuations.
1000     if (prev && prev->firstChild() && prev->firstChild()->isInline() && prev->firstChild()->isRunIn())
1001         return false;
1002 
1003     if ((prev && (prev->isRubyRun() || prev->isRubyBase()))
1004         || (next && (next->isRubyRun() || next->isRubyBase())))
1005         return false;
1006 
1007     if (!prev || !next)
1008         return true;
1009 
1010     // Make sure the types of the anonymous blocks match up.
1011     return prev->isAnonymousColumnsBlock() == next->isAnonymousColumnsBlock()
1012            && prev->isAnonymousColumnSpanBlock() == next->isAnonymousColumnSpanBlock();
1013 }
1014 
removeChild(RenderObject * oldChild)1015 void RenderBlock::removeChild(RenderObject* oldChild)
1016 {
1017     // If this child is a block, and if our previous and next siblings are
1018     // both anonymous blocks with inline content, then we can go ahead and
1019     // fold the inline content back together.
1020     RenderObject* prev = oldChild->previousSibling();
1021     RenderObject* next = oldChild->nextSibling();
1022     bool canMergeAnonymousBlocks = canMergeContiguousAnonymousBlocks(oldChild, prev, next);
1023     if (canMergeAnonymousBlocks && prev && next) {
1024         prev->setNeedsLayoutAndPrefWidthsRecalc();
1025         RenderBlock* nextBlock = toRenderBlock(next);
1026         RenderBlock* prevBlock = toRenderBlock(prev);
1027 
1028         if (prev->childrenInline() != next->childrenInline()) {
1029             RenderBlock* inlineChildrenBlock = prev->childrenInline() ? prevBlock : nextBlock;
1030             RenderBlock* blockChildrenBlock = prev->childrenInline() ? nextBlock : prevBlock;
1031 
1032             // Place the inline children block inside of the block children block instead of deleting it.
1033             // In order to reuse it, we have to reset it to just be a generic anonymous block.  Make sure
1034             // to clear out inherited column properties by just making a new style, and to also clear the
1035             // column span flag if it is set.
1036             ASSERT(!inlineChildrenBlock->continuation());
1037             RefPtr<RenderStyle> newStyle = RenderStyle::createAnonymousStyle(style());
1038             children()->removeChildNode(this, inlineChildrenBlock, inlineChildrenBlock->hasLayer());
1039             inlineChildrenBlock->setStyle(newStyle);
1040 
1041             // Now just put the inlineChildrenBlock inside the blockChildrenBlock.
1042             blockChildrenBlock->children()->insertChildNode(blockChildrenBlock, inlineChildrenBlock, prev == inlineChildrenBlock ? blockChildrenBlock->firstChild() : 0,
1043                                                             inlineChildrenBlock->hasLayer() || blockChildrenBlock->hasLayer());
1044             next->setNeedsLayoutAndPrefWidthsRecalc();
1045 
1046             // inlineChildrenBlock got reparented to blockChildrenBlock, so it is no longer a child
1047             // of "this". we null out prev or next so that is not used later in the function.
1048             if (inlineChildrenBlock == prevBlock)
1049                 prev = 0;
1050             else
1051                 next = 0;
1052         } else {
1053             // Take all the children out of the |next| block and put them in
1054             // the |prev| block.
1055             nextBlock->moveAllChildrenTo(prevBlock, nextBlock->hasLayer() || prevBlock->hasLayer());
1056 
1057             // Delete the now-empty block's lines and nuke it.
1058             nextBlock->deleteLineBoxTree();
1059             nextBlock->destroy();
1060             next = 0;
1061         }
1062     }
1063 
1064     RenderBox::removeChild(oldChild);
1065 
1066     RenderObject* child = prev ? prev : next;
1067     if (canMergeAnonymousBlocks && child && !child->previousSibling() && !child->nextSibling() && !isFlexibleBox()) {
1068         // The removal has knocked us down to containing only a single anonymous
1069         // box.  We can go ahead and pull the content right back up into our
1070         // box.
1071         setNeedsLayoutAndPrefWidthsRecalc();
1072         setChildrenInline(child->childrenInline());
1073         RenderBlock* anonBlock = toRenderBlock(children()->removeChildNode(this, child, child->hasLayer()));
1074         anonBlock->moveAllChildrenTo(this, child->hasLayer());
1075         // Delete the now-empty block's lines and nuke it.
1076         anonBlock->deleteLineBoxTree();
1077         anonBlock->destroy();
1078     }
1079 
1080     if (!firstChild() && !documentBeingDestroyed()) {
1081         // If this was our last child be sure to clear out our line boxes.
1082         if (childrenInline())
1083             lineBoxes()->deleteLineBoxes(renderArena());
1084     }
1085 }
1086 
isSelfCollapsingBlock() const1087 bool RenderBlock::isSelfCollapsingBlock() const
1088 {
1089     // We are not self-collapsing if we
1090     // (a) have a non-zero height according to layout (an optimization to avoid wasting time)
1091     // (b) are a table,
1092     // (c) have border/padding,
1093     // (d) have a min-height
1094     // (e) have specified that one of our margins can't collapse using a CSS extension
1095     if (logicalHeight() > 0
1096         || isTable() || borderAndPaddingLogicalHeight()
1097         || style()->logicalMinHeight().isPositive()
1098         || style()->marginBeforeCollapse() == MSEPARATE || style()->marginAfterCollapse() == MSEPARATE)
1099         return false;
1100 
1101     Length logicalHeightLength = style()->logicalHeight();
1102     bool hasAutoHeight = logicalHeightLength.isAuto();
1103     if (logicalHeightLength.isPercent() && !document()->inQuirksMode()) {
1104         hasAutoHeight = true;
1105         for (RenderBlock* cb = containingBlock(); !cb->isRenderView(); cb = cb->containingBlock()) {
1106             if (cb->style()->logicalHeight().isFixed() || cb->isTableCell())
1107                 hasAutoHeight = false;
1108         }
1109     }
1110 
1111     // If the height is 0 or auto, then whether or not we are a self-collapsing block depends
1112     // on whether we have content that is all self-collapsing or not.
1113     if (hasAutoHeight || ((logicalHeightLength.isFixed() || logicalHeightLength.isPercent()) && logicalHeightLength.isZero())) {
1114         // If the block has inline children, see if we generated any line boxes.  If we have any
1115         // line boxes, then we can't be self-collapsing, since we have content.
1116         if (childrenInline())
1117             return !firstLineBox();
1118 
1119         // Whether or not we collapse is dependent on whether all our normal flow children
1120         // are also self-collapsing.
1121         for (RenderBox* child = firstChildBox(); child; child = child->nextSiblingBox()) {
1122             if (child->isFloatingOrPositioned())
1123                 continue;
1124             if (!child->isSelfCollapsingBlock())
1125                 return false;
1126         }
1127         return true;
1128     }
1129     return false;
1130 }
1131 
startDelayUpdateScrollInfo()1132 void RenderBlock::startDelayUpdateScrollInfo()
1133 {
1134     if (gDelayUpdateScrollInfo == 0) {
1135         ASSERT(!gDelayedUpdateScrollInfoSet);
1136         gDelayedUpdateScrollInfoSet = new DelayedUpdateScrollInfoSet;
1137     }
1138     ASSERT(gDelayedUpdateScrollInfoSet);
1139     ++gDelayUpdateScrollInfo;
1140 }
1141 
finishDelayUpdateScrollInfo()1142 void RenderBlock::finishDelayUpdateScrollInfo()
1143 {
1144     --gDelayUpdateScrollInfo;
1145     ASSERT(gDelayUpdateScrollInfo >= 0);
1146     if (gDelayUpdateScrollInfo == 0) {
1147         ASSERT(gDelayedUpdateScrollInfoSet);
1148 
1149         OwnPtr<DelayedUpdateScrollInfoSet> infoSet(gDelayedUpdateScrollInfoSet);
1150         gDelayedUpdateScrollInfoSet = 0;
1151 
1152         for (DelayedUpdateScrollInfoSet::iterator it = infoSet->begin(); it != infoSet->end(); ++it) {
1153             RenderBlock* block = *it;
1154             if (block->hasOverflowClip()) {
1155                 block->layer()->updateScrollInfoAfterLayout();
1156             }
1157         }
1158     }
1159 }
1160 
updateScrollInfoAfterLayout()1161 void RenderBlock::updateScrollInfoAfterLayout()
1162 {
1163     if (hasOverflowClip()) {
1164         if (gDelayUpdateScrollInfo)
1165             gDelayedUpdateScrollInfoSet->add(this);
1166         else
1167             layer()->updateScrollInfoAfterLayout();
1168     }
1169 }
1170 
layout()1171 void RenderBlock::layout()
1172 {
1173     // Update our first letter info now.
1174     updateFirstLetter();
1175 
1176     // Table cells call layoutBlock directly, so don't add any logic here.  Put code into
1177     // layoutBlock().
1178     layoutBlock(false);
1179 
1180     // It's safe to check for control clip here, since controls can never be table cells.
1181     // If we have a lightweight clip, there can never be any overflow from children.
1182     if (hasControlClip() && m_overflow)
1183         clearLayoutOverflow();
1184 }
1185 
layoutBlock(bool relayoutChildren,int pageLogicalHeight)1186 void RenderBlock::layoutBlock(bool relayoutChildren, int pageLogicalHeight)
1187 {
1188     ASSERT(needsLayout());
1189 
1190     if (isInline() && !isInlineBlockOrInlineTable()) // Inline <form>s inside various table elements can
1191         return;                                      // cause us to come in here.  Just bail.
1192 
1193     if (!relayoutChildren && simplifiedLayout())
1194         return;
1195 
1196     LayoutRepainter repainter(*this, m_everHadLayout && checkForRepaintDuringLayout());
1197 
1198     int oldWidth = logicalWidth();
1199     int oldColumnWidth = desiredColumnWidth();
1200 
1201     computeLogicalWidth();
1202     calcColumnWidth();
1203 
1204     m_overflow.clear();
1205 
1206     if (oldWidth != logicalWidth() || oldColumnWidth != desiredColumnWidth())
1207         relayoutChildren = true;
1208 
1209 #ifdef ANDROID_LAYOUT
1210     checkAndSetRelayoutChildren(&relayoutChildren);
1211 #endif
1212 
1213     clearFloats();
1214 
1215     int previousHeight = logicalHeight();
1216     setLogicalHeight(0);
1217     bool hasSpecifiedPageLogicalHeight = false;
1218     bool pageLogicalHeightChanged = false;
1219     ColumnInfo* colInfo = columnInfo();
1220     if (hasColumns()) {
1221         if (!pageLogicalHeight) {
1222             // We need to go ahead and set our explicit page height if one exists, so that we can
1223             // avoid doing two layout passes.
1224             computeLogicalHeight();
1225             int columnHeight = contentLogicalHeight();
1226             if (columnHeight > 0) {
1227                 pageLogicalHeight = columnHeight;
1228                 hasSpecifiedPageLogicalHeight = true;
1229             }
1230             setLogicalHeight(0);
1231         }
1232         if (colInfo->columnHeight() != pageLogicalHeight && m_everHadLayout) {
1233             colInfo->setColumnHeight(pageLogicalHeight);
1234             pageLogicalHeightChanged = true;
1235         }
1236 
1237         if (!hasSpecifiedPageLogicalHeight && !pageLogicalHeight)
1238             colInfo->clearForcedBreaks();
1239     }
1240 
1241     LayoutStateMaintainer statePusher(view(), this, IntSize(x(), y()), hasColumns() || hasTransform() || hasReflection() || style()->isFlippedBlocksWritingMode(), pageLogicalHeight, pageLogicalHeightChanged, colInfo);
1242 
1243     // We use four values, maxTopPos, maxTopNeg, maxBottomPos, and maxBottomNeg, to track
1244     // our current maximal positive and negative margins.  These values are used when we
1245     // are collapsed with adjacent blocks, so for example, if you have block A and B
1246     // collapsing together, then you'd take the maximal positive margin from both A and B
1247     // and subtract it from the maximal negative margin from both A and B to get the
1248     // true collapsed margin.  This algorithm is recursive, so when we finish layout()
1249     // our block knows its current maximal positive/negative values.
1250     //
1251     // Start out by setting our margin values to our current margins.  Table cells have
1252     // no margins, so we don't fill in the values for table cells.
1253     bool isCell = isTableCell();
1254     if (!isCell) {
1255         initMaxMarginValues();
1256 
1257         setMarginBeforeQuirk(style()->marginBefore().quirk());
1258         setMarginAfterQuirk(style()->marginAfter().quirk());
1259 
1260         Node* n = node();
1261         if (n && n->hasTagName(formTag) && static_cast<HTMLFormElement*>(n)->isMalformed()) {
1262             // See if this form is malformed (i.e., unclosed). If so, don't give the form
1263             // a bottom margin.
1264             setMaxMarginAfterValues(0, 0);
1265         }
1266 
1267         setPaginationStrut(0);
1268     }
1269 
1270     // For overflow:scroll blocks, ensure we have both scrollbars in place always.
1271     if (scrollsOverflow()) {
1272         if (style()->overflowX() == OSCROLL)
1273             layer()->setHasHorizontalScrollbar(true);
1274         if (style()->overflowY() == OSCROLL)
1275             layer()->setHasVerticalScrollbar(true);
1276     }
1277 
1278     int repaintLogicalTop = 0;
1279     int repaintLogicalBottom = 0;
1280     int maxFloatLogicalBottom = 0;
1281     if (!firstChild() && !isAnonymousBlock())
1282         setChildrenInline(true);
1283     if (childrenInline())
1284         layoutInlineChildren(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
1285     else
1286         layoutBlockChildren(relayoutChildren, maxFloatLogicalBottom);
1287 
1288     // Expand our intrinsic height to encompass floats.
1289     int toAdd = borderAfter() + paddingAfter() + scrollbarLogicalHeight();
1290     if (lowestFloatLogicalBottom() > (logicalHeight() - toAdd) && expandsToEncloseOverhangingFloats())
1291         setLogicalHeight(lowestFloatLogicalBottom() + toAdd);
1292 
1293     if (layoutColumns(hasSpecifiedPageLogicalHeight, pageLogicalHeight, statePusher))
1294         return;
1295 
1296     // Calculate our new height.
1297     int oldHeight = logicalHeight();
1298     int oldClientAfterEdge = clientLogicalBottom();
1299     computeLogicalHeight();
1300     int newHeight = logicalHeight();
1301     if (oldHeight != newHeight) {
1302         if (oldHeight > newHeight && maxFloatLogicalBottom > newHeight && !childrenInline()) {
1303             // One of our children's floats may have become an overhanging float for us. We need to look for it.
1304             for (RenderObject* child = firstChild(); child; child = child->nextSibling()) {
1305                 if (child->isBlockFlow() && !child->isFloatingOrPositioned()) {
1306                     RenderBlock* block = toRenderBlock(child);
1307                     if (block->lowestFloatLogicalBottom() + block->logicalTop() > newHeight)
1308                         addOverhangingFloats(block, -block->logicalLeft(), -block->logicalTop(), false);
1309                 }
1310             }
1311         }
1312     }
1313 
1314     if (previousHeight != newHeight)
1315         relayoutChildren = true;
1316 
1317     layoutPositionedObjects(relayoutChildren || isRoot());
1318 
1319     // Add overflow from children (unless we're multi-column, since in that case all our child overflow is clipped anyway).
1320     computeOverflow(oldClientAfterEdge);
1321 
1322     statePusher.pop();
1323 
1324     if (view()->layoutState()->m_pageLogicalHeight)
1325         setPageLogicalOffset(view()->layoutState()->pageLogicalOffset(logicalTop()));
1326 
1327     updateLayerTransform();
1328 
1329     // Update our scroll information if we're overflow:auto/scroll/hidden now that we know if
1330     // we overflow or not.
1331     updateScrollInfoAfterLayout();
1332 
1333     // Repaint with our new bounds if they are different from our old bounds.
1334     bool didFullRepaint = repainter.repaintAfterLayout();
1335     if (!didFullRepaint && repaintLogicalTop != repaintLogicalBottom && (style()->visibility() == VISIBLE || enclosingLayer()->hasVisibleContent())) {
1336         // FIXME: We could tighten up the left and right invalidation points if we let layoutInlineChildren fill them in based off the particular lines
1337         // it had to lay out.  We wouldn't need the hasOverflowClip() hack in that case either.
1338         int repaintLogicalLeft = logicalLeftVisualOverflow();
1339         int repaintLogicalRight = logicalRightVisualOverflow();
1340         if (hasOverflowClip()) {
1341             // If we have clipped overflow, we should use layout overflow as well, since visual overflow from lines didn't propagate to our block's overflow.
1342             // Note the old code did this as well but even for overflow:visible.  The addition of hasOverflowClip() at least tightens up the hack a bit.
1343             // layoutInlineChildren should be patched to compute the entire repaint rect.
1344             repaintLogicalLeft = min(repaintLogicalLeft, logicalLeftLayoutOverflow());
1345             repaintLogicalRight = max(repaintLogicalRight, logicalRightLayoutOverflow());
1346         }
1347 
1348         IntRect repaintRect;
1349         if (isHorizontalWritingMode())
1350             repaintRect = IntRect(repaintLogicalLeft, repaintLogicalTop, repaintLogicalRight - repaintLogicalLeft, repaintLogicalBottom - repaintLogicalTop);
1351         else
1352             repaintRect = IntRect(repaintLogicalTop, repaintLogicalLeft, repaintLogicalBottom - repaintLogicalTop, repaintLogicalRight - repaintLogicalLeft);
1353 
1354         // The repaint rect may be split across columns, in which case adjustRectForColumns() will return the union.
1355         adjustRectForColumns(repaintRect);
1356 
1357         repaintRect.inflate(maximalOutlineSize(PaintPhaseOutline));
1358 
1359         if (hasOverflowClip()) {
1360             // Adjust repaint rect for scroll offset
1361             repaintRect.move(-layer()->scrolledContentOffset());
1362 
1363             // Don't allow this rect to spill out of our overflow box.
1364             repaintRect.intersect(IntRect(0, 0, width(), height()));
1365         }
1366 
1367         // Make sure the rect is still non-empty after intersecting for overflow above
1368         if (!repaintRect.isEmpty()) {
1369             repaintRectangle(repaintRect); // We need to do a partial repaint of our content.
1370             if (hasReflection())
1371                 repaintRectangle(reflectedRect(repaintRect));
1372         }
1373     }
1374     setNeedsLayout(false);
1375 }
1376 
addOverflowFromChildren()1377 void RenderBlock::addOverflowFromChildren()
1378 {
1379     if (!hasColumns()) {
1380         if (childrenInline())
1381             addOverflowFromInlineChildren();
1382         else
1383             addOverflowFromBlockChildren();
1384     } else {
1385         ColumnInfo* colInfo = columnInfo();
1386         if (columnCount(colInfo)) {
1387             IntRect lastRect = columnRectAt(colInfo, columnCount(colInfo) - 1);
1388             if (isHorizontalWritingMode()) {
1389                 int overflowLeft = !style()->isLeftToRightDirection() ? min(0, lastRect.x()) : 0;
1390                 int overflowRight = style()->isLeftToRightDirection() ? max(width(), lastRect.maxX()) : 0;
1391                 int overflowHeight = borderBefore() + paddingBefore() + colInfo->columnHeight();
1392                 addLayoutOverflow(IntRect(overflowLeft, 0, overflowRight - overflowLeft, overflowHeight));
1393                 if (!hasOverflowClip())
1394                     addVisualOverflow(IntRect(overflowLeft, 0, overflowRight - overflowLeft, overflowHeight));
1395             } else {
1396                 IntRect lastRect = columnRectAt(colInfo, columnCount(colInfo) - 1);
1397                 int overflowTop = !style()->isLeftToRightDirection() ? min(0, lastRect.y()) : 0;
1398                 int overflowBottom = style()->isLeftToRightDirection() ? max(height(), lastRect.maxY()) : 0;
1399                 int overflowWidth = borderBefore() + paddingBefore() + colInfo->columnHeight();
1400                 addLayoutOverflow(IntRect(0, overflowTop, overflowWidth, overflowBottom - overflowTop));
1401                 if (!hasOverflowClip())
1402                     addVisualOverflow(IntRect(0, overflowTop, overflowWidth, overflowBottom - overflowTop));
1403             }
1404         }
1405     }
1406 }
1407 
computeOverflow(int oldClientAfterEdge,bool recomputeFloats)1408 void RenderBlock::computeOverflow(int oldClientAfterEdge, bool recomputeFloats)
1409 {
1410     // Add overflow from children.
1411     addOverflowFromChildren();
1412 
1413     if (!hasColumns() && (recomputeFloats || isRoot() || expandsToEncloseOverhangingFloats() || hasSelfPaintingLayer()))
1414         addOverflowFromFloats();
1415 
1416     // Add in the overflow from positioned objects.
1417     addOverflowFromPositionedObjects();
1418 
1419     if (hasOverflowClip()) {
1420         // When we have overflow clip, propagate the original spillout since it will include collapsed bottom margins
1421         // and bottom padding.  Set the axis we don't care about to be 1, since we want this overflow to always
1422         // be considered reachable.
1423         IntRect clientRect(clientBoxRect());
1424         IntRect rectToApply;
1425         if (isHorizontalWritingMode())
1426             rectToApply = IntRect(clientRect.x(), clientRect.y(), 1, max(0, oldClientAfterEdge - clientRect.y()));
1427         else
1428             rectToApply = IntRect(clientRect.x(), clientRect.y(), max(0, oldClientAfterEdge - clientRect.x()), 1);
1429         addLayoutOverflow(rectToApply);
1430     }
1431 
1432     // Add visual overflow from box-shadow and reflections.
1433     addShadowOverflow();
1434 }
1435 
addOverflowFromBlockChildren()1436 void RenderBlock::addOverflowFromBlockChildren()
1437 {
1438     for (RenderBox* child = firstChildBox(); child; child = child->nextSiblingBox()) {
1439         if (!child->isFloatingOrPositioned())
1440             addOverflowFromChild(child);
1441     }
1442 }
1443 
addOverflowFromFloats()1444 void RenderBlock::addOverflowFromFloats()
1445 {
1446     if (!m_floatingObjects)
1447         return;
1448 
1449     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
1450     FloatingObjectSetIterator end = floatingObjectSet.end();
1451     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
1452         FloatingObject* r = *it;
1453         if (r->m_isDescendant)
1454             addOverflowFromChild(r->m_renderer, IntSize(xPositionForFloatIncludingMargin(r), yPositionForFloatIncludingMargin(r)));
1455     }
1456     return;
1457 }
1458 
addOverflowFromPositionedObjects()1459 void RenderBlock::addOverflowFromPositionedObjects()
1460 {
1461     if (!m_positionedObjects)
1462         return;
1463 
1464     RenderBox* positionedObject;
1465     Iterator end = m_positionedObjects->end();
1466     for (Iterator it = m_positionedObjects->begin(); it != end; ++it) {
1467         positionedObject = *it;
1468 
1469         // Fixed positioned elements don't contribute to layout overflow, since they don't scroll with the content.
1470         if (positionedObject->style()->position() != FixedPosition)
1471             addOverflowFromChild(positionedObject);
1472     }
1473 }
1474 
expandsToEncloseOverhangingFloats() const1475 bool RenderBlock::expandsToEncloseOverhangingFloats() const
1476 {
1477     return isInlineBlockOrInlineTable() || isFloatingOrPositioned() || hasOverflowClip() || (parent() && parent()->isFlexibleBox())
1478            || hasColumns() || isTableCell() || isFieldset() || isWritingModeRoot();
1479 }
1480 
adjustPositionedBlock(RenderBox * child,const MarginInfo & marginInfo)1481 void RenderBlock::adjustPositionedBlock(RenderBox* child, const MarginInfo& marginInfo)
1482 {
1483     bool isHorizontal = isHorizontalWritingMode();
1484     bool hasStaticBlockPosition = child->style()->hasStaticBlockPosition(isHorizontal);
1485     RenderLayer* childLayer = child->layer();
1486 
1487     childLayer->setStaticInlinePosition(borderAndPaddingStart());
1488 
1489     int logicalTop = logicalHeight();
1490     if (!marginInfo.canCollapseWithMarginBefore()) {
1491         child->computeBlockDirectionMargins(this);
1492         int marginBefore = marginBeforeForChild(child);
1493         int collapsedBeforePos = marginInfo.positiveMargin();
1494         int collapsedBeforeNeg = marginInfo.negativeMargin();
1495         if (marginBefore > 0) {
1496             if (marginBefore > collapsedBeforePos)
1497                 collapsedBeforePos = marginBefore;
1498         } else {
1499             if (-marginBefore > collapsedBeforeNeg)
1500                 collapsedBeforeNeg = -marginBefore;
1501         }
1502         logicalTop += (collapsedBeforePos - collapsedBeforeNeg) - marginBefore;
1503     }
1504     if (childLayer->staticBlockPosition() != logicalTop) {
1505         childLayer->setStaticBlockPosition(logicalTop);
1506         if (hasStaticBlockPosition)
1507             child->setChildNeedsLayout(true, false);
1508     }
1509 }
1510 
adjustFloatingBlock(const MarginInfo & marginInfo)1511 void RenderBlock::adjustFloatingBlock(const MarginInfo& marginInfo)
1512 {
1513     // The float should be positioned taking into account the bottom margin
1514     // of the previous flow.  We add that margin into the height, get the
1515     // float positioned properly, and then subtract the margin out of the
1516     // height again.  In the case of self-collapsing blocks, we always just
1517     // use the top margins, since the self-collapsing block collapsed its
1518     // own bottom margin into its top margin.
1519     //
1520     // Note also that the previous flow may collapse its margin into the top of
1521     // our block.  If this is the case, then we do not add the margin in to our
1522     // height when computing the position of the float.   This condition can be tested
1523     // for by simply calling canCollapseWithMarginBefore.  See
1524     // http://www.hixie.ch/tests/adhoc/css/box/block/margin-collapse/046.html for
1525     // an example of this scenario.
1526     int marginOffset = marginInfo.canCollapseWithMarginBefore() ? 0 : marginInfo.margin();
1527     setLogicalHeight(logicalHeight() + marginOffset);
1528     positionNewFloats();
1529     setLogicalHeight(logicalHeight() - marginOffset);
1530 }
1531 
handleSpecialChild(RenderBox * child,const MarginInfo & marginInfo)1532 bool RenderBlock::handleSpecialChild(RenderBox* child, const MarginInfo& marginInfo)
1533 {
1534     // Handle in the given order
1535     return handlePositionedChild(child, marginInfo)
1536         || handleFloatingChild(child, marginInfo)
1537         || handleRunInChild(child);
1538 }
1539 
1540 
handlePositionedChild(RenderBox * child,const MarginInfo & marginInfo)1541 bool RenderBlock::handlePositionedChild(RenderBox* child, const MarginInfo& marginInfo)
1542 {
1543     if (child->isPositioned()) {
1544         child->containingBlock()->insertPositionedObject(child);
1545         adjustPositionedBlock(child, marginInfo);
1546         return true;
1547     }
1548     return false;
1549 }
1550 
handleFloatingChild(RenderBox * child,const MarginInfo & marginInfo)1551 bool RenderBlock::handleFloatingChild(RenderBox* child, const MarginInfo& marginInfo)
1552 {
1553     if (child->isFloating()) {
1554         insertFloatingObject(child);
1555         adjustFloatingBlock(marginInfo);
1556         return true;
1557     }
1558     return false;
1559 }
1560 
handleRunInChild(RenderBox * child)1561 bool RenderBlock::handleRunInChild(RenderBox* child)
1562 {
1563     // See if we have a run-in element with inline children.  If the
1564     // children aren't inline, then just treat the run-in as a normal
1565     // block.
1566     if (!child->isRunIn() || !child->childrenInline())
1567         return false;
1568     // FIXME: We don't handle non-block elements with run-in for now.
1569     if (!child->isRenderBlock())
1570         return false;
1571 
1572     RenderBlock* blockRunIn = toRenderBlock(child);
1573     RenderObject* curr = blockRunIn->nextSibling();
1574     if (!curr || !curr->isRenderBlock() || !curr->childrenInline() || curr->isRunIn() || curr->isAnonymous() || curr->isFloatingOrPositioned())
1575         return false;
1576 
1577     RenderBlock* currBlock = toRenderBlock(curr);
1578 
1579     // First we destroy any :before/:after content. It will be regenerated by the new inline.
1580     // Exception is if the run-in itself is generated.
1581     if (child->style()->styleType() != BEFORE && child->style()->styleType() != AFTER) {
1582         RenderObject* generatedContent;
1583         if (child->getCachedPseudoStyle(BEFORE) && (generatedContent = child->beforePseudoElementRenderer()))
1584             generatedContent->destroy();
1585         if (child->getCachedPseudoStyle(AFTER) && (generatedContent = child->afterPseudoElementRenderer()))
1586             generatedContent->destroy();
1587     }
1588 
1589     // Remove the old child.
1590     children()->removeChildNode(this, blockRunIn);
1591 
1592     // Create an inline.
1593     Node* runInNode = blockRunIn->node();
1594     RenderInline* inlineRunIn = new (renderArena()) RenderInline(runInNode ? runInNode : document());
1595     inlineRunIn->setStyle(blockRunIn->style());
1596 
1597     // Move the nodes from the old child to the new child
1598     for (RenderObject* runInChild = blockRunIn->firstChild(); runInChild;) {
1599         RenderObject* nextSibling = runInChild->nextSibling();
1600         blockRunIn->children()->removeChildNode(blockRunIn, runInChild, false);
1601         inlineRunIn->addChild(runInChild); // Use addChild instead of appendChildNode since it handles correct placement of the children relative to :after-generated content.
1602         runInChild = nextSibling;
1603     }
1604 
1605     // Now insert the new child under |currBlock|. Use addChild instead of insertChildNode since it handles correct placement of the children, esp where we cannot insert
1606     // anything before the first child. e.g. details tag. See https://bugs.webkit.org/show_bug.cgi?id=58228.
1607     currBlock->addChild(inlineRunIn, currBlock->firstChild());
1608 
1609     // If the run-in had an element, we need to set the new renderer.
1610     if (runInNode)
1611         runInNode->setRenderer(inlineRunIn);
1612 
1613     // Destroy the block run-in, which includes deleting its line box tree.
1614     blockRunIn->deleteLineBoxTree();
1615     blockRunIn->destroy();
1616 
1617     // The block acts like an inline, so just null out its
1618     // position.
1619 
1620     return true;
1621 }
1622 
collapseMargins(RenderBox * child,MarginInfo & marginInfo)1623 int RenderBlock::collapseMargins(RenderBox* child, MarginInfo& marginInfo)
1624 {
1625     // Get the four margin values for the child and cache them.
1626     const MarginValues childMargins = marginValuesForChild(child);
1627 
1628     // Get our max pos and neg top margins.
1629     int posTop = childMargins.positiveMarginBefore();
1630     int negTop = childMargins.negativeMarginBefore();
1631 
1632     // For self-collapsing blocks, collapse our bottom margins into our
1633     // top to get new posTop and negTop values.
1634     if (child->isSelfCollapsingBlock()) {
1635         posTop = max(posTop, childMargins.positiveMarginAfter());
1636         negTop = max(negTop, childMargins.negativeMarginAfter());
1637     }
1638 
1639     // See if the top margin is quirky. We only care if this child has
1640     // margins that will collapse with us.
1641     bool topQuirk = child->isMarginBeforeQuirk() || style()->marginBeforeCollapse() == MDISCARD;
1642 
1643     if (marginInfo.canCollapseWithMarginBefore()) {
1644         // This child is collapsing with the top of the
1645         // block.  If it has larger margin values, then we need to update
1646         // our own maximal values.
1647         if (!document()->inQuirksMode() || !marginInfo.quirkContainer() || !topQuirk)
1648             setMaxMarginBeforeValues(max(posTop, maxPositiveMarginBefore()), max(negTop, maxNegativeMarginBefore()));
1649 
1650         // The minute any of the margins involved isn't a quirk, don't
1651         // collapse it away, even if the margin is smaller (www.webreference.com
1652         // has an example of this, a <dt> with 0.8em author-specified inside
1653         // a <dl> inside a <td>.
1654         if (!marginInfo.determinedMarginBeforeQuirk() && !topQuirk && (posTop - negTop)) {
1655             setMarginBeforeQuirk(false);
1656             marginInfo.setDeterminedMarginBeforeQuirk(true);
1657         }
1658 
1659         if (!marginInfo.determinedMarginBeforeQuirk() && topQuirk && !marginBefore())
1660             // We have no top margin and our top child has a quirky margin.
1661             // We will pick up this quirky margin and pass it through.
1662             // This deals with the <td><div><p> case.
1663             // Don't do this for a block that split two inlines though.  You do
1664             // still apply margins in this case.
1665             setMarginBeforeQuirk(true);
1666     }
1667 
1668     if (marginInfo.quirkContainer() && marginInfo.atBeforeSideOfBlock() && (posTop - negTop))
1669         marginInfo.setMarginBeforeQuirk(topQuirk);
1670 
1671     int beforeCollapseLogicalTop = logicalHeight();
1672     int logicalTop = beforeCollapseLogicalTop;
1673     if (child->isSelfCollapsingBlock()) {
1674         // This child has no height.  We need to compute our
1675         // position before we collapse the child's margins together,
1676         // so that we can get an accurate position for the zero-height block.
1677         int collapsedBeforePos = max(marginInfo.positiveMargin(), childMargins.positiveMarginBefore());
1678         int collapsedBeforeNeg = max(marginInfo.negativeMargin(), childMargins.negativeMarginBefore());
1679         marginInfo.setMargin(collapsedBeforePos, collapsedBeforeNeg);
1680 
1681         // Now collapse the child's margins together, which means examining our
1682         // bottom margin values as well.
1683         marginInfo.setPositiveMarginIfLarger(childMargins.positiveMarginAfter());
1684         marginInfo.setNegativeMarginIfLarger(childMargins.negativeMarginAfter());
1685 
1686         if (!marginInfo.canCollapseWithMarginBefore())
1687             // We need to make sure that the position of the self-collapsing block
1688             // is correct, since it could have overflowing content
1689             // that needs to be positioned correctly (e.g., a block that
1690             // had a specified height of 0 but that actually had subcontent).
1691             logicalTop = logicalHeight() + collapsedBeforePos - collapsedBeforeNeg;
1692     }
1693     else {
1694         if (child->style()->marginBeforeCollapse() == MSEPARATE) {
1695             setLogicalHeight(logicalHeight() + marginInfo.margin() + marginBeforeForChild(child));
1696             logicalTop = logicalHeight();
1697         }
1698         else if (!marginInfo.atBeforeSideOfBlock() ||
1699             (!marginInfo.canCollapseMarginBeforeWithChildren()
1700              && (!document()->inQuirksMode() || !marginInfo.quirkContainer() || !marginInfo.marginBeforeQuirk()))) {
1701             // We're collapsing with a previous sibling's margins and not
1702             // with the top of the block.
1703             setLogicalHeight(logicalHeight() + max(marginInfo.positiveMargin(), posTop) - max(marginInfo.negativeMargin(), negTop));
1704             logicalTop = logicalHeight();
1705         }
1706 
1707         marginInfo.setPositiveMargin(childMargins.positiveMarginAfter());
1708         marginInfo.setNegativeMargin(childMargins.negativeMarginAfter());
1709 
1710         if (marginInfo.margin())
1711             marginInfo.setMarginAfterQuirk(child->isMarginAfterQuirk() || style()->marginAfterCollapse() == MDISCARD);
1712     }
1713 
1714     // If margins would pull us past the top of the next page, then we need to pull back and pretend like the margins
1715     // collapsed into the page edge.
1716     bool paginated = view()->layoutState()->isPaginated();
1717     if (paginated && logicalTop > beforeCollapseLogicalTop) {
1718         int oldLogicalTop = logicalTop;
1719         logicalTop = min(logicalTop, nextPageLogicalTop(beforeCollapseLogicalTop));
1720         setLogicalHeight(logicalHeight() + (logicalTop - oldLogicalTop));
1721     }
1722     return logicalTop;
1723 }
1724 
clearFloatsIfNeeded(RenderBox * child,MarginInfo & marginInfo,int oldTopPosMargin,int oldTopNegMargin,int yPos)1725 int RenderBlock::clearFloatsIfNeeded(RenderBox* child, MarginInfo& marginInfo, int oldTopPosMargin, int oldTopNegMargin, int yPos)
1726 {
1727     int heightIncrease = getClearDelta(child, yPos);
1728     if (!heightIncrease)
1729         return yPos;
1730 
1731     if (child->isSelfCollapsingBlock()) {
1732         // For self-collapsing blocks that clear, they can still collapse their
1733         // margins with following siblings.  Reset the current margins to represent
1734         // the self-collapsing block's margins only.
1735         // CSS2.1 states:
1736         // "An element that has had clearance applied to it never collapses its top margin with its parent block's bottom margin.
1737         // Therefore if we are at the bottom of the block, let's go ahead and reset margins to only include the
1738         // self-collapsing block's bottom margin.
1739         bool atBottomOfBlock = true;
1740         for (RenderBox* curr = child->nextSiblingBox(); curr && atBottomOfBlock; curr = curr->nextSiblingBox()) {
1741             if (!curr->isFloatingOrPositioned())
1742                 atBottomOfBlock = false;
1743         }
1744 
1745         MarginValues childMargins = marginValuesForChild(child);
1746         if (atBottomOfBlock) {
1747             marginInfo.setPositiveMargin(childMargins.positiveMarginAfter());
1748             marginInfo.setNegativeMargin(childMargins.negativeMarginAfter());
1749         } else {
1750             marginInfo.setPositiveMargin(max(childMargins.positiveMarginBefore(), childMargins.positiveMarginAfter()));
1751             marginInfo.setNegativeMargin(max(childMargins.negativeMarginBefore(), childMargins.negativeMarginAfter()));
1752         }
1753 
1754         // Adjust our height such that we are ready to be collapsed with subsequent siblings (or the bottom
1755         // of the parent block).
1756         setLogicalHeight(child->y() - max(0, marginInfo.margin()));
1757     } else
1758         // Increase our height by the amount we had to clear.
1759         setLogicalHeight(height() + heightIncrease);
1760 
1761     if (marginInfo.canCollapseWithMarginBefore()) {
1762         // We can no longer collapse with the top of the block since a clear
1763         // occurred.  The empty blocks collapse into the cleared block.
1764         // FIXME: This isn't quite correct.  Need clarification for what to do
1765         // if the height the cleared block is offset by is smaller than the
1766         // margins involved.
1767         setMaxMarginBeforeValues(oldTopPosMargin, oldTopNegMargin);
1768         marginInfo.setAtBeforeSideOfBlock(false);
1769     }
1770 
1771     return yPos + heightIncrease;
1772 }
1773 
estimateLogicalTopPosition(RenderBox * child,const MarginInfo & marginInfo)1774 int RenderBlock::estimateLogicalTopPosition(RenderBox* child, const MarginInfo& marginInfo)
1775 {
1776     // FIXME: We need to eliminate the estimation of vertical position, because when it's wrong we sometimes trigger a pathological
1777     // relayout if there are intruding floats.
1778     int logicalTopEstimate = logicalHeight();
1779     if (!marginInfo.canCollapseWithMarginBefore()) {
1780         int childMarginBefore = child->selfNeedsLayout() ? marginBeforeForChild(child) : collapsedMarginBeforeForChild(child);
1781         logicalTopEstimate += max(marginInfo.margin(), childMarginBefore);
1782     }
1783 
1784     bool paginated = view()->layoutState()->isPaginated();
1785 
1786     // Adjust logicalTopEstimate down to the next page if the margins are so large that we don't fit on the current
1787     // page.
1788     if (paginated && logicalTopEstimate > logicalHeight())
1789         logicalTopEstimate = min(logicalTopEstimate, nextPageLogicalTop(logicalHeight()));
1790 
1791     logicalTopEstimate += getClearDelta(child, logicalTopEstimate);
1792 
1793     if (paginated) {
1794         // If the object has a page or column break value of "before", then we should shift to the top of the next page.
1795         logicalTopEstimate = applyBeforeBreak(child, logicalTopEstimate);
1796 
1797         // For replaced elements and scrolled elements, we want to shift them to the next page if they don't fit on the current one.
1798         logicalTopEstimate = adjustForUnsplittableChild(child, logicalTopEstimate);
1799 
1800         if (!child->selfNeedsLayout() && child->isRenderBlock())
1801             logicalTopEstimate += toRenderBlock(child)->paginationStrut();
1802     }
1803 
1804     return logicalTopEstimate;
1805 }
1806 
determineLogicalLeftPositionForChild(RenderBox * child)1807 void RenderBlock::determineLogicalLeftPositionForChild(RenderBox* child)
1808 {
1809     int startPosition = borderStart() + paddingStart();
1810     int totalAvailableLogicalWidth = borderAndPaddingLogicalWidth() + availableLogicalWidth();
1811 
1812     // Add in our start margin.
1813     int childMarginStart = marginStartForChild(child);
1814     int newPosition = startPosition + childMarginStart;
1815 
1816     // Some objects (e.g., tables, horizontal rules, overflow:auto blocks) avoid floats.  They need
1817     // to shift over as necessary to dodge any floats that might get in the way.
1818     if (child->avoidsFloats()) {
1819         int startOff = style()->isLeftToRightDirection() ? logicalLeftOffsetForLine(logicalHeight(), false) : totalAvailableLogicalWidth - logicalRightOffsetForLine(logicalHeight(), false);
1820         if (style()->textAlign() != WEBKIT_CENTER && !child->style()->marginStartUsing(style()).isAuto()) {
1821             if (childMarginStart < 0)
1822                 startOff += childMarginStart;
1823             newPosition = max(newPosition, startOff); // Let the float sit in the child's margin if it can fit.
1824         } else if (startOff != startPosition) {
1825             // The object is shifting to the "end" side of the block. The object might be centered, so we need to
1826             // recalculate our inline direction margins. Note that the containing block content
1827             // width computation will take into account the delta between |startOff| and |startPosition|
1828             // so that we can just pass the content width in directly to the |computeMarginsInContainingBlockInlineDirection|
1829             // function.
1830             child->computeInlineDirectionMargins(this, availableLogicalWidthForLine(logicalTopForChild(child), false), logicalWidthForChild(child));
1831             newPosition = startOff + marginStartForChild(child);
1832         }
1833     }
1834 
1835     setLogicalLeftForChild(child, style()->isLeftToRightDirection() ? newPosition : totalAvailableLogicalWidth - newPosition - logicalWidthForChild(child), ApplyLayoutDelta);
1836 }
1837 
setCollapsedBottomMargin(const MarginInfo & marginInfo)1838 void RenderBlock::setCollapsedBottomMargin(const MarginInfo& marginInfo)
1839 {
1840     if (marginInfo.canCollapseWithMarginAfter() && !marginInfo.canCollapseWithMarginBefore()) {
1841         // Update our max pos/neg bottom margins, since we collapsed our bottom margins
1842         // with our children.
1843         setMaxMarginAfterValues(max(maxPositiveMarginAfter(), marginInfo.positiveMargin()), max(maxNegativeMarginAfter(), marginInfo.negativeMargin()));
1844 
1845         if (!marginInfo.marginAfterQuirk())
1846             setMarginAfterQuirk(false);
1847 
1848         if (marginInfo.marginAfterQuirk() && marginAfter() == 0)
1849             // We have no bottom margin and our last child has a quirky margin.
1850             // We will pick up this quirky margin and pass it through.
1851             // This deals with the <td><div><p> case.
1852             setMarginAfterQuirk(true);
1853     }
1854 }
1855 
handleAfterSideOfBlock(int beforeSide,int afterSide,MarginInfo & marginInfo)1856 void RenderBlock::handleAfterSideOfBlock(int beforeSide, int afterSide, MarginInfo& marginInfo)
1857 {
1858     marginInfo.setAtAfterSideOfBlock(true);
1859 
1860     // If we can't collapse with children then go ahead and add in the bottom margin.
1861     if (!marginInfo.canCollapseWithMarginAfter() && !marginInfo.canCollapseWithMarginBefore()
1862         && (!document()->inQuirksMode() || !marginInfo.quirkContainer() || !marginInfo.marginAfterQuirk()))
1863         setLogicalHeight(logicalHeight() + marginInfo.margin());
1864 
1865     // Now add in our bottom border/padding.
1866     setLogicalHeight(logicalHeight() + afterSide);
1867 
1868     // Negative margins can cause our height to shrink below our minimal height (border/padding).
1869     // If this happens, ensure that the computed height is increased to the minimal height.
1870     setLogicalHeight(max(logicalHeight(), beforeSide + afterSide));
1871 
1872     // Update our bottom collapsed margin info.
1873     setCollapsedBottomMargin(marginInfo);
1874 }
1875 
setLogicalLeftForChild(RenderBox * child,int logicalLeft,ApplyLayoutDeltaMode applyDelta)1876 void RenderBlock::setLogicalLeftForChild(RenderBox* child, int logicalLeft, ApplyLayoutDeltaMode applyDelta)
1877 {
1878     if (isHorizontalWritingMode()) {
1879         if (applyDelta == ApplyLayoutDelta)
1880             view()->addLayoutDelta(IntSize(child->x() - logicalLeft, 0));
1881         child->setX(logicalLeft);
1882     } else {
1883         if (applyDelta == ApplyLayoutDelta)
1884             view()->addLayoutDelta(IntSize(0, child->y() - logicalLeft));
1885         child->setY(logicalLeft);
1886     }
1887 }
1888 
setLogicalTopForChild(RenderBox * child,int logicalTop,ApplyLayoutDeltaMode applyDelta)1889 void RenderBlock::setLogicalTopForChild(RenderBox* child, int logicalTop, ApplyLayoutDeltaMode applyDelta)
1890 {
1891     if (isHorizontalWritingMode()) {
1892         if (applyDelta == ApplyLayoutDelta)
1893             view()->addLayoutDelta(IntSize(0, child->y() - logicalTop));
1894         child->setY(logicalTop);
1895     } else {
1896         if (applyDelta == ApplyLayoutDelta)
1897             view()->addLayoutDelta(IntSize(child->x() - logicalTop, 0));
1898         child->setX(logicalTop);
1899     }
1900 }
1901 
layoutBlockChildren(bool relayoutChildren,int & maxFloatLogicalBottom)1902 void RenderBlock::layoutBlockChildren(bool relayoutChildren, int& maxFloatLogicalBottom)
1903 {
1904     if (gPercentHeightDescendantsMap) {
1905         if (HashSet<RenderBox*>* descendants = gPercentHeightDescendantsMap->get(this)) {
1906             HashSet<RenderBox*>::iterator end = descendants->end();
1907             for (HashSet<RenderBox*>::iterator it = descendants->begin(); it != end; ++it) {
1908                 RenderBox* box = *it;
1909                 while (box != this) {
1910                     if (box->normalChildNeedsLayout())
1911                         break;
1912                     box->setChildNeedsLayout(true, false);
1913                     box = box->containingBlock();
1914                     ASSERT(box);
1915                     if (!box)
1916                         break;
1917                 }
1918             }
1919         }
1920     }
1921 
1922     int beforeEdge = borderBefore() + paddingBefore();
1923     int afterEdge = borderAfter() + paddingAfter() + scrollbarLogicalHeight();
1924 
1925     setLogicalHeight(beforeEdge);
1926 
1927     // The margin struct caches all our current margin collapsing state.  The compact struct caches state when we encounter compacts,
1928     MarginInfo marginInfo(this, beforeEdge, afterEdge);
1929 
1930     // Fieldsets need to find their legend and position it inside the border of the object.
1931     // The legend then gets skipped during normal layout.  The same is true for ruby text.
1932     // It doesn't get included in the normal layout process but is instead skipped.
1933     RenderObject* childToExclude = layoutSpecialExcludedChild(relayoutChildren);
1934 
1935     int previousFloatLogicalBottom = 0;
1936     maxFloatLogicalBottom = 0;
1937 
1938     RenderBox* next = firstChildBox();
1939 
1940     while (next) {
1941         RenderBox* child = next;
1942         next = child->nextSiblingBox();
1943 
1944         if (childToExclude == child)
1945             continue; // Skip this child, since it will be positioned by the specialized subclass (fieldsets and ruby runs).
1946 
1947         // Make sure we layout children if they need it.
1948         // FIXME: Technically percentage height objects only need a relayout if their percentage isn't going to be turned into
1949         // an auto value.  Add a method to determine this, so that we can avoid the relayout.
1950         if (relayoutChildren || ((child->style()->logicalHeight().isPercent() || child->style()->logicalMinHeight().isPercent() || child->style()->logicalMaxHeight().isPercent()) && !isRenderView()))
1951             child->setChildNeedsLayout(true, false);
1952 
1953         // If relayoutChildren is set and the child has percentage padding, we also need to invalidate the child's pref widths.
1954         if (relayoutChildren && (child->style()->paddingStart().isPercent() || child->style()->paddingEnd().isPercent()))
1955             child->setPreferredLogicalWidthsDirty(true, false);
1956 
1957         // Handle the four types of special elements first.  These include positioned content, floating content, compacts and
1958         // run-ins.  When we encounter these four types of objects, we don't actually lay them out as normal flow blocks.
1959         if (handleSpecialChild(child, marginInfo))
1960             continue;
1961 
1962         // Lay out the child.
1963         layoutBlockChild(child, marginInfo, previousFloatLogicalBottom, maxFloatLogicalBottom);
1964     }
1965 
1966     // Now do the handling of the bottom of the block, adding in our bottom border/padding and
1967     // determining the correct collapsed bottom margin information.
1968     handleAfterSideOfBlock(beforeEdge, afterEdge, marginInfo);
1969 }
1970 
layoutBlockChild(RenderBox * child,MarginInfo & marginInfo,int & previousFloatLogicalBottom,int & maxFloatLogicalBottom)1971 void RenderBlock::layoutBlockChild(RenderBox* child, MarginInfo& marginInfo, int& previousFloatLogicalBottom, int& maxFloatLogicalBottom)
1972 {
1973     int oldPosMarginBefore = maxPositiveMarginBefore();
1974     int oldNegMarginBefore = maxNegativeMarginBefore();
1975 
1976     // The child is a normal flow object.  Compute the margins we will use for collapsing now.
1977     child->computeBlockDirectionMargins(this);
1978 
1979     // Do not allow a collapse if the margin-before-collapse style is set to SEPARATE.
1980     if (child->style()->marginBeforeCollapse() == MSEPARATE) {
1981         marginInfo.setAtBeforeSideOfBlock(false);
1982         marginInfo.clearMargin();
1983     }
1984 
1985     // Try to guess our correct logical top position.  In most cases this guess will
1986     // be correct.  Only if we're wrong (when we compute the real logical top position)
1987     // will we have to potentially relayout.
1988     int logicalTopEstimate = estimateLogicalTopPosition(child, marginInfo);
1989 
1990     // Cache our old rect so that we can dirty the proper repaint rects if the child moves.
1991     IntRect oldRect(child->x(), child->y() , child->width(), child->height());
1992     int oldLogicalTop = logicalTopForChild(child);
1993 
1994 #ifndef NDEBUG
1995     IntSize oldLayoutDelta = view()->layoutDelta();
1996 #endif
1997     // Go ahead and position the child as though it didn't collapse with the top.
1998     setLogicalTopForChild(child, logicalTopEstimate, ApplyLayoutDelta);
1999 
2000     RenderBlock* childRenderBlock = child->isRenderBlock() ? toRenderBlock(child) : 0;
2001     bool markDescendantsWithFloats = false;
2002     if (logicalTopEstimate != oldLogicalTop && !child->avoidsFloats() && childRenderBlock && childRenderBlock->containsFloats())
2003         markDescendantsWithFloats = true;
2004     else if (!child->avoidsFloats() || child->shrinkToAvoidFloats()) {
2005         // If an element might be affected by the presence of floats, then always mark it for
2006         // layout.
2007         int fb = max(previousFloatLogicalBottom, lowestFloatLogicalBottom());
2008         if (fb > logicalTopEstimate)
2009             markDescendantsWithFloats = true;
2010     }
2011 
2012     if (childRenderBlock) {
2013         if (markDescendantsWithFloats)
2014             childRenderBlock->markAllDescendantsWithFloatsForLayout();
2015         if (!child->isWritingModeRoot())
2016             previousFloatLogicalBottom = max(previousFloatLogicalBottom, oldLogicalTop + childRenderBlock->lowestFloatLogicalBottom());
2017     }
2018 
2019     if (!child->needsLayout())
2020         child->markForPaginationRelayoutIfNeeded();
2021 
2022     bool childHadLayout = child->m_everHadLayout;
2023     bool childNeededLayout = child->needsLayout();
2024     if (childNeededLayout)
2025         child->layout();
2026 
2027     // Cache if we are at the top of the block right now.
2028     bool atBeforeSideOfBlock = marginInfo.atBeforeSideOfBlock();
2029 
2030     // Now determine the correct ypos based off examination of collapsing margin
2031     // values.
2032     int logicalTopBeforeClear = collapseMargins(child, marginInfo);
2033 
2034     // Now check for clear.
2035     int logicalTopAfterClear = clearFloatsIfNeeded(child, marginInfo, oldPosMarginBefore, oldNegMarginBefore, logicalTopBeforeClear);
2036 
2037     bool paginated = view()->layoutState()->isPaginated();
2038     if (paginated) {
2039         int oldTop = logicalTopAfterClear;
2040 
2041         // If the object has a page or column break value of "before", then we should shift to the top of the next page.
2042         logicalTopAfterClear = applyBeforeBreak(child, logicalTopAfterClear);
2043 
2044         // For replaced elements and scrolled elements, we want to shift them to the next page if they don't fit on the current one.
2045         int logicalTopBeforeUnsplittableAdjustment = logicalTopAfterClear;
2046         int logicalTopAfterUnsplittableAdjustment = adjustForUnsplittableChild(child, logicalTopAfterClear);
2047 
2048         int paginationStrut = 0;
2049         int unsplittableAdjustmentDelta = logicalTopAfterUnsplittableAdjustment - logicalTopBeforeUnsplittableAdjustment;
2050         if (unsplittableAdjustmentDelta)
2051             paginationStrut = unsplittableAdjustmentDelta;
2052         else if (childRenderBlock && childRenderBlock->paginationStrut())
2053             paginationStrut = childRenderBlock->paginationStrut();
2054 
2055         if (paginationStrut) {
2056             // We are willing to propagate out to our parent block as long as we were at the top of the block prior
2057             // to collapsing our margins, and as long as we didn't clear or move as a result of other pagination.
2058             if (atBeforeSideOfBlock && oldTop == logicalTopBeforeClear && !isPositioned() && !isTableCell()) {
2059                 // FIXME: Should really check if we're exceeding the page height before propagating the strut, but we don't
2060                 // have all the information to do so (the strut only has the remaining amount to push).  Gecko gets this wrong too
2061                 // and pushes to the next page anyway, so not too concerned about it.
2062                 setPaginationStrut(logicalTopAfterClear + paginationStrut);
2063                 if (childRenderBlock)
2064                     childRenderBlock->setPaginationStrut(0);
2065             } else
2066                 logicalTopAfterClear += paginationStrut;
2067         }
2068 
2069         // Similar to how we apply clearance.  Go ahead and boost height() to be the place where we're going to position the child.
2070         setLogicalHeight(logicalHeight() + (logicalTopAfterClear - oldTop));
2071     }
2072 
2073     setLogicalTopForChild(child, logicalTopAfterClear, ApplyLayoutDelta);
2074 
2075     // Now we have a final top position.  See if it really does end up being different from our estimate.
2076     if (logicalTopAfterClear != logicalTopEstimate) {
2077         if (child->shrinkToAvoidFloats()) {
2078             // The child's width depends on the line width.
2079             // When the child shifts to clear an item, its width can
2080             // change (because it has more available line width).
2081             // So go ahead and mark the item as dirty.
2082             child->setChildNeedsLayout(true, false);
2083         }
2084         if (childRenderBlock) {
2085             if (!child->avoidsFloats() && childRenderBlock->containsFloats())
2086                 childRenderBlock->markAllDescendantsWithFloatsForLayout();
2087             if (!child->needsLayout())
2088                 child->markForPaginationRelayoutIfNeeded();
2089         }
2090 
2091         // Our guess was wrong. Make the child lay itself out again.
2092         child->layoutIfNeeded();
2093     }
2094 
2095     // We are no longer at the top of the block if we encounter a non-empty child.
2096     // This has to be done after checking for clear, so that margins can be reset if a clear occurred.
2097     if (marginInfo.atBeforeSideOfBlock() && !child->isSelfCollapsingBlock())
2098         marginInfo.setAtBeforeSideOfBlock(false);
2099 
2100     // Now place the child in the correct left position
2101     determineLogicalLeftPositionForChild(child);
2102 
2103     // Update our height now that the child has been placed in the correct position.
2104     setLogicalHeight(logicalHeight() + logicalHeightForChild(child));
2105     if (child->style()->marginAfterCollapse() == MSEPARATE) {
2106         setLogicalHeight(logicalHeight() + marginAfterForChild(child));
2107         marginInfo.clearMargin();
2108     }
2109     // If the child has overhanging floats that intrude into following siblings (or possibly out
2110     // of this block), then the parent gets notified of the floats now.
2111     if (childRenderBlock && childRenderBlock->containsFloats())
2112         maxFloatLogicalBottom = max(maxFloatLogicalBottom, addOverhangingFloats(toRenderBlock(child), -child->logicalLeft(), -child->logicalTop(), !childNeededLayout));
2113 
2114     IntSize childOffset(child->x() - oldRect.x(), child->y() - oldRect.y());
2115     if (childOffset.width() || childOffset.height()) {
2116         view()->addLayoutDelta(childOffset);
2117 
2118         // If the child moved, we have to repaint it as well as any floating/positioned
2119         // descendants.  An exception is if we need a layout.  In this case, we know we're going to
2120         // repaint ourselves (and the child) anyway.
2121         if (childHadLayout && !selfNeedsLayout() && child->checkForRepaintDuringLayout())
2122             child->repaintDuringLayoutIfMoved(oldRect);
2123     }
2124 
2125     if (!childHadLayout && child->checkForRepaintDuringLayout()) {
2126         child->repaint();
2127         child->repaintOverhangingFloats(true);
2128     }
2129 
2130     if (paginated) {
2131         // Check for an after page/column break.
2132         int newHeight = applyAfterBreak(child, logicalHeight(), marginInfo);
2133         if (newHeight != height())
2134             setLogicalHeight(newHeight);
2135     }
2136 
2137     ASSERT(oldLayoutDelta == view()->layoutDelta());
2138 }
2139 
simplifiedNormalFlowLayout()2140 void RenderBlock::simplifiedNormalFlowLayout()
2141 {
2142     if (childrenInline()) {
2143         ListHashSet<RootInlineBox*> lineBoxes;
2144         bool endOfInline = false;
2145         RenderObject* o = bidiFirst(this, 0, false);
2146         while (o) {
2147             if (!o->isPositioned() && (o->isReplaced() || o->isFloating())) {
2148                 o->layoutIfNeeded();
2149                 if (toRenderBox(o)->inlineBoxWrapper()) {
2150                     RootInlineBox* box = toRenderBox(o)->inlineBoxWrapper()->root();
2151                     lineBoxes.add(box);
2152                 }
2153             } else if (o->isText() || (o->isRenderInline() && !endOfInline))
2154                 o->setNeedsLayout(false);
2155             o = bidiNext(this, o, 0, false, &endOfInline);
2156         }
2157 
2158         // FIXME: Glyph overflow will get lost in this case, but not really a big deal.
2159         GlyphOverflowAndFallbackFontsMap textBoxDataMap;
2160         for (ListHashSet<RootInlineBox*>::const_iterator it = lineBoxes.begin(); it != lineBoxes.end(); ++it) {
2161             RootInlineBox* box = *it;
2162             box->computeOverflow(box->lineTop(), box->lineBottom(), textBoxDataMap);
2163         }
2164     } else {
2165         for (RenderBox* box = firstChildBox(); box; box = box->nextSiblingBox()) {
2166             if (!box->isPositioned())
2167                 box->layoutIfNeeded();
2168         }
2169     }
2170 }
2171 
simplifiedLayout()2172 bool RenderBlock::simplifiedLayout()
2173 {
2174     if ((!posChildNeedsLayout() && !needsSimplifiedNormalFlowLayout()) || normalChildNeedsLayout() || selfNeedsLayout())
2175         return false;
2176 
2177     LayoutStateMaintainer statePusher(view(), this, IntSize(x(), y()), hasColumns() || hasTransform() || hasReflection() || style()->isFlippedBlocksWritingMode());
2178 
2179     if (needsPositionedMovementLayout() && !tryLayoutDoingPositionedMovementOnly())
2180         return false;
2181 
2182     // Lay out positioned descendants or objects that just need to recompute overflow.
2183     if (needsSimplifiedNormalFlowLayout())
2184         simplifiedNormalFlowLayout();
2185 
2186     // Lay out our positioned objects if our positioned child bit is set.
2187     if (posChildNeedsLayout())
2188         layoutPositionedObjects(false);
2189 
2190     // Recompute our overflow information.
2191     // FIXME: We could do better here by computing a temporary overflow object from layoutPositionedObjects and only
2192     // updating our overflow if we either used to have overflow or if the new temporary object has overflow.
2193     // For now just always recompute overflow.  This is no worse performance-wise than the old code that called rightmostPosition and
2194     // lowestPosition on every relayout so it's not a regression.
2195     m_overflow.clear();
2196     computeOverflow(clientLogicalBottom(), true);
2197 
2198     statePusher.pop();
2199 
2200     updateLayerTransform();
2201 
2202     updateScrollInfoAfterLayout();
2203 
2204     setNeedsLayout(false);
2205     return true;
2206 }
2207 
layoutPositionedObjects(bool relayoutChildren)2208 void RenderBlock::layoutPositionedObjects(bool relayoutChildren)
2209 {
2210     if (!m_positionedObjects)
2211         return;
2212 
2213     if (hasColumns())
2214         view()->layoutState()->clearPaginationInformation(); // Positioned objects are not part of the column flow, so they don't paginate with the columns.
2215 
2216     RenderBox* r;
2217     Iterator end = m_positionedObjects->end();
2218     for (Iterator it = m_positionedObjects->begin(); it != end; ++it) {
2219         r = *it;
2220         // When a non-positioned block element moves, it may have positioned children that are implicitly positioned relative to the
2221         // non-positioned block.  Rather than trying to detect all of these movement cases, we just always lay out positioned
2222         // objects that are positioned implicitly like this.  Such objects are rare, and so in typical DHTML menu usage (where everything is
2223         // positioned explicitly) this should not incur a performance penalty.
2224         if (relayoutChildren || (r->style()->hasStaticBlockPosition(isHorizontalWritingMode()) && r->parent() != this && r->parent()->isBlockFlow()))
2225             r->setChildNeedsLayout(true, false);
2226 
2227         // If relayoutChildren is set and we have percentage padding, we also need to invalidate the child's pref widths.
2228         if (relayoutChildren && (r->style()->paddingStart().isPercent() || r->style()->paddingEnd().isPercent()))
2229             r->setPreferredLogicalWidthsDirty(true, false);
2230 
2231         if (!r->needsLayout())
2232             r->markForPaginationRelayoutIfNeeded();
2233 
2234         // We don't have to do a full layout.  We just have to update our position. Try that first. If we have shrink-to-fit width
2235         // and we hit the available width constraint, the layoutIfNeeded() will catch it and do a full layout.
2236         if (r->needsPositionedMovementLayoutOnly() && r->tryLayoutDoingPositionedMovementOnly())
2237             r->setNeedsLayout(false);
2238         r->layoutIfNeeded();
2239     }
2240 
2241     if (hasColumns())
2242         view()->layoutState()->m_columnInfo = columnInfo(); // FIXME: Kind of gross. We just put this back into the layout state so that pop() will work.
2243 }
2244 
markPositionedObjectsForLayout()2245 void RenderBlock::markPositionedObjectsForLayout()
2246 {
2247     if (m_positionedObjects) {
2248         RenderBox* r;
2249         Iterator end = m_positionedObjects->end();
2250         for (Iterator it = m_positionedObjects->begin(); it != end; ++it) {
2251             r = *it;
2252             r->setChildNeedsLayout(true);
2253         }
2254     }
2255 }
2256 
markForPaginationRelayoutIfNeeded()2257 void RenderBlock::markForPaginationRelayoutIfNeeded()
2258 {
2259     ASSERT(!needsLayout());
2260     if (needsLayout())
2261         return;
2262 
2263     if (view()->layoutState()->pageLogicalHeightChanged() || (view()->layoutState()->pageLogicalHeight() && view()->layoutState()->pageLogicalOffset(logicalTop()) != pageLogicalOffset()))
2264         setChildNeedsLayout(true, false);
2265 }
2266 
repaintOverhangingFloats(bool paintAllDescendants)2267 void RenderBlock::repaintOverhangingFloats(bool paintAllDescendants)
2268 {
2269     // Repaint any overhanging floats (if we know we're the one to paint them).
2270     // Otherwise, bail out.
2271     if (!hasOverhangingFloats())
2272         return;
2273 
2274     // FIXME: Avoid disabling LayoutState. At the very least, don't disable it for floats originating
2275     // in this block. Better yet would be to push extra state for the containers of other floats.
2276     view()->disableLayoutState();
2277     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2278     FloatingObjectSetIterator end = floatingObjectSet.end();
2279     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
2280         FloatingObject* r = *it;
2281         // Only repaint the object if it is overhanging, is not in its own layer, and
2282         // is our responsibility to paint (m_shouldPaint is set). When paintAllDescendants is true, the latter
2283         // condition is replaced with being a descendant of us.
2284         if (logicalBottomForFloat(r) > logicalHeight() && ((paintAllDescendants && r->m_renderer->isDescendantOf(this)) || r->m_shouldPaint) && !r->m_renderer->hasSelfPaintingLayer()) {
2285             r->m_renderer->repaint();
2286             r->m_renderer->repaintOverhangingFloats();
2287         }
2288     }
2289     view()->enableLayoutState();
2290 }
2291 
paint(PaintInfo & paintInfo,int tx,int ty)2292 void RenderBlock::paint(PaintInfo& paintInfo, int tx, int ty)
2293 {
2294     tx += x();
2295     ty += y();
2296 
2297     PaintPhase phase = paintInfo.phase;
2298 
2299     // Check if we need to do anything at all.
2300     // FIXME: Could eliminate the isRoot() check if we fix background painting so that the RenderView
2301     // paints the root's background.
2302     if (!isRoot()) {
2303         IntRect overflowBox = visualOverflowRect();
2304         flipForWritingMode(overflowBox);
2305         overflowBox.inflate(maximalOutlineSize(paintInfo.phase));
2306         overflowBox.move(tx, ty);
2307         if (!overflowBox.intersects(paintInfo.rect))
2308             return;
2309     }
2310 
2311     bool pushedClip = pushContentsClip(paintInfo, tx, ty);
2312     paintObject(paintInfo, tx, ty);
2313     if (pushedClip)
2314         popContentsClip(paintInfo, phase, tx, ty);
2315 
2316     // Our scrollbar widgets paint exactly when we tell them to, so that they work properly with
2317     // z-index.  We paint after we painted the background/border, so that the scrollbars will
2318     // sit above the background/border.
2319     if (hasOverflowClip() && style()->visibility() == VISIBLE && (phase == PaintPhaseBlockBackground || phase == PaintPhaseChildBlockBackground) && paintInfo.shouldPaintWithinRoot(this))
2320         layer()->paintOverflowControls(paintInfo.context, tx, ty, paintInfo.rect);
2321 }
2322 
paintColumnRules(PaintInfo & paintInfo,int tx,int ty)2323 void RenderBlock::paintColumnRules(PaintInfo& paintInfo, int tx, int ty)
2324 {
2325     const Color& ruleColor = style()->visitedDependentColor(CSSPropertyWebkitColumnRuleColor);
2326     bool ruleTransparent = style()->columnRuleIsTransparent();
2327     EBorderStyle ruleStyle = style()->columnRuleStyle();
2328     int ruleWidth = style()->columnRuleWidth();
2329     int colGap = columnGap();
2330     bool renderRule = ruleStyle > BHIDDEN && !ruleTransparent && ruleWidth <= colGap;
2331     if (!renderRule)
2332         return;
2333 
2334     // We need to do multiple passes, breaking up our child painting into strips.
2335     ColumnInfo* colInfo = columnInfo();
2336     unsigned colCount = columnCount(colInfo);
2337     int currLogicalLeftOffset = style()->isLeftToRightDirection() ? 0 : contentLogicalWidth();
2338     int ruleAdd = logicalLeftOffsetForContent();
2339     int ruleLogicalLeft = style()->isLeftToRightDirection() ? 0 : contentLogicalWidth();
2340     for (unsigned i = 0; i < colCount; i++) {
2341         IntRect colRect = columnRectAt(colInfo, i);
2342 
2343         int inlineDirectionSize = isHorizontalWritingMode() ? colRect.width() : colRect.height();
2344 
2345         // Move to the next position.
2346         if (style()->isLeftToRightDirection()) {
2347             ruleLogicalLeft += inlineDirectionSize + colGap / 2;
2348             currLogicalLeftOffset += inlineDirectionSize + colGap;
2349         } else {
2350             ruleLogicalLeft -= (inlineDirectionSize + colGap / 2);
2351             currLogicalLeftOffset -= (inlineDirectionSize + colGap);
2352         }
2353 
2354         // Now paint the column rule.
2355         if (i < colCount - 1) {
2356             int ruleLeft = isHorizontalWritingMode() ? tx + ruleLogicalLeft - ruleWidth / 2 + ruleAdd : tx + borderBefore() + paddingBefore();
2357             int ruleRight = isHorizontalWritingMode() ? ruleLeft + ruleWidth : ruleLeft + contentWidth();
2358             int ruleTop = isHorizontalWritingMode() ? ty + borderTop() + paddingTop() : ty + ruleLogicalLeft - ruleWidth / 2 + ruleAdd;
2359             int ruleBottom = isHorizontalWritingMode() ? ruleTop + contentHeight() : ruleTop + ruleWidth;
2360             drawLineForBoxSide(paintInfo.context, ruleLeft, ruleTop, ruleRight, ruleBottom,
2361                                style()->isLeftToRightDirection() ? BSLeft : BSRight, ruleColor, ruleStyle, 0, 0);
2362         }
2363 
2364         ruleLogicalLeft = currLogicalLeftOffset;
2365     }
2366 }
2367 
paintColumnContents(PaintInfo & paintInfo,int tx,int ty,bool paintingFloats)2368 void RenderBlock::paintColumnContents(PaintInfo& paintInfo, int tx, int ty, bool paintingFloats)
2369 {
2370     // We need to do multiple passes, breaking up our child painting into strips.
2371     GraphicsContext* context = paintInfo.context;
2372     ColumnInfo* colInfo = columnInfo();
2373     unsigned colCount = columnCount(colInfo);
2374     if (!colCount)
2375         return;
2376     int currLogicalTopOffset = 0;
2377     for (unsigned i = 0; i < colCount; i++) {
2378         // For each rect, we clip to the rect, and then we adjust our coords.
2379         IntRect colRect = columnRectAt(colInfo, i);
2380         flipForWritingMode(colRect);
2381         int logicalLeftOffset = (isHorizontalWritingMode() ? colRect.x() : colRect.y()) - logicalLeftOffsetForContent();
2382         IntSize offset = isHorizontalWritingMode() ? IntSize(logicalLeftOffset, currLogicalTopOffset) : IntSize(currLogicalTopOffset, logicalLeftOffset);
2383         colRect.move(tx, ty);
2384         PaintInfo info(paintInfo);
2385         info.rect.intersect(colRect);
2386 
2387         if (!info.rect.isEmpty()) {
2388             context->save();
2389 
2390             // Each strip pushes a clip, since column boxes are specified as being
2391             // like overflow:hidden.
2392             context->clip(colRect);
2393 
2394             // Adjust our x and y when painting.
2395             int finalX = tx + offset.width();
2396             int finalY = ty + offset.height();
2397             if (paintingFloats)
2398                 paintFloats(info, finalX, finalY, paintInfo.phase == PaintPhaseSelection || paintInfo.phase == PaintPhaseTextClip);
2399             else
2400                 paintContents(info, finalX, finalY);
2401 
2402             context->restore();
2403         }
2404 
2405         int blockDelta = (isHorizontalWritingMode() ? colRect.height() : colRect.width());
2406         if (style()->isFlippedBlocksWritingMode())
2407             currLogicalTopOffset += blockDelta;
2408         else
2409             currLogicalTopOffset -= blockDelta;
2410     }
2411 }
2412 
paintContents(PaintInfo & paintInfo,int tx,int ty)2413 void RenderBlock::paintContents(PaintInfo& paintInfo, int tx, int ty)
2414 {
2415     // Avoid painting descendants of the root element when stylesheets haven't loaded.  This eliminates FOUC.
2416     // It's ok not to draw, because later on, when all the stylesheets do load, updateStyleSelector on the Document
2417     // will do a full repaint().
2418     if (document()->didLayoutWithPendingStylesheets() && !isRenderView())
2419         return;
2420 
2421     if (childrenInline())
2422         m_lineBoxes.paint(this, paintInfo, tx, ty);
2423     else
2424         paintChildren(paintInfo, tx, ty);
2425 }
2426 
paintChildren(PaintInfo & paintInfo,int tx,int ty)2427 void RenderBlock::paintChildren(PaintInfo& paintInfo, int tx, int ty)
2428 {
2429     PaintPhase newPhase = (paintInfo.phase == PaintPhaseChildOutlines) ? PaintPhaseOutline : paintInfo.phase;
2430     newPhase = (newPhase == PaintPhaseChildBlockBackgrounds) ? PaintPhaseChildBlockBackground : newPhase;
2431 
2432     // We don't paint our own background, but we do let the kids paint their backgrounds.
2433     PaintInfo info(paintInfo);
2434     info.phase = newPhase;
2435     info.updatePaintingRootForChildren(this);
2436 
2437     // FIXME: Paint-time pagination is obsolete and is now only used by embedded WebViews inside AppKit
2438     // NSViews.  Do not add any more code for this.
2439     RenderView* renderView = view();
2440     bool usePrintRect = !renderView->printRect().isEmpty();
2441 
2442     for (RenderBox* child = firstChildBox(); child; child = child->nextSiblingBox()) {
2443         // Check for page-break-before: always, and if it's set, break and bail.
2444         bool checkBeforeAlways = !childrenInline() && (usePrintRect && child->style()->pageBreakBefore() == PBALWAYS);
2445         if (checkBeforeAlways
2446             && (ty + child->y()) > paintInfo.rect.y()
2447             && (ty + child->y()) < paintInfo.rect.maxY()) {
2448             view()->setBestTruncatedAt(ty + child->y(), this, true);
2449             return;
2450         }
2451 
2452         if (!child->isFloating() && child->isReplaced() && usePrintRect && child->height() <= renderView->printRect().height()) {
2453             // Paginate block-level replaced elements.
2454             if (ty + child->y() + child->height() > renderView->printRect().maxY()) {
2455                 if (ty + child->y() < renderView->truncatedAt())
2456                     renderView->setBestTruncatedAt(ty + child->y(), child);
2457                 // If we were able to truncate, don't paint.
2458                 if (ty + child->y() >= renderView->truncatedAt())
2459                     break;
2460             }
2461         }
2462 
2463         IntPoint childPoint = flipForWritingMode(child, IntPoint(tx, ty), ParentToChildFlippingAdjustment);
2464         if (!child->hasSelfPaintingLayer() && !child->isFloating())
2465             child->paint(info, childPoint.x(), childPoint.y());
2466 
2467         // Check for page-break-after: always, and if it's set, break and bail.
2468         bool checkAfterAlways = !childrenInline() && (usePrintRect && child->style()->pageBreakAfter() == PBALWAYS);
2469         if (checkAfterAlways
2470             && (ty + child->y() + child->height()) > paintInfo.rect.y()
2471             && (ty + child->y() + child->height()) < paintInfo.rect.maxY()) {
2472             view()->setBestTruncatedAt(ty + child->y() + child->height() + max(0, child->collapsedMarginAfter()), this, true);
2473             return;
2474         }
2475     }
2476 }
2477 
paintCaret(PaintInfo & paintInfo,int tx,int ty,CaretType type)2478 void RenderBlock::paintCaret(PaintInfo& paintInfo, int tx, int ty, CaretType type)
2479 {
2480     SelectionController* selection = type == CursorCaret ? frame()->selection() : frame()->page()->dragCaretController();
2481 
2482     // Paint the caret if the SelectionController says so or if caret browsing is enabled
2483     bool caretBrowsing = frame()->settings() && frame()->settings()->caretBrowsingEnabled();
2484     RenderObject* caretPainter = selection->caretRenderer();
2485     if (caretPainter == this && (selection->isContentEditable() || caretBrowsing)) {
2486         // Convert the painting offset into the local coordinate system of this renderer,
2487         // to match the localCaretRect computed by the SelectionController
2488         offsetForContents(tx, ty);
2489 
2490         if (type == CursorCaret)
2491             frame()->selection()->paintCaret(paintInfo.context, tx, ty, paintInfo.rect);
2492         else
2493             frame()->selection()->paintDragCaret(paintInfo.context, tx, ty, paintInfo.rect);
2494     }
2495 }
2496 
paintObject(PaintInfo & paintInfo,int tx,int ty)2497 void RenderBlock::paintObject(PaintInfo& paintInfo, int tx, int ty)
2498 {
2499     PaintPhase paintPhase = paintInfo.phase;
2500 
2501     // 1. paint background, borders etc
2502     if ((paintPhase == PaintPhaseBlockBackground || paintPhase == PaintPhaseChildBlockBackground
2503 #if PLATFORM(ANDROID)
2504          || paintPhase == PaintPhaseBlockBackgroundDecorations
2505 #endif
2506         ) && style()->visibility() == VISIBLE) {
2507         if (hasBoxDecorations())
2508             paintBoxDecorations(paintInfo, tx, ty);
2509         if (hasColumns())
2510             paintColumnRules(paintInfo, tx, ty);
2511     }
2512 
2513     if (paintPhase == PaintPhaseMask && style()->visibility() == VISIBLE) {
2514         paintMask(paintInfo, tx, ty);
2515         return;
2516     }
2517 
2518     // We're done.  We don't bother painting any children.
2519     if (paintPhase == PaintPhaseBlockBackground)
2520         return;
2521 
2522     // Adjust our painting position if we're inside a scrolled layer (e.g., an overflow:auto div).
2523     int scrolledX = tx;
2524     int scrolledY = ty;
2525     if (hasOverflowClip()) {
2526         IntSize offset = layer()->scrolledContentOffset();
2527         scrolledX -= offset.width();
2528         scrolledY -= offset.height();
2529     }
2530 
2531     // 2. paint contents
2532     if (paintPhase != PaintPhaseSelfOutline) {
2533         if (hasColumns())
2534             paintColumnContents(paintInfo, scrolledX, scrolledY);
2535         else
2536             paintContents(paintInfo, scrolledX, scrolledY);
2537     }
2538 
2539     // 3. paint selection
2540     // FIXME: Make this work with multi column layouts.  For now don't fill gaps.
2541     bool isPrinting = document()->printing();
2542     if (!isPrinting && !hasColumns())
2543         paintSelection(paintInfo, scrolledX, scrolledY); // Fill in gaps in selection on lines and between blocks.
2544 
2545     // 4. paint floats.
2546     if (paintPhase == PaintPhaseFloat || paintPhase == PaintPhaseSelection || paintPhase == PaintPhaseTextClip) {
2547         if (hasColumns())
2548             paintColumnContents(paintInfo, scrolledX, scrolledY, true);
2549         else
2550             paintFloats(paintInfo, scrolledX, scrolledY, paintPhase == PaintPhaseSelection || paintPhase == PaintPhaseTextClip);
2551     }
2552 
2553     // 5. paint outline.
2554     if ((paintPhase == PaintPhaseOutline || paintPhase == PaintPhaseSelfOutline) && hasOutline() && style()->visibility() == VISIBLE)
2555         paintOutline(paintInfo.context, tx, ty, width(), height());
2556 
2557     // 6. paint continuation outlines.
2558     if ((paintPhase == PaintPhaseOutline || paintPhase == PaintPhaseChildOutlines)) {
2559         RenderInline* inlineCont = inlineElementContinuation();
2560         if (inlineCont && inlineCont->hasOutline() && inlineCont->style()->visibility() == VISIBLE) {
2561             RenderInline* inlineRenderer = toRenderInline(inlineCont->node()->renderer());
2562             RenderBlock* cb = containingBlock();
2563 
2564             bool inlineEnclosedInSelfPaintingLayer = false;
2565             for (RenderBoxModelObject* box = inlineRenderer; box != cb; box = box->parent()->enclosingBoxModelObject()) {
2566                 if (box->hasSelfPaintingLayer()) {
2567                     inlineEnclosedInSelfPaintingLayer = true;
2568                     break;
2569                 }
2570             }
2571 
2572             if (!inlineEnclosedInSelfPaintingLayer)
2573                 cb->addContinuationWithOutline(inlineRenderer);
2574             else if (!inlineRenderer->firstLineBox())
2575                 inlineRenderer->paintOutline(paintInfo.context, tx - x() + inlineRenderer->containingBlock()->x(),
2576                                              ty - y() + inlineRenderer->containingBlock()->y());
2577         }
2578         paintContinuationOutlines(paintInfo, tx, ty);
2579     }
2580 
2581     // 7. paint caret.
2582     // If the caret's node's render object's containing block is this block, and the paint action is PaintPhaseForeground,
2583     // then paint the caret.
2584     if (paintPhase == PaintPhaseForeground) {
2585         paintCaret(paintInfo, scrolledX, scrolledY, CursorCaret);
2586         paintCaret(paintInfo, scrolledX, scrolledY, DragCaret);
2587     }
2588 }
2589 
flipFloatForWritingMode(const FloatingObject * child,const IntPoint & point) const2590 IntPoint RenderBlock::flipFloatForWritingMode(const FloatingObject* child, const IntPoint& point) const
2591 {
2592     if (!style()->isFlippedBlocksWritingMode())
2593         return point;
2594 
2595     // This is similar to the ParentToChildFlippingAdjustment in RenderBox::flipForWritingMode.  We have to subtract out our left/top offsets twice, since
2596     // it's going to get added back in.  We hide this complication here so that the calling code looks normal for the unflipped
2597     // case.
2598     if (isHorizontalWritingMode())
2599         return IntPoint(point.x(), point.y() + height() - child->renderer()->height() - 2 * yPositionForFloatIncludingMargin(child));
2600     return IntPoint(point.x() + width() - child->width() - 2 * xPositionForFloatIncludingMargin(child), point.y());
2601 }
2602 
paintFloats(PaintInfo & paintInfo,int tx,int ty,bool preservePhase)2603 void RenderBlock::paintFloats(PaintInfo& paintInfo, int tx, int ty, bool preservePhase)
2604 {
2605     if (!m_floatingObjects)
2606         return;
2607 
2608     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2609     FloatingObjectSetIterator end = floatingObjectSet.end();
2610     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
2611         FloatingObject* r = *it;
2612         // Only paint the object if our m_shouldPaint flag is set.
2613         if (r->m_shouldPaint && !r->m_renderer->hasSelfPaintingLayer()) {
2614             PaintInfo currentPaintInfo(paintInfo);
2615             currentPaintInfo.phase = preservePhase ? paintInfo.phase : PaintPhaseBlockBackground;
2616             IntPoint childPoint = flipFloatForWritingMode(r, IntPoint(tx + xPositionForFloatIncludingMargin(r) - r->m_renderer->x(), ty + yPositionForFloatIncludingMargin(r) - r->m_renderer->y()));
2617             r->m_renderer->paint(currentPaintInfo, childPoint.x(), childPoint.y());
2618             if (!preservePhase) {
2619                 currentPaintInfo.phase = PaintPhaseChildBlockBackgrounds;
2620                 r->m_renderer->paint(currentPaintInfo, childPoint.x(), childPoint.y());
2621                 currentPaintInfo.phase = PaintPhaseFloat;
2622                 r->m_renderer->paint(currentPaintInfo, childPoint.x(), childPoint.y());
2623                 currentPaintInfo.phase = PaintPhaseForeground;
2624                 r->m_renderer->paint(currentPaintInfo, childPoint.x(), childPoint.y());
2625                 currentPaintInfo.phase = PaintPhaseOutline;
2626                 r->m_renderer->paint(currentPaintInfo, childPoint.x(), childPoint.y());
2627             }
2628         }
2629     }
2630 }
2631 
paintEllipsisBoxes(PaintInfo & paintInfo,int tx,int ty)2632 void RenderBlock::paintEllipsisBoxes(PaintInfo& paintInfo, int tx, int ty)
2633 {
2634     if (!paintInfo.shouldPaintWithinRoot(this) || !firstLineBox())
2635         return;
2636 
2637     if (style()->visibility() == VISIBLE && paintInfo.phase == PaintPhaseForeground) {
2638         // We can check the first box and last box and avoid painting if we don't
2639         // intersect.
2640         int yPos = ty + firstLineBox()->y();
2641         int h = lastLineBox()->y() + lastLineBox()->logicalHeight() - firstLineBox()->y();
2642         if (yPos >= paintInfo.rect.maxY() || yPos + h <= paintInfo.rect.y())
2643             return;
2644 
2645         // See if our boxes intersect with the dirty rect.  If so, then we paint
2646         // them.  Note that boxes can easily overlap, so we can't make any assumptions
2647         // based off positions of our first line box or our last line box.
2648         for (RootInlineBox* curr = firstRootBox(); curr; curr = curr->nextRootBox()) {
2649             yPos = ty + curr->y();
2650             h = curr->logicalHeight();
2651             if (curr->ellipsisBox() && yPos < paintInfo.rect.maxY() && yPos + h > paintInfo.rect.y())
2652                 curr->paintEllipsisBox(paintInfo, tx, ty, curr->lineTop(), curr->lineBottom());
2653         }
2654     }
2655 }
2656 
inlineElementContinuation() const2657 RenderInline* RenderBlock::inlineElementContinuation() const
2658 {
2659     RenderBoxModelObject* continuation = this->continuation();
2660     return continuation && continuation->isInline() ? toRenderInline(continuation) : 0;
2661 }
2662 
blockElementContinuation() const2663 RenderBlock* RenderBlock::blockElementContinuation() const
2664 {
2665     RenderBoxModelObject* currentContinuation = continuation();
2666     if (!currentContinuation || currentContinuation->isInline())
2667         return 0;
2668     RenderBlock* nextContinuation = toRenderBlock(currentContinuation);
2669     if (nextContinuation->isAnonymousBlock())
2670         return nextContinuation->blockElementContinuation();
2671     return nextContinuation;
2672 }
2673 
continuationOutlineTable()2674 static ContinuationOutlineTableMap* continuationOutlineTable()
2675 {
2676     DEFINE_STATIC_LOCAL(ContinuationOutlineTableMap, table, ());
2677     return &table;
2678 }
2679 
addContinuationWithOutline(RenderInline * flow)2680 void RenderBlock::addContinuationWithOutline(RenderInline* flow)
2681 {
2682     // We can't make this work if the inline is in a layer.  We'll just rely on the broken
2683     // way of painting.
2684     ASSERT(!flow->layer() && !flow->isInlineElementContinuation());
2685 
2686     ContinuationOutlineTableMap* table = continuationOutlineTable();
2687     ListHashSet<RenderInline*>* continuations = table->get(this);
2688     if (!continuations) {
2689         continuations = new ListHashSet<RenderInline*>;
2690         table->set(this, continuations);
2691     }
2692 
2693     continuations->add(flow);
2694 }
2695 
paintsContinuationOutline(RenderInline * flow)2696 bool RenderBlock::paintsContinuationOutline(RenderInline* flow)
2697 {
2698     ContinuationOutlineTableMap* table = continuationOutlineTable();
2699     if (table->isEmpty())
2700         return false;
2701 
2702     ListHashSet<RenderInline*>* continuations = table->get(this);
2703     if (!continuations)
2704         return false;
2705 
2706     return continuations->contains(flow);
2707 }
2708 
paintContinuationOutlines(PaintInfo & info,int tx,int ty)2709 void RenderBlock::paintContinuationOutlines(PaintInfo& info, int tx, int ty)
2710 {
2711     ContinuationOutlineTableMap* table = continuationOutlineTable();
2712     if (table->isEmpty())
2713         return;
2714 
2715     ListHashSet<RenderInline*>* continuations = table->get(this);
2716     if (!continuations)
2717         return;
2718 
2719     // Paint each continuation outline.
2720     ListHashSet<RenderInline*>::iterator end = continuations->end();
2721     for (ListHashSet<RenderInline*>::iterator it = continuations->begin(); it != end; ++it) {
2722         // Need to add in the coordinates of the intervening blocks.
2723         RenderInline* flow = *it;
2724         RenderBlock* block = flow->containingBlock();
2725         for ( ; block && block != this; block = block->containingBlock()) {
2726             tx += block->x();
2727             ty += block->y();
2728         }
2729         ASSERT(block);
2730         flow->paintOutline(info.context, tx, ty);
2731     }
2732 
2733     // Delete
2734     delete continuations;
2735     table->remove(this);
2736 }
2737 
shouldPaintSelectionGaps() const2738 bool RenderBlock::shouldPaintSelectionGaps() const
2739 {
2740     return selectionState() != SelectionNone && style()->visibility() == VISIBLE && isSelectionRoot();
2741 }
2742 
isSelectionRoot() const2743 bool RenderBlock::isSelectionRoot() const
2744 {
2745     if (!node())
2746         return false;
2747 
2748     // FIXME: Eventually tables should have to learn how to fill gaps between cells, at least in simple non-spanning cases.
2749     if (isTable())
2750         return false;
2751 
2752     if (isBody() || isRoot() || hasOverflowClip() || isRelPositioned() ||
2753         isFloatingOrPositioned() || isTableCell() || isInlineBlockOrInlineTable() || hasTransform() ||
2754         hasReflection() || hasMask() || isWritingModeRoot())
2755         return true;
2756 
2757     if (view() && view()->selectionStart()) {
2758         Node* startElement = view()->selectionStart()->node();
2759         if (startElement && startElement->rootEditableElement() == node())
2760             return true;
2761     }
2762 
2763     return false;
2764 }
2765 
selectionGapRectsForRepaint(RenderBoxModelObject * repaintContainer)2766 GapRects RenderBlock::selectionGapRectsForRepaint(RenderBoxModelObject* repaintContainer)
2767 {
2768     ASSERT(!needsLayout());
2769 
2770     if (!shouldPaintSelectionGaps())
2771         return GapRects();
2772 
2773     // FIXME: this is broken with transforms
2774     TransformState transformState(TransformState::ApplyTransformDirection, FloatPoint());
2775     mapLocalToContainer(repaintContainer, false, false, transformState);
2776     IntPoint offsetFromRepaintContainer = roundedIntPoint(transformState.mappedPoint());
2777 
2778     if (hasOverflowClip())
2779         offsetFromRepaintContainer -= layer()->scrolledContentOffset();
2780 
2781     int lastTop = 0;
2782     int lastLeft = logicalLeftSelectionOffset(this, lastTop);
2783     int lastRight = logicalRightSelectionOffset(this, lastTop);
2784 
2785     return selectionGaps(this, offsetFromRepaintContainer, IntSize(), lastTop, lastLeft, lastRight);
2786 }
2787 
paintSelection(PaintInfo & paintInfo,int tx,int ty)2788 void RenderBlock::paintSelection(PaintInfo& paintInfo, int tx, int ty)
2789 {
2790     if (shouldPaintSelectionGaps() && paintInfo.phase == PaintPhaseForeground) {
2791         int lastTop = 0;
2792         int lastLeft = logicalLeftSelectionOffset(this, lastTop);
2793         int lastRight = logicalRightSelectionOffset(this, lastTop);
2794         paintInfo.context->save();
2795         IntRect gapRectsBounds = selectionGaps(this, IntPoint(tx, ty), IntSize(), lastTop, lastLeft, lastRight, &paintInfo);
2796         if (!gapRectsBounds.isEmpty()) {
2797             if (RenderLayer* layer = enclosingLayer()) {
2798                 gapRectsBounds.move(IntSize(-tx, -ty));
2799                 if (!hasLayer()) {
2800                     IntRect localBounds(gapRectsBounds);
2801                     flipForWritingMode(localBounds);
2802                     gapRectsBounds = localToContainerQuad(FloatRect(localBounds), layer->renderer()).enclosingBoundingBox();
2803                     gapRectsBounds.move(layer->scrolledContentOffset());
2804                 }
2805                 layer->addBlockSelectionGapsBounds(gapRectsBounds);
2806             }
2807         }
2808         paintInfo.context->restore();
2809     }
2810 }
2811 
clipOutPositionedObjects(const PaintInfo * paintInfo,const IntPoint & offset,RenderBlock::PositionedObjectsListHashSet * positionedObjects)2812 static void clipOutPositionedObjects(const PaintInfo* paintInfo, const IntPoint& offset, RenderBlock::PositionedObjectsListHashSet* positionedObjects)
2813 {
2814     if (!positionedObjects)
2815         return;
2816 
2817     RenderBlock::PositionedObjectsListHashSet::const_iterator end = positionedObjects->end();
2818     for (RenderBlock::PositionedObjectsListHashSet::const_iterator it = positionedObjects->begin(); it != end; ++it) {
2819         RenderBox* r = *it;
2820         paintInfo->context->clipOut(IntRect(offset.x() + r->x(), offset.y() + r->y(), r->width(), r->height()));
2821     }
2822 }
2823 
blockDirectionOffset(RenderBlock * rootBlock,const IntSize & offsetFromRootBlock)2824 static int blockDirectionOffset(RenderBlock* rootBlock, const IntSize& offsetFromRootBlock)
2825 {
2826     return rootBlock->isHorizontalWritingMode() ? offsetFromRootBlock.height() : offsetFromRootBlock.width();
2827 }
2828 
inlineDirectionOffset(RenderBlock * rootBlock,const IntSize & offsetFromRootBlock)2829 static int inlineDirectionOffset(RenderBlock* rootBlock, const IntSize& offsetFromRootBlock)
2830 {
2831     return rootBlock->isHorizontalWritingMode() ? offsetFromRootBlock.width() : offsetFromRootBlock.height();
2832 }
2833 
logicalRectToPhysicalRect(const IntPoint & rootBlockPhysicalPosition,const IntRect & logicalRect)2834 IntRect RenderBlock::logicalRectToPhysicalRect(const IntPoint& rootBlockPhysicalPosition, const IntRect& logicalRect)
2835 {
2836     IntRect result;
2837     if (isHorizontalWritingMode())
2838         result = logicalRect;
2839     else
2840         result = IntRect(logicalRect.y(), logicalRect.x(), logicalRect.height(), logicalRect.width());
2841     flipForWritingMode(result);
2842     result.move(rootBlockPhysicalPosition.x(), rootBlockPhysicalPosition.y());
2843     return result;
2844 }
2845 
selectionGaps(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,int & lastLogicalTop,int & lastLogicalLeft,int & lastLogicalRight,const PaintInfo * paintInfo)2846 GapRects RenderBlock::selectionGaps(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
2847                                     int& lastLogicalTop, int& lastLogicalLeft, int& lastLogicalRight, const PaintInfo* paintInfo)
2848 {
2849     // IMPORTANT: Callers of this method that intend for painting to happen need to do a save/restore.
2850     // Clip out floating and positioned objects when painting selection gaps.
2851     if (paintInfo) {
2852         // Note that we don't clip out overflow for positioned objects.  We just stick to the border box.
2853         IntRect flippedBlockRect = IntRect(offsetFromRootBlock.width(), offsetFromRootBlock.height(), width(), height());
2854         rootBlock->flipForWritingMode(flippedBlockRect);
2855         flippedBlockRect.move(rootBlockPhysicalPosition.x(), rootBlockPhysicalPosition.y());
2856         clipOutPositionedObjects(paintInfo, flippedBlockRect.location(), m_positionedObjects.get());
2857         if (isBody() || isRoot()) // The <body> must make sure to examine its containingBlock's positioned objects.
2858             for (RenderBlock* cb = containingBlock(); cb && !cb->isRenderView(); cb = cb->containingBlock())
2859                 clipOutPositionedObjects(paintInfo, IntPoint(cb->x(), cb->y()), cb->m_positionedObjects.get()); // FIXME: Not right for flipped writing modes.
2860         if (m_floatingObjects) {
2861             FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2862             FloatingObjectSetIterator end = floatingObjectSet.end();
2863             for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
2864                 FloatingObject* r = *it;
2865                 IntRect floatBox = IntRect(offsetFromRootBlock.width() + xPositionForFloatIncludingMargin(r),
2866                                            offsetFromRootBlock.height() + yPositionForFloatIncludingMargin(r),
2867                                            r->m_renderer->width(), r->m_renderer->height());
2868                 rootBlock->flipForWritingMode(floatBox);
2869                 floatBox.move(rootBlockPhysicalPosition.x(), rootBlockPhysicalPosition.y());
2870                 paintInfo->context->clipOut(floatBox);
2871             }
2872         }
2873     }
2874 
2875     // FIXME: overflow: auto/scroll regions need more math here, since painting in the border box is different from painting in the padding box (one is scrolled, the other is
2876     // fixed).
2877     GapRects result;
2878     if (!isBlockFlow()) // FIXME: Make multi-column selection gap filling work someday.
2879         return result;
2880 
2881     if (hasColumns() || hasTransform() || style()->columnSpan()) {
2882         // FIXME: We should learn how to gap fill multiple columns and transforms eventually.
2883         lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalHeight();
2884         lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, logicalHeight());
2885         lastLogicalRight = logicalRightSelectionOffset(rootBlock, logicalHeight());
2886         return result;
2887     }
2888 
2889     if (childrenInline())
2890         result = inlineSelectionGaps(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight, paintInfo);
2891     else
2892         result = blockSelectionGaps(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight, paintInfo);
2893 
2894     // Go ahead and fill the vertical gap all the way to the bottom of our block if the selection extends past our block.
2895     if (rootBlock == this && (selectionState() != SelectionBoth && selectionState() != SelectionEnd))
2896         result.uniteCenter(blockSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight,
2897                                              logicalHeight(), paintInfo));
2898     return result;
2899 }
2900 
inlineSelectionGaps(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,int & lastLogicalTop,int & lastLogicalLeft,int & lastLogicalRight,const PaintInfo * paintInfo)2901 GapRects RenderBlock::inlineSelectionGaps(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
2902                                           int& lastLogicalTop, int& lastLogicalLeft, int& lastLogicalRight, const PaintInfo* paintInfo)
2903 {
2904     GapRects result;
2905 
2906     bool containsStart = selectionState() == SelectionStart || selectionState() == SelectionBoth;
2907 
2908     if (!firstLineBox()) {
2909         if (containsStart) {
2910             // Go ahead and update our lastLogicalTop to be the bottom of the block.  <hr>s or empty blocks with height can trip this
2911             // case.
2912             lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalHeight();
2913             lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, logicalHeight());
2914             lastLogicalRight = logicalRightSelectionOffset(rootBlock, logicalHeight());
2915         }
2916         return result;
2917     }
2918 
2919     RootInlineBox* lastSelectedLine = 0;
2920     RootInlineBox* curr;
2921     for (curr = firstRootBox(); curr && !curr->hasSelectedChildren(); curr = curr->nextRootBox()) { }
2922 
2923     // Now paint the gaps for the lines.
2924     for (; curr && curr->hasSelectedChildren(); curr = curr->nextRootBox()) {
2925         int selTop =  curr->selectionTop();
2926         int selHeight = curr->selectionHeight();
2927 
2928         if (!containsStart && !lastSelectedLine &&
2929             selectionState() != SelectionStart && selectionState() != SelectionBoth)
2930             result.uniteCenter(blockSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight,
2931                                                  selTop, paintInfo));
2932 
2933         IntRect logicalRect(curr->logicalLeft(), selTop, curr->logicalWidth(), selTop + selHeight);
2934         logicalRect.move(isHorizontalWritingMode() ? offsetFromRootBlock : IntSize(offsetFromRootBlock.height(), offsetFromRootBlock.width()));
2935         IntRect physicalRect = rootBlock->logicalRectToPhysicalRect(rootBlockPhysicalPosition, logicalRect);
2936         if (!paintInfo || (isHorizontalWritingMode() && physicalRect.y() < paintInfo->rect.maxY() && physicalRect.maxY() > paintInfo->rect.y())
2937             || (!isHorizontalWritingMode() && physicalRect.x() < paintInfo->rect.maxX() && physicalRect.maxX() > paintInfo->rect.x()))
2938             result.unite(curr->lineSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, selTop, selHeight, paintInfo));
2939 
2940         lastSelectedLine = curr;
2941     }
2942 
2943     if (containsStart && !lastSelectedLine)
2944         // VisibleSelection must start just after our last line.
2945         lastSelectedLine = lastRootBox();
2946 
2947     if (lastSelectedLine && selectionState() != SelectionEnd && selectionState() != SelectionBoth) {
2948         // Go ahead and update our lastY to be the bottom of the last selected line.
2949         lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + lastSelectedLine->selectionBottom();
2950         lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, lastSelectedLine->selectionBottom());
2951         lastLogicalRight = logicalRightSelectionOffset(rootBlock, lastSelectedLine->selectionBottom());
2952     }
2953     return result;
2954 }
2955 
blockSelectionGaps(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,int & lastLogicalTop,int & lastLogicalLeft,int & lastLogicalRight,const PaintInfo * paintInfo)2956 GapRects RenderBlock::blockSelectionGaps(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
2957                                          int& lastLogicalTop, int& lastLogicalLeft, int& lastLogicalRight, const PaintInfo* paintInfo)
2958 {
2959     GapRects result;
2960 
2961     // Go ahead and jump right to the first block child that contains some selected objects.
2962     RenderBox* curr;
2963     for (curr = firstChildBox(); curr && curr->selectionState() == SelectionNone; curr = curr->nextSiblingBox()) { }
2964 
2965     for (bool sawSelectionEnd = false; curr && !sawSelectionEnd; curr = curr->nextSiblingBox()) {
2966         SelectionState childState = curr->selectionState();
2967         if (childState == SelectionBoth || childState == SelectionEnd)
2968             sawSelectionEnd = true;
2969 
2970         if (curr->isFloatingOrPositioned())
2971             continue; // We must be a normal flow object in order to even be considered.
2972 
2973         if (curr->isRelPositioned() && curr->hasLayer()) {
2974             // If the relposition offset is anything other than 0, then treat this just like an absolute positioned element.
2975             // Just disregard it completely.
2976             IntSize relOffset = curr->layer()->relativePositionOffset();
2977             if (relOffset.width() || relOffset.height())
2978                 continue;
2979         }
2980 
2981         bool paintsOwnSelection = curr->shouldPaintSelectionGaps() || curr->isTable(); // FIXME: Eventually we won't special-case table like this.
2982         bool fillBlockGaps = paintsOwnSelection || (curr->canBeSelectionLeaf() && childState != SelectionNone);
2983         if (fillBlockGaps) {
2984             // We need to fill the vertical gap above this object.
2985             if (childState == SelectionEnd || childState == SelectionInside)
2986                 // Fill the gap above the object.
2987                 result.uniteCenter(blockSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight,
2988                                                      curr->logicalTop(), paintInfo));
2989 
2990             // Only fill side gaps for objects that paint their own selection if we know for sure the selection is going to extend all the way *past*
2991             // our object.  We know this if the selection did not end inside our object.
2992             if (paintsOwnSelection && (childState == SelectionStart || sawSelectionEnd))
2993                 childState = SelectionNone;
2994 
2995             // Fill side gaps on this object based off its state.
2996             bool leftGap, rightGap;
2997             getSelectionGapInfo(childState, leftGap, rightGap);
2998 
2999             if (leftGap)
3000                 result.uniteLeft(logicalLeftSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, this, curr->logicalLeft(), curr->logicalTop(), curr->logicalHeight(), paintInfo));
3001             if (rightGap)
3002                 result.uniteRight(logicalRightSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, this, curr->logicalRight(), curr->logicalTop(), curr->logicalHeight(), paintInfo));
3003 
3004             // Update lastLogicalTop to be just underneath the object.  lastLogicalLeft and lastLogicalRight extend as far as
3005             // they can without bumping into floating or positioned objects.  Ideally they will go right up
3006             // to the border of the root selection block.
3007             lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + curr->logicalBottom();
3008             lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, curr->logicalBottom());
3009             lastLogicalRight = logicalRightSelectionOffset(rootBlock, curr->logicalBottom());
3010         } else if (childState != SelectionNone)
3011             // We must be a block that has some selected object inside it.  Go ahead and recur.
3012             result.unite(toRenderBlock(curr)->selectionGaps(rootBlock, rootBlockPhysicalPosition, IntSize(offsetFromRootBlock.width() + curr->x(), offsetFromRootBlock.height() + curr->y()),
3013                                                             lastLogicalTop, lastLogicalLeft, lastLogicalRight, paintInfo));
3014     }
3015     return result;
3016 }
3017 
blockSelectionGap(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,int lastLogicalTop,int lastLogicalLeft,int lastLogicalRight,int logicalBottom,const PaintInfo * paintInfo)3018 IntRect RenderBlock::blockSelectionGap(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
3019                                        int lastLogicalTop, int lastLogicalLeft, int lastLogicalRight, int logicalBottom, const PaintInfo* paintInfo)
3020 {
3021     int logicalTop = lastLogicalTop;
3022     int logicalHeight = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalBottom - logicalTop;
3023     if (logicalHeight <= 0)
3024         return IntRect();
3025 
3026     // Get the selection offsets for the bottom of the gap
3027     int logicalLeft = max(lastLogicalLeft, logicalLeftSelectionOffset(rootBlock, logicalBottom));
3028     int logicalRight = min(lastLogicalRight, logicalRightSelectionOffset(rootBlock, logicalBottom));
3029     int logicalWidth = logicalRight - logicalLeft;
3030     if (logicalWidth <= 0)
3031         return IntRect();
3032 
3033     IntRect gapRect = rootBlock->logicalRectToPhysicalRect(rootBlockPhysicalPosition, IntRect(logicalLeft, logicalTop, logicalWidth, logicalHeight));
3034     if (paintInfo)
3035         paintInfo->context->fillRect(gapRect, selectionBackgroundColor(), style()->colorSpace());
3036     return gapRect;
3037 }
3038 
logicalLeftSelectionGap(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,RenderObject * selObj,int logicalLeft,int logicalTop,int logicalHeight,const PaintInfo * paintInfo)3039 IntRect RenderBlock::logicalLeftSelectionGap(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
3040                                              RenderObject* selObj, int logicalLeft, int logicalTop, int logicalHeight, const PaintInfo* paintInfo)
3041 {
3042     int rootBlockLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalTop;
3043     int rootBlockLogicalLeft = max(logicalLeftSelectionOffset(rootBlock, logicalTop), logicalLeftSelectionOffset(rootBlock, logicalTop + logicalHeight));
3044     int rootBlockLogicalRight = min(inlineDirectionOffset(rootBlock, offsetFromRootBlock) + logicalLeft, min(logicalRightSelectionOffset(rootBlock, logicalTop), logicalRightSelectionOffset(rootBlock, logicalTop + logicalHeight)));
3045     int rootBlockLogicalWidth = rootBlockLogicalRight - rootBlockLogicalLeft;
3046     if (rootBlockLogicalWidth <= 0)
3047         return IntRect();
3048 
3049     IntRect gapRect = rootBlock->logicalRectToPhysicalRect(rootBlockPhysicalPosition, IntRect(rootBlockLogicalLeft, rootBlockLogicalTop, rootBlockLogicalWidth, logicalHeight));
3050     if (paintInfo)
3051         paintInfo->context->fillRect(gapRect, selObj->selectionBackgroundColor(), selObj->style()->colorSpace());
3052     return gapRect;
3053 }
3054 
logicalRightSelectionGap(RenderBlock * rootBlock,const IntPoint & rootBlockPhysicalPosition,const IntSize & offsetFromRootBlock,RenderObject * selObj,int logicalRight,int logicalTop,int logicalHeight,const PaintInfo * paintInfo)3055 IntRect RenderBlock::logicalRightSelectionGap(RenderBlock* rootBlock, const IntPoint& rootBlockPhysicalPosition, const IntSize& offsetFromRootBlock,
3056                                               RenderObject* selObj, int logicalRight, int logicalTop, int logicalHeight, const PaintInfo* paintInfo)
3057 {
3058     int rootBlockLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalTop;
3059     int rootBlockLogicalLeft = max(inlineDirectionOffset(rootBlock, offsetFromRootBlock) + logicalRight, max(logicalLeftSelectionOffset(rootBlock, logicalTop), logicalLeftSelectionOffset(rootBlock, logicalTop + logicalHeight)));
3060     int rootBlockLogicalRight = min(logicalRightSelectionOffset(rootBlock, logicalTop), logicalRightSelectionOffset(rootBlock, logicalTop + logicalHeight));
3061     int rootBlockLogicalWidth = rootBlockLogicalRight - rootBlockLogicalLeft;
3062     if (rootBlockLogicalWidth <= 0)
3063         return IntRect();
3064 
3065     IntRect gapRect = rootBlock->logicalRectToPhysicalRect(rootBlockPhysicalPosition, IntRect(rootBlockLogicalLeft, rootBlockLogicalTop, rootBlockLogicalWidth, logicalHeight));
3066     if (paintInfo)
3067         paintInfo->context->fillRect(gapRect, selObj->selectionBackgroundColor(), selObj->style()->colorSpace());
3068     return gapRect;
3069 }
3070 
getSelectionGapInfo(SelectionState state,bool & leftGap,bool & rightGap)3071 void RenderBlock::getSelectionGapInfo(SelectionState state, bool& leftGap, bool& rightGap)
3072 {
3073     bool ltr = style()->isLeftToRightDirection();
3074     leftGap = (state == RenderObject::SelectionInside) ||
3075               (state == RenderObject::SelectionEnd && ltr) ||
3076               (state == RenderObject::SelectionStart && !ltr);
3077     rightGap = (state == RenderObject::SelectionInside) ||
3078                (state == RenderObject::SelectionStart && ltr) ||
3079                (state == RenderObject::SelectionEnd && !ltr);
3080 }
3081 
logicalLeftSelectionOffset(RenderBlock * rootBlock,int position)3082 int RenderBlock::logicalLeftSelectionOffset(RenderBlock* rootBlock, int position)
3083 {
3084     int logicalLeft = logicalLeftOffsetForLine(position, false);
3085     if (logicalLeft == logicalLeftOffsetForContent()) {
3086         if (rootBlock != this)
3087             // The border can potentially be further extended by our containingBlock().
3088             return containingBlock()->logicalLeftSelectionOffset(rootBlock, position + logicalTop());
3089         return logicalLeft;
3090     } else {
3091         RenderBlock* cb = this;
3092         while (cb != rootBlock) {
3093             logicalLeft += cb->logicalLeft();
3094             cb = cb->containingBlock();
3095         }
3096     }
3097     return logicalLeft;
3098 }
3099 
logicalRightSelectionOffset(RenderBlock * rootBlock,int position)3100 int RenderBlock::logicalRightSelectionOffset(RenderBlock* rootBlock, int position)
3101 {
3102     int logicalRight = logicalRightOffsetForLine(position, false);
3103     if (logicalRight == logicalRightOffsetForContent()) {
3104         if (rootBlock != this)
3105             // The border can potentially be further extended by our containingBlock().
3106             return containingBlock()->logicalRightSelectionOffset(rootBlock, position + logicalTop());
3107         return logicalRight;
3108     } else {
3109         RenderBlock* cb = this;
3110         while (cb != rootBlock) {
3111             logicalRight += cb->logicalLeft();
3112             cb = cb->containingBlock();
3113         }
3114     }
3115     return logicalRight;
3116 }
3117 
insertPositionedObject(RenderBox * o)3118 void RenderBlock::insertPositionedObject(RenderBox* o)
3119 {
3120     // Create the list of special objects if we don't aleady have one
3121     if (!m_positionedObjects)
3122         m_positionedObjects = adoptPtr(new PositionedObjectsListHashSet);
3123 
3124     m_positionedObjects->add(o);
3125 }
3126 
removePositionedObject(RenderBox * o)3127 void RenderBlock::removePositionedObject(RenderBox* o)
3128 {
3129     if (m_positionedObjects)
3130         m_positionedObjects->remove(o);
3131 }
3132 
removePositionedObjects(RenderBlock * o)3133 void RenderBlock::removePositionedObjects(RenderBlock* o)
3134 {
3135     if (!m_positionedObjects)
3136         return;
3137 
3138     RenderBox* r;
3139 
3140     Iterator end = m_positionedObjects->end();
3141 
3142     Vector<RenderBox*, 16> deadObjects;
3143 
3144     for (Iterator it = m_positionedObjects->begin(); it != end; ++it) {
3145         r = *it;
3146         if (!o || r->isDescendantOf(o)) {
3147             if (o)
3148                 r->setChildNeedsLayout(true, false);
3149 
3150             // It is parent blocks job to add positioned child to positioned objects list of its containing block
3151             // Parent layout needs to be invalidated to ensure this happens.
3152             RenderObject* p = r->parent();
3153             while (p && !p->isRenderBlock())
3154                 p = p->parent();
3155             if (p)
3156                 p->setChildNeedsLayout(true);
3157 
3158             deadObjects.append(r);
3159         }
3160     }
3161 
3162     for (unsigned i = 0; i < deadObjects.size(); i++)
3163         m_positionedObjects->remove(deadObjects.at(i));
3164 }
3165 
insertFloatingObject(RenderBox * o)3166 RenderBlock::FloatingObject* RenderBlock::insertFloatingObject(RenderBox* o)
3167 {
3168     ASSERT(o->isFloating());
3169 
3170     // Create the list of special objects if we don't aleady have one
3171     if (!m_floatingObjects)
3172         m_floatingObjects = adoptPtr(new FloatingObjects);
3173     else {
3174         // Don't insert the object again if it's already in the list
3175         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3176         FloatingObjectSetIterator it = floatingObjectSet.find<RenderBox*, FloatingObjectHashTranslator>(o);
3177         if (it != floatingObjectSet.end())
3178             return *it;
3179     }
3180 
3181     // Create the special object entry & append it to the list
3182 
3183     FloatingObject* newObj = new FloatingObject(o->style()->floating() == FLEFT ? FloatingObject::FloatLeft : FloatingObject::FloatRight);
3184 
3185     // Our location is irrelevant if we're unsplittable or no pagination is in effect.
3186     // Just go ahead and lay out the float.
3187     bool isChildRenderBlock = o->isRenderBlock();
3188     if (isChildRenderBlock && !o->needsLayout() && view()->layoutState()->pageLogicalHeightChanged())
3189         o->setChildNeedsLayout(true, false);
3190 
3191     bool affectedByPagination = isChildRenderBlock && view()->layoutState()->m_pageLogicalHeight;
3192     if (!affectedByPagination || isWritingModeRoot()) // We are unsplittable if we're a block flow root.
3193         o->layoutIfNeeded();
3194     else {
3195         o->computeLogicalWidth();
3196         o->computeBlockDirectionMargins(this);
3197     }
3198     setLogicalWidthForFloat(newObj, logicalWidthForChild(o) + marginStartForChild(o) + marginEndForChild(o));
3199 
3200     newObj->m_shouldPaint = !o->hasSelfPaintingLayer(); // If a layer exists, the float will paint itself.  Otherwise someone else will.
3201     newObj->m_isDescendant = true;
3202     newObj->m_renderer = o;
3203 
3204     m_floatingObjects->increaseObjectsCount(newObj->type());
3205     m_floatingObjects->set().add(newObj);
3206 
3207     return newObj;
3208 }
3209 
removeFloatingObject(RenderBox * o)3210 void RenderBlock::removeFloatingObject(RenderBox* o)
3211 {
3212     if (m_floatingObjects) {
3213         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3214         FloatingObjectSet::iterator it = floatingObjectSet.find<RenderBox*, FloatingObjectHashTranslator>(o);
3215         if (it != floatingObjectSet.end()) {
3216             FloatingObject* r = *it;
3217             if (childrenInline()) {
3218                 int logicalTop = logicalTopForFloat(r);
3219                 int logicalBottom = logicalBottomForFloat(r);
3220 
3221                 // Fix for https://bugs.webkit.org/show_bug.cgi?id=54995.
3222                 if (logicalBottom < 0 || logicalBottom < logicalTop || logicalTop == numeric_limits<int>::max())
3223                     logicalBottom = numeric_limits<int>::max();
3224                 else {
3225                     // Special-case zero- and less-than-zero-height floats: those don't touch
3226                     // the line that they're on, but it still needs to be dirtied. This is
3227                     // accomplished by pretending they have a height of 1.
3228                     logicalBottom = max(logicalBottom, logicalTop + 1);
3229                 }
3230                 if (r->m_originatingLine) {
3231                     ASSERT(r->m_originatingLine->renderer() == this);
3232                     r->m_originatingLine->markDirty();
3233 #if !ASSERT_DISABLED
3234                     r->m_originatingLine = 0;
3235 #endif
3236                 }
3237                 markLinesDirtyInBlockRange(0, logicalBottom);
3238             }
3239             m_floatingObjects->decreaseObjectsCount(r->type());
3240             floatingObjectSet.remove(it);
3241             ASSERT(!r->m_originatingLine);
3242             delete r;
3243         }
3244     }
3245 }
3246 
removeFloatingObjectsBelow(FloatingObject * lastFloat,int logicalOffset)3247 void RenderBlock::removeFloatingObjectsBelow(FloatingObject* lastFloat, int logicalOffset)
3248 {
3249     if (!m_floatingObjects)
3250         return;
3251 
3252     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3253     FloatingObject* curr = floatingObjectSet.last();
3254     while (curr != lastFloat && (!curr->isPlaced() || logicalTopForFloat(curr) >= logicalOffset)) {
3255         m_floatingObjects->decreaseObjectsCount(curr->type());
3256         floatingObjectSet.removeLast();
3257         ASSERT(!curr->m_originatingLine);
3258         delete curr;
3259         curr = floatingObjectSet.last();
3260     }
3261 }
3262 
positionNewFloats()3263 bool RenderBlock::positionNewFloats()
3264 {
3265     if (!m_floatingObjects)
3266         return false;
3267 
3268     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3269     if (floatingObjectSet.isEmpty())
3270         return false;
3271 
3272     // If all floats have already been positioned, then we have no work to do.
3273     if (floatingObjectSet.last()->isPlaced())
3274         return false;
3275 
3276     // Move backwards through our floating object list until we find a float that has
3277     // already been positioned.  Then we'll be able to move forward, positioning all of
3278     // the new floats that need it.
3279     FloatingObjectSetIterator it = floatingObjectSet.end();
3280     --it; // Go to last item.
3281     FloatingObjectSetIterator begin = floatingObjectSet.begin();
3282     FloatingObject* lastPlacedFloatingObject = 0;
3283     while (it != begin) {
3284         --it;
3285         if ((*it)->isPlaced()) {
3286             lastPlacedFloatingObject = *it;
3287             ++it;
3288             break;
3289         }
3290     }
3291 
3292     int logicalTop = logicalHeight();
3293 
3294     // The float cannot start above the top position of the last positioned float.
3295     if (lastPlacedFloatingObject)
3296         logicalTop = max(logicalTopForFloat(lastPlacedFloatingObject), logicalTop);
3297 
3298     FloatingObjectSetIterator end = floatingObjectSet.end();
3299     // Now walk through the set of unpositioned floats and place them.
3300     for (; it != end; ++it) {
3301          FloatingObject* floatingObject = *it;
3302         // The containing block is responsible for positioning floats, so if we have floats in our
3303         // list that come from somewhere else, do not attempt to position them.
3304         if (floatingObject->renderer()->containingBlock() != this)
3305             continue;
3306 
3307         RenderBox* childBox = floatingObject->renderer();
3308         int childLogicalLeftMargin = style()->isLeftToRightDirection() ? marginStartForChild(childBox) : marginEndForChild(childBox);
3309 
3310         int rightOffset = logicalRightOffsetForContent(); // Constant part of right offset.
3311         int leftOffset = logicalLeftOffsetForContent(); // Constant part of left offset.
3312         int floatLogicalWidth = logicalWidthForFloat(floatingObject); // The width we look for.
3313         if (rightOffset - leftOffset < floatLogicalWidth)
3314             floatLogicalWidth = rightOffset - leftOffset; // Never look for more than what will be available.
3315 
3316         IntRect oldRect(childBox->x(), childBox->y() , childBox->width(), childBox->height());
3317 
3318         if (childBox->style()->clear() & CLEFT)
3319             logicalTop = max(lowestFloatLogicalBottom(FloatingObject::FloatLeft), logicalTop);
3320         if (childBox->style()->clear() & CRIGHT)
3321             logicalTop = max(lowestFloatLogicalBottom(FloatingObject::FloatRight), logicalTop);
3322 
3323         int floatLogicalLeft;
3324         if (childBox->style()->floating() == FLEFT) {
3325             int heightRemainingLeft = 1;
3326             int heightRemainingRight = 1;
3327             floatLogicalLeft = logicalLeftOffsetForLine(logicalTop, leftOffset, false, &heightRemainingLeft);
3328             while (logicalRightOffsetForLine(logicalTop, rightOffset, false, &heightRemainingRight) - floatLogicalLeft < floatLogicalWidth) {
3329                 logicalTop += min(heightRemainingLeft, heightRemainingRight);
3330                 floatLogicalLeft = logicalLeftOffsetForLine(logicalTop, leftOffset, false, &heightRemainingLeft);
3331             }
3332             floatLogicalLeft = max(0, floatLogicalLeft);
3333         } else {
3334             int heightRemainingLeft = 1;
3335             int heightRemainingRight = 1;
3336             floatLogicalLeft = logicalRightOffsetForLine(logicalTop, rightOffset, false, &heightRemainingRight);
3337             while (floatLogicalLeft - logicalLeftOffsetForLine(logicalTop, leftOffset, false, &heightRemainingLeft) < floatLogicalWidth) {
3338                 logicalTop += min(heightRemainingLeft, heightRemainingRight);
3339                 floatLogicalLeft = logicalRightOffsetForLine(logicalTop, rightOffset, false, &heightRemainingRight);
3340             }
3341             floatLogicalLeft -= logicalWidthForFloat(floatingObject); // Use the original width of the float here, since the local variable
3342                                                                       // |floatLogicalWidth| was capped to the available line width.
3343                                                                       // See fast/block/float/clamped-right-float.html.
3344         }
3345 
3346         setLogicalLeftForFloat(floatingObject, floatLogicalLeft);
3347         setLogicalLeftForChild(childBox, floatLogicalLeft + childLogicalLeftMargin);
3348         setLogicalTopForChild(childBox, logicalTop + marginBeforeForChild(childBox));
3349 
3350         if (view()->layoutState()->isPaginated()) {
3351             RenderBlock* childBlock = childBox->isRenderBlock() ? toRenderBlock(childBox) : 0;
3352 
3353             if (!childBox->needsLayout())
3354                 childBox->markForPaginationRelayoutIfNeeded();;
3355             childBox->layoutIfNeeded();
3356 
3357             // If we are unsplittable and don't fit, then we need to move down.
3358             // We include our margins as part of the unsplittable area.
3359             int newLogicalTop = adjustForUnsplittableChild(childBox, logicalTop, true);
3360 
3361             // See if we have a pagination strut that is making us move down further.
3362             // Note that an unsplittable child can't also have a pagination strut, so this is
3363             // exclusive with the case above.
3364             if (childBlock && childBlock->paginationStrut()) {
3365                 newLogicalTop += childBlock->paginationStrut();
3366                 childBlock->setPaginationStrut(0);
3367             }
3368 
3369             if (newLogicalTop != logicalTop) {
3370                 floatingObject->m_paginationStrut = newLogicalTop - logicalTop;
3371                 logicalTop = newLogicalTop;
3372                 setLogicalTopForChild(childBox, logicalTop + marginBeforeForChild(childBox));
3373                 if (childBlock)
3374                     childBlock->setChildNeedsLayout(true, false);
3375                 childBox->layoutIfNeeded();
3376             }
3377         }
3378 
3379         setLogicalTopForFloat(floatingObject, logicalTop);
3380         setLogicalHeightForFloat(floatingObject, logicalHeightForChild(childBox) + marginBeforeForChild(childBox) + marginAfterForChild(childBox));
3381 
3382         floatingObject->setIsPlaced();
3383 
3384         // If the child moved, we have to repaint it.
3385         if (childBox->checkForRepaintDuringLayout())
3386             childBox->repaintDuringLayoutIfMoved(oldRect);
3387     }
3388     return true;
3389 }
3390 
newLine(EClear clear)3391 void RenderBlock::newLine(EClear clear)
3392 {
3393     positionNewFloats();
3394     // set y position
3395     int newY = 0;
3396     switch (clear)
3397     {
3398         case CLEFT:
3399             newY = lowestFloatLogicalBottom(FloatingObject::FloatLeft);
3400             break;
3401         case CRIGHT:
3402             newY = lowestFloatLogicalBottom(FloatingObject::FloatRight);
3403             break;
3404         case CBOTH:
3405             newY = lowestFloatLogicalBottom();
3406         default:
3407             break;
3408     }
3409     if (height() < newY)
3410         setLogicalHeight(newY);
3411 }
3412 
addPercentHeightDescendant(RenderBox * descendant)3413 void RenderBlock::addPercentHeightDescendant(RenderBox* descendant)
3414 {
3415     if (!gPercentHeightDescendantsMap) {
3416         gPercentHeightDescendantsMap = new PercentHeightDescendantsMap;
3417         gPercentHeightContainerMap = new PercentHeightContainerMap;
3418     }
3419 
3420     HashSet<RenderBox*>* descendantSet = gPercentHeightDescendantsMap->get(this);
3421     if (!descendantSet) {
3422         descendantSet = new HashSet<RenderBox*>;
3423         gPercentHeightDescendantsMap->set(this, descendantSet);
3424     }
3425     bool added = descendantSet->add(descendant).second;
3426     if (!added) {
3427         ASSERT(gPercentHeightContainerMap->get(descendant));
3428         ASSERT(gPercentHeightContainerMap->get(descendant)->contains(this));
3429         return;
3430     }
3431 
3432     HashSet<RenderBlock*>* containerSet = gPercentHeightContainerMap->get(descendant);
3433     if (!containerSet) {
3434         containerSet = new HashSet<RenderBlock*>;
3435         gPercentHeightContainerMap->set(descendant, containerSet);
3436     }
3437     ASSERT(!containerSet->contains(this));
3438     containerSet->add(this);
3439 }
3440 
removePercentHeightDescendant(RenderBox * descendant)3441 void RenderBlock::removePercentHeightDescendant(RenderBox* descendant)
3442 {
3443     if (!gPercentHeightContainerMap)
3444         return;
3445 
3446     HashSet<RenderBlock*>* containerSet = gPercentHeightContainerMap->take(descendant);
3447     if (!containerSet)
3448         return;
3449 
3450     HashSet<RenderBlock*>::iterator end = containerSet->end();
3451     for (HashSet<RenderBlock*>::iterator it = containerSet->begin(); it != end; ++it) {
3452         RenderBlock* container = *it;
3453         HashSet<RenderBox*>* descendantSet = gPercentHeightDescendantsMap->get(container);
3454         ASSERT(descendantSet);
3455         if (!descendantSet)
3456             continue;
3457         ASSERT(descendantSet->contains(descendant));
3458         descendantSet->remove(descendant);
3459         if (descendantSet->isEmpty()) {
3460             gPercentHeightDescendantsMap->remove(container);
3461             delete descendantSet;
3462         }
3463     }
3464 
3465     delete containerSet;
3466 }
3467 
percentHeightDescendants() const3468 HashSet<RenderBox*>* RenderBlock::percentHeightDescendants() const
3469 {
3470     return gPercentHeightDescendantsMap ? gPercentHeightDescendantsMap->get(this) : 0;
3471 }
3472 
3473 // FIXME: The logicalLeftOffsetForLine/logicalRightOffsetForLine functions are very slow if there are many floats
3474 // present. We need to add a structure to floating objects to represent "lines" of floats.  Then instead of checking
3475 // each float individually, we'd just walk backwards through the "lines" and stop when we hit a line that is fully above
3476 // the vertical offset that we'd like to check.  Computing the "lines" would be rather complicated, but could replace the left
3477 // objects and right objects count hack that is currently used here.
logicalLeftOffsetForLine(int logicalTop,int fixedOffset,bool applyTextIndent,int * heightRemaining) const3478 int RenderBlock::logicalLeftOffsetForLine(int logicalTop, int fixedOffset, bool applyTextIndent, int* heightRemaining) const
3479 {
3480     int left = fixedOffset;
3481     if (m_floatingObjects && m_floatingObjects->hasLeftObjects()) {
3482         if (heightRemaining)
3483             *heightRemaining = 1;
3484 
3485         // We know the list is non-empty, since we have "left" objects to search for.
3486         // Therefore we can assume that begin != end, and that we can do at least one
3487         // decrement.
3488         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3489         FloatingObjectSetIterator begin = floatingObjectSet.begin();
3490         FloatingObjectSetIterator it = floatingObjectSet.end();
3491         do {
3492             --it;
3493             FloatingObject* r = *it;
3494             if (r->isPlaced() && logicalTopForFloat(r) <= logicalTop && logicalBottomForFloat(r) > logicalTop
3495                 && r->type() == FloatingObject::FloatLeft
3496                 && logicalRightForFloat(r) > left) {
3497                 left = max(left, logicalRightForFloat(r));
3498                 if (heightRemaining)
3499                     *heightRemaining = logicalBottomForFloat(r) - logicalTop;
3500             }
3501         } while (it != begin);
3502     }
3503 
3504     if (applyTextIndent && style()->isLeftToRightDirection()) {
3505         int cw = 0;
3506         if (style()->textIndent().isPercent())
3507             cw = containingBlock()->availableLogicalWidth();
3508         left += style()->textIndent().calcMinValue(cw);
3509     }
3510 
3511     return left;
3512 }
3513 
logicalRightOffsetForLine(int logicalTop,int fixedOffset,bool applyTextIndent,int * heightRemaining) const3514 int RenderBlock::logicalRightOffsetForLine(int logicalTop, int fixedOffset, bool applyTextIndent, int* heightRemaining) const
3515 {
3516     int right = fixedOffset;
3517 
3518     if (m_floatingObjects && m_floatingObjects->hasRightObjects()) {
3519         if (heightRemaining)
3520             *heightRemaining = 1;
3521 
3522         // We know the list is non-empty, since we have "right" objects to search for.
3523         // Therefore we can assume that begin != end, and that we can do at least one
3524         // decrement.
3525         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3526         FloatingObjectSetIterator begin = floatingObjectSet.begin();
3527         FloatingObjectSetIterator it = floatingObjectSet.end();
3528         do {
3529             --it;
3530             FloatingObject* r = *it;
3531             if (r->isPlaced() && logicalTopForFloat(r) <= logicalTop && logicalBottomForFloat(r) > logicalTop
3532                 && r->type() == FloatingObject::FloatRight
3533                 && logicalLeftForFloat(r) < right) {
3534                 right = min(right, logicalLeftForFloat(r));
3535                 if (heightRemaining)
3536                     *heightRemaining = logicalBottomForFloat(r) - logicalTop;
3537             }
3538         } while (it != begin);
3539     }
3540 
3541     if (applyTextIndent && !style()->isLeftToRightDirection()) {
3542         int cw = 0;
3543         if (style()->textIndent().isPercent())
3544             cw = containingBlock()->availableLogicalWidth();
3545         right -= style()->textIndent().calcMinValue(cw);
3546     }
3547 
3548     return right;
3549 }
3550 
availableLogicalWidthForLine(int position,bool firstLine) const3551 int RenderBlock::availableLogicalWidthForLine(int position, bool firstLine) const
3552 {
3553     int result = logicalRightOffsetForLine(position, firstLine) - logicalLeftOffsetForLine(position, firstLine);
3554     return (result < 0) ? 0 : result;
3555 }
3556 
nextFloatLogicalBottomBelow(int logicalHeight) const3557 int RenderBlock::nextFloatLogicalBottomBelow(int logicalHeight) const
3558 {
3559     if (!m_floatingObjects)
3560         return 0;
3561 
3562     int bottom = INT_MAX;
3563     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3564     FloatingObjectSetIterator end = floatingObjectSet.end();
3565     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
3566         FloatingObject* r = *it;
3567         int floatBottom = logicalBottomForFloat(r);
3568         if (floatBottom > logicalHeight)
3569             bottom = min(floatBottom, bottom);
3570     }
3571 
3572     return bottom == INT_MAX ? 0 : bottom;
3573 }
3574 
lowestFloatLogicalBottom(FloatingObject::Type floatType) const3575 int RenderBlock::lowestFloatLogicalBottom(FloatingObject::Type floatType) const
3576 {
3577     if (!m_floatingObjects)
3578         return 0;
3579     int lowestFloatBottom = 0;
3580     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3581     FloatingObjectSetIterator end = floatingObjectSet.end();
3582     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
3583         FloatingObject* r = *it;
3584         if (r->isPlaced() && r->type() & floatType)
3585             lowestFloatBottom = max(lowestFloatBottom, logicalBottomForFloat(r));
3586     }
3587     return lowestFloatBottom;
3588 }
3589 
markLinesDirtyInBlockRange(int logicalTop,int logicalBottom,RootInlineBox * highest)3590 void RenderBlock::markLinesDirtyInBlockRange(int logicalTop, int logicalBottom, RootInlineBox* highest)
3591 {
3592     if (logicalTop >= logicalBottom)
3593         return;
3594 
3595     RootInlineBox* lowestDirtyLine = lastRootBox();
3596     RootInlineBox* afterLowest = lowestDirtyLine;
3597     while (lowestDirtyLine && lowestDirtyLine->blockLogicalHeight() >= logicalBottom && logicalBottom < numeric_limits<int>::max()) {
3598         afterLowest = lowestDirtyLine;
3599         lowestDirtyLine = lowestDirtyLine->prevRootBox();
3600     }
3601 
3602     while (afterLowest && afterLowest != highest && (afterLowest->blockLogicalHeight() >= logicalTop || afterLowest->blockLogicalHeight() < 0)) {
3603         afterLowest->markDirty();
3604         afterLowest = afterLowest->prevRootBox();
3605     }
3606 }
3607 
clearFloats()3608 void RenderBlock::clearFloats()
3609 {
3610     // Inline blocks are covered by the isReplaced() check in the avoidFloats method.
3611     if (avoidsFloats() || isRoot() || isRenderView() || isFloatingOrPositioned() || isTableCell()) {
3612         if (m_floatingObjects) {
3613             deleteAllValues(m_floatingObjects->set());
3614             m_floatingObjects->clear();
3615         }
3616         return;
3617     }
3618 
3619     typedef HashMap<RenderObject*, FloatingObject*> RendererToFloatInfoMap;
3620     RendererToFloatInfoMap floatMap;
3621 
3622     if (m_floatingObjects) {
3623         FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3624         if (childrenInline()) {
3625             FloatingObjectSet::iterator end = floatingObjectSet.end();
3626             for (FloatingObjectSet::iterator it = floatingObjectSet.begin(); it != end; ++it) {
3627                 FloatingObject* f = *it;
3628                 floatMap.add(f->m_renderer, f);
3629             }
3630         } else
3631             deleteAllValues(floatingObjectSet);
3632         m_floatingObjects->clear();
3633     }
3634 
3635     // We should not process floats if the parent node is not a RenderBlock. Otherwise, we will add
3636     // floats in an invalid context. This will cause a crash arising from a bad cast on the parent.
3637     // See <rdar://problem/8049753>, where float property is applied on a text node in a SVG.
3638     if (!parent() || !parent()->isRenderBlock())
3639         return;
3640 
3641     // Attempt to locate a previous sibling with overhanging floats.  We skip any elements that are
3642     // out of flow (like floating/positioned elements), and we also skip over any objects that may have shifted
3643     // to avoid floats.
3644     bool parentHasFloats = false;
3645     RenderBlock* parentBlock = toRenderBlock(parent());
3646     RenderObject* prev = previousSibling();
3647     while (prev && (prev->isFloatingOrPositioned() || !prev->isBox() || !prev->isRenderBlock() || toRenderBlock(prev)->avoidsFloats())) {
3648         if (prev->isFloating())
3649             parentHasFloats = true;
3650          prev = prev->previousSibling();
3651     }
3652 
3653     // First add in floats from the parent.
3654     int logicalTopOffset = logicalTop();
3655     if (parentHasFloats)
3656         addIntrudingFloats(parentBlock, parentBlock->logicalLeftOffsetForContent(), logicalTopOffset);
3657 
3658     int logicalLeftOffset = 0;
3659     if (prev)
3660         logicalTopOffset -= toRenderBox(prev)->logicalTop();
3661     else {
3662         prev = parentBlock;
3663         logicalLeftOffset += parentBlock->logicalLeftOffsetForContent();
3664     }
3665 
3666     // Add overhanging floats from the previous RenderBlock, but only if it has a float that intrudes into our space.
3667     if (!prev || !prev->isRenderBlock())
3668         return;
3669 
3670     RenderBlock* block = toRenderBlock(prev);
3671     if (block->m_floatingObjects && block->lowestFloatLogicalBottom() > logicalTopOffset)
3672         addIntrudingFloats(block, logicalLeftOffset, logicalTopOffset);
3673 
3674     if (childrenInline()) {
3675         int changeLogicalTop = numeric_limits<int>::max();
3676         int changeLogicalBottom = numeric_limits<int>::min();
3677         if (m_floatingObjects) {
3678             FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3679             FloatingObjectSetIterator end = floatingObjectSet.end();
3680             for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
3681                 FloatingObject* f = *it;
3682                 FloatingObject* oldFloatingObject = floatMap.get(f->m_renderer);
3683                 int logicalBottom = logicalBottomForFloat(f);
3684                 if (oldFloatingObject) {
3685                     int oldLogicalBottom = logicalBottomForFloat(oldFloatingObject);
3686                     if (logicalWidthForFloat(f) != logicalWidthForFloat(oldFloatingObject) || logicalLeftForFloat(f) != logicalLeftForFloat(oldFloatingObject)) {
3687                         changeLogicalTop = 0;
3688                         changeLogicalBottom = max(changeLogicalBottom, max(logicalBottom, oldLogicalBottom));
3689                     } else if (logicalBottom != oldLogicalBottom) {
3690                         changeLogicalTop = min(changeLogicalTop, min(logicalBottom, oldLogicalBottom));
3691                         changeLogicalBottom = max(changeLogicalBottom, max(logicalBottom, oldLogicalBottom));
3692                     }
3693 
3694                     floatMap.remove(f->m_renderer);
3695                     if (oldFloatingObject->m_originatingLine) {
3696                         ASSERT(oldFloatingObject->m_originatingLine->renderer() == this);
3697                         oldFloatingObject->m_originatingLine->markDirty();
3698                     }
3699                     delete oldFloatingObject;
3700                 } else {
3701                     changeLogicalTop = 0;
3702                     changeLogicalBottom = max(changeLogicalBottom, logicalBottom);
3703                 }
3704             }
3705         }
3706 
3707         RendererToFloatInfoMap::iterator end = floatMap.end();
3708         for (RendererToFloatInfoMap::iterator it = floatMap.begin(); it != end; ++it) {
3709             FloatingObject* floatingObject = (*it).second;
3710             if (!floatingObject->m_isDescendant) {
3711                 changeLogicalTop = 0;
3712                 changeLogicalBottom = max(changeLogicalBottom, logicalBottomForFloat(floatingObject));
3713             }
3714         }
3715         deleteAllValues(floatMap);
3716 
3717         markLinesDirtyInBlockRange(changeLogicalTop, changeLogicalBottom);
3718     }
3719 }
3720 
addOverhangingFloats(RenderBlock * child,int logicalLeftOffset,int logicalTopOffset,bool makeChildPaintOtherFloats)3721 int RenderBlock::addOverhangingFloats(RenderBlock* child, int logicalLeftOffset, int logicalTopOffset, bool makeChildPaintOtherFloats)
3722 {
3723     // Prevent floats from being added to the canvas by the root element, e.g., <html>.
3724     if (child->hasOverflowClip() || !child->containsFloats() || child->isRoot() || child->hasColumns() || child->isWritingModeRoot())
3725         return 0;
3726 
3727     int childLogicalTop = child->logicalTop();
3728     int lowestFloatLogicalBottom = 0;
3729 
3730     // Floats that will remain the child's responsibility to paint should factor into its
3731     // overflow.
3732     FloatingObjectSetIterator childEnd = child->m_floatingObjects->set().end();
3733     for (FloatingObjectSetIterator childIt = child->m_floatingObjects->set().begin(); childIt != childEnd; ++childIt) {
3734         FloatingObject* r = *childIt;
3735         int logicalBottomForFloat = min(this->logicalBottomForFloat(r), numeric_limits<int>::max() - childLogicalTop);
3736         int logicalBottom = childLogicalTop + logicalBottomForFloat;
3737         lowestFloatLogicalBottom = max(lowestFloatLogicalBottom, logicalBottom);
3738 
3739         if (logicalBottom > logicalHeight()) {
3740             // If the object is not in the list, we add it now.
3741             if (!containsFloat(r->m_renderer)) {
3742                 int leftOffset = isHorizontalWritingMode() ? logicalLeftOffset : logicalTopOffset;
3743                 int topOffset = isHorizontalWritingMode() ? logicalTopOffset : logicalLeftOffset;
3744                 FloatingObject* floatingObj = new FloatingObject(r->type(), IntRect(r->x() - leftOffset, r->y() - topOffset, r->width(), r->height()));
3745                 floatingObj->m_renderer = r->m_renderer;
3746 
3747                 // The nearest enclosing layer always paints the float (so that zindex and stacking
3748                 // behaves properly).  We always want to propagate the desire to paint the float as
3749                 // far out as we can, to the outermost block that overlaps the float, stopping only
3750                 // if we hit a self-painting layer boundary.
3751                 if (r->m_renderer->enclosingFloatPaintingLayer() == enclosingFloatPaintingLayer())
3752                     r->m_shouldPaint = false;
3753                 else
3754                     floatingObj->m_shouldPaint = false;
3755 
3756                 floatingObj->m_isDescendant = true;
3757 
3758                 // We create the floating object list lazily.
3759                 if (!m_floatingObjects)
3760                     m_floatingObjects = adoptPtr(new FloatingObjects);
3761 
3762                 m_floatingObjects->increaseObjectsCount(floatingObj->type());
3763                 m_floatingObjects->set().add(floatingObj);
3764             }
3765         } else {
3766             if (makeChildPaintOtherFloats && !r->m_shouldPaint && !r->m_renderer->hasSelfPaintingLayer() &&
3767                 r->m_renderer->isDescendantOf(child) && r->m_renderer->enclosingFloatPaintingLayer() == child->enclosingFloatPaintingLayer()) {
3768                 // The float is not overhanging from this block, so if it is a descendant of the child, the child should
3769                 // paint it (the other case is that it is intruding into the child), unless it has its own layer or enclosing
3770                 // layer.
3771                 // If makeChildPaintOtherFloats is false, it means that the child must already know about all the floats
3772                 // it should paint.
3773                 r->m_shouldPaint = true;
3774             }
3775 
3776             // Since the float doesn't overhang, it didn't get put into our list.  We need to go ahead and add its overflow in to the
3777             // child now.
3778             if (r->m_isDescendant)
3779                 child->addOverflowFromChild(r->m_renderer, IntSize(xPositionForFloatIncludingMargin(r), yPositionForFloatIncludingMargin(r)));
3780         }
3781     }
3782     return lowestFloatLogicalBottom;
3783 }
3784 
hasOverhangingFloat(RenderBox * renderer)3785 bool RenderBlock::hasOverhangingFloat(RenderBox* renderer)
3786 {
3787     if (!m_floatingObjects || hasColumns() || !parent())
3788         return false;
3789 
3790     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3791     FloatingObjectSetIterator it = floatingObjectSet.find<RenderBox*, FloatingObjectHashTranslator>(renderer);
3792     if (it == floatingObjectSet.end())
3793         return false;
3794 
3795     return logicalBottomForFloat(*it) > logicalHeight();
3796 }
3797 
addIntrudingFloats(RenderBlock * prev,int logicalLeftOffset,int logicalTopOffset)3798 void RenderBlock::addIntrudingFloats(RenderBlock* prev, int logicalLeftOffset, int logicalTopOffset)
3799 {
3800     // If the parent or previous sibling doesn't have any floats to add, don't bother.
3801     if (!prev->m_floatingObjects)
3802         return;
3803 
3804     logicalLeftOffset += (isHorizontalWritingMode() ? marginLeft() : marginTop());
3805 
3806     FloatingObjectSet& prevSet = prev->m_floatingObjects->set();
3807     FloatingObjectSetIterator prevEnd = prevSet.end();
3808     for (FloatingObjectSetIterator prevIt = prevSet.begin(); prevIt != prevEnd; ++prevIt) {
3809         FloatingObject* r = *prevIt;
3810         if (logicalBottomForFloat(r) > logicalTopOffset) {
3811             if (!m_floatingObjects || !m_floatingObjects->set().contains(r)) {
3812                 int leftOffset = isHorizontalWritingMode() ? logicalLeftOffset : logicalTopOffset;
3813                 int topOffset = isHorizontalWritingMode() ? logicalTopOffset : logicalLeftOffset;
3814 
3815                 FloatingObject* floatingObj = new FloatingObject(r->type(), IntRect(r->x() - leftOffset, r->y() - topOffset, r->width(), r->height()));
3816 
3817                 // Applying the child's margin makes no sense in the case where the child was passed in.
3818                 // since this margin was added already through the modification of the |logicalLeftOffset| variable
3819                 // above.  |logicalLeftOffset| will equal the margin in this case, so it's already been taken
3820                 // into account.  Only apply this code if prev is the parent, since otherwise the left margin
3821                 // will get applied twice.
3822                 if (prev != parent()) {
3823                     if (isHorizontalWritingMode())
3824                         floatingObj->setX(floatingObj->x() + prev->marginLeft());
3825                     else
3826                         floatingObj->setY(floatingObj->y() + prev->marginTop());
3827                 }
3828 
3829                 floatingObj->m_shouldPaint = false;  // We are not in the direct inheritance chain for this float. We will never paint it.
3830                 floatingObj->m_renderer = r->m_renderer;
3831 
3832                 // We create the floating object list lazily.
3833                 if (!m_floatingObjects)
3834                     m_floatingObjects = adoptPtr(new FloatingObjects);
3835                 m_floatingObjects->increaseObjectsCount(floatingObj->type());
3836                 m_floatingObjects->set().add(floatingObj);
3837             }
3838         }
3839     }
3840 }
3841 
avoidsFloats() const3842 bool RenderBlock::avoidsFloats() const
3843 {
3844     // Floats can't intrude into our box if we have a non-auto column count or width.
3845     return RenderBox::avoidsFloats() || !style()->hasAutoColumnCount() || !style()->hasAutoColumnWidth();
3846 }
3847 
containsFloat(RenderBox * renderer)3848 bool RenderBlock::containsFloat(RenderBox* renderer)
3849 {
3850     return m_floatingObjects && m_floatingObjects->set().contains<RenderBox*, FloatingObjectHashTranslator>(renderer);
3851 }
3852 
markAllDescendantsWithFloatsForLayout(RenderBox * floatToRemove,bool inLayout)3853 void RenderBlock::markAllDescendantsWithFloatsForLayout(RenderBox* floatToRemove, bool inLayout)
3854 {
3855     if (!m_everHadLayout)
3856         return;
3857 
3858     setChildNeedsLayout(true, !inLayout);
3859 
3860     if (floatToRemove)
3861         removeFloatingObject(floatToRemove);
3862 
3863     // Iterate over our children and mark them as needed.
3864     if (!childrenInline()) {
3865         for (RenderObject* child = firstChild(); child; child = child->nextSibling()) {
3866             if ((!floatToRemove && child->isFloatingOrPositioned()) || !child->isRenderBlock())
3867                 continue;
3868             RenderBlock* childBlock = toRenderBlock(child);
3869             if ((floatToRemove ? childBlock->containsFloat(floatToRemove) : childBlock->containsFloats()) || childBlock->shrinkToAvoidFloats())
3870                 childBlock->markAllDescendantsWithFloatsForLayout(floatToRemove, inLayout);
3871         }
3872     }
3873 }
3874 
markSiblingsWithFloatsForLayout()3875 void RenderBlock::markSiblingsWithFloatsForLayout()
3876 {
3877     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3878     FloatingObjectSetIterator end = floatingObjectSet.end();
3879     for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
3880         if (logicalBottomForFloat(*it) > logicalHeight()) {
3881             RenderBox* floatingBox = (*it)->renderer();
3882 
3883             RenderObject* next = nextSibling();
3884             while (next) {
3885                 if (next->isRenderBlock() && !next->isFloatingOrPositioned() && !toRenderBlock(next)->avoidsFloats()) {
3886                     RenderBlock* nextBlock = toRenderBlock(next);
3887                     if (nextBlock->containsFloat(floatingBox))
3888                         nextBlock->markAllDescendantsWithFloatsForLayout(floatingBox);
3889                     else
3890                         break;
3891                 }
3892 
3893                 next = next->nextSibling();
3894             }
3895         }
3896     }
3897 }
3898 
getClearDelta(RenderBox * child,int yPos)3899 int RenderBlock::getClearDelta(RenderBox* child, int yPos)
3900 {
3901     // There is no need to compute clearance if we have no floats.
3902     if (!containsFloats())
3903         return 0;
3904 
3905     // At least one float is present.  We need to perform the clearance computation.
3906     bool clearSet = child->style()->clear() != CNONE;
3907     int bottom = 0;
3908     switch (child->style()->clear()) {
3909         case CNONE:
3910             break;
3911         case CLEFT:
3912             bottom = lowestFloatLogicalBottom(FloatingObject::FloatLeft);
3913             break;
3914         case CRIGHT:
3915             bottom = lowestFloatLogicalBottom(FloatingObject::FloatRight);
3916             break;
3917         case CBOTH:
3918             bottom = lowestFloatLogicalBottom();
3919             break;
3920     }
3921 
3922     // We also clear floats if we are too big to sit on the same line as a float (and wish to avoid floats by default).
3923     int result = clearSet ? max(0, bottom - yPos) : 0;
3924     if (!result && child->avoidsFloats()) {
3925         int y = yPos;
3926         while (true) {
3927             int widthAtY = availableLogicalWidthForLine(y, false);
3928             if (widthAtY == availableLogicalWidth())
3929                 return y - yPos;
3930 
3931             int oldChildY = child->y();
3932             int oldChildWidth = child->width();
3933             child->setY(y);
3934             child->computeLogicalWidth();
3935             int childWidthAtY = child->width();
3936             child->setY(oldChildY);
3937             child->setWidth(oldChildWidth);
3938 
3939             if (childWidthAtY <= widthAtY)
3940                 return y - yPos;
3941 
3942             y = nextFloatLogicalBottomBelow(y);
3943             ASSERT(y >= yPos);
3944             if (y < yPos)
3945                 break;
3946         }
3947         ASSERT_NOT_REACHED();
3948     }
3949     return result;
3950 }
3951 
isPointInOverflowControl(HitTestResult & result,int _x,int _y,int _tx,int _ty)3952 bool RenderBlock::isPointInOverflowControl(HitTestResult& result, int _x, int _y, int _tx, int _ty)
3953 {
3954     if (!scrollsOverflow())
3955         return false;
3956 
3957     return layer()->hitTestOverflowControls(result, IntPoint(_x - _tx, _y - _ty));
3958 }
3959 
nodeAtPoint(const HitTestRequest & request,HitTestResult & result,int _x,int _y,int _tx,int _ty,HitTestAction hitTestAction)3960 bool RenderBlock::nodeAtPoint(const HitTestRequest& request, HitTestResult& result, int _x, int _y, int _tx, int _ty, HitTestAction hitTestAction)
3961 {
3962     int tx = _tx + x();
3963     int ty = _ty + y();
3964 
3965     if (!isRenderView()) {
3966         // Check if we need to do anything at all.
3967         IntRect overflowBox = visualOverflowRect();
3968         overflowBox.move(tx, ty);
3969         if (!overflowBox.intersects(result.rectForPoint(_x, _y)))
3970             return false;
3971     }
3972 
3973     if ((hitTestAction == HitTestBlockBackground || hitTestAction == HitTestChildBlockBackground) && isPointInOverflowControl(result, _x, _y, tx, ty)) {
3974         updateHitTestResult(result, IntPoint(_x - tx, _y - ty));
3975         // FIXME: isPointInOverflowControl() doesn't handle rect-based tests yet.
3976         if (!result.addNodeToRectBasedTestResult(node(), _x, _y))
3977            return true;
3978     }
3979 
3980     // If we have clipping, then we can't have any spillout.
3981     bool useOverflowClip = hasOverflowClip() && !hasSelfPaintingLayer();
3982     bool useClip = (hasControlClip() || useOverflowClip);
3983     IntRect hitTestArea(result.rectForPoint(_x, _y));
3984     bool checkChildren = !useClip || (hasControlClip() ? controlClipRect(tx, ty).intersects(hitTestArea) : overflowClipRect(tx, ty, IncludeOverlayScrollbarSize).intersects(hitTestArea));
3985     if (checkChildren) {
3986         // Hit test descendants first.
3987         int scrolledX = tx;
3988         int scrolledY = ty;
3989         if (hasOverflowClip()) {
3990             IntSize offset = layer()->scrolledContentOffset();
3991             scrolledX -= offset.width();
3992             scrolledY -= offset.height();
3993         }
3994 
3995         // Hit test contents if we don't have columns.
3996         if (!hasColumns()) {
3997             if (hitTestContents(request, result, _x, _y, scrolledX, scrolledY, hitTestAction)) {
3998                 updateHitTestResult(result, IntPoint(_x - tx, _y - ty));
3999                 return true;
4000             }
4001             if (hitTestAction == HitTestFloat && hitTestFloats(request, result, _x, _y, scrolledX, scrolledY))
4002                 return true;
4003         } else if (hitTestColumns(request, result, _x, _y, scrolledX, scrolledY, hitTestAction)) {
4004             updateHitTestResult(result, IntPoint(_x - tx, _y - ty));
4005             return true;
4006         }
4007     }
4008 
4009     // Now hit test our background
4010     if (hitTestAction == HitTestBlockBackground || hitTestAction == HitTestChildBlockBackground) {
4011         IntRect boundsRect(tx, ty, width(), height());
4012         if (visibleToHitTesting() && boundsRect.intersects(result.rectForPoint(_x, _y))) {
4013             updateHitTestResult(result, flipForWritingMode(IntPoint(_x - tx, _y - ty)));
4014             if (!result.addNodeToRectBasedTestResult(node(), _x, _y, boundsRect))
4015                 return true;
4016         }
4017     }
4018 
4019     return false;
4020 }
4021 
hitTestFloats(const HitTestRequest & request,HitTestResult & result,int x,int y,int tx,int ty)4022 bool RenderBlock::hitTestFloats(const HitTestRequest& request, HitTestResult& result, int x, int y, int tx, int ty)
4023 {
4024     if (!m_floatingObjects)
4025         return false;
4026 
4027     if (isRenderView()) {
4028         tx += toRenderView(this)->frameView()->scrollX();
4029         ty += toRenderView(this)->frameView()->scrollY();
4030     }
4031 
4032     FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
4033     FloatingObjectSetIterator begin = floatingObjectSet.begin();
4034     for (FloatingObjectSetIterator it = floatingObjectSet.end(); it != begin;) {
4035         --it;
4036         FloatingObject* floatingObject = *it;
4037         if (floatingObject->m_shouldPaint && !floatingObject->m_renderer->hasSelfPaintingLayer()) {
4038             int xOffset = xPositionForFloatIncludingMargin(floatingObject) - floatingObject->m_renderer->x();
4039             int yOffset = yPositionForFloatIncludingMargin(floatingObject) - floatingObject->m_renderer->y();
4040             IntPoint childPoint = flipFloatForWritingMode(floatingObject, IntPoint(tx + xOffset, ty + yOffset));
4041             if (floatingObject->m_renderer->hitTest(request, result, IntPoint(x, y), childPoint.x(), childPoint.y())) {
4042                 updateHitTestResult(result, IntPoint(x - childPoint.x(), y - childPoint.y()));
4043                 return true;
4044             }
4045         }
4046     }
4047 
4048     return false;
4049 }
4050 
hitTestColumns(const HitTestRequest & request,HitTestResult & result,int x,int y,int tx,int ty,HitTestAction hitTestAction)4051 bool RenderBlock::hitTestColumns(const HitTestRequest& request, HitTestResult& result, int x, int y, int tx, int ty, HitTestAction hitTestAction)
4052 {
4053     // We need to do multiple passes, breaking up our hit testing into strips.
4054     ColumnInfo* colInfo = columnInfo();
4055     int colCount = columnCount(colInfo);
4056     if (!colCount)
4057         return false;
4058     int logicalLeft = logicalLeftOffsetForContent();
4059     int currLogicalTopOffset = 0;
4060     int i;
4061     bool isHorizontal = isHorizontalWritingMode();
4062     for (i = 0; i < colCount; i++) {
4063         IntRect colRect = columnRectAt(colInfo, i);
4064         int blockDelta =  (isHorizontal ? colRect.height() : colRect.width());
4065         if (style()->isFlippedBlocksWritingMode())
4066             currLogicalTopOffset += blockDelta;
4067         else
4068             currLogicalTopOffset -= blockDelta;
4069     }
4070     for (i = colCount - 1; i >= 0; i--) {
4071         IntRect colRect = columnRectAt(colInfo, i);
4072         flipForWritingMode(colRect);
4073         int currLogicalLeftOffset = (isHorizontal ? colRect.x() : colRect.y()) - logicalLeft;
4074         int blockDelta =  (isHorizontal ? colRect.height() : colRect.width());
4075         if (style()->isFlippedBlocksWritingMode())
4076             currLogicalTopOffset -= blockDelta;
4077         else
4078             currLogicalTopOffset += blockDelta;
4079         colRect.move(tx, ty);
4080 
4081         if (colRect.intersects(result.rectForPoint(x, y))) {
4082             // The point is inside this column.
4083             // Adjust tx and ty to change where we hit test.
4084 
4085             IntSize offset = isHorizontal ? IntSize(currLogicalLeftOffset, currLogicalTopOffset) : IntSize(currLogicalTopOffset, currLogicalLeftOffset);
4086             int finalX = tx + offset.width();
4087             int finalY = ty + offset.height();
4088             if (result.isRectBasedTest() && !colRect.contains(result.rectForPoint(x, y)))
4089                 hitTestContents(request, result, x, y, finalX, finalY, hitTestAction);
4090             else
4091                 return hitTestContents(request, result, x, y, finalX, finalY, hitTestAction) || (hitTestAction == HitTestFloat && hitTestFloats(request, result, x, y, finalX, finalY));
4092         }
4093     }
4094 
4095     return false;
4096 }
4097 
hitTestContents(const HitTestRequest & request,HitTestResult & result,int x,int y,int tx,int ty,HitTestAction hitTestAction)4098 bool RenderBlock::hitTestContents(const HitTestRequest& request, HitTestResult& result, int x, int y, int tx, int ty, HitTestAction hitTestAction)
4099 {
4100     if (childrenInline() && !isTable()) {
4101         // We have to hit-test our line boxes.
4102         if (m_lineBoxes.hitTest(this, request, result, x, y, tx, ty, hitTestAction))
4103             return true;
4104     } else {
4105         // Hit test our children.
4106         HitTestAction childHitTest = hitTestAction;
4107         if (hitTestAction == HitTestChildBlockBackgrounds)
4108             childHitTest = HitTestChildBlockBackground;
4109         for (RenderBox* child = lastChildBox(); child; child = child->previousSiblingBox()) {
4110             IntPoint childPoint = flipForWritingMode(child, IntPoint(tx, ty), ParentToChildFlippingAdjustment);
4111             if (!child->hasSelfPaintingLayer() && !child->isFloating() && child->nodeAtPoint(request, result, x, y, childPoint.x(), childPoint.y(), childHitTest))
4112                 return true;
4113         }
4114     }
4115 
4116     return false;
4117 }
4118 
positionForBox(InlineBox * box,bool start) const4119 Position RenderBlock::positionForBox(InlineBox *box, bool start) const
4120 {
4121     if (!box)
4122         return Position();
4123 
4124     if (!box->renderer()->node())
4125         return Position(node(), start ? caretMinOffset() : caretMaxOffset());
4126 
4127     if (!box->isInlineTextBox())
4128         return Position(box->renderer()->node(), start ? box->renderer()->caretMinOffset() : box->renderer()->caretMaxOffset());
4129 
4130     InlineTextBox *textBox = static_cast<InlineTextBox *>(box);
4131     return Position(box->renderer()->node(), start ? textBox->start() : textBox->start() + textBox->len());
4132 }
4133 
4134 // FIXME: This function should go on RenderObject as an instance method. Then
4135 // all cases in which positionForPoint recurs could call this instead to
4136 // prevent crossing editable boundaries. This would require many tests.
positionForPointRespectingEditingBoundaries(RenderBlock * parent,RenderBox * child,const IntPoint & pointInParentCoordinates)4137 static VisiblePosition positionForPointRespectingEditingBoundaries(RenderBlock* parent, RenderBox* child, const IntPoint& pointInParentCoordinates)
4138 {
4139     // FIXME: This is wrong if the child's writing-mode is different from the parent's.
4140     IntPoint pointInChildCoordinates(pointInParentCoordinates - child->location());
4141 
4142     // If this is an anonymous renderer, we just recur normally
4143     Node* childNode = child->node();
4144     if (!childNode)
4145         return child->positionForPoint(pointInChildCoordinates);
4146 
4147     // Otherwise, first make sure that the editability of the parent and child agree.
4148     // If they don't agree, then we return a visible position just before or after the child
4149     RenderObject* ancestor = parent;
4150     while (ancestor && !ancestor->node())
4151         ancestor = ancestor->parent();
4152 
4153     // If we can't find an ancestor to check editability on, or editability is unchanged, we recur like normal
4154     if (!ancestor || ancestor->node()->rendererIsEditable() == childNode->rendererIsEditable())
4155         return child->positionForPoint(pointInChildCoordinates);
4156 
4157     // Otherwise return before or after the child, depending on if the click was to the logical left or logical right of the child
4158     int childMiddle = parent->logicalWidthForChild(child) / 2;
4159     int logicalLeft = parent->isHorizontalWritingMode() ? pointInChildCoordinates.x() : pointInChildCoordinates.y();
4160     if (logicalLeft < childMiddle)
4161         return ancestor->createVisiblePosition(childNode->nodeIndex(), DOWNSTREAM);
4162     return ancestor->createVisiblePosition(childNode->nodeIndex() + 1, UPSTREAM);
4163 }
4164 
positionForPointWithInlineChildren(const IntPoint & pointInLogicalContents)4165 VisiblePosition RenderBlock::positionForPointWithInlineChildren(const IntPoint& pointInLogicalContents)
4166 {
4167     ASSERT(childrenInline());
4168 
4169     if (!firstRootBox())
4170         return createVisiblePosition(0, DOWNSTREAM);
4171 
4172     // look for the closest line box in the root box which is at the passed-in y coordinate
4173     InlineBox* closestBox = 0;
4174     RootInlineBox* firstRootBoxWithChildren = 0;
4175     RootInlineBox* lastRootBoxWithChildren = 0;
4176     for (RootInlineBox* root = firstRootBox(); root; root = root->nextRootBox()) {
4177         if (!root->firstLeafChild())
4178             continue;
4179         if (!firstRootBoxWithChildren)
4180             firstRootBoxWithChildren = root;
4181         lastRootBoxWithChildren = root;
4182 
4183         // check if this root line box is located at this y coordinate
4184         if (pointInLogicalContents.y() < root->selectionBottom()) {
4185             closestBox = root->closestLeafChildForLogicalLeftPosition(pointInLogicalContents.x());
4186             if (closestBox)
4187                 break;
4188         }
4189     }
4190 
4191     bool moveCaretToBoundary = document()->frame()->editor()->behavior().shouldMoveCaretToHorizontalBoundaryWhenPastTopOrBottom();
4192 
4193     if (!moveCaretToBoundary && !closestBox && lastRootBoxWithChildren) {
4194         // y coordinate is below last root line box, pretend we hit it
4195         closestBox = lastRootBoxWithChildren->closestLeafChildForLogicalLeftPosition(pointInLogicalContents.x());
4196     }
4197 
4198     if (closestBox) {
4199         if (moveCaretToBoundary && pointInLogicalContents.y() < firstRootBoxWithChildren->selectionTop()) {
4200             // y coordinate is above first root line box, so return the start of the first
4201             return VisiblePosition(positionForBox(firstRootBoxWithChildren->firstLeafChild(), true), DOWNSTREAM);
4202         }
4203 
4204         // pass the box a top position that is inside it
4205         IntPoint point(pointInLogicalContents.x(), closestBox->logicalTop());
4206         if (!isHorizontalWritingMode())
4207             point = point.transposedPoint();
4208         if (closestBox->renderer()->isReplaced())
4209             return positionForPointRespectingEditingBoundaries(this, toRenderBox(closestBox->renderer()), point);
4210         return closestBox->renderer()->positionForPoint(point);
4211     }
4212 
4213     if (lastRootBoxWithChildren) {
4214         // We hit this case for Mac behavior when the Y coordinate is below the last box.
4215         ASSERT(moveCaretToBoundary);
4216         InlineBox* logicallyLastBox;
4217         if (lastRootBoxWithChildren->getLogicalEndBoxWithNode(logicallyLastBox))
4218             return VisiblePosition(positionForBox(logicallyLastBox, false), DOWNSTREAM);
4219     }
4220 
4221     // Can't reach this. We have a root line box, but it has no kids.
4222     // FIXME: This should ASSERT_NOT_REACHED(), but clicking on placeholder text
4223     // seems to hit this code path.
4224     return createVisiblePosition(0, DOWNSTREAM);
4225 }
4226 
isChildHitTestCandidate(RenderBox * box)4227 static inline bool isChildHitTestCandidate(RenderBox* box)
4228 {
4229     return box->height() && box->style()->visibility() == VISIBLE && !box->isFloatingOrPositioned();
4230 }
4231 
positionForPoint(const IntPoint & point)4232 VisiblePosition RenderBlock::positionForPoint(const IntPoint& point)
4233 {
4234     if (isTable())
4235         return RenderBox::positionForPoint(point);
4236 
4237     if (isReplaced()) {
4238         // FIXME: This seems wrong when the object's writing-mode doesn't match the line's writing-mode.
4239         int pointLogicalLeft = isHorizontalWritingMode() ? point.x() : point.y();
4240         int pointLogicalTop = isHorizontalWritingMode() ? point.y() : point.x();
4241 
4242         if (pointLogicalTop < 0 || (pointLogicalTop < logicalHeight() && pointLogicalLeft < 0))
4243             return createVisiblePosition(caretMinOffset(), DOWNSTREAM);
4244         if (pointLogicalTop >= logicalHeight() || (pointLogicalTop >= 0 && pointLogicalLeft >= logicalWidth()))
4245             return createVisiblePosition(caretMaxOffset(), DOWNSTREAM);
4246     }
4247 
4248     int contentsX = point.x();
4249     int contentsY = point.y();
4250     offsetForContents(contentsX, contentsY);
4251     IntPoint pointInContents(contentsX, contentsY);
4252     IntPoint pointInLogicalContents(pointInContents);
4253     if (!isHorizontalWritingMode())
4254         pointInLogicalContents = pointInLogicalContents.transposedPoint();
4255 
4256     if (childrenInline())
4257         return positionForPointWithInlineChildren(pointInLogicalContents);
4258 
4259     if (lastChildBox() && pointInContents.y() > lastChildBox()->logicalTop()) {
4260         for (RenderBox* childBox = lastChildBox(); childBox; childBox = childBox->previousSiblingBox()) {
4261             if (isChildHitTestCandidate(childBox))
4262                 return positionForPointRespectingEditingBoundaries(this, childBox, pointInContents);
4263         }
4264     } else {
4265         for (RenderBox* childBox = firstChildBox(); childBox; childBox = childBox->nextSiblingBox()) {
4266             // We hit child if our click is above the bottom of its padding box (like IE6/7 and FF3).
4267             if (isChildHitTestCandidate(childBox) && pointInContents.y() < childBox->logicalBottom())
4268                 return positionForPointRespectingEditingBoundaries(this, childBox, pointInContents);
4269         }
4270     }
4271 
4272     // We only get here if there are no hit test candidate children below the click.
4273     return RenderBox::positionForPoint(point);
4274 }
4275 
offsetForContents(int & tx,int & ty) const4276 void RenderBlock::offsetForContents(int& tx, int& ty) const
4277 {
4278     IntPoint contentsPoint(tx, ty);
4279 
4280     if (hasOverflowClip())
4281         contentsPoint += layer()->scrolledContentOffset();
4282 
4283     if (hasColumns())
4284         adjustPointToColumnContents(contentsPoint);
4285 
4286     tx = contentsPoint.x();
4287     ty = contentsPoint.y();
4288 }
4289 
availableLogicalWidth() const4290 int RenderBlock::availableLogicalWidth() const
4291 {
4292     // If we have multiple columns, then the available logical width is reduced to our column width.
4293     if (hasColumns())
4294         return desiredColumnWidth();
4295     return RenderBox::availableLogicalWidth();
4296 }
4297 
columnGap() const4298 int RenderBlock::columnGap() const
4299 {
4300     if (style()->hasNormalColumnGap())
4301         return style()->fontDescription().computedPixelSize(); // "1em" is recommended as the normal gap setting. Matches <p> margins.
4302     return static_cast<int>(style()->columnGap());
4303 }
4304 
calcColumnWidth()4305 void RenderBlock::calcColumnWidth()
4306 {
4307     // Calculate our column width and column count.
4308     unsigned desiredColumnCount = 1;
4309     int desiredColumnWidth = contentLogicalWidth();
4310 
4311     // For now, we don't support multi-column layouts when printing, since we have to do a lot of work for proper pagination.
4312     if (document()->paginated() || (style()->hasAutoColumnCount() && style()->hasAutoColumnWidth())) {
4313         setDesiredColumnCountAndWidth(desiredColumnCount, desiredColumnWidth);
4314         return;
4315     }
4316 
4317     int availWidth = desiredColumnWidth;
4318     int colGap = columnGap();
4319     int colWidth = max(1, static_cast<int>(style()->columnWidth()));
4320     int colCount = max(1, static_cast<int>(style()->columnCount()));
4321 
4322     if (style()->hasAutoColumnWidth() && !style()->hasAutoColumnCount()) {
4323         desiredColumnCount = colCount;
4324         desiredColumnWidth = max<int>(0, (availWidth - ((desiredColumnCount - 1) * colGap)) / desiredColumnCount);
4325     } else if (!style()->hasAutoColumnWidth() && style()->hasAutoColumnCount()) {
4326         desiredColumnCount = max<int>(1, (float)(availWidth + colGap) / (colWidth + colGap));
4327         desiredColumnWidth = ((availWidth + colGap) / desiredColumnCount) - colGap;
4328     } else {
4329         desiredColumnCount = max(min<int>(colCount, (float)(availWidth + colGap) / (colWidth + colGap)), 1);
4330         desiredColumnWidth = ((availWidth + colGap) / desiredColumnCount) - colGap;
4331     }
4332     setDesiredColumnCountAndWidth(desiredColumnCount, desiredColumnWidth);
4333 }
4334 
setDesiredColumnCountAndWidth(int count,int width)4335 void RenderBlock::setDesiredColumnCountAndWidth(int count, int width)
4336 {
4337     bool destroyColumns = !firstChild()
4338                           || (count == 1 && style()->hasAutoColumnWidth())
4339                           || firstChild()->isAnonymousColumnsBlock()
4340                           || firstChild()->isAnonymousColumnSpanBlock();
4341     if (destroyColumns) {
4342         if (hasColumns()) {
4343             delete gColumnInfoMap->take(this);
4344             setHasColumns(false);
4345         }
4346     } else {
4347         ColumnInfo* info;
4348         if (hasColumns())
4349             info = gColumnInfoMap->get(this);
4350         else {
4351             if (!gColumnInfoMap)
4352                 gColumnInfoMap = new ColumnInfoMap;
4353             info = new ColumnInfo;
4354             gColumnInfoMap->add(this, info);
4355             setHasColumns(true);
4356         }
4357         info->setDesiredColumnCount(count);
4358         info->setDesiredColumnWidth(width);
4359     }
4360 }
4361 
desiredColumnWidth() const4362 int RenderBlock::desiredColumnWidth() const
4363 {
4364     if (!hasColumns())
4365         return contentLogicalWidth();
4366     return gColumnInfoMap->get(this)->desiredColumnWidth();
4367 }
4368 
desiredColumnCount() const4369 unsigned RenderBlock::desiredColumnCount() const
4370 {
4371     if (!hasColumns())
4372         return 1;
4373     return gColumnInfoMap->get(this)->desiredColumnCount();
4374 }
4375 
columnInfo() const4376 ColumnInfo* RenderBlock::columnInfo() const
4377 {
4378     if (!hasColumns())
4379         return 0;
4380     return gColumnInfoMap->get(this);
4381 }
4382 
columnCount(ColumnInfo * colInfo) const4383 unsigned RenderBlock::columnCount(ColumnInfo* colInfo) const
4384 {
4385     ASSERT(hasColumns() && gColumnInfoMap->get(this) == colInfo);
4386     return colInfo->columnCount();
4387 }
4388 
columnRectAt(ColumnInfo * colInfo,unsigned index) const4389 IntRect RenderBlock::columnRectAt(ColumnInfo* colInfo, unsigned index) const
4390 {
4391     ASSERT(hasColumns() && gColumnInfoMap->get(this) == colInfo);
4392 
4393     // Compute the appropriate rect based off our information.
4394     int colLogicalWidth = colInfo->desiredColumnWidth();
4395     int colLogicalHeight = colInfo->columnHeight();
4396     int colLogicalTop = borderBefore() + paddingBefore();
4397     int colGap = columnGap();
4398     int colLogicalLeft = style()->isLeftToRightDirection() ?
4399                           logicalLeftOffsetForContent() + (index * (colLogicalWidth + colGap))
4400                         : logicalLeftOffsetForContent() + contentLogicalWidth() - colLogicalWidth - (index * (colLogicalWidth + colGap));
4401     IntRect rect(colLogicalLeft, colLogicalTop, colLogicalWidth, colLogicalHeight);
4402     if (isHorizontalWritingMode())
4403         return IntRect(colLogicalLeft, colLogicalTop, colLogicalWidth, colLogicalHeight);
4404     return IntRect(colLogicalTop, colLogicalLeft, colLogicalHeight, colLogicalWidth);
4405 }
4406 
layoutColumns(bool hasSpecifiedPageLogicalHeight,int pageLogicalHeight,LayoutStateMaintainer & statePusher)4407 bool RenderBlock::layoutColumns(bool hasSpecifiedPageLogicalHeight, int pageLogicalHeight, LayoutStateMaintainer& statePusher)
4408 {
4409     if (!hasColumns())
4410         return false;
4411 
4412     // FIXME: We don't balance properly at all in the presence of forced page breaks.  We need to understand what
4413     // the distance between forced page breaks is so that we can avoid making the minimum column height too tall.
4414     ColumnInfo* colInfo = columnInfo();
4415     int desiredColumnCount = colInfo->desiredColumnCount();
4416     if (!hasSpecifiedPageLogicalHeight) {
4417         int columnHeight = pageLogicalHeight;
4418         int minColumnCount = colInfo->forcedBreaks() + 1;
4419         if (minColumnCount >= desiredColumnCount) {
4420             // The forced page breaks are in control of the balancing.  Just set the column height to the
4421             // maximum page break distance.
4422             if (!pageLogicalHeight) {
4423                 int distanceBetweenBreaks = max(colInfo->maximumDistanceBetweenForcedBreaks(),
4424                                                 view()->layoutState()->pageLogicalOffset(borderBefore() + paddingBefore() + contentLogicalHeight()) - colInfo->forcedBreakOffset());
4425                 columnHeight = max(colInfo->minimumColumnHeight(), distanceBetweenBreaks);
4426             }
4427         } else if (contentLogicalHeight() > pageLogicalHeight * desiredColumnCount) {
4428             // Now that we know the intrinsic height of the columns, we have to rebalance them.
4429             columnHeight = max(colInfo->minimumColumnHeight(), (int)ceilf((float)contentLogicalHeight() / desiredColumnCount));
4430         }
4431 
4432         if (columnHeight && columnHeight != pageLogicalHeight) {
4433             statePusher.pop();
4434             m_everHadLayout = true;
4435             layoutBlock(false, columnHeight);
4436             return true;
4437         }
4438     }
4439 
4440     if (pageLogicalHeight)
4441         colInfo->setColumnCountAndHeight(ceilf((float)contentLogicalHeight() / pageLogicalHeight), pageLogicalHeight);
4442 
4443     if (columnCount(colInfo)) {
4444         setLogicalHeight(borderBefore() + paddingBefore() + colInfo->columnHeight() + borderAfter() + paddingAfter() + scrollbarLogicalHeight());
4445         m_overflow.clear();
4446     }
4447 
4448     return false;
4449 }
4450 
adjustPointToColumnContents(IntPoint & point) const4451 void RenderBlock::adjustPointToColumnContents(IntPoint& point) const
4452 {
4453     // Just bail if we have no columns.
4454     if (!hasColumns())
4455         return;
4456 
4457     ColumnInfo* colInfo = columnInfo();
4458     if (!columnCount(colInfo))
4459         return;
4460 
4461     // Determine which columns we intersect.
4462     int colGap = columnGap();
4463     int halfColGap = colGap / 2;
4464     IntPoint columnPoint(columnRectAt(colInfo, 0).location());
4465     int logicalOffset = 0;
4466     for (unsigned i = 0; i < colInfo->columnCount(); i++) {
4467         // Add in half the column gap to the left and right of the rect.
4468         IntRect colRect = columnRectAt(colInfo, i);
4469         if (isHorizontalWritingMode()) {
4470             IntRect gapAndColumnRect(colRect.x() - halfColGap, colRect.y(), colRect.width() + colGap, colRect.height());
4471             if (point.x() >= gapAndColumnRect.x() && point.x() < gapAndColumnRect.maxX()) {
4472                 // FIXME: The clamping that follows is not completely right for right-to-left
4473                 // content.
4474                 // Clamp everything above the column to its top left.
4475                 if (point.y() < gapAndColumnRect.y())
4476                     point = gapAndColumnRect.location();
4477                 // Clamp everything below the column to the next column's top left. If there is
4478                 // no next column, this still maps to just after this column.
4479                 else if (point.y() >= gapAndColumnRect.maxY()) {
4480                     point = gapAndColumnRect.location();
4481                     point.move(0, gapAndColumnRect.height());
4482                 }
4483 
4484                 // We're inside the column.  Translate the x and y into our column coordinate space.
4485                 point.move(columnPoint.x() - colRect.x(), logicalOffset);
4486                 return;
4487             }
4488 
4489             // Move to the next position.
4490             logicalOffset += colRect.height();
4491         } else {
4492             IntRect gapAndColumnRect(colRect.x(), colRect.y() - halfColGap, colRect.width(), colRect.height() + colGap);
4493             if (point.y() >= gapAndColumnRect.y() && point.y() < gapAndColumnRect.maxY()) {
4494                 // FIXME: The clamping that follows is not completely right for right-to-left
4495                 // content.
4496                 // Clamp everything above the column to its top left.
4497                 if (point.x() < gapAndColumnRect.x())
4498                     point = gapAndColumnRect.location();
4499                 // Clamp everything below the column to the next column's top left. If there is
4500                 // no next column, this still maps to just after this column.
4501                 else if (point.x() >= gapAndColumnRect.maxX()) {
4502                     point = gapAndColumnRect.location();
4503                     point.move(gapAndColumnRect.width(), 0);
4504                 }
4505 
4506                 // We're inside the column.  Translate the x and y into our column coordinate space.
4507                 point.move(logicalOffset, columnPoint.y() - colRect.y());
4508                 return;
4509             }
4510 
4511             // Move to the next position.
4512             logicalOffset += colRect.width();
4513         }
4514     }
4515 }
4516 
adjustRectForColumns(IntRect & r) const4517 void RenderBlock::adjustRectForColumns(IntRect& r) const
4518 {
4519     // Just bail if we have no columns.
4520     if (!hasColumns())
4521         return;
4522 
4523     ColumnInfo* colInfo = columnInfo();
4524 
4525     // Begin with a result rect that is empty.
4526     IntRect result;
4527 
4528     // Determine which columns we intersect.
4529     unsigned colCount = columnCount(colInfo);
4530     if (!colCount)
4531         return;
4532 
4533     int logicalLeft = logicalLeftOffsetForContent();
4534     int currLogicalOffset = 0;
4535 
4536     for (unsigned i = 0; i < colCount; i++) {
4537         IntRect colRect = columnRectAt(colInfo, i);
4538         IntRect repaintRect = r;
4539         if (isHorizontalWritingMode()) {
4540             int currXOffset = colRect.x() - logicalLeft;
4541             repaintRect.move(currXOffset, currLogicalOffset);
4542             currLogicalOffset -= colRect.height();
4543         } else {
4544             int currYOffset = colRect.y() - logicalLeft;
4545             repaintRect.move(currLogicalOffset, currYOffset);
4546             currLogicalOffset -= colRect.width();
4547         }
4548         repaintRect.intersect(colRect);
4549         result.unite(repaintRect);
4550     }
4551 
4552     r = result;
4553 }
4554 
flipForWritingModeIncludingColumns(const IntPoint & point) const4555 IntPoint RenderBlock::flipForWritingModeIncludingColumns(const IntPoint& point) const
4556 {
4557     ASSERT(hasColumns());
4558     if (!hasColumns() || !style()->isFlippedBlocksWritingMode())
4559         return point;
4560     ColumnInfo* colInfo = columnInfo();
4561     int columnLogicalHeight = colInfo->columnHeight();
4562     int expandedLogicalHeight = borderBefore() + paddingBefore() + columnCount(colInfo) * columnLogicalHeight + borderAfter() + paddingAfter() + scrollbarLogicalHeight();
4563     if (isHorizontalWritingMode())
4564         return IntPoint(point.x(), expandedLogicalHeight - point.y());
4565     return IntPoint(expandedLogicalHeight - point.x(), point.y());
4566 }
4567 
flipForWritingModeIncludingColumns(IntRect & rect) const4568 void RenderBlock::flipForWritingModeIncludingColumns(IntRect& rect) const
4569 {
4570     ASSERT(hasColumns());
4571     if (!hasColumns() || !style()->isFlippedBlocksWritingMode())
4572         return;
4573 
4574     ColumnInfo* colInfo = columnInfo();
4575     int columnLogicalHeight = colInfo->columnHeight();
4576     int expandedLogicalHeight = borderBefore() + paddingBefore() + columnCount(colInfo) * columnLogicalHeight + borderAfter() + paddingAfter() + scrollbarLogicalHeight();
4577     if (isHorizontalWritingMode())
4578         rect.setY(expandedLogicalHeight - rect.maxY());
4579     else
4580         rect.setX(expandedLogicalHeight - rect.maxX());
4581 }
4582 
adjustForColumns(IntSize & offset,const IntPoint & point) const4583 void RenderBlock::adjustForColumns(IntSize& offset, const IntPoint& point) const
4584 {
4585     if (!hasColumns())
4586         return;
4587 
4588     ColumnInfo* colInfo = columnInfo();
4589 
4590     int logicalLeft = logicalLeftOffsetForContent();
4591     size_t colCount = columnCount(colInfo);
4592     int colLogicalWidth = colInfo->desiredColumnWidth();
4593     int colLogicalHeight = colInfo->columnHeight();
4594 
4595     for (size_t i = 0; i < colCount; ++i) {
4596         // Compute the edges for a given column in the block progression direction.
4597         IntRect sliceRect = IntRect(logicalLeft, borderBefore() + paddingBefore() + i * colLogicalHeight, colLogicalWidth, colLogicalHeight);
4598         if (!isHorizontalWritingMode())
4599             sliceRect = sliceRect.transposedRect();
4600 
4601         // If we have a flipped blocks writing mode, then convert the column so that it's coming from the after edge (either top or left edge).
4602         flipForWritingModeIncludingColumns(sliceRect);
4603 
4604         int logicalOffset = style()->isFlippedBlocksWritingMode() ? (colCount - 1 - i) * colLogicalHeight : i * colLogicalHeight;
4605 
4606         // Now we're in the same coordinate space as the point.  See if it is inside the rectangle.
4607         if (isHorizontalWritingMode()) {
4608             if (point.y() >= sliceRect.y() && point.y() < sliceRect.maxY()) {
4609                 offset.expand(columnRectAt(colInfo, i).x() - logicalLeft, -logicalOffset);
4610                 return;
4611             }
4612         } else {
4613             if (point.x() >= sliceRect.x() && point.x() < sliceRect.maxX()) {
4614                 offset.expand(-logicalOffset, columnRectAt(colInfo, i).y() - logicalLeft);
4615                 return;
4616             }
4617         }
4618     }
4619 }
4620 
computePreferredLogicalWidths()4621 void RenderBlock::computePreferredLogicalWidths()
4622 {
4623     ASSERT(preferredLogicalWidthsDirty());
4624 
4625     updateFirstLetter();
4626 
4627     if (!isTableCell() && style()->logicalWidth().isFixed() && style()->logicalWidth().value() > 0)
4628         m_minPreferredLogicalWidth = m_maxPreferredLogicalWidth = computeContentBoxLogicalWidth(style()->logicalWidth().value());
4629     else {
4630         m_minPreferredLogicalWidth = 0;
4631         m_maxPreferredLogicalWidth = 0;
4632 
4633         if (childrenInline())
4634             computeInlinePreferredLogicalWidths();
4635         else
4636             computeBlockPreferredLogicalWidths();
4637 
4638         m_maxPreferredLogicalWidth = max(m_minPreferredLogicalWidth, m_maxPreferredLogicalWidth);
4639 
4640         if (!style()->autoWrap() && childrenInline()) {
4641             m_minPreferredLogicalWidth = m_maxPreferredLogicalWidth;
4642 
4643             // A horizontal marquee with inline children has no minimum width.
4644             if (layer() && layer()->marquee() && layer()->marquee()->isHorizontal())
4645                 m_minPreferredLogicalWidth = 0;
4646         }
4647 
4648         int scrollbarWidth = 0;
4649         if (hasOverflowClip() && style()->overflowY() == OSCROLL) {
4650             layer()->setHasVerticalScrollbar(true);
4651             scrollbarWidth = verticalScrollbarWidth();
4652             m_maxPreferredLogicalWidth += scrollbarWidth;
4653         }
4654 
4655         if (isTableCell()) {
4656             Length w = toRenderTableCell(this)->styleOrColLogicalWidth();
4657             if (w.isFixed() && w.value() > 0) {
4658                 m_maxPreferredLogicalWidth = max(m_minPreferredLogicalWidth, computeContentBoxLogicalWidth(w.value()));
4659                 scrollbarWidth = 0;
4660             }
4661         }
4662 
4663         m_minPreferredLogicalWidth += scrollbarWidth;
4664     }
4665 
4666     if (style()->logicalMinWidth().isFixed() && style()->logicalMinWidth().value() > 0) {
4667         m_maxPreferredLogicalWidth = max(m_maxPreferredLogicalWidth, computeContentBoxLogicalWidth(style()->logicalMinWidth().value()));
4668         m_minPreferredLogicalWidth = max(m_minPreferredLogicalWidth, computeContentBoxLogicalWidth(style()->logicalMinWidth().value()));
4669     }
4670 
4671     if (style()->logicalMaxWidth().isFixed() && style()->logicalMaxWidth().value() != undefinedLength) {
4672         m_maxPreferredLogicalWidth = min(m_maxPreferredLogicalWidth, computeContentBoxLogicalWidth(style()->logicalMaxWidth().value()));
4673         m_minPreferredLogicalWidth = min(m_minPreferredLogicalWidth, computeContentBoxLogicalWidth(style()->logicalMaxWidth().value()));
4674     }
4675 
4676     int borderAndPadding = borderAndPaddingLogicalWidth();
4677     m_minPreferredLogicalWidth += borderAndPadding;
4678     m_maxPreferredLogicalWidth += borderAndPadding;
4679 
4680     setPreferredLogicalWidthsDirty(false);
4681 }
4682 
4683 struct InlineMinMaxIterator {
4684 /* InlineMinMaxIterator is a class that will iterate over all render objects that contribute to
4685    inline min/max width calculations.  Note the following about the way it walks:
4686    (1) Positioned content is skipped (since it does not contribute to min/max width of a block)
4687    (2) We do not drill into the children of floats or replaced elements, since you can't break
4688        in the middle of such an element.
4689    (3) Inline flows (e.g., <a>, <span>, <i>) are walked twice, since each side can have
4690        distinct borders/margin/padding that contribute to the min/max width.
4691 */
4692     RenderObject* parent;
4693     RenderObject* current;
4694     bool endOfInline;
4695 
InlineMinMaxIteratorWebCore::InlineMinMaxIterator4696     InlineMinMaxIterator(RenderObject* p, bool end = false)
4697         :parent(p), current(p), endOfInline(end) {}
4698 
4699     RenderObject* next();
4700 };
4701 
next()4702 RenderObject* InlineMinMaxIterator::next()
4703 {
4704     RenderObject* result = 0;
4705     bool oldEndOfInline = endOfInline;
4706     endOfInline = false;
4707     while (current || current == parent) {
4708         if (!oldEndOfInline &&
4709             (current == parent ||
4710              (!current->isFloating() && !current->isReplaced() && !current->isPositioned())))
4711             result = current->firstChild();
4712         if (!result) {
4713             // We hit the end of our inline. (It was empty, e.g., <span></span>.)
4714             if (!oldEndOfInline && current->isRenderInline()) {
4715                 result = current;
4716                 endOfInline = true;
4717                 break;
4718             }
4719 
4720             while (current && current != parent) {
4721                 result = current->nextSibling();
4722                 if (result) break;
4723                 current = current->parent();
4724                 if (current && current != parent && current->isRenderInline()) {
4725                     result = current;
4726                     endOfInline = true;
4727                     break;
4728                 }
4729             }
4730         }
4731 
4732         if (!result)
4733             break;
4734 
4735         if (!result->isPositioned() && (result->isText() || result->isFloating() || result->isReplaced() || result->isRenderInline()))
4736              break;
4737 
4738         current = result;
4739         result = 0;
4740     }
4741 
4742     // Update our position.
4743     current = result;
4744     return current;
4745 }
4746 
getBPMWidth(int childValue,Length cssUnit)4747 static int getBPMWidth(int childValue, Length cssUnit)
4748 {
4749     if (cssUnit.type() != Auto)
4750         return (cssUnit.isFixed() ? cssUnit.value() : childValue);
4751     return 0;
4752 }
4753 
getBorderPaddingMargin(const RenderBoxModelObject * child,bool endOfInline)4754 static int getBorderPaddingMargin(const RenderBoxModelObject* child, bool endOfInline)
4755 {
4756     RenderStyle* cstyle = child->style();
4757     if (endOfInline)
4758         return getBPMWidth(child->marginEnd(), cstyle->marginEnd()) +
4759                getBPMWidth(child->paddingEnd(), cstyle->paddingEnd()) +
4760                child->borderEnd();
4761     return getBPMWidth(child->marginStart(), cstyle->marginStart()) +
4762                getBPMWidth(child->paddingStart(), cstyle->paddingStart()) +
4763                child->borderStart();
4764 }
4765 
stripTrailingSpace(float & inlineMax,float & inlineMin,RenderObject * trailingSpaceChild)4766 static inline void stripTrailingSpace(float& inlineMax, float& inlineMin,
4767                                       RenderObject* trailingSpaceChild)
4768 {
4769     if (trailingSpaceChild && trailingSpaceChild->isText()) {
4770         // Collapse away the trailing space at the end of a block.
4771         RenderText* t = toRenderText(trailingSpaceChild);
4772         const UChar space = ' ';
4773         const Font& font = t->style()->font(); // FIXME: This ignores first-line.
4774         float spaceWidth = font.width(TextRun(&space, 1));
4775         inlineMax -= spaceWidth + font.wordSpacing();
4776         if (inlineMin > inlineMax)
4777             inlineMin = inlineMax;
4778     }
4779 }
4780 
updatePreferredWidth(int & preferredWidth,float & result)4781 static inline void updatePreferredWidth(int& preferredWidth, float& result)
4782 {
4783     int snappedResult = ceilf(result);
4784     preferredWidth = max(snappedResult, preferredWidth);
4785 }
4786 
computeInlinePreferredLogicalWidths()4787 void RenderBlock::computeInlinePreferredLogicalWidths()
4788 {
4789     float inlineMax = 0;
4790     float inlineMin = 0;
4791 
4792     int cw = containingBlock()->contentLogicalWidth();
4793 
4794     // If we are at the start of a line, we want to ignore all white-space.
4795     // Also strip spaces if we previously had text that ended in a trailing space.
4796     bool stripFrontSpaces = true;
4797     RenderObject* trailingSpaceChild = 0;
4798 
4799     // Firefox and Opera will allow a table cell to grow to fit an image inside it under
4800     // very specific cirucumstances (in order to match common WinIE renderings).
4801     // Not supporting the quirk has caused us to mis-render some real sites. (See Bugzilla 10517.)
4802     bool allowImagesToBreak = !document()->inQuirksMode() || !isTableCell() || !style()->logicalWidth().isIntrinsicOrAuto();
4803 
4804     bool autoWrap, oldAutoWrap;
4805     autoWrap = oldAutoWrap = style()->autoWrap();
4806 
4807     InlineMinMaxIterator childIterator(this);
4808     bool addedTextIndent = false; // Only gets added in once.
4809     RenderObject* prevFloat = 0;
4810     while (RenderObject* child = childIterator.next()) {
4811         autoWrap = child->isReplaced() ? child->parent()->style()->autoWrap() :
4812             child->style()->autoWrap();
4813 
4814         if (!child->isBR()) {
4815             // Step One: determine whether or not we need to go ahead and
4816             // terminate our current line.  Each discrete chunk can become
4817             // the new min-width, if it is the widest chunk seen so far, and
4818             // it can also become the max-width.
4819 
4820             // Children fall into three categories:
4821             // (1) An inline flow object.  These objects always have a min/max of 0,
4822             // and are included in the iteration solely so that their margins can
4823             // be added in.
4824             //
4825             // (2) An inline non-text non-flow object, e.g., an inline replaced element.
4826             // These objects can always be on a line by themselves, so in this situation
4827             // we need to go ahead and break the current line, and then add in our own
4828             // margins and min/max width on its own line, and then terminate the line.
4829             //
4830             // (3) A text object.  Text runs can have breakable characters at the start,
4831             // the middle or the end.  They may also lose whitespace off the front if
4832             // we're already ignoring whitespace.  In order to compute accurate min-width
4833             // information, we need three pieces of information.
4834             // (a) the min-width of the first non-breakable run.  Should be 0 if the text string
4835             // starts with whitespace.
4836             // (b) the min-width of the last non-breakable run. Should be 0 if the text string
4837             // ends with whitespace.
4838             // (c) the min/max width of the string (trimmed for whitespace).
4839             //
4840             // If the text string starts with whitespace, then we need to go ahead and
4841             // terminate our current line (unless we're already in a whitespace stripping
4842             // mode.
4843             //
4844             // If the text string has a breakable character in the middle, but didn't start
4845             // with whitespace, then we add the width of the first non-breakable run and
4846             // then end the current line.  We then need to use the intermediate min/max width
4847             // values (if any of them are larger than our current min/max).  We then look at
4848             // the width of the last non-breakable run and use that to start a new line
4849             // (unless we end in whitespace).
4850             RenderStyle* cstyle = child->style();
4851             float childMin = 0;
4852             float childMax = 0;
4853 
4854             if (!child->isText()) {
4855                 // Case (1) and (2).  Inline replaced and inline flow elements.
4856                 if (child->isRenderInline()) {
4857                     // Add in padding/border/margin from the appropriate side of
4858                     // the element.
4859                     float bpm = getBorderPaddingMargin(toRenderInline(child), childIterator.endOfInline);
4860                     childMin += bpm;
4861                     childMax += bpm;
4862 
4863                     inlineMin += childMin;
4864                     inlineMax += childMax;
4865 
4866                     child->setPreferredLogicalWidthsDirty(false);
4867                 } else {
4868                     // Inline replaced elts add in their margins to their min/max values.
4869                     float margins = 0;
4870                     Length startMargin = cstyle->marginStart();
4871                     Length endMargin = cstyle->marginEnd();
4872                     if (startMargin.isFixed())
4873                         margins += startMargin.value();
4874                     if (endMargin.isFixed())
4875                         margins += endMargin.value();
4876                     childMin += margins;
4877                     childMax += margins;
4878                 }
4879             }
4880 
4881             if (!child->isRenderInline() && !child->isText()) {
4882                 // Case (2). Inline replaced elements and floats.
4883                 // Go ahead and terminate the current line as far as
4884                 // minwidth is concerned.
4885                 childMin += child->minPreferredLogicalWidth();
4886                 childMax += child->maxPreferredLogicalWidth();
4887 
4888                 bool clearPreviousFloat;
4889                 if (child->isFloating()) {
4890                     clearPreviousFloat = (prevFloat
4891                         && ((prevFloat->style()->floating() == FLEFT && (child->style()->clear() & CLEFT))
4892                             || (prevFloat->style()->floating() == FRIGHT && (child->style()->clear() & CRIGHT))));
4893                     prevFloat = child;
4894                 } else
4895                     clearPreviousFloat = false;
4896 
4897                 bool canBreakReplacedElement = !child->isImage() || allowImagesToBreak;
4898                 if ((canBreakReplacedElement && (autoWrap || oldAutoWrap)) || clearPreviousFloat) {
4899                     updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
4900                     inlineMin = 0;
4901                 }
4902 
4903                 // If we're supposed to clear the previous float, then terminate maxwidth as well.
4904                 if (clearPreviousFloat) {
4905                     updatePreferredWidth(m_maxPreferredLogicalWidth, inlineMax);
4906                     inlineMax = 0;
4907                 }
4908 
4909                 // Add in text-indent.  This is added in only once.
4910                 int ti = 0;
4911                 if (!addedTextIndent) {
4912                     addedTextIndent = true;
4913                     ti = style()->textIndent().calcMinValue(cw);
4914                     childMin += ti;
4915                     childMax += ti;
4916                 }
4917 
4918                 // Add our width to the max.
4919                 inlineMax += childMax;
4920 
4921                 if (!autoWrap || !canBreakReplacedElement) {
4922                     if (child->isFloating())
4923                         updatePreferredWidth(m_minPreferredLogicalWidth, childMin);
4924                     else
4925                         inlineMin += childMin;
4926                 } else {
4927                     // Now check our line.
4928                     updatePreferredWidth(m_minPreferredLogicalWidth, childMin);
4929 
4930                     // Now start a new line.
4931                     inlineMin = 0;
4932                 }
4933 
4934                 // We are no longer stripping whitespace at the start of
4935                 // a line.
4936                 if (!child->isFloating()) {
4937                     stripFrontSpaces = false;
4938                     trailingSpaceChild = 0;
4939                 }
4940             } else if (child->isText()) {
4941                 // Case (3). Text.
4942                 RenderText* t = toRenderText(child);
4943 
4944                 if (t->isWordBreak()) {
4945                     updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
4946                     inlineMin = 0;
4947                     continue;
4948                 }
4949 
4950                 if (t->style()->hasTextCombine() && t->isCombineText())
4951                     toRenderCombineText(t)->combineText();
4952 
4953                 // Determine if we have a breakable character.  Pass in
4954                 // whether or not we should ignore any spaces at the front
4955                 // of the string.  If those are going to be stripped out,
4956                 // then they shouldn't be considered in the breakable char
4957                 // check.
4958                 bool hasBreakableChar, hasBreak;
4959                 float beginMin, endMin;
4960                 bool beginWS, endWS;
4961                 float beginMax, endMax;
4962                 t->trimmedPrefWidths(inlineMax, beginMin, beginWS, endMin, endWS,
4963                                      hasBreakableChar, hasBreak, beginMax, endMax,
4964                                      childMin, childMax, stripFrontSpaces);
4965 
4966                 // This text object will not be rendered, but it may still provide a breaking opportunity.
4967                 if (!hasBreak && childMax == 0) {
4968                     if (autoWrap && (beginWS || endWS)) {
4969                         updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
4970                         inlineMin = 0;
4971                     }
4972                     continue;
4973                 }
4974 
4975                 if (stripFrontSpaces)
4976                     trailingSpaceChild = child;
4977                 else
4978                     trailingSpaceChild = 0;
4979 
4980                 // Add in text-indent.  This is added in only once.
4981                 int ti = 0;
4982                 if (!addedTextIndent) {
4983                     addedTextIndent = true;
4984                     ti = style()->textIndent().calcMinValue(cw);
4985                     childMin+=ti; beginMin += ti;
4986                     childMax+=ti; beginMax += ti;
4987                 }
4988 
4989                 // If we have no breakable characters at all,
4990                 // then this is the easy case. We add ourselves to the current
4991                 // min and max and continue.
4992                 if (!hasBreakableChar) {
4993                     inlineMin += childMin;
4994                 } else {
4995                     // We have a breakable character.  Now we need to know if
4996                     // we start and end with whitespace.
4997                     if (beginWS)
4998                         // Go ahead and end the current line.
4999                         updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5000                     else {
5001                         inlineMin += beginMin;
5002                         updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5003                         childMin -= ti;
5004                     }
5005 
5006                     inlineMin = childMin;
5007 
5008                     if (endWS) {
5009                         // We end in whitespace, which means we can go ahead
5010                         // and end our current line.
5011                         updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5012                         inlineMin = 0;
5013                     } else {
5014                         updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5015                         inlineMin = endMin;
5016                     }
5017                 }
5018 
5019                 if (hasBreak) {
5020                     inlineMax += beginMax;
5021                     updatePreferredWidth(m_maxPreferredLogicalWidth, inlineMax);
5022                     updatePreferredWidth(m_maxPreferredLogicalWidth, childMax);
5023                     inlineMax = endMax;
5024                 } else
5025                     inlineMax += childMax;
5026             }
5027 
5028             // Ignore spaces after a list marker.
5029             if (child->isListMarker())
5030                 stripFrontSpaces = true;
5031         } else {
5032             updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5033             updatePreferredWidth(m_maxPreferredLogicalWidth, inlineMax);
5034             inlineMin = inlineMax = 0;
5035             stripFrontSpaces = true;
5036             trailingSpaceChild = 0;
5037         }
5038 
5039         oldAutoWrap = autoWrap;
5040     }
5041 
5042     if (style()->collapseWhiteSpace())
5043         stripTrailingSpace(inlineMax, inlineMin, trailingSpaceChild);
5044 
5045     updatePreferredWidth(m_minPreferredLogicalWidth, inlineMin);
5046     updatePreferredWidth(m_maxPreferredLogicalWidth, inlineMax);
5047 }
5048 
5049 // Use a very large value (in effect infinite).
5050 #define BLOCK_MAX_WIDTH 15000
5051 
computeBlockPreferredLogicalWidths()5052 void RenderBlock::computeBlockPreferredLogicalWidths()
5053 {
5054     bool nowrap = style()->whiteSpace() == NOWRAP;
5055 
5056     RenderObject *child = firstChild();
5057     int floatLeftWidth = 0, floatRightWidth = 0;
5058     while (child) {
5059         // Positioned children don't affect the min/max width
5060         if (child->isPositioned()) {
5061             child = child->nextSibling();
5062             continue;
5063         }
5064 
5065         if (child->isFloating() || (child->isBox() && toRenderBox(child)->avoidsFloats())) {
5066             int floatTotalWidth = floatLeftWidth + floatRightWidth;
5067             if (child->style()->clear() & CLEFT) {
5068                 m_maxPreferredLogicalWidth = max(floatTotalWidth, m_maxPreferredLogicalWidth);
5069                 floatLeftWidth = 0;
5070             }
5071             if (child->style()->clear() & CRIGHT) {
5072                 m_maxPreferredLogicalWidth = max(floatTotalWidth, m_maxPreferredLogicalWidth);
5073                 floatRightWidth = 0;
5074             }
5075         }
5076 
5077         // A margin basically has three types: fixed, percentage, and auto (variable).
5078         // Auto and percentage margins simply become 0 when computing min/max width.
5079         // Fixed margins can be added in as is.
5080         Length startMarginLength = child->style()->marginStart();
5081         Length endMarginLength = child->style()->marginEnd();
5082         int margin = 0;
5083         int marginStart = 0;
5084         int marginEnd = 0;
5085         if (startMarginLength.isFixed())
5086             marginStart += startMarginLength.value();
5087         if (endMarginLength.isFixed())
5088             marginEnd += endMarginLength.value();
5089         margin = marginStart + marginEnd;
5090 
5091         int w = child->minPreferredLogicalWidth() + margin;
5092         m_minPreferredLogicalWidth = max(w, m_minPreferredLogicalWidth);
5093 
5094         // IE ignores tables for calculation of nowrap. Makes some sense.
5095         if (nowrap && !child->isTable())
5096             m_maxPreferredLogicalWidth = max(w, m_maxPreferredLogicalWidth);
5097 
5098         w = child->maxPreferredLogicalWidth() + margin;
5099 
5100         if (!child->isFloating()) {
5101             if (child->isBox() && toRenderBox(child)->avoidsFloats()) {
5102                 // Determine a left and right max value based off whether or not the floats can fit in the
5103                 // margins of the object.  For negative margins, we will attempt to overlap the float if the negative margin
5104                 // is smaller than the float width.
5105                 bool ltr = containingBlock()->style()->isLeftToRightDirection();
5106                 int marginLogicalLeft = ltr ? marginStart : marginEnd;
5107                 int marginLogicalRight = ltr ? marginEnd : marginStart;
5108                 int maxLeft = marginLogicalLeft > 0 ? max(floatLeftWidth, marginLogicalLeft) : floatLeftWidth + marginLogicalLeft;
5109                 int maxRight = marginLogicalRight > 0 ? max(floatRightWidth, marginLogicalRight) : floatRightWidth + marginLogicalRight;
5110                 w = child->maxPreferredLogicalWidth() + maxLeft + maxRight;
5111                 w = max(w, floatLeftWidth + floatRightWidth);
5112             }
5113             else
5114                 m_maxPreferredLogicalWidth = max(floatLeftWidth + floatRightWidth, m_maxPreferredLogicalWidth);
5115             floatLeftWidth = floatRightWidth = 0;
5116         }
5117 
5118         if (child->isFloating()) {
5119             if (style()->floating() == FLEFT)
5120                 floatLeftWidth += w;
5121             else
5122                 floatRightWidth += w;
5123         } else
5124             m_maxPreferredLogicalWidth = max(w, m_maxPreferredLogicalWidth);
5125 
5126         // A very specific WinIE quirk.
5127         // Example:
5128         /*
5129            <div style="position:absolute; width:100px; top:50px;">
5130               <div style="position:absolute;left:0px;top:50px;height:50px;background-color:green">
5131                 <table style="width:100%"><tr><td></table>
5132               </div>
5133            </div>
5134         */
5135         // In the above example, the inner absolute positioned block should have a computed width
5136         // of 100px because of the table.
5137         // We can achieve this effect by making the maxwidth of blocks that contain tables
5138         // with percentage widths be infinite (as long as they are not inside a table cell).
5139         if (document()->inQuirksMode() && child->style()->logicalWidth().isPercent() &&
5140             !isTableCell() && child->isTable() && m_maxPreferredLogicalWidth < BLOCK_MAX_WIDTH) {
5141             RenderBlock* cb = containingBlock();
5142             while (!cb->isRenderView() && !cb->isTableCell())
5143                 cb = cb->containingBlock();
5144             if (!cb->isTableCell())
5145                 m_maxPreferredLogicalWidth = BLOCK_MAX_WIDTH;
5146         }
5147 
5148         child = child->nextSibling();
5149     }
5150 
5151     // Always make sure these values are non-negative.
5152     m_minPreferredLogicalWidth = max(0, m_minPreferredLogicalWidth);
5153     m_maxPreferredLogicalWidth = max(0, m_maxPreferredLogicalWidth);
5154 
5155     m_maxPreferredLogicalWidth = max(floatLeftWidth + floatRightWidth, m_maxPreferredLogicalWidth);
5156 }
5157 
hasLineIfEmpty() const5158 bool RenderBlock::hasLineIfEmpty() const
5159 {
5160     if (!node())
5161         return false;
5162 
5163     if (node()->rendererIsEditable() && node()->rootEditableElement() == node())
5164         return true;
5165 
5166     if (node()->isShadowRoot() && (node()->shadowHost()->hasTagName(inputTag)))
5167         return true;
5168 
5169     return false;
5170 }
5171 
lineHeight(bool firstLine,LineDirectionMode direction,LinePositionMode linePositionMode) const5172 int RenderBlock::lineHeight(bool firstLine, LineDirectionMode direction, LinePositionMode linePositionMode) const
5173 {
5174     // Inline blocks are replaced elements. Otherwise, just pass off to
5175     // the base class.  If we're being queried as though we're the root line
5176     // box, then the fact that we're an inline-block is irrelevant, and we behave
5177     // just like a block.
5178     if (isReplaced() && linePositionMode == PositionOnContainingLine)
5179         return RenderBox::lineHeight(firstLine, direction, linePositionMode);
5180 
5181     if (firstLine && document()->usesFirstLineRules()) {
5182         RenderStyle* s = style(firstLine);
5183         if (s != style())
5184             return s->computedLineHeight();
5185     }
5186 
5187     if (m_lineHeight == -1)
5188         m_lineHeight = style()->computedLineHeight();
5189 
5190     return m_lineHeight;
5191 }
5192 
baselinePosition(FontBaseline baselineType,bool firstLine,LineDirectionMode direction,LinePositionMode linePositionMode) const5193 int RenderBlock::baselinePosition(FontBaseline baselineType, bool firstLine, LineDirectionMode direction, LinePositionMode linePositionMode) const
5194 {
5195     // Inline blocks are replaced elements. Otherwise, just pass off to
5196     // the base class.  If we're being queried as though we're the root line
5197     // box, then the fact that we're an inline-block is irrelevant, and we behave
5198     // just like a block.
5199     if (isReplaced() && linePositionMode == PositionOnContainingLine) {
5200         // For "leaf" theme objects, let the theme decide what the baseline position is.
5201         // FIXME: Might be better to have a custom CSS property instead, so that if the theme
5202         // is turned off, checkboxes/radios will still have decent baselines.
5203         // FIXME: Need to patch form controls to deal with vertical lines.
5204         if (style()->hasAppearance() && !theme()->isControlContainer(style()->appearance()))
5205             return theme()->baselinePosition(this);
5206 
5207         // CSS2.1 states that the baseline of an inline block is the baseline of the last line box in
5208         // the normal flow.  We make an exception for marquees, since their baselines are meaningless
5209         // (the content inside them moves).  This matches WinIE as well, which just bottom-aligns them.
5210         // We also give up on finding a baseline if we have a vertical scrollbar, or if we are scrolled
5211         // vertically (e.g., an overflow:hidden block that has had scrollTop moved) or if the baseline is outside
5212         // of our content box.
5213         bool ignoreBaseline = (layer() && (layer()->marquee() || (direction == HorizontalLine ? (layer()->verticalScrollbar() || layer()->scrollYOffset() != 0)
5214             : (layer()->horizontalScrollbar() || layer()->scrollXOffset() != 0)))) || (isWritingModeRoot() && !isRubyRun());
5215 
5216         int baselinePos = ignoreBaseline ? -1 : lastLineBoxBaseline();
5217 
5218         int bottomOfContent = direction == HorizontalLine ? borderTop() + paddingTop() + contentHeight() : borderRight() + paddingRight() + contentWidth();
5219         if (baselinePos != -1 && baselinePos <= bottomOfContent)
5220             return direction == HorizontalLine ? marginTop() + baselinePos : marginRight() + baselinePos;
5221 
5222         return RenderBox::baselinePosition(baselineType, firstLine, direction, linePositionMode);
5223     }
5224 
5225     const FontMetrics& fontMetrics = style(firstLine)->fontMetrics();
5226     return fontMetrics.ascent(baselineType) + (lineHeight(firstLine, direction, linePositionMode) - fontMetrics.height()) / 2;
5227 }
5228 
firstLineBoxBaseline() const5229 int RenderBlock::firstLineBoxBaseline() const
5230 {
5231     if (!isBlockFlow() || (isWritingModeRoot() && !isRubyRun()))
5232         return -1;
5233 
5234     if (childrenInline()) {
5235         if (firstLineBox())
5236             return firstLineBox()->logicalTop() + style(true)->fontMetrics().ascent(firstRootBox()->baselineType());
5237         else
5238             return -1;
5239     }
5240     else {
5241         for (RenderBox* curr = firstChildBox(); curr; curr = curr->nextSiblingBox()) {
5242             if (!curr->isFloatingOrPositioned()) {
5243                 int result = curr->firstLineBoxBaseline();
5244                 if (result != -1)
5245                     return curr->logicalTop() + result; // Translate to our coordinate space.
5246             }
5247         }
5248     }
5249 
5250     return -1;
5251 }
5252 
lastLineBoxBaseline() const5253 int RenderBlock::lastLineBoxBaseline() const
5254 {
5255     if (!isBlockFlow() || (isWritingModeRoot() && !isRubyRun()))
5256         return -1;
5257 
5258     LineDirectionMode lineDirection = isHorizontalWritingMode() ? HorizontalLine : VerticalLine;
5259 
5260     if (childrenInline()) {
5261         if (!firstLineBox() && hasLineIfEmpty()) {
5262             const FontMetrics& fontMetrics = firstLineStyle()->fontMetrics();
5263             return fontMetrics.ascent()
5264                  + (lineHeight(true, lineDirection, PositionOfInteriorLineBoxes) - fontMetrics.height()) / 2
5265                  + (lineDirection == HorizontalLine ? borderTop() + paddingTop() : borderRight() + paddingRight());
5266         }
5267         if (lastLineBox())
5268             return lastLineBox()->logicalTop() + style(lastLineBox() == firstLineBox())->fontMetrics().ascent(lastRootBox()->baselineType());
5269         return -1;
5270     } else {
5271         bool haveNormalFlowChild = false;
5272         for (RenderBox* curr = lastChildBox(); curr; curr = curr->previousSiblingBox()) {
5273             if (!curr->isFloatingOrPositioned()) {
5274                 haveNormalFlowChild = true;
5275                 int result = curr->lastLineBoxBaseline();
5276                 if (result != -1)
5277                     return curr->logicalTop() + result; // Translate to our coordinate space.
5278             }
5279         }
5280         if (!haveNormalFlowChild && hasLineIfEmpty()) {
5281             const FontMetrics& fontMetrics = firstLineStyle()->fontMetrics();
5282             return fontMetrics.ascent()
5283                  + (lineHeight(true, lineDirection, PositionOfInteriorLineBoxes) - fontMetrics.height()) / 2
5284                  + (lineDirection == HorizontalLine ? borderTop() + paddingTop() : borderRight() + paddingRight());
5285         }
5286     }
5287 
5288     return -1;
5289 }
5290 
containsNonZeroBidiLevel() const5291 bool RenderBlock::containsNonZeroBidiLevel() const
5292 {
5293     for (RootInlineBox* root = firstRootBox(); root; root = root->nextRootBox()) {
5294         for (InlineBox* box = root->firstLeafChild(); box; box = box->nextLeafChild()) {
5295             if (box->bidiLevel())
5296                 return true;
5297         }
5298     }
5299     return false;
5300 }
5301 
firstLineBlock() const5302 RenderBlock* RenderBlock::firstLineBlock() const
5303 {
5304     RenderBlock* firstLineBlock = const_cast<RenderBlock*>(this);
5305     bool hasPseudo = false;
5306     while (true) {
5307         hasPseudo = firstLineBlock->style()->hasPseudoStyle(FIRST_LINE);
5308         if (hasPseudo)
5309             break;
5310         RenderObject* parentBlock = firstLineBlock->parent();
5311         if (firstLineBlock->isReplaced() || firstLineBlock->isFloating() ||
5312             !parentBlock || parentBlock->firstChild() != firstLineBlock || !parentBlock->isBlockFlow())
5313             break;
5314         ASSERT(parentBlock->isRenderBlock());
5315         firstLineBlock = toRenderBlock(parentBlock);
5316     }
5317 
5318     if (!hasPseudo)
5319         return 0;
5320 
5321     return firstLineBlock;
5322 }
5323 
styleForFirstLetter(RenderObject * firstLetterBlock,RenderObject * firstLetterContainer)5324 static RenderStyle* styleForFirstLetter(RenderObject* firstLetterBlock, RenderObject* firstLetterContainer)
5325 {
5326     RenderStyle* pseudoStyle = firstLetterBlock->getCachedPseudoStyle(FIRST_LETTER, firstLetterContainer->firstLineStyle());
5327     // Force inline display (except for floating first-letters).
5328     pseudoStyle->setDisplay(pseudoStyle->isFloating() ? BLOCK : INLINE);
5329     // CSS2 says first-letter can't be positioned.
5330     pseudoStyle->setPosition(StaticPosition);
5331     return pseudoStyle;
5332 }
5333 
5334 // CSS 2.1 http://www.w3.org/TR/CSS21/selector.html#first-letter
5335 // "Punctuation (i.e, characters defined in Unicode [UNICODE] in the "open" (Ps), "close" (Pe),
5336 // "initial" (Pi). "final" (Pf) and "other" (Po) punctuation classes), that precedes or follows the first letter should be included"
isPunctuationForFirstLetter(UChar c)5337 static inline bool isPunctuationForFirstLetter(UChar c)
5338 {
5339     CharCategory charCategory = category(c);
5340     return charCategory == Punctuation_Open
5341         || charCategory == Punctuation_Close
5342         || charCategory == Punctuation_InitialQuote
5343         || charCategory == Punctuation_FinalQuote
5344         || charCategory == Punctuation_Other;
5345 }
5346 
shouldSkipForFirstLetter(UChar c)5347 static inline bool shouldSkipForFirstLetter(UChar c)
5348 {
5349     return isSpaceOrNewline(c) || c == noBreakSpace || isPunctuationForFirstLetter(c);
5350 }
5351 
updateFirstLetter()5352 void RenderBlock::updateFirstLetter()
5353 {
5354     if (!document()->usesFirstLetterRules())
5355         return;
5356     // Don't recur
5357     if (style()->styleType() == FIRST_LETTER)
5358         return;
5359 
5360     // FIXME: We need to destroy the first-letter object if it is no longer the first child.  Need to find
5361     // an efficient way to check for that situation though before implementing anything.
5362     RenderObject* firstLetterBlock = this;
5363     bool hasPseudoStyle = false;
5364     while (true) {
5365         // We only honor first-letter if the firstLetterBlock can have children in the DOM. This correctly
5366         // prevents form controls from honoring first-letter.
5367         hasPseudoStyle = firstLetterBlock->style()->hasPseudoStyle(FIRST_LETTER)
5368             && firstLetterBlock->canHaveChildren();
5369         if (hasPseudoStyle)
5370             break;
5371         RenderObject* parentBlock = firstLetterBlock->parent();
5372         if (firstLetterBlock->isReplaced() || !parentBlock || parentBlock->firstChild() != firstLetterBlock ||
5373             !parentBlock->isBlockFlow())
5374             break;
5375         firstLetterBlock = parentBlock;
5376     }
5377 
5378     if (!hasPseudoStyle)
5379         return;
5380 
5381     // Drill into inlines looking for our first text child.
5382     RenderObject* currChild = firstLetterBlock->firstChild();
5383     while (currChild) {
5384         if (currChild->isText())
5385             break;
5386         if (currChild->isListMarker())
5387             currChild = currChild->nextSibling();
5388         else if (currChild->isFloatingOrPositioned()) {
5389             if (currChild->style()->styleType() == FIRST_LETTER) {
5390                 currChild = currChild->firstChild();
5391                 break;
5392             }
5393             currChild = currChild->nextSibling();
5394         } else if (currChild->isReplaced() || currChild->isRenderButton() || currChild->isMenuList())
5395             break;
5396         else if (currChild->style()->hasPseudoStyle(FIRST_LETTER) && currChild->canHaveChildren())  {
5397             // We found a lower-level node with first-letter, which supersedes the higher-level style
5398             firstLetterBlock = currChild;
5399             currChild = currChild->firstChild();
5400         }
5401         else
5402             currChild = currChild->firstChild();
5403     }
5404 
5405     if (!currChild)
5406         return;
5407 
5408     // If the child already has style, then it has already been created, so we just want
5409     // to update it.
5410     if (currChild->parent()->style()->styleType() == FIRST_LETTER) {
5411         RenderObject* firstLetter = currChild->parent();
5412         RenderObject* firstLetterContainer = firstLetter->parent();
5413         RenderStyle* pseudoStyle = styleForFirstLetter(firstLetterBlock, firstLetterContainer);
5414 
5415         if (Node::diff(firstLetter->style(), pseudoStyle) == Node::Detach) {
5416             // The first-letter renderer needs to be replaced. Create a new renderer of the right type.
5417             RenderObject* newFirstLetter;
5418             if (pseudoStyle->display() == INLINE)
5419                 newFirstLetter = new (renderArena()) RenderInline(document());
5420             else
5421                 newFirstLetter = new (renderArena()) RenderBlock(document());
5422             newFirstLetter->setStyle(pseudoStyle);
5423 
5424             // Move the first letter into the new renderer.
5425             view()->disableLayoutState();
5426             while (RenderObject* child = firstLetter->firstChild()) {
5427                 if (child->isText())
5428                     toRenderText(child)->removeAndDestroyTextBoxes();
5429                 firstLetter->removeChild(child);
5430                 newFirstLetter->addChild(child, 0);
5431             }
5432 
5433             RenderTextFragment* remainingText = 0;
5434             RenderObject* nextSibling = firstLetter->nextSibling();
5435             RenderObject* next = nextSibling;
5436             while (next) {
5437                 if (next->isText() && toRenderText(next)->isTextFragment()) {
5438                     remainingText = toRenderTextFragment(next);
5439                     break;
5440                 }
5441                 next = next->nextSibling();
5442             }
5443             if (remainingText) {
5444                 ASSERT(remainingText->node()->renderer() == remainingText);
5445                 // Replace the old renderer with the new one.
5446                 remainingText->setFirstLetter(newFirstLetter);
5447             }
5448             firstLetter->destroy();
5449             firstLetter = newFirstLetter;
5450             firstLetterContainer->addChild(firstLetter, nextSibling);
5451             view()->enableLayoutState();
5452         } else
5453             firstLetter->setStyle(pseudoStyle);
5454 
5455         for (RenderObject* genChild = firstLetter->firstChild(); genChild; genChild = genChild->nextSibling()) {
5456             if (genChild->isText())
5457                 genChild->setStyle(pseudoStyle);
5458         }
5459 
5460         return;
5461     }
5462 
5463     if (!currChild->isText() || currChild->isBR())
5464         return;
5465 
5466     // If the child does not already have style, we create it here.
5467     RenderObject* firstLetterContainer = currChild->parent();
5468 
5469     // Our layout state is not valid for the repaints we are going to trigger by
5470     // adding and removing children of firstLetterContainer.
5471     view()->disableLayoutState();
5472 
5473     RenderText* textObj = toRenderText(currChild);
5474 
5475     // Create our pseudo style now that we have our firstLetterContainer determined.
5476     RenderStyle* pseudoStyle = styleForFirstLetter(firstLetterBlock, firstLetterContainer);
5477 
5478     RenderObject* firstLetter = 0;
5479     if (pseudoStyle->display() == INLINE)
5480         firstLetter = new (renderArena()) RenderInline(document());
5481     else
5482         firstLetter = new (renderArena()) RenderBlock(document());
5483     firstLetter->setStyle(pseudoStyle);
5484     firstLetterContainer->addChild(firstLetter, currChild);
5485 
5486     // The original string is going to be either a generated content string or a DOM node's
5487     // string.  We want the original string before it got transformed in case first-letter has
5488     // no text-transform or a different text-transform applied to it.
5489     RefPtr<StringImpl> oldText = textObj->originalText();
5490     ASSERT(oldText);
5491 
5492     if (oldText && oldText->length() > 0) {
5493         unsigned length = 0;
5494 
5495         // Account for leading spaces and punctuation.
5496         while (length < oldText->length() && shouldSkipForFirstLetter((*oldText)[length]))
5497             length++;
5498 
5499         // Account for first letter.
5500         length++;
5501 
5502         // Keep looking for whitespace and allowed punctuation, but avoid
5503         // accumulating just whitespace into the :first-letter.
5504         for (unsigned scanLength = length; scanLength < oldText->length(); ++scanLength) {
5505             UChar c = (*oldText)[scanLength];
5506 
5507             if (!shouldSkipForFirstLetter(c))
5508                 break;
5509 
5510             if (isPunctuationForFirstLetter(c))
5511                 length = scanLength + 1;
5512          }
5513 
5514         // Construct a text fragment for the text after the first letter.
5515         // This text fragment might be empty.
5516         RenderTextFragment* remainingText =
5517             new (renderArena()) RenderTextFragment(textObj->node() ? textObj->node() : textObj->document(), oldText.get(), length, oldText->length() - length);
5518         remainingText->setStyle(textObj->style());
5519         if (remainingText->node())
5520             remainingText->node()->setRenderer(remainingText);
5521 
5522         firstLetterContainer->addChild(remainingText, textObj);
5523         firstLetterContainer->removeChild(textObj);
5524         remainingText->setFirstLetter(firstLetter);
5525 
5526         // construct text fragment for the first letter
5527         RenderTextFragment* letter =
5528             new (renderArena()) RenderTextFragment(remainingText->node() ? remainingText->node() : remainingText->document(), oldText.get(), 0, length);
5529         letter->setStyle(pseudoStyle);
5530         firstLetter->addChild(letter);
5531 
5532         textObj->destroy();
5533     }
5534     view()->enableLayoutState();
5535 }
5536 
5537 // Helper methods for obtaining the last line, computing line counts and heights for line counts
5538 // (crawling into blocks).
shouldCheckLines(RenderObject * obj)5539 static bool shouldCheckLines(RenderObject* obj)
5540 {
5541     return !obj->isFloatingOrPositioned() && !obj->isRunIn() &&
5542             obj->isBlockFlow() && obj->style()->height().isAuto() &&
5543             (!obj->isFlexibleBox() || obj->style()->boxOrient() == VERTICAL);
5544 }
5545 
getLineAtIndex(RenderBlock * block,int i,int & count)5546 static RootInlineBox* getLineAtIndex(RenderBlock* block, int i, int& count)
5547 {
5548     if (block->style()->visibility() == VISIBLE) {
5549         if (block->childrenInline()) {
5550             for (RootInlineBox* box = block->firstRootBox(); box; box = box->nextRootBox()) {
5551                 if (count++ == i)
5552                     return box;
5553             }
5554         }
5555         else {
5556             for (RenderObject* obj = block->firstChild(); obj; obj = obj->nextSibling()) {
5557                 if (shouldCheckLines(obj)) {
5558                     RootInlineBox *box = getLineAtIndex(toRenderBlock(obj), i, count);
5559                     if (box)
5560                         return box;
5561                 }
5562             }
5563         }
5564     }
5565     return 0;
5566 }
5567 
getHeightForLineCount(RenderBlock * block,int l,bool includeBottom,int & count)5568 static int getHeightForLineCount(RenderBlock* block, int l, bool includeBottom, int& count)
5569 {
5570     if (block->style()->visibility() == VISIBLE) {
5571         if (block->childrenInline()) {
5572             for (RootInlineBox* box = block->firstRootBox(); box; box = box->nextRootBox()) {
5573                 if (++count == l)
5574                     return box->lineBottom() + (includeBottom ? (block->borderBottom() + block->paddingBottom()) : 0);
5575             }
5576         }
5577         else {
5578             RenderBox* normalFlowChildWithoutLines = 0;
5579             for (RenderBox* obj = block->firstChildBox(); obj; obj = obj->nextSiblingBox()) {
5580                 if (shouldCheckLines(obj)) {
5581                     int result = getHeightForLineCount(toRenderBlock(obj), l, false, count);
5582                     if (result != -1)
5583                         return result + obj->y() + (includeBottom ? (block->borderBottom() + block->paddingBottom()) : 0);
5584                 }
5585                 else if (!obj->isFloatingOrPositioned() && !obj->isRunIn())
5586                     normalFlowChildWithoutLines = obj;
5587             }
5588             if (normalFlowChildWithoutLines && l == 0)
5589                 return normalFlowChildWithoutLines->y() + normalFlowChildWithoutLines->height();
5590         }
5591     }
5592 
5593     return -1;
5594 }
5595 
lineAtIndex(int i)5596 RootInlineBox* RenderBlock::lineAtIndex(int i)
5597 {
5598     int count = 0;
5599     return getLineAtIndex(this, i, count);
5600 }
5601 
lineCount()5602 int RenderBlock::lineCount()
5603 {
5604     int count = 0;
5605     if (style()->visibility() == VISIBLE) {
5606         if (childrenInline())
5607             for (RootInlineBox* box = firstRootBox(); box; box = box->nextRootBox())
5608                 count++;
5609         else
5610             for (RenderObject* obj = firstChild(); obj; obj = obj->nextSibling())
5611                 if (shouldCheckLines(obj))
5612                     count += toRenderBlock(obj)->lineCount();
5613     }
5614     return count;
5615 }
5616 
heightForLineCount(int l)5617 int RenderBlock::heightForLineCount(int l)
5618 {
5619     int count = 0;
5620     return getHeightForLineCount(this, l, true, count);
5621 }
5622 
adjustForBorderFit(int x,int & left,int & right) const5623 void RenderBlock::adjustForBorderFit(int x, int& left, int& right) const
5624 {
5625     // We don't deal with relative positioning.  Our assumption is that you shrink to fit the lines without accounting
5626     // for either overflow or translations via relative positioning.
5627     if (style()->visibility() == VISIBLE) {
5628         if (childrenInline()) {
5629             for (RootInlineBox* box = firstRootBox(); box; box = box->nextRootBox()) {
5630                 if (box->firstChild())
5631                     left = min(left, x + static_cast<int>(box->firstChild()->x()));
5632                 if (box->lastChild())
5633                     right = max(right, x + static_cast<int>(ceilf(box->lastChild()->logicalRight())));
5634             }
5635         }
5636         else {
5637             for (RenderBox* obj = firstChildBox(); obj; obj = obj->nextSiblingBox()) {
5638                 if (!obj->isFloatingOrPositioned()) {
5639                     if (obj->isBlockFlow() && !obj->hasOverflowClip())
5640                         toRenderBlock(obj)->adjustForBorderFit(x + obj->x(), left, right);
5641                     else if (obj->style()->visibility() == VISIBLE) {
5642                         // We are a replaced element or some kind of non-block-flow object.
5643                         left = min(left, x + obj->x());
5644                         right = max(right, x + obj->x() + obj->width());
5645                     }
5646                 }
5647             }
5648         }
5649 
5650         if (m_floatingObjects) {
5651             FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
5652             FloatingObjectSetIterator end = floatingObjectSet.end();
5653             for (FloatingObjectSetIterator it = floatingObjectSet.begin(); it != end; ++it) {
5654                 FloatingObject* r = *it;
5655                 // Only examine the object if our m_shouldPaint flag is set.
5656                 if (r->m_shouldPaint) {
5657                     int floatLeft = xPositionForFloatIncludingMargin(r) - r->m_renderer->x();
5658                     int floatRight = floatLeft + r->m_renderer->width();
5659                     left = min(left, floatLeft);
5660                     right = max(right, floatRight);
5661                 }
5662             }
5663         }
5664     }
5665 }
5666 
borderFitAdjust(int & x,int & w) const5667 void RenderBlock::borderFitAdjust(int& x, int& w) const
5668 {
5669     if (style()->borderFit() == BorderFitBorder)
5670         return;
5671 
5672     // Walk any normal flow lines to snugly fit.
5673     int left = INT_MAX;
5674     int right = INT_MIN;
5675     int oldWidth = w;
5676     adjustForBorderFit(0, left, right);
5677     if (left != INT_MAX) {
5678         left -= (borderLeft() + paddingLeft());
5679         if (left > 0) {
5680             x += left;
5681             w -= left;
5682         }
5683     }
5684     if (right != INT_MIN) {
5685         right += (borderRight() + paddingRight());
5686         if (right < oldWidth)
5687             w -= (oldWidth - right);
5688     }
5689 }
5690 
clearTruncation()5691 void RenderBlock::clearTruncation()
5692 {
5693     if (style()->visibility() == VISIBLE) {
5694         if (childrenInline() && hasMarkupTruncation()) {
5695             setHasMarkupTruncation(false);
5696             for (RootInlineBox* box = firstRootBox(); box; box = box->nextRootBox())
5697                 box->clearTruncation();
5698         }
5699         else
5700             for (RenderObject* obj = firstChild(); obj; obj = obj->nextSibling())
5701                 if (shouldCheckLines(obj))
5702                     toRenderBlock(obj)->clearTruncation();
5703     }
5704 }
5705 
setMaxMarginBeforeValues(int pos,int neg)5706 void RenderBlock::setMaxMarginBeforeValues(int pos, int neg)
5707 {
5708     if (!m_rareData) {
5709         if (pos == RenderBlockRareData::positiveMarginBeforeDefault(this) && neg == RenderBlockRareData::negativeMarginBeforeDefault(this))
5710             return;
5711         m_rareData = new RenderBlockRareData(this);
5712     }
5713     m_rareData->m_margins.setPositiveMarginBefore(pos);
5714     m_rareData->m_margins.setNegativeMarginBefore(neg);
5715 }
5716 
setMaxMarginAfterValues(int pos,int neg)5717 void RenderBlock::setMaxMarginAfterValues(int pos, int neg)
5718 {
5719     if (!m_rareData) {
5720         if (pos == RenderBlockRareData::positiveMarginAfterDefault(this) && neg == RenderBlockRareData::negativeMarginAfterDefault(this))
5721             return;
5722         m_rareData = new RenderBlockRareData(this);
5723     }
5724     m_rareData->m_margins.setPositiveMarginAfter(pos);
5725     m_rareData->m_margins.setNegativeMarginAfter(neg);
5726 }
5727 
setPaginationStrut(int strut)5728 void RenderBlock::setPaginationStrut(int strut)
5729 {
5730     if (!m_rareData) {
5731         if (!strut)
5732             return;
5733         m_rareData = new RenderBlockRareData(this);
5734     }
5735     m_rareData->m_paginationStrut = strut;
5736 }
5737 
setPageLogicalOffset(int logicalOffset)5738 void RenderBlock::setPageLogicalOffset(int logicalOffset)
5739 {
5740     if (!m_rareData) {
5741         if (!logicalOffset)
5742             return;
5743         m_rareData = new RenderBlockRareData(this);
5744     }
5745     m_rareData->m_pageLogicalOffset = logicalOffset;
5746 }
5747 
absoluteRects(Vector<IntRect> & rects,int tx,int ty)5748 void RenderBlock::absoluteRects(Vector<IntRect>& rects, int tx, int ty)
5749 {
5750     // For blocks inside inlines, we go ahead and include margins so that we run right up to the
5751     // inline boxes above and below us (thus getting merged with them to form a single irregular
5752     // shape).
5753     if (isAnonymousBlockContinuation()) {
5754         // FIXME: This is wrong for block-flows that are horizontal.
5755         // https://bugs.webkit.org/show_bug.cgi?id=46781
5756         rects.append(IntRect(tx, ty - collapsedMarginBefore(),
5757                              width(), height() + collapsedMarginBefore() + collapsedMarginAfter()));
5758         continuation()->absoluteRects(rects,
5759                                       tx - x() + inlineElementContinuation()->containingBlock()->x(),
5760                                       ty - y() + inlineElementContinuation()->containingBlock()->y());
5761     } else
5762         rects.append(IntRect(tx, ty, width(), height()));
5763 }
5764 
absoluteQuads(Vector<FloatQuad> & quads)5765 void RenderBlock::absoluteQuads(Vector<FloatQuad>& quads)
5766 {
5767     // For blocks inside inlines, we go ahead and include margins so that we run right up to the
5768     // inline boxes above and below us (thus getting merged with them to form a single irregular
5769     // shape).
5770     if (isAnonymousBlockContinuation()) {
5771         // FIXME: This is wrong for block-flows that are horizontal.
5772         // https://bugs.webkit.org/show_bug.cgi?id=46781
5773         FloatRect localRect(0, -collapsedMarginBefore(),
5774                             width(), height() + collapsedMarginBefore() + collapsedMarginAfter());
5775         quads.append(localToAbsoluteQuad(localRect));
5776         continuation()->absoluteQuads(quads);
5777     } else
5778         quads.append(RenderBox::localToAbsoluteQuad(FloatRect(0, 0, width(), height())));
5779 }
5780 
rectWithOutlineForRepaint(RenderBoxModelObject * repaintContainer,int outlineWidth)5781 IntRect RenderBlock::rectWithOutlineForRepaint(RenderBoxModelObject* repaintContainer, int outlineWidth)
5782 {
5783     IntRect r(RenderBox::rectWithOutlineForRepaint(repaintContainer, outlineWidth));
5784     if (isAnonymousBlockContinuation())
5785         r.inflateY(collapsedMarginBefore()); // FIXME: This is wrong for block-flows that are horizontal.
5786     return r;
5787 }
5788 
hoverAncestor() const5789 RenderObject* RenderBlock::hoverAncestor() const
5790 {
5791     return isAnonymousBlockContinuation() ? continuation() : RenderBox::hoverAncestor();
5792 }
5793 
updateDragState(bool dragOn)5794 void RenderBlock::updateDragState(bool dragOn)
5795 {
5796     RenderBox::updateDragState(dragOn);
5797     if (continuation())
5798         continuation()->updateDragState(dragOn);
5799 }
5800 
outlineStyleForRepaint() const5801 RenderStyle* RenderBlock::outlineStyleForRepaint() const
5802 {
5803     return isAnonymousBlockContinuation() ? continuation()->style() : style();
5804 }
5805 
childBecameNonInline(RenderObject *)5806 void RenderBlock::childBecameNonInline(RenderObject*)
5807 {
5808     makeChildrenNonInline();
5809     if (isAnonymousBlock() && parent() && parent()->isRenderBlock())
5810         toRenderBlock(parent())->removeLeftoverAnonymousBlock(this);
5811     // |this| may be dead here
5812 }
5813 
updateHitTestResult(HitTestResult & result,const IntPoint & point)5814 void RenderBlock::updateHitTestResult(HitTestResult& result, const IntPoint& point)
5815 {
5816     if (result.innerNode())
5817         return;
5818 
5819     Node* n = node();
5820     if (isAnonymousBlockContinuation())
5821         // We are in the margins of block elements that are part of a continuation.  In
5822         // this case we're actually still inside the enclosing element that was
5823         // split.  Go ahead and set our inner node accordingly.
5824         n = continuation()->node();
5825 
5826     if (n) {
5827         result.setInnerNode(n);
5828         if (!result.innerNonSharedNode())
5829             result.setInnerNonSharedNode(n);
5830         result.setLocalPoint(point);
5831     }
5832 }
5833 
localCaretRect(InlineBox * inlineBox,int caretOffset,int * extraWidthToEndOfLine)5834 IntRect RenderBlock::localCaretRect(InlineBox* inlineBox, int caretOffset, int* extraWidthToEndOfLine)
5835 {
5836     // Do the normal calculation in most cases.
5837     if (firstChild())
5838         return RenderBox::localCaretRect(inlineBox, caretOffset, extraWidthToEndOfLine);
5839 
5840     // This is a special case:
5841     // The element is not an inline element, and it's empty. So we have to
5842     // calculate a fake position to indicate where objects are to be inserted.
5843 
5844     // FIXME: This does not take into account either :first-line or :first-letter
5845     // However, as soon as some content is entered, the line boxes will be
5846     // constructed and this kludge is not called any more. So only the caret size
5847     // of an empty :first-line'd block is wrong. I think we can live with that.
5848     RenderStyle* currentStyle = firstLineStyle();
5849     int height = lineHeight(true, currentStyle->isHorizontalWritingMode() ? HorizontalLine : VerticalLine);
5850 
5851     enum CaretAlignment { alignLeft, alignRight, alignCenter };
5852 
5853     CaretAlignment alignment = alignLeft;
5854 
5855     switch (currentStyle->textAlign()) {
5856         case TAAUTO:
5857         case JUSTIFY:
5858             if (!currentStyle->isLeftToRightDirection())
5859                 alignment = alignRight;
5860             break;
5861         case LEFT:
5862         case WEBKIT_LEFT:
5863             break;
5864         case CENTER:
5865         case WEBKIT_CENTER:
5866             alignment = alignCenter;
5867             break;
5868         case RIGHT:
5869         case WEBKIT_RIGHT:
5870             alignment = alignRight;
5871             break;
5872         case TASTART:
5873             if (!currentStyle->isLeftToRightDirection())
5874                 alignment = alignRight;
5875             break;
5876         case TAEND:
5877             if (currentStyle->isLeftToRightDirection())
5878                 alignment = alignRight;
5879             break;
5880     }
5881 
5882     int x = borderLeft() + paddingLeft();
5883     int w = width();
5884 
5885     switch (alignment) {
5886         case alignLeft:
5887             break;
5888         case alignCenter:
5889             x = (x + w - (borderRight() + paddingRight())) / 2;
5890             break;
5891         case alignRight:
5892             x = w - (borderRight() + paddingRight()) - caretWidth;
5893             break;
5894     }
5895 
5896     if (extraWidthToEndOfLine) {
5897         if (isRenderBlock()) {
5898             *extraWidthToEndOfLine = w - (x + caretWidth);
5899         } else {
5900             // FIXME: This code looks wrong.
5901             // myRight and containerRight are set up, but then clobbered.
5902             // So *extraWidthToEndOfLine will always be 0 here.
5903 
5904             int myRight = x + caretWidth;
5905             // FIXME: why call localToAbsoluteForContent() twice here, too?
5906             FloatPoint absRightPoint = localToAbsolute(FloatPoint(myRight, 0));
5907 
5908             int containerRight = containingBlock()->x() + containingBlockLogicalWidthForContent();
5909             FloatPoint absContainerPoint = localToAbsolute(FloatPoint(containerRight, 0));
5910 
5911             *extraWidthToEndOfLine = absContainerPoint.x() - absRightPoint.x();
5912         }
5913     }
5914 
5915     int y = paddingTop() + borderTop();
5916 
5917     return IntRect(x, y, caretWidth, height);
5918 }
5919 
addFocusRingRects(Vector<IntRect> & rects,int tx,int ty)5920 void RenderBlock::addFocusRingRects(Vector<IntRect>& rects, int tx, int ty)
5921 {
5922     // For blocks inside inlines, we go ahead and include margins so that we run right up to the
5923     // inline boxes above and below us (thus getting merged with them to form a single irregular
5924     // shape).
5925     if (inlineElementContinuation()) {
5926         // FIXME: This check really isn't accurate.
5927         bool nextInlineHasLineBox = inlineElementContinuation()->firstLineBox();
5928         // FIXME: This is wrong. The principal renderer may not be the continuation preceding this block.
5929         // FIXME: This is wrong for block-flows that are horizontal.
5930         // https://bugs.webkit.org/show_bug.cgi?id=46781
5931         bool prevInlineHasLineBox = toRenderInline(inlineElementContinuation()->node()->renderer())->firstLineBox();
5932         int topMargin = prevInlineHasLineBox ? collapsedMarginBefore() : 0;
5933         int bottomMargin = nextInlineHasLineBox ? collapsedMarginAfter() : 0;
5934         IntRect rect(tx, ty - topMargin, width(), height() + topMargin + bottomMargin);
5935         if (!rect.isEmpty())
5936             rects.append(rect);
5937     } else if (width() && height())
5938         rects.append(IntRect(tx, ty, width(), height()));
5939 
5940     if (!hasOverflowClip() && !hasControlClip()) {
5941         for (RootInlineBox* curr = firstRootBox(); curr; curr = curr->nextRootBox()) {
5942             int top = max(curr->lineTop(), curr->logicalTop());
5943             int bottom = min(curr->lineBottom(), curr->logicalTop() + curr->logicalHeight());
5944             IntRect rect(tx + curr->x(), ty + top, curr->logicalWidth(), bottom - top);
5945             if (!rect.isEmpty())
5946                 rects.append(rect);
5947         }
5948 
5949         for (RenderObject* curr = firstChild(); curr; curr = curr->nextSibling()) {
5950             if (!curr->isText() && !curr->isListMarker() && curr->isBox()) {
5951                 RenderBox* box = toRenderBox(curr);
5952                 FloatPoint pos;
5953                 // FIXME: This doesn't work correctly with transforms.
5954                 if (box->layer())
5955                     pos = curr->localToAbsolute();
5956                 else
5957                     pos = FloatPoint(tx + box->x(), ty + box->y());
5958                 box->addFocusRingRects(rects, pos.x(), pos.y());
5959             }
5960         }
5961     }
5962 
5963     if (inlineElementContinuation())
5964         inlineElementContinuation()->addFocusRingRects(rects,
5965                                                        tx - x() + inlineElementContinuation()->containingBlock()->x(),
5966                                                        ty - y() + inlineElementContinuation()->containingBlock()->y());
5967 }
5968 
createAnonymousBlock(bool isFlexibleBox) const5969 RenderBlock* RenderBlock::createAnonymousBlock(bool isFlexibleBox) const
5970 {
5971     RefPtr<RenderStyle> newStyle = RenderStyle::createAnonymousStyle(style());
5972 
5973     RenderBlock* newBox = 0;
5974     if (isFlexibleBox) {
5975         newStyle->setDisplay(BOX);
5976         newBox = new (renderArena()) RenderFlexibleBox(document() /* anonymous box */);
5977     } else {
5978         newStyle->setDisplay(BLOCK);
5979         newBox = new (renderArena()) RenderBlock(document() /* anonymous box */);
5980     }
5981 
5982     newBox->setStyle(newStyle.release());
5983     return newBox;
5984 }
5985 
createAnonymousBlockWithSameTypeAs(RenderBlock * otherAnonymousBlock) const5986 RenderBlock* RenderBlock::createAnonymousBlockWithSameTypeAs(RenderBlock* otherAnonymousBlock) const
5987 {
5988     if (otherAnonymousBlock->isAnonymousColumnsBlock())
5989         return createAnonymousColumnsBlock();
5990     if (otherAnonymousBlock->isAnonymousColumnSpanBlock())
5991         return createAnonymousColumnSpanBlock();
5992     return createAnonymousBlock(otherAnonymousBlock->style()->display() == BOX);
5993 }
5994 
createAnonymousColumnsBlock() const5995 RenderBlock* RenderBlock::createAnonymousColumnsBlock() const
5996 {
5997     RefPtr<RenderStyle> newStyle = RenderStyle::createAnonymousStyle(style());
5998     newStyle->inheritColumnPropertiesFrom(style());
5999     newStyle->setDisplay(BLOCK);
6000 
6001     RenderBlock* newBox = new (renderArena()) RenderBlock(document() /* anonymous box */);
6002     newBox->setStyle(newStyle.release());
6003     return newBox;
6004 }
6005 
createAnonymousColumnSpanBlock() const6006 RenderBlock* RenderBlock::createAnonymousColumnSpanBlock() const
6007 {
6008     RefPtr<RenderStyle> newStyle = RenderStyle::createAnonymousStyle(style());
6009     newStyle->setColumnSpan(true);
6010     newStyle->setDisplay(BLOCK);
6011 
6012     RenderBlock* newBox = new (renderArena()) RenderBlock(document() /* anonymous box */);
6013     newBox->setStyle(newStyle.release());
6014     return newBox;
6015 }
6016 
nextPageLogicalTop(int logicalOffset) const6017 int RenderBlock::nextPageLogicalTop(int logicalOffset) const
6018 {
6019     LayoutState* layoutState = view()->layoutState();
6020     if (!layoutState->m_pageLogicalHeight)
6021         return logicalOffset;
6022 
6023     // The logicalOffset is in our coordinate space.  We can add in our pushed offset.
6024     int pageLogicalHeight = layoutState->m_pageLogicalHeight;
6025     IntSize delta = layoutState->m_layoutOffset - layoutState->m_pageOffset;
6026     int offset = isHorizontalWritingMode() ? delta.height() : delta.width();
6027     int remainingLogicalHeight = (pageLogicalHeight - (offset + logicalOffset) % pageLogicalHeight) % pageLogicalHeight;
6028     return logicalOffset + remainingLogicalHeight;
6029 }
6030 
inNormalFlow(RenderBox * child)6031 static bool inNormalFlow(RenderBox* child)
6032 {
6033     RenderBlock* curr = child->containingBlock();
6034     RenderBlock* initialBlock = child->view();
6035     while (curr && curr != initialBlock) {
6036         if (curr->hasColumns())
6037             return true;
6038         if (curr->isFloatingOrPositioned())
6039             return false;
6040         curr = curr->containingBlock();
6041     }
6042     return true;
6043 }
6044 
applyBeforeBreak(RenderBox * child,int logicalOffset)6045 int RenderBlock::applyBeforeBreak(RenderBox* child, int logicalOffset)
6046 {
6047     // FIXME: Add page break checking here when we support printing.
6048     bool checkColumnBreaks = view()->layoutState()->isPaginatingColumns();
6049     bool checkPageBreaks = !checkColumnBreaks && view()->layoutState()->m_pageLogicalHeight; // FIXME: Once columns can print we have to check this.
6050     bool checkBeforeAlways = (checkColumnBreaks && child->style()->columnBreakBefore() == PBALWAYS) || (checkPageBreaks && child->style()->pageBreakBefore() == PBALWAYS);
6051     if (checkBeforeAlways && inNormalFlow(child)) {
6052         if (checkColumnBreaks)
6053             view()->layoutState()->addForcedColumnBreak(logicalOffset);
6054         return nextPageLogicalTop(logicalOffset);
6055     }
6056     return logicalOffset;
6057 }
6058 
applyAfterBreak(RenderBox * child,int logicalOffset,MarginInfo & marginInfo)6059 int RenderBlock::applyAfterBreak(RenderBox* child, int logicalOffset, MarginInfo& marginInfo)
6060 {
6061     // FIXME: Add page break checking here when we support printing.
6062     bool checkColumnBreaks = view()->layoutState()->isPaginatingColumns();
6063     bool checkPageBreaks = !checkColumnBreaks && view()->layoutState()->m_pageLogicalHeight; // FIXME: Once columns can print we have to check this.
6064     bool checkAfterAlways = (checkColumnBreaks && child->style()->columnBreakAfter() == PBALWAYS) || (checkPageBreaks && child->style()->pageBreakAfter() == PBALWAYS);
6065     if (checkAfterAlways && inNormalFlow(child)) {
6066         marginInfo.setMarginAfterQuirk(true); // Cause margins to be discarded for any following content.
6067         if (checkColumnBreaks)
6068             view()->layoutState()->addForcedColumnBreak(logicalOffset);
6069         return nextPageLogicalTop(logicalOffset);
6070     }
6071     return logicalOffset;
6072 }
6073 
adjustForUnsplittableChild(RenderBox * child,int logicalOffset,bool includeMargins)6074 int RenderBlock::adjustForUnsplittableChild(RenderBox* child, int logicalOffset, bool includeMargins)
6075 {
6076     bool isUnsplittable = child->isReplaced() || child->scrollsOverflow();
6077     if (!isUnsplittable)
6078         return logicalOffset;
6079     int childLogicalHeight = logicalHeightForChild(child) + (includeMargins ? marginBeforeForChild(child) + marginAfterForChild(child) : 0);
6080     LayoutState* layoutState = view()->layoutState();
6081     if (layoutState->m_columnInfo)
6082         layoutState->m_columnInfo->updateMinimumColumnHeight(childLogicalHeight);
6083     int pageLogicalHeight = layoutState->m_pageLogicalHeight;
6084     if (!pageLogicalHeight || childLogicalHeight > pageLogicalHeight)
6085         return logicalOffset;
6086     IntSize delta = layoutState->m_layoutOffset - layoutState->m_pageOffset;
6087     int offset = isHorizontalWritingMode() ? delta.height() : delta.width();
6088     int remainingLogicalHeight = (pageLogicalHeight - (offset + logicalOffset) % pageLogicalHeight) % pageLogicalHeight;
6089     if (remainingLogicalHeight < childLogicalHeight)
6090         return logicalOffset + remainingLogicalHeight;
6091     return logicalOffset;
6092 }
6093 
adjustLinePositionForPagination(RootInlineBox * lineBox,int & delta)6094 void RenderBlock::adjustLinePositionForPagination(RootInlineBox* lineBox, int& delta)
6095 {
6096     // FIXME: For now we paginate using line overflow.  This ensures that lines don't overlap at all when we
6097     // put a strut between them for pagination purposes.  However, this really isn't the desired rendering, since
6098     // the line on the top of the next page will appear too far down relative to the same kind of line at the top
6099     // of the first column.
6100     //
6101     // The rendering we would like to see is one where the lineTop is at the top of the column, and any line overflow
6102     // simply spills out above the top of the column.  This effect would match what happens at the top of the first column.
6103     // We can't achieve this rendering, however, until we stop columns from clipping to the column bounds (thus allowing
6104     // for overflow to occur), and then cache visible overflow for each column rect.
6105     //
6106     // Furthermore, the paint we have to do when a column has overflow has to be special.  We need to exclude
6107     // content that paints in a previous column (and content that paints in the following column).
6108     //
6109     // FIXME: Another problem with simply moving lines is that the available line width may change (because of floats).
6110     // Technically if the location we move the line to has a different line width than our old position, then we need to dirty the
6111     // line and all following lines.
6112     LayoutState* layoutState = view()->layoutState();
6113     int pageLogicalHeight = layoutState->m_pageLogicalHeight;
6114     IntRect logicalVisualOverflow = lineBox->logicalVisualOverflowRect(lineBox->lineTop(), lineBox->lineBottom());
6115     int logicalOffset = logicalVisualOverflow.y();
6116     int lineHeight = logicalVisualOverflow.maxY() - logicalOffset;
6117     if (layoutState->m_columnInfo)
6118         layoutState->m_columnInfo->updateMinimumColumnHeight(lineHeight);
6119     logicalOffset += delta;
6120     lineBox->setPaginationStrut(0);
6121     if (!pageLogicalHeight || lineHeight > pageLogicalHeight)
6122         return;
6123     IntSize offsetDelta = layoutState->m_layoutOffset - layoutState->m_pageOffset;
6124     int offset = isHorizontalWritingMode() ? offsetDelta.height() : offsetDelta.width();
6125     int remainingLogicalHeight = pageLogicalHeight - (offset + logicalOffset) % pageLogicalHeight;
6126     if (remainingLogicalHeight < lineHeight) {
6127         int totalLogicalHeight = lineHeight + max(0, logicalOffset);
6128         if (lineBox == firstRootBox() && totalLogicalHeight < pageLogicalHeight && !isPositioned() && !isTableCell())
6129             setPaginationStrut(remainingLogicalHeight + max(0, logicalOffset));
6130         else {
6131             delta += remainingLogicalHeight;
6132             lineBox->setPaginationStrut(remainingLogicalHeight);
6133         }
6134     }
6135 }
6136 
collapsedMarginBeforeForChild(RenderBox * child) const6137 int RenderBlock::collapsedMarginBeforeForChild(RenderBox* child) const
6138 {
6139     // If the child has the same directionality as we do, then we can just return its
6140     // collapsed margin.
6141     if (!child->isWritingModeRoot())
6142         return child->collapsedMarginBefore();
6143 
6144     // The child has a different directionality.  If the child is parallel, then it's just
6145     // flipped relative to us.  We can use the collapsed margin for the opposite edge.
6146     if (child->isHorizontalWritingMode() == isHorizontalWritingMode())
6147         return child->collapsedMarginAfter();
6148 
6149     // The child is perpendicular to us, which means its margins don't collapse but are on the
6150     // "logical left/right" sides of the child box.  We can just return the raw margin in this case.
6151     return marginBeforeForChild(child);
6152 }
6153 
collapsedMarginAfterForChild(RenderBox * child) const6154 int RenderBlock::collapsedMarginAfterForChild(RenderBox* child) const
6155 {
6156     // If the child has the same directionality as we do, then we can just return its
6157     // collapsed margin.
6158     if (!child->isWritingModeRoot())
6159         return child->collapsedMarginAfter();
6160 
6161     // The child has a different directionality.  If the child is parallel, then it's just
6162     // flipped relative to us.  We can use the collapsed margin for the opposite edge.
6163     if (child->isHorizontalWritingMode() == isHorizontalWritingMode())
6164         return child->collapsedMarginBefore();
6165 
6166     // The child is perpendicular to us, which means its margins don't collapse but are on the
6167     // "logical left/right" side of the child box.  We can just return the raw margin in this case.
6168     return marginAfterForChild(child);
6169 }
6170 
marginBeforeForChild(RenderBoxModelObject * child) const6171 int RenderBlock::marginBeforeForChild(RenderBoxModelObject* child) const
6172 {
6173     switch (style()->writingMode()) {
6174     case TopToBottomWritingMode:
6175         return child->marginTop();
6176     case BottomToTopWritingMode:
6177         return child->marginBottom();
6178     case LeftToRightWritingMode:
6179         return child->marginLeft();
6180     case RightToLeftWritingMode:
6181         return child->marginRight();
6182     }
6183     ASSERT_NOT_REACHED();
6184     return child->marginTop();
6185 }
6186 
marginAfterForChild(RenderBoxModelObject * child) const6187 int RenderBlock::marginAfterForChild(RenderBoxModelObject* child) const
6188 {
6189     switch (style()->writingMode()) {
6190     case TopToBottomWritingMode:
6191         return child->marginBottom();
6192     case BottomToTopWritingMode:
6193         return child->marginTop();
6194     case LeftToRightWritingMode:
6195         return child->marginRight();
6196     case RightToLeftWritingMode:
6197         return child->marginLeft();
6198     }
6199     ASSERT_NOT_REACHED();
6200     return child->marginBottom();
6201 }
6202 
marginStartForChild(RenderBoxModelObject * child) const6203 int RenderBlock::marginStartForChild(RenderBoxModelObject* child) const
6204 {
6205     if (isHorizontalWritingMode())
6206         return style()->isLeftToRightDirection() ? child->marginLeft() : child->marginRight();
6207     return style()->isLeftToRightDirection() ? child->marginTop() : child->marginBottom();
6208 }
6209 
marginEndForChild(RenderBoxModelObject * child) const6210 int RenderBlock::marginEndForChild(RenderBoxModelObject* child) const
6211 {
6212     if (isHorizontalWritingMode())
6213         return style()->isLeftToRightDirection() ? child->marginRight() : child->marginLeft();
6214     return style()->isLeftToRightDirection() ? child->marginBottom() : child->marginTop();
6215 }
6216 
setMarginStartForChild(RenderBox * child,int margin)6217 void RenderBlock::setMarginStartForChild(RenderBox* child, int margin)
6218 {
6219     if (isHorizontalWritingMode()) {
6220         if (style()->isLeftToRightDirection())
6221             child->setMarginLeft(margin);
6222         else
6223             child->setMarginRight(margin);
6224     } else {
6225         if (style()->isLeftToRightDirection())
6226             child->setMarginTop(margin);
6227         else
6228             child->setMarginBottom(margin);
6229     }
6230 }
6231 
setMarginEndForChild(RenderBox * child,int margin)6232 void RenderBlock::setMarginEndForChild(RenderBox* child, int margin)
6233 {
6234     if (isHorizontalWritingMode()) {
6235         if (style()->isLeftToRightDirection())
6236             child->setMarginRight(margin);
6237         else
6238             child->setMarginLeft(margin);
6239     } else {
6240         if (style()->isLeftToRightDirection())
6241             child->setMarginBottom(margin);
6242         else
6243             child->setMarginTop(margin);
6244     }
6245 }
6246 
setMarginBeforeForChild(RenderBox * child,int margin)6247 void RenderBlock::setMarginBeforeForChild(RenderBox* child, int margin)
6248 {
6249     switch (style()->writingMode()) {
6250     case TopToBottomWritingMode:
6251         child->setMarginTop(margin);
6252         break;
6253     case BottomToTopWritingMode:
6254         child->setMarginBottom(margin);
6255         break;
6256     case LeftToRightWritingMode:
6257         child->setMarginLeft(margin);
6258         break;
6259     case RightToLeftWritingMode:
6260         child->setMarginRight(margin);
6261         break;
6262     }
6263 }
6264 
setMarginAfterForChild(RenderBox * child,int margin)6265 void RenderBlock::setMarginAfterForChild(RenderBox* child, int margin)
6266 {
6267     switch (style()->writingMode()) {
6268     case TopToBottomWritingMode:
6269         child->setMarginBottom(margin);
6270         break;
6271     case BottomToTopWritingMode:
6272         child->setMarginTop(margin);
6273         break;
6274     case LeftToRightWritingMode:
6275         child->setMarginRight(margin);
6276         break;
6277     case RightToLeftWritingMode:
6278         child->setMarginLeft(margin);
6279         break;
6280     }
6281 }
6282 
marginValuesForChild(RenderBox * child)6283 RenderBlock::MarginValues RenderBlock::marginValuesForChild(RenderBox* child)
6284 {
6285     int childBeforePositive = 0;
6286     int childBeforeNegative = 0;
6287     int childAfterPositive = 0;
6288     int childAfterNegative = 0;
6289 
6290     int beforeMargin = 0;
6291     int afterMargin = 0;
6292 
6293     RenderBlock* childRenderBlock = child->isRenderBlock() ? toRenderBlock(child) : 0;
6294 
6295     // If the child has the same directionality as we do, then we can just return its
6296     // margins in the same direction.
6297     if (!child->isWritingModeRoot()) {
6298         if (childRenderBlock) {
6299             childBeforePositive = childRenderBlock->maxPositiveMarginBefore();
6300             childBeforeNegative = childRenderBlock->maxNegativeMarginBefore();
6301             childAfterPositive = childRenderBlock->maxPositiveMarginAfter();
6302             childAfterNegative = childRenderBlock->maxNegativeMarginAfter();
6303         } else {
6304             beforeMargin = child->marginBefore();
6305             afterMargin = child->marginAfter();
6306         }
6307     } else if (child->isHorizontalWritingMode() == isHorizontalWritingMode()) {
6308         // The child has a different directionality.  If the child is parallel, then it's just
6309         // flipped relative to us.  We can use the margins for the opposite edges.
6310         if (childRenderBlock) {
6311             childBeforePositive = childRenderBlock->maxPositiveMarginAfter();
6312             childBeforeNegative = childRenderBlock->maxNegativeMarginAfter();
6313             childAfterPositive = childRenderBlock->maxPositiveMarginBefore();
6314             childAfterNegative = childRenderBlock->maxNegativeMarginBefore();
6315         } else {
6316             beforeMargin = child->marginAfter();
6317             afterMargin = child->marginBefore();
6318         }
6319     } else {
6320         // The child is perpendicular to us, which means its margins don't collapse but are on the
6321         // "logical left/right" sides of the child box.  We can just return the raw margin in this case.
6322         beforeMargin = marginBeforeForChild(child);
6323         afterMargin = marginAfterForChild(child);
6324     }
6325 
6326     // Resolve uncollapsing margins into their positive/negative buckets.
6327     if (beforeMargin) {
6328         if (beforeMargin > 0)
6329             childBeforePositive = beforeMargin;
6330         else
6331             childBeforeNegative = -beforeMargin;
6332     }
6333     if (afterMargin) {
6334         if (afterMargin > 0)
6335             childAfterPositive = afterMargin;
6336         else
6337             childAfterNegative = -afterMargin;
6338     }
6339 
6340     return MarginValues(childBeforePositive, childBeforeNegative, childAfterPositive, childAfterNegative);
6341 }
6342 
renderName() const6343 const char* RenderBlock::renderName() const
6344 {
6345     if (isBody())
6346         return "RenderBody"; // FIXME: Temporary hack until we know that the regression tests pass.
6347 
6348     if (isFloating())
6349         return "RenderBlock (floating)";
6350     if (isPositioned())
6351         return "RenderBlock (positioned)";
6352     if (isAnonymousColumnsBlock())
6353         return "RenderBlock (anonymous multi-column)";
6354     if (isAnonymousColumnSpanBlock())
6355         return "RenderBlock (anonymous multi-column span)";
6356     if (isAnonymousBlock())
6357         return "RenderBlock (anonymous)";
6358     else if (isAnonymous())
6359         return "RenderBlock (generated)";
6360     if (isRelPositioned())
6361         return "RenderBlock (relative positioned)";
6362     if (isRunIn())
6363         return "RenderBlock (run-in)";
6364     return "RenderBlock";
6365 }
6366 
clear()6367 inline void RenderBlock::FloatingObjects::clear()
6368 {
6369     m_set.clear();
6370     m_leftObjectsCount = 0;
6371     m_rightObjectsCount = 0;
6372 }
6373 
increaseObjectsCount(FloatingObject::Type type)6374 inline void RenderBlock::FloatingObjects::increaseObjectsCount(FloatingObject::Type type)
6375 {
6376     if (type == FloatingObject::FloatLeft)
6377         m_leftObjectsCount++;
6378     else
6379         m_rightObjectsCount++;
6380 }
6381 
decreaseObjectsCount(FloatingObject::Type type)6382 inline void RenderBlock::FloatingObjects::decreaseObjectsCount(FloatingObject::Type type)
6383 {
6384     if (type == FloatingObject::FloatLeft)
6385         m_leftObjectsCount--;
6386     else
6387         m_rightObjectsCount--;
6388 }
6389 
6390 } // namespace WebCore
6391