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

Side by Side Diff: Source/core/rendering/RenderMultiColumnFlowThread.cpp

Issue 296413007: [New Multicolumn] Add support for column-span:all (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@359976
Patch Set: Created 6 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2012 Apple Inc. All rights reserved. 2 * Copyright (C) 2012 Apple Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions 5 * modification, are permitted provided that the following conditions
6 * are met: 6 * are met:
7 * 1. Redistributions of source code must retain the above copyright 7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer. 8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright 9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the 10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution. 11 * documentation and/or other materials provided with the distribution.
12 * 12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS IN..0TERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 20 * PROFITS; OR BUSINESS IN..0TERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */ 24 */
25 25
26 #include "config.h" 26 #include "config.h"
27 #include "core/rendering/RenderMultiColumnFlowThread.h" 27 #include "core/rendering/RenderMultiColumnFlowThread.h"
28 28
29 #include "core/rendering/RenderMultiColumnSet.h" 29 #include "core/rendering/RenderMultiColumnSet.h"
30 #include "core/rendering/RenderMultiColumnSpannerPlaceholder.h"
30 31
31 namespace WebCore { 32 namespace WebCore {
32 33
33 RenderMultiColumnFlowThread::RenderMultiColumnFlowThread() 34 RenderMultiColumnFlowThread::RenderMultiColumnFlowThread()
34 : m_columnCount(1) 35 : m_lastSetWorkedOn(0)
36 , m_columnCount(1)
35 , m_columnHeightAvailable(0) 37 , m_columnHeightAvailable(0)
38 , m_inLayout(false)
36 , m_inBalancingPass(false) 39 , m_inBalancingPass(false)
37 , m_needsColumnHeightsRecalculation(false) 40 , m_needsColumnHeightsRecalculation(false)
38 , m_progressionIsInline(true) 41 , m_progressionIsInline(true)
42 , m_beingEvacuated(false)
39 { 43 {
40 setFlowThreadState(InsideInFlowThread); 44 setFlowThreadState(InsideInFlowThread);
41 } 45 }
42 46
43 RenderMultiColumnFlowThread::~RenderMultiColumnFlowThread() 47 RenderMultiColumnFlowThread::~RenderMultiColumnFlowThread()
44 { 48 {
45 } 49 }
46 50
47 RenderMultiColumnFlowThread* RenderMultiColumnFlowThread::createAnonymous(Docume nt& document, RenderStyle* parentStyle) 51 RenderMultiColumnFlowThread* RenderMultiColumnFlowThread::createAnonymous(Docume nt& document, RenderStyle* parentStyle)
48 { 52 {
(...skipping 14 matching lines...) Expand all
63 67
64 RenderMultiColumnSet* RenderMultiColumnFlowThread::lastMultiColumnSet() const 68 RenderMultiColumnSet* RenderMultiColumnFlowThread::lastMultiColumnSet() const
65 { 69 {
66 for (RenderObject* sibling = multiColumnBlockFlow()->lastChild(); sibling; s ibling = sibling->previousSibling()) { 70 for (RenderObject* sibling = multiColumnBlockFlow()->lastChild(); sibling; s ibling = sibling->previousSibling()) {
67 if (sibling->isRenderMultiColumnSet()) 71 if (sibling->isRenderMultiColumnSet())
68 return toRenderMultiColumnSet(sibling); 72 return toRenderMultiColumnSet(sibling);
69 } 73 }
70 return 0; 74 return 0;
71 } 75 }
72 76
73 void RenderMultiColumnFlowThread::addChild(RenderObject* newChild, RenderObject* beforeChild) 77 RenderBox* RenderMultiColumnFlowThread::firstColumnSetOrSpanner() const
74 { 78 {
75 RenderBlockFlow::addChild(newChild, beforeChild); 79 if (RenderObject* sibling = nextSibling()) {
76 if (firstMultiColumnSet()) 80 ASSERT(sibling->isBox());
77 return; 81 ASSERT(sibling->isRenderMultiColumnSet() || findColumnSpannerPlaceholder (toRenderBox(sibling)));
82 return toRenderBox(sibling);
83 }
84 return 0;
85 }
78 86
79 // For now we only create one column set. It's created as soon as the multic ol container gets 87 RenderBox* RenderMultiColumnFlowThread::nextColumnSetOrSpannerSiblingOf(const Re nderBox* child)
80 // any content at all. 88 {
81 RenderMultiColumnSet* newSet = RenderMultiColumnSet::createAnonymous(this, m ultiColumnBlockFlow()->style()); 89 if (!child)
90 return 0;
91 if (RenderObject* sibling = child->nextSibling()) {
92 ASSERT(sibling->isBox());
93 return toRenderBox(sibling);
94 }
95 return 0;
96 }
82 97
83 // Need to skip RenderBlockFlow's implementation of addChild(), or we'd get redirected right 98 RenderBox* RenderMultiColumnFlowThread::previousColumnSetOrSpannerSiblingOf(cons t RenderBox* child)
84 // back here. 99 {
85 multiColumnBlockFlow()->RenderBlock::addChild(newSet); 100 if (!child)
101 return 0;
102 if (RenderObject* sibling = child->previousSibling()) {
103 ASSERT(sibling->isBox());
104 if (sibling->isRenderFlowThread())
105 return 0;
106 return toRenderBox(sibling);
107 }
108 return 0;
109 }
86 110
87 invalidateRegions(); 111 RenderMultiColumnSet* RenderMultiColumnFlowThread::findSetRendering(RenderObject * renderer) const
112 {
113 for (RenderMultiColumnSet* multicolSet = firstMultiColumnSet(); multicolSet; multicolSet = multicolSet->nextSiblingMultiColumnSet()) {
114 if (multicolSet->renders(renderer))
115 return multicolSet;
116 }
117 return 0;
88 } 118 }
89 119
90 void RenderMultiColumnFlowThread::populate() 120 void RenderMultiColumnFlowThread::populate()
91 { 121 {
92 RenderBlockFlow* multicolContainer = multiColumnBlockFlow(); 122 RenderBlockFlow* multicolContainer = multiColumnBlockFlow();
93 ASSERT(!nextSibling()); 123 ASSERT(!nextSibling());
94 // Reparent children preceding the flow thread into the flow thread. It's mu lticol content 124 // Reparent children preceding the flow thread into the flow thread. It's mu lticol content
95 // now. At this point there's obviously nothing after the flow thread, but r enderers (column 125 // now. At this point there's obviously nothing after the flow thread, but r enderers (column
96 // sets and spanners) will be inserted there as we insert elements into the flow thread. 126 // sets and spanners) will be inserted there as we insert elements into the flow thread.
97 multicolContainer->moveChildrenTo(this, multicolContainer->firstChild(), thi s, true); 127 multicolContainer->moveChildrenTo(this, multicolContainer->firstChild(), thi s, true);
98 } 128 }
99 129
100 void RenderMultiColumnFlowThread::evacuateAndDestroy() 130 void RenderMultiColumnFlowThread::evacuateAndDestroy()
101 { 131 {
102 RenderBlockFlow* multicolContainer = multiColumnBlockFlow(); 132 RenderBlockFlow* multicolContainer = multiColumnBlockFlow();
133 m_beingEvacuated = true;
134
135 // First promote all children of the flow thread. Before we move them to the flow thread's
136 // container, we need to unregister the flow thread, so that they aren't jus t re-added again to
137 // the flow thread that we're trying to empty.
138 multicolContainer->resetMultiColumnFlowThread();
139 moveAllChildrenTo(multicolContainer, true);
140
141 // Move spanners back to their original DOM position in the tree, and destro y the placeholders.
142 SpannerMap::iterator it;
143 while ((it = m_spannerMap.begin()) != m_spannerMap.end()) {
144 RenderBox* spanner = it->key;
145 RenderMultiColumnSpannerPlaceholder* placeholder = it->value;
146 RenderBlockFlow* originalContainer = toRenderBlockFlow(placeholder->pare nt());
147 multicolContainer->removeChild(spanner);
148 originalContainer->addChild(spanner, placeholder);
149 placeholder->destroy();
150 m_spannerMap.remove(it);
151 }
103 152
104 // Remove all sets. 153 // Remove all sets.
105 while (RenderMultiColumnSet* columnSet = firstMultiColumnSet()) 154 while (RenderMultiColumnSet* columnSet = firstMultiColumnSet())
106 columnSet->destroy(); 155 columnSet->destroy();
107 156
108 ASSERT(!previousSibling());
109 ASSERT(!nextSibling());
110
111 // Finally we can promote all flow thread's children. Before we move them to the flow thread's
112 // container, we need to unregister the flow thread, so that they aren't jus t re-added again to
113 // the flow thread that we're trying to empty.
114 multicolContainer->resetMultiColumnFlowThread();
115 moveAllChildrenTo(multicolContainer, true);
116
117 // FIXME: it's scary that neither destroy() nor the move*Children* methods t ake care of this, 157 // FIXME: it's scary that neither destroy() nor the move*Children* methods t ake care of this,
118 // and instead leave you with dangling root line box pointers. But since thi s is how it is done 158 // and instead leave you with dangling root line box pointers. But since thi s is how it is done
119 // in other parts of the code that deal with reparenting renderers, let's do the cleanup on our 159 // in other parts of the code that deal with reparenting renderers, let's do the cleanup on our
120 // own here as well. 160 // own here as well.
121 deleteLineBoxTree(); 161 deleteLineBoxTree();
122 162
123 destroy(); 163 destroy();
124 } 164 }
125 165
126 LayoutSize RenderMultiColumnFlowThread::columnOffset(const LayoutPoint& point) c onst 166 LayoutSize RenderMultiColumnFlowThread::columnOffset(const LayoutPoint& point) c onst
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after
160 m_needsColumnHeightsRecalculation = false; 200 m_needsColumnHeightsRecalculation = false;
161 return; 201 return;
162 } 202 }
163 203
164 for (RenderMultiColumnSet* columnSet = firstMultiColumnSet(); columnSet; col umnSet = columnSet->nextSiblingMultiColumnSet()) { 204 for (RenderMultiColumnSet* columnSet = firstMultiColumnSet(); columnSet; col umnSet = columnSet->nextSiblingMultiColumnSet()) {
165 if (!m_inBalancingPass) { 205 if (!m_inBalancingPass) {
166 // This is the initial layout pass. We need to reset the column heig ht, because contents 206 // This is the initial layout pass. We need to reset the column heig ht, because contents
167 // typically have changed. 207 // typically have changed.
168 columnSet->resetColumnHeight(); 208 columnSet->resetColumnHeight();
169 } 209 }
210 columnSet->resetFlow();
170 } 211 }
171 212
172 invalidateRegions(); 213 invalidateRegions();
173 m_needsColumnHeightsRecalculation = heightIsAuto(); 214 m_needsColumnHeightsRecalculation = true;
174 layout(); 215 layout();
175 } 216 }
176 217
177 bool RenderMultiColumnFlowThread::recalculateColumnHeights() 218 bool RenderMultiColumnFlowThread::recalculateColumnHeights()
178 { 219 {
179 // All column sets that needed layout have now been laid out, so we can fina lly validate them. 220 // All column sets that needed layout have now been laid out, so we can fina lly validate them.
180 validateRegions(); 221 validateRegions();
181 222
182 if (!m_needsColumnHeightsRecalculation) 223 if (!m_needsColumnHeightsRecalculation)
183 return false; 224 return false;
(...skipping 66 matching lines...) Expand 10 before | Expand all | Expand 10 after
250 { 291 {
251 // Detach all column sets from the flow thread. Cannot destroy them at this point, since they 292 // Detach all column sets from the flow thread. Cannot destroy them at this point, since they
252 // are siblings of this object, and there may be pointers to this object's s ibling somewhere 293 // are siblings of this object, and there may be pointers to this object's s ibling somewhere
253 // further up on the call stack. 294 // further up on the call stack.
254 for (RenderMultiColumnSet* columnSet = firstMultiColumnSet(); columnSet; col umnSet = columnSet->nextSiblingMultiColumnSet()) 295 for (RenderMultiColumnSet* columnSet = firstMultiColumnSet(); columnSet; col umnSet = columnSet->nextSiblingMultiColumnSet())
255 columnSet->detachRegion(); 296 columnSet->detachRegion();
256 multiColumnBlockFlow()->resetMultiColumnFlowThread(); 297 multiColumnBlockFlow()->resetMultiColumnFlowThread();
257 RenderFlowThread::willBeRemovedFromTree(); 298 RenderFlowThread::willBeRemovedFromTree();
258 } 299 }
259 300
301 RenderObject* RenderMultiColumnFlowThread::resolveMovedChild(RenderObject* child ) const
302 {
303 if (child->style()->columnSpan() != ColumnSpanAll || !child->isBox()) {
304 // We only need to resolve for column spanners.
305 return child;
306 }
307 // The renderer for the actual DOM node that establishes a spanner is moved from its original
308 // location in the render tree to becoming a sibling of the column sets. In other words, it's
309 // moved out from the flow thread (and becomes a sibling of it). When we for instance want to
310 // create and insert a renderer for the sibling node immediately preceding t he spanner, we need
311 // to map that spanner renderer to the spanner's placeholder, which is where the new inserted
312 // renderer belongs.
313 if (RenderMultiColumnSpannerPlaceholder* placeholder = findColumnSpannerPlac eholder(toRenderBox(child)))
314 return placeholder;
315
316 // This is an invalid spanner, or its placeholder hasn't been created yet. T his happens when
317 // moving an entire subtree into the flow thread, when we are processing the insertion of this
318 // spanner's preceding sibling, and we obviously haven't got as far as proce ssing this spanner
319 // yet.
320 return child;
321 }
322
323 static bool isValidColumnSpanner(RenderMultiColumnFlowThread* flowThread, Render Object* descendant)
rune 2014/06/19 14:24:32 Why not make this a member method of the RenderMul
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done.
rune 2014/08/27 08:17:38 Acknowledged.
324 {
325 // We assume that we're inside the flow thread. This function is not to be c alled otherwise.
326 ASSERT(descendant->isDescendantOf(flowThread));
327
328 if (flowThread->isRenderPagedFlowThread())
329 return false; // Spanners are for true multicol only.
330
331 // First make sure that the renderer itself has the right properties for bec oming a spanner.
332 RenderStyle* style = descendant->style();
333 if (style->columnSpan() != ColumnSpanAll || !descendant->isBox() || descenda nt->isFloatingOrOutOfFlowPositioned())
334 return false;
335
336 RenderBlock* container = descendant->containingBlock();
337 if (!container->isRenderBlockFlow() || container->childrenInline()) {
338 // Needs to be block-level.
339 return false;
340 }
341
342 // This looks like a spanner, but if we're inside something unbreakable, it' s not to be treated as one.
343 for (RenderBox* ancestor = toRenderBox(descendant)->parentBox(); ancestor; a ncestor = ancestor->parentBox()) {
344 if (ancestor->isRenderFlowThread()) {
345 // Don't allow any intervening non-multicol fragmentation contexts. The spec doesn't say
346 // anything about disallowing this, but it's just going to be too co mplicated to
347 // implement (not to mention specify behavior).
348 return ancestor == flowThread;
349 }
350 ASSERT(ancestor->style()->columnSpan() != ColumnSpanAll || !isValidColum nSpanner(flowThread, ancestor));
351 if (ancestor->isUnsplittableForPagination())
352 return false;
353 }
354 ASSERT_NOT_REACHED();
355 return false;
356 }
357
358 void RenderMultiColumnFlowThread::flowThreadDescendantInserted(RenderObject* des cendant)
359 {
360 if (m_beingEvacuated)
361 return;
362 RenderObject* subtreeRoot = descendant;
363 for (; descendant; descendant = descendant->nextInPreOrder(subtreeRoot)) {
364 if (descendant->isRenderMultiColumnSpannerPlaceholder()) {
365 // A spanner's placeholder has been inserted. The actual spanner ren derer is moved from
366 // where it wound otherwise occur (if it weren't a spanner) to becom ing a sibling of the
rune 2014/06/19 14:24:33 "where it _would_ otherwise ..."
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done.
rune 2014/08/27 08:17:38 Acknowledged.
367 // column sets.
368 RenderMultiColumnSpannerPlaceholder* placeholder = toRenderMultiColu mnSpannerPlaceholder(descendant);
369 ASSERT(!m_spannerMap.get(placeholder->spanner()));
370 m_spannerMap.add(placeholder->spanner(), placeholder);
371 ASSERT(!placeholder->slowFirstChild()); // There should be no childr en here, but if there are, we ought to skip them.
372 continue;
373 }
374 RenderBlockFlow* multicolContainer = multiColumnBlockFlow();
375 RenderObject* nextRendererInFlowThread = descendant->nextInPreOrderAfter Children(this);
376 RenderObject* insertBeforeMulticolChild = 0;
377 if (isValidColumnSpanner(this, descendant)) {
378 // This is a spanner (column-span:all). Such renderers are moved fro m where they would
379 // otherwise occur in the render tree to becoming a direct child of the multicol container,
380 // so that they live among the column sets. This simplifies the layo ut implementation, and
381 // basically just relies on regular block layout done by the RenderB lockFlow that
382 // establishes the multicol container.
383 RenderBlockFlow* container = toRenderBlockFlow(descendant->parent()) ;
384 RenderMultiColumnSet* setToSplit = 0;
385 if (nextRendererInFlowThread) {
386 setToSplit = findSetRendering(descendant);
387 if (setToSplit) {
388 setToSplit->setNeedsLayoutAndFullRepaint();
389 insertBeforeMulticolChild = setToSplit->nextSibling();
390 }
391 }
392 // Moving a spanner's renderer so that it becomes a sibling of the c olumn sets requires us
393 // to insert an anonymous placeholder in the tree where the spanner' s renderer otherwise
394 // would have been. This is needed for a two reasons: We need a way of separating inline
rune 2014/06/19 14:24:32 "needed for a two reasons"?
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done.
rune 2014/08/27 08:17:38 Acknowledged.
395 // content before and after the spanner, so that it becomes separate line boxes. Secondly,
396 // this placeholder serves as a break point for column sets, so that , when encountered, we
397 // end flowing one column set and move to the next one.
rune 2014/06/19 14:24:33 "move _on_ to the next one" perhaps?
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done.
rune 2014/08/27 08:17:38 Acknowledged.
398 RenderMultiColumnSpannerPlaceholder* placeholder = RenderMultiColumn SpannerPlaceholder::createAnonymous(this, toRenderBox(descendant), container->st yle());
399 container->addChild(placeholder, descendant->nextSibling());
400 container->removeChild(descendant);
401 multicolContainer->RenderBlock::addChild(descendant, insertBeforeMul ticolChild);
402
403 // The spanner has now been moved out from the flow thread, but we d on't want to
404 // examine its children anyway. They are all part of the spanner and shouldn't trigger
405 // creation of column sets or anything like that. Continue at its or iginal position in
406 // the tree, i.e. where the placeholder was just put.
407 if (subtreeRoot == descendant)
408 subtreeRoot = placeholder;
409 descendant = placeholder;
410 } else {
411 // This is regular multicol content, i.e. not part of a spanner.
412 if (nextRendererInFlowThread && nextRendererInFlowThread->isRenderMu ltiColumnSpannerPlaceholder()) {
413 // Inserted right before a spanner. Is there a set for us there?
414 RenderMultiColumnSpannerPlaceholder* placeholder = toRenderMulti ColumnSpannerPlaceholder(nextRendererInFlowThread);
415 if (RenderObject* previous = placeholder->spanner()->previousSib ling()) {
416 if (previous->isRenderMultiColumnSet())
417 continue; // There's already a set there. Nothing to do.
418 }
419 insertBeforeMulticolChild = placeholder->spanner();
420 } else if (RenderMultiColumnSet* lastSet = lastMultiColumnSet()) {
421 // This child is not an immediate predecessor of a spanner, whic h means that if this
422 // child precedes a spanner at all, there has to be a column set created for us there
423 // already. If it doesn't precede any spanner at all, on the oth er hand, we need a
424 // column set at the end of the multicol container. We don't rea lly check here if the
425 // child inserted precedes any spanner or not (as that's an expe nsive operation). Just
426 // make sure we have a column set at the end. It's no big deal i f it remains unused.
427 if (!lastSet->nextSibling())
428 continue;
429 }
430 }
431 // Need to create a new column set when there's no set already created. We also always insert
432 // another column set after a spanner. Even if it turns out that there a re no renderers
433 // following the spanner, there may be bottom margins there, which take up space.
434 RenderMultiColumnSet* newSet = RenderMultiColumnSet::createAnonymous(thi s, multicolContainer->style());
435 multicolContainer->RenderBlock::addChild(newSet, insertBeforeMulticolChi ld);
436 invalidateRegions();
437
438 // We cannot handle immediate column set siblings at the moment (and the re's no need for
439 // it, either). There has to be at least one spanner separating them.
440 ASSERT(!previousColumnSetOrSpannerSiblingOf(newSet) || !previousColumnSe tOrSpannerSiblingOf(newSet)->isRenderMultiColumnSet());
441 ASSERT(!nextColumnSetOrSpannerSiblingOf(newSet) || !nextColumnSetOrSpann erSiblingOf(newSet)->isRenderMultiColumnSet());
442 }
443 }
444
445 void RenderMultiColumnFlowThread::flowThreadRelativeWillBeRemoved(RenderObject* relative)
446 {
447 if (m_beingEvacuated)
448 return;
449 invalidateRegions();
450 if (relative->isRenderMultiColumnSpannerPlaceholder()) {
451 // Remove the map entry for this spanner, but leave the actual spanner r enderer alone. Also
452 // keep the reference to the spanner, since the placeholder may be about to be re-inserted
453 // in the tree.
rune 2014/06/19 14:24:32 "into the tree"
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done.
rune 2014/08/27 08:17:38 Acknowledged.
454 ASSERT(relative->isDescendantOf(this));
455 m_spannerMap.remove(toRenderMultiColumnSpannerPlaceholder(relative)->spa nner());
456 return;
457 }
458 if (relative->style()->columnSpan() == ColumnSpanAll) {
459 if (relative->parent() != parent())
460 return; // not a valid spanner.
461
462 // The placeholder may already have been removed, but if it hasn't, do s o now.
463 if (RenderMultiColumnSpannerPlaceholder* placeholder = m_spannerMap.get( toRenderBox(relative))) {
464 placeholder->containingBlock()->RenderBlock::removeChild(placeholder );
465 m_spannerMap.remove(toRenderBox(relative));
466 }
467
468 if (RenderObject* next = relative->nextSibling()) {
469 if (RenderObject* previous = relative->previousSibling()) {
470 if (previous->isRenderMultiColumnSet() && next->isRenderMultiCol umnSet()) {
471 // Merge two sets that no longer will be separated by a span ner.
472 next->destroy();
473 previous->setNeedsLayoutAndFullRepaint();
474 }
475 }
476 }
477 }
478 // Note that we might end up with empty column sets if all column content is removed. That's no
479 // big deal though (and locating them would be expensive), and they will be found and re-used if
480 // content is added again later.
481 }
482
483 void RenderMultiColumnFlowThread::flowThreadDescendantBoxLaidOut(RenderBox* desc endant, LayoutUnit logicalBottomInFlowThread)
484 {
485 if (!descendant->isRenderMultiColumnSpannerPlaceholder())
486 return;
487 RenderMultiColumnSpannerPlaceholder* placeholder = toRenderMultiColumnSpanne rPlaceholder(descendant);
488
489 for (RenderBox* prev = previousColumnSetOrSpannerSiblingOf(placeholder->span ner()); prev; prev = previousColumnSetOrSpannerSiblingOf(prev)) {
490 if (prev->isRenderMultiColumnSet()) {
491 toRenderMultiColumnSet(prev)->endFlow(logicalBottomInFlowThread);
492 break;
493 }
494 }
495
496 for (RenderBox* next = nextColumnSetOrSpannerSiblingOf(placeholder->spanner( )); next; next = nextColumnSetOrSpannerSiblingOf(next)) {
497 if (next->isRenderMultiColumnSet()) {
498 m_lastSetWorkedOn = toRenderMultiColumnSet(next);
499 m_lastSetWorkedOn->beginFlow(logicalBottomInFlowThread);
500 break;
501 }
502 }
503 }
504
260 void RenderMultiColumnFlowThread::computeLogicalHeight(LayoutUnit logicalHeight, LayoutUnit logicalTop, LogicalExtentComputedValues& computedValues) const 505 void RenderMultiColumnFlowThread::computeLogicalHeight(LayoutUnit logicalHeight, LayoutUnit logicalTop, LogicalExtentComputedValues& computedValues) const
261 { 506 {
262 // We simply remain at our intrinsic height. 507 // We simply remain at our intrinsic height.
263 computedValues.m_extent = logicalHeight; 508 computedValues.m_extent = logicalHeight;
264 computedValues.m_position = logicalTop; 509 computedValues.m_position = logicalTop;
265 } 510 }
266 511
267 void RenderMultiColumnFlowThread::updateLogicalWidth() 512 void RenderMultiColumnFlowThread::updateLogicalWidth()
268 { 513 {
269 LayoutUnit columnWidth; 514 LayoutUnit columnWidth;
270 calculateColumnCountAndWidth(columnWidth, m_columnCount); 515 calculateColumnCountAndWidth(columnWidth, m_columnCount);
271 setLogicalWidth(columnWidth); 516 setLogicalWidth(columnWidth);
272 } 517 }
273 518
274 void RenderMultiColumnFlowThread::layout() 519 void RenderMultiColumnFlowThread::layout()
275 { 520 {
521 ASSERT(!m_inLayout);
522 m_inLayout = true;
523 m_lastSetWorkedOn = 0;
524 if (RenderBox* first = firstColumnSetOrSpanner()) {
525 if (first->isRenderMultiColumnSet()) {
526 m_lastSetWorkedOn = toRenderMultiColumnSet(first);
527 m_lastSetWorkedOn->beginFlow(LayoutUnit());
528 }
529 }
276 RenderFlowThread::layout(); 530 RenderFlowThread::layout();
277 if (RenderMultiColumnSet* lastSet = lastMultiColumnSet()) 531 if (RenderMultiColumnSet* lastSet = lastMultiColumnSet()) {
532 if (!nextColumnSetOrSpannerSiblingOf(lastSet))
533 lastSet->endFlow(logicalHeight());
278 lastSet->expandToEncompassFlowThreadContentsIfNeeded(); 534 lastSet->expandToEncompassFlowThreadContentsIfNeeded();
535 }
536 m_inLayout = false;
537 m_lastSetWorkedOn = 0;
279 } 538 }
280 539
281 void RenderMultiColumnFlowThread::setPageBreak(LayoutUnit offset, LayoutUnit spa ceShortage) 540 void RenderMultiColumnFlowThread::setPageBreak(LayoutUnit offset, LayoutUnit spa ceShortage)
282 { 541 {
283 // Only positive values are interesting (and allowed) here. Zero space short age may be reported 542 // Only positive values are interesting (and allowed) here. Zero space short age may be reported
284 // when we're at the top of a column and the element has zero height. Ignore this, and also 543 // when we're at the top of a column and the element has zero height. Ignore this, and also
285 // ignore any negative values, which may occur when we set an early break in order to honor 544 // ignore any negative values, which may occur when we set an early break in order to honor
286 // widows in the next column. 545 // widows in the next column.
287 if (spaceShortage <= 0) 546 if (spaceShortage <= 0)
288 return; 547 return;
289 548
290 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset))) 549 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset)))
291 multicolSet->recordSpaceShortage(spaceShortage); 550 multicolSet->recordSpaceShortage(spaceShortage);
292 } 551 }
293 552
294 void RenderMultiColumnFlowThread::updateMinimumPageHeight(LayoutUnit offset, Lay outUnit minHeight) 553 void RenderMultiColumnFlowThread::updateMinimumPageHeight(LayoutUnit offset, Lay outUnit minHeight)
295 { 554 {
296 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset))) 555 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset)))
297 multicolSet->updateMinimumColumnHeight(minHeight); 556 multicolSet->updateMinimumColumnHeight(minHeight);
298 } 557 }
299 558
300 RenderRegion* RenderMultiColumnFlowThread::regionAtBlockOffset(LayoutUnit /*offs et*/) const 559 RenderRegion* RenderMultiColumnFlowThread::regionAtBlockOffset(LayoutUnit offset ) const
301 { 560 {
302 // For now there's only one column set, so this is easy: 561 if (!m_inLayout)
303 return firstMultiColumnSet(); 562 return RenderFlowThread::regionAtBlockOffset(offset);
rune 2014/06/19 14:24:33 A single set is still the case for RenderPagedFlow
mstensho (USE GERRIT) 2014/08/26 09:43:58 Yes.
rune 2014/08/27 08:17:38 Acknowledged.
563
564 // Layout in progress. We are calculating the set heights as we speak, so th e region range
565 // information is not up-to-date.
566
567 RenderMultiColumnSet* columnSet = m_lastSetWorkedOn ? m_lastSetWorkedOn : fi rstMultiColumnSet();
568 if (!columnSet) {
569 // If there's no set, bail. This multicol is empty or only consists of s panners. There
570 // are no regions.
571 return 0;
572 }
573 // The last set worked on is a good guess. But if we're not within the bound s, search for the
574 // right one.
575 if (offset < columnSet->logicalTopInFlowThread()) {
576 do {
577 if (RenderMultiColumnSet* prev = columnSet->previousSiblingMultiColu mnSet())
578 columnSet = prev;
579 else
580 break;
581 } while (offset < columnSet->logicalTopInFlowThread());
582 } else {
583 while (offset >= columnSet->logicalBottomInFlowThread()) {
584 RenderMultiColumnSet* next = columnSet->nextSiblingMultiColumnSet();
585 if (!next || !next->hasBeenFlowed())
586 break;
rune 2014/06/19 14:24:33 If I understand this correctly, if we ask for a re
mstensho (USE GERRIT) 2014/08/26 09:43:58 Done. I'm not aware of any usecase for searching
rune 2014/08/27 08:17:38 Acknowledged.
587 columnSet = next;
588 }
589 }
590 return columnSet;
304 } 591 }
305 592
306 bool RenderMultiColumnFlowThread::addForcedRegionBreak(LayoutUnit offset, Render Object* /*breakChild*/, bool /*isBefore*/, LayoutUnit* offsetBreakAdjustment) 593 bool RenderMultiColumnFlowThread::addForcedRegionBreak(LayoutUnit offset, Render Object* breakChild, bool /*isBefore*/, LayoutUnit* offsetBreakAdjustment)
307 { 594 {
308 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset))) { 595 if (RenderMultiColumnSet* multicolSet = toRenderMultiColumnSet(regionAtBlock Offset(offset))) {
309 multicolSet->addContentRun(offset); 596 // Spanner placeholders force a break (to make sure that the unused part of the last column
597 // is empty), but this break should not affect column balancing.
598 if (!breakChild->isRenderMultiColumnSpannerPlaceholder())
599 multicolSet->addContentRun(offset);
310 if (offsetBreakAdjustment) 600 if (offsetBreakAdjustment)
311 *offsetBreakAdjustment = pageLogicalHeightForOffset(offset) ? pageRe mainingLogicalHeightForOffset(offset, IncludePageBoundary) : LayoutUnit(); 601 *offsetBreakAdjustment = pageLogicalHeightForOffset(offset) ? pageRe mainingLogicalHeightForOffset(offset, IncludePageBoundary) : LayoutUnit();
312 return true; 602 return true;
313 } 603 }
314 return false; 604 return false;
315 } 605 }
316 606
317 bool RenderMultiColumnFlowThread::isPageLogicalHeightKnown() const 607 bool RenderMultiColumnFlowThread::isPageLogicalHeightKnown() const
318 { 608 {
319 if (RenderMultiColumnSet* columnSet = lastMultiColumnSet()) 609 if (RenderMultiColumnSet* columnSet = lastMultiColumnSet())
320 return columnSet->pageLogicalHeight(); 610 return columnSet->pageLogicalHeight();
321 return false; 611 return false;
322 } 612 }
323 613
324 } 614 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698