blob: f90cb8ad35c943bf88ebe8281f8d933819926466 [file] [log] [blame]
hyatt@apple.com5388e672013-09-06 20:54:47 +00001/*
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)
bfulgham@apple.comb5953432015-02-13 21:56:01 +00005 * Copyright (C) 2003-2015 Apple Inc. All rights reserved.
hyatt@apple.com5388e672013-09-06 20:54:47 +00006 * 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 "RenderBlockFlow.h"
26
antti@apple.com50b36fde2019-08-11 11:02:01 +000027#include "ComplexLineLayout.h"
weinig@apple.com611b9292013-10-20 22:57:54 +000028#include "Editor.h"
bjonesbe@adobe.com67478092013-09-09 22:18:17 +000029#include "FloatingObjects.h"
weinig@apple.com611b9292013-10-20 22:57:54 +000030#include "Frame.h"
zalan@apple.come36543a2014-07-29 01:45:54 +000031#include "FrameSelection.h"
darin@apple.com15708b12014-03-16 16:38:58 +000032#include "HTMLElement.h"
cdumez@apple.coma92c4d62016-05-20 01:50:51 +000033#include "HTMLInputElement.h"
darin@apple.com5917cb12017-11-23 17:32:42 +000034#include "HTMLParserIdioms.h"
cdumez@apple.coma92c4d62016-05-20 01:50:51 +000035#include "HTMLTextAreaElement.h"
bjonesbe@adobe.com24199752013-10-08 23:20:42 +000036#include "HitTestLocation.h"
weinig@apple.com611b9292013-10-20 22:57:54 +000037#include "InlineTextBox.h"
antti@apple.comd4dda742019-12-03 15:41:43 +000038#include "LayoutIntegrationLineLayout.h"
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +000039#include "LayoutRepainter.h"
simon.fraser@apple.com36676e52016-05-07 00:05:58 +000040#include "Logging.h"
zalan@apple.comac6956c2014-09-05 14:18:06 +000041#include "RenderCombineText.h"
hyatt@apple.com531e35d2017-04-13 16:37:00 +000042#include "RenderFlexibleBox.h"
zalan@apple.comac6956c2014-09-05 14:18:06 +000043#include "RenderInline.h"
akling@apple.comf3028052013-11-04 08:46:06 +000044#include "RenderIterator.h"
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +000045#include "RenderLayer.h"
antti@apple.comf41183e2018-12-07 20:10:14 +000046#include "RenderLayoutState.h"
zalan@apple.com8bf2a912015-04-10 03:15:50 +000047#include "RenderLineBreak.h"
antti@apple.com0b3dffe2014-03-24 16:30:52 +000048#include "RenderListItem.h"
hyatt@apple.com73715ca2014-05-06 21:35:52 +000049#include "RenderMarquee.h"
hyatt@apple.com4e0bf862017-09-27 20:54:17 +000050#include "RenderMultiColumnFlow.h"
hyatt@apple.comd4be3772014-01-24 19:55:33 +000051#include "RenderMultiColumnSet.h"
hyatt@apple.com73715ca2014-05-06 21:35:52 +000052#include "RenderTableCell.h"
antti@apple.com940f5872013-10-24 20:31:11 +000053#include "RenderText.h"
antti@apple.com89b8fff2017-12-16 19:07:55 +000054#include "RenderTreeBuilder.h"
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +000055#include "RenderView.h"
bfulgham@apple.comb5953432015-02-13 21:56:01 +000056#include "Settings.h"
antti@apple.com940f5872013-10-24 20:31:11 +000057#include "SimpleLineLayoutFunctions.h"
zalan@apple.com5cd09e52017-02-25 02:14:40 +000058#include "SimpleLineLayoutPagination.h"
zalan@apple.com95cae4f2018-04-23 14:47:00 +000059#include "SimpleLineLayoutResolver.h"
antti@apple.com4918b4d2017-08-14 13:20:47 +000060#include "TextAutoSizing.h"
hyatt@apple.com3cd5c772013-09-27 18:22:50 +000061#include "VerticalPositionCache.h"
weinig@apple.com611b9292013-10-20 22:57:54 +000062#include "VisiblePosition.h"
fpizlo@apple.comd03c5942017-11-07 19:21:52 +000063#include <wtf/IsoMallocInlines.h>
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +000064
hyatt@apple.com5388e672013-09-06 20:54:47 +000065namespace WebCore {
66
fpizlo@apple.comd03c5942017-11-07 19:21:52 +000067WTF_MAKE_ISO_ALLOCATED_IMPL(RenderBlockFlow);
68
bjonesbe@adobe.com24199752013-10-08 23:20:42 +000069bool RenderBlock::s_canPropagateFloatIntoSibling = false;
70
hyatt@apple.com1807b5b2013-09-11 19:50:03 +000071struct SameSizeAsMarginInfo {
72 uint32_t bitfields : 16;
73 LayoutUnit margins[2];
74};
75
76COMPILE_ASSERT(sizeof(RenderBlockFlow::MarginValues) == sizeof(LayoutUnit[4]), MarginValues_should_stay_small);
akling@apple.com42e10632013-10-14 17:55:52 +000077COMPILE_ASSERT(sizeof(RenderBlockFlow::MarginInfo) == sizeof(SameSizeAsMarginInfo), MarginInfo_should_stay_small);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +000078
79// Our MarginInfo state used when laying out block children.
hyatt@apple.com9a79c622015-09-15 18:38:18 +000080RenderBlockFlow::MarginInfo::MarginInfo(const RenderBlockFlow& block, LayoutUnit beforeBorderPadding, LayoutUnit afterBorderPadding)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +000081 : m_atBeforeSideOfBlock(true)
82 , m_atAfterSideOfBlock(false)
83 , m_hasMarginBeforeQuirk(false)
84 , m_hasMarginAfterQuirk(false)
85 , m_determinedMarginBeforeQuirk(false)
86 , m_discardMargin(false)
87{
akling@apple.com827be9c2013-10-29 02:58:43 +000088 const RenderStyle& blockStyle = block.style();
weinig@apple.com12840dc2013-10-22 23:59:08 +000089 ASSERT(block.isRenderView() || block.parent());
jfernandez@igalia.com136f1702014-12-08 19:13:16 +000090 m_canCollapseWithChildren = !block.createsNewFormattingContext() && !block.isRenderView();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +000091
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +000092 m_canCollapseMarginBeforeWithChildren = m_canCollapseWithChildren && !beforeBorderPadding && blockStyle.marginBeforeCollapse() != MarginCollapse::Separate;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +000093
94 // If any height other than auto is specified in CSS, then we don't collapse our bottom
95 // margins with our children's margins. To do otherwise would be to risk odd visual
96 // effects when the children overflow out of the parent block and yet still collapse
97 // with it. We also don't collapse if we have any bottom border/padding.
98 m_canCollapseMarginAfterWithChildren = m_canCollapseWithChildren && !afterBorderPadding
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +000099 && (blockStyle.logicalHeight().isAuto() && !blockStyle.logicalHeight().value()) && blockStyle.marginAfterCollapse() != MarginCollapse::Separate;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000100
weinig@apple.com12840dc2013-10-22 23:59:08 +0000101 m_quirkContainer = block.isTableCell() || block.isBody();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000102
weinig@apple.com12840dc2013-10-22 23:59:08 +0000103 m_discardMargin = m_canCollapseMarginBeforeWithChildren && block.mustDiscardMarginBefore();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000104
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +0000105 m_positiveMargin = (m_canCollapseMarginBeforeWithChildren && !block.mustDiscardMarginBefore()) ? block.maxPositiveMarginBefore() : 0_lu;
106 m_negativeMargin = (m_canCollapseMarginBeforeWithChildren && !block.mustDiscardMarginBefore()) ? block.maxNegativeMarginBefore() : 0_lu;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000107}
108
antti@apple.com454418f2016-04-25 19:49:23 +0000109RenderBlockFlow::RenderBlockFlow(Element& element, RenderStyle&& style)
aestes@apple.com13aae082016-01-02 08:03:08 +0000110 : RenderBlock(element, WTFMove(style), RenderBlockFlowFlag)
dbates@webkit.org102013c2016-09-26 21:51:25 +0000111#if ENABLE(TEXT_AUTOSIZING)
aestes@apple.com6751d842014-01-12 02:51:25 +0000112 , m_widthForTextAutosizing(-1)
113 , m_lineCountForTextAutosizing(NOT_SET)
114#endif
hyatt@apple.com5388e672013-09-06 20:54:47 +0000115{
weinig@apple.com611b9292013-10-20 22:57:54 +0000116 setChildrenInline(true);
akling@apple.com42e10632013-10-14 17:55:52 +0000117}
118
antti@apple.com454418f2016-04-25 19:49:23 +0000119RenderBlockFlow::RenderBlockFlow(Document& document, RenderStyle&& style)
aestes@apple.com13aae082016-01-02 08:03:08 +0000120 : RenderBlock(document, WTFMove(style), RenderBlockFlowFlag)
dbates@webkit.org102013c2016-09-26 21:51:25 +0000121#if ENABLE(TEXT_AUTOSIZING)
aestes@apple.com6751d842014-01-12 02:51:25 +0000122 , m_widthForTextAutosizing(-1)
123 , m_lineCountForTextAutosizing(NOT_SET)
124#endif
akling@apple.com42e10632013-10-14 17:55:52 +0000125{
weinig@apple.com611b9292013-10-20 22:57:54 +0000126 setChildrenInline(true);
hyatt@apple.com5388e672013-09-06 20:54:47 +0000127}
128
129RenderBlockFlow::~RenderBlockFlow()
130{
simon.fraser@apple.com9812a452017-03-20 17:28:55 +0000131 // Do not add any code here. Add it to willBeDestroyed() instead.
hyatt@apple.com5388e672013-09-06 20:54:47 +0000132}
133
zalan@apple.comfde57e42018-02-22 00:20:40 +0000134void RenderBlockFlow::willBeDestroyed()
hyatt@apple.com3cd5c772013-09-27 18:22:50 +0000135{
simon.fraser@apple.comb59c6902017-03-17 00:20:45 +0000136 if (!renderTreeBeingDestroyed()) {
akling@apple.comee3c8df2013-11-06 08:09:44 +0000137 if (firstRootBox()) {
weinig@apple.com611b9292013-10-20 22:57:54 +0000138 // We can't wait for RenderBox::destroy to clear the selection,
139 // because by then we will have nuked the line boxes.
weinig@apple.com611b9292013-10-20 22:57:54 +0000140 if (isSelectionBorder())
jacob_uphoff@apple.com5a6faad2020-03-30 21:13:50 +0000141 frame().selection().setNeedsSelectionUpdate();
weinig@apple.com611b9292013-10-20 22:57:54 +0000142
143 // If we are an anonymous block, then our line boxes might have children
144 // that will outlast this block. In the non-anonymous block case those
145 // children will be destroyed by the time we return from this function.
146 if (isAnonymousBlock()) {
cdumez@apple.comc1d54fa2015-10-13 19:15:55 +0000147 for (auto* box = firstRootBox(); box; box = box->nextRootBox()) {
weinig@apple.com611b9292013-10-20 22:57:54 +0000148 while (auto childBox = box->firstChild())
149 childBox->removeFromParent();
150 }
151 }
akling@apple.coma1986f52014-12-08 22:17:55 +0000152 } else if (parent())
153 parent()->dirtyLinesFromChangedChild(*this);
weinig@apple.com611b9292013-10-20 22:57:54 +0000154 }
155
antti@apple.com8aa23022019-11-25 23:18:05 +0000156 if (complexLineLayout())
157 complexLineLayout()->lineBoxes().deleteLineBoxes();
weinig@apple.com611b9292013-10-20 22:57:54 +0000158
simon.fraser@apple.com9812a452017-03-20 17:28:55 +0000159 blockWillBeDestroyed();
weinig@apple.com611b9292013-10-20 22:57:54 +0000160
161 // NOTE: This jumps down to RenderBox, bypassing RenderBlock since it would do duplicate work.
zalan@apple.comfde57e42018-02-22 00:20:40 +0000162 RenderBox::willBeDestroyed();
hyatt@apple.com3cd5c772013-09-27 18:22:50 +0000163}
164
ggaren@apple.com91592ea2019-05-29 21:15:54 +0000165RenderMultiColumnFlow* RenderBlockFlow::multiColumnFlowSlowCase() const
166{
167 return rareBlockFlowData()->m_multiColumnFlow.get();
168}
169
jfernandez@igalia.com93f23d22014-12-09 17:44:40 +0000170RenderBlockFlow* RenderBlockFlow::previousSiblingWithOverhangingFloats(bool& parentHasFloats) const
171{
172 // Attempt to locate a previous sibling with overhanging floats. We skip any elements that are
173 // out of flow (like floating/positioned elements), and we also skip over any objects that may have shifted
174 // to avoid floats.
175 parentHasFloats = false;
176 for (RenderObject* sibling = previousSibling(); sibling; sibling = sibling->previousSibling()) {
177 if (is<RenderBlockFlow>(*sibling)) {
178 auto& siblingBlock = downcast<RenderBlockFlow>(*sibling);
179 if (!siblingBlock.avoidsFloats())
180 return &siblingBlock;
181 }
182 if (sibling->isFloating())
183 parentHasFloats = true;
184 }
185 return nullptr;
186}
187
bjonesbe@adobe.comf9f10402014-02-20 19:40:28 +0000188void RenderBlockFlow::rebuildFloatingObjectSetFromIntrudingFloats()
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000189{
190 if (m_floatingObjects)
191 m_floatingObjects->setHorizontalWritingMode(isHorizontalWritingMode());
192
193 HashSet<RenderBox*> oldIntrudingFloatSet;
194 if (!childrenInline() && m_floatingObjects) {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000195 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
196 auto end = floatingObjectSet.end();
197 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
198 FloatingObject* floatingObject = it->get();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000199 if (!floatingObject->isDescendant())
darin@apple.com7cad7042013-09-24 05:53:55 +0000200 oldIntrudingFloatSet.add(&floatingObject->renderer());
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000201 }
202 }
203
204 // Inline blocks are covered by the isReplaced() check in the avoidFloats method.
simon.fraser@apple.comf63871d2015-10-10 02:20:52 +0000205 if (avoidsFloats() || isDocumentElementRenderer() || isRenderView() || isFloatingOrOutOfFlowPositioned() || isTableCell()) {
mihnea@adobe.combe79cf12013-10-17 09:02:19 +0000206 if (m_floatingObjects)
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000207 m_floatingObjects->clear();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000208 if (!oldIntrudingFloatSet.isEmpty())
209 markAllDescendantsWithFloatsForLayout();
210 return;
211 }
212
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000213 RendererToFloatInfoMap floatMap;
214
215 if (m_floatingObjects) {
bjonesbe@adobe.com0434768a2013-09-16 22:01:38 +0000216 if (childrenInline())
217 m_floatingObjects->moveAllToFloatInfoMap(floatMap);
218 else
219 m_floatingObjects->clear();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000220 }
221
222 // We should not process floats if the parent node is not a RenderBlock. Otherwise, we will add
223 // floats in an invalid context. This will cause a crash arising from a bad cast on the parent.
224 // See <rdar://problem/8049753>, where float property is applied on a text node in a SVG.
antti@apple.comae85e112017-08-31 23:27:02 +0000225 if (!is<RenderBlockFlow>(parent()))
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000226 return;
227
robert@webkit.org97037ef2013-11-20 19:26:10 +0000228 // First add in floats from the parent. Self-collapsing blocks let their parent track any floats that intrude into
229 // them (as opposed to floats they contain themselves) so check for those here too.
antti@apple.comae85e112017-08-31 23:27:02 +0000230 auto& parentBlock = downcast<RenderBlockFlow>(*parent());
231 bool parentHasFloats = false;
232 RenderBlockFlow* previousBlock = previousSiblingWithOverhangingFloats(parentHasFloats);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000233 LayoutUnit logicalTopOffset = logicalTop();
jfernandez@igalia.com93f23d22014-12-09 17:44:40 +0000234 if (parentHasFloats || (parentBlock.lowestFloatLogicalBottom() > logicalTopOffset && previousBlock && previousBlock->isSelfCollapsingBlock()))
hyatt@apple.com21c60802015-04-01 18:10:32 +0000235 addIntrudingFloats(&parentBlock, &parentBlock, parentBlock.logicalLeftOffsetForContent(), logicalTopOffset);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000236
ross.kirsling@sony.combd744282018-11-18 03:14:31 +0000237 LayoutUnit logicalLeftOffset;
jfernandez@igalia.com93f23d22014-12-09 17:44:40 +0000238 if (previousBlock)
239 logicalTopOffset -= previousBlock->logicalTop();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000240 else {
jfernandez@igalia.com93f23d22014-12-09 17:44:40 +0000241 previousBlock = &parentBlock;
cdumez@apple.com34e77ab2014-10-09 16:17:06 +0000242 logicalLeftOffset += parentBlock.logicalLeftOffsetForContent();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000243 }
244
245 // Add overhanging floats from the previous RenderBlock, but only if it has a float that intrudes into our space.
jfernandez@igalia.com93f23d22014-12-09 17:44:40 +0000246 if (previousBlock->m_floatingObjects && previousBlock->lowestFloatLogicalBottom() > logicalTopOffset)
hyatt@apple.com21c60802015-04-01 18:10:32 +0000247 addIntrudingFloats(previousBlock, &parentBlock, logicalLeftOffset, logicalTopOffset);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000248
249 if (childrenInline()) {
250 LayoutUnit changeLogicalTop = LayoutUnit::max();
251 LayoutUnit changeLogicalBottom = LayoutUnit::min();
252 if (m_floatingObjects) {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000253 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
254 auto end = floatingObjectSet.end();
255 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +0000256 const auto& floatingObject = *it->get();
257 std::unique_ptr<FloatingObject> oldFloatingObject = floatMap.take(&floatingObject.renderer());
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000258 LayoutUnit logicalBottom = logicalBottomForFloat(floatingObject);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000259 if (oldFloatingObject) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +0000260 LayoutUnit oldLogicalBottom = logicalBottomForFloat(*oldFloatingObject);
261 if (logicalWidthForFloat(floatingObject) != logicalWidthForFloat(*oldFloatingObject) || logicalLeftForFloat(floatingObject) != logicalLeftForFloat(*oldFloatingObject)) {
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000262 changeLogicalTop = 0;
andersca@apple.com86298632013-11-10 19:32:33 +0000263 changeLogicalBottom = std::max(changeLogicalBottom, std::max(logicalBottom, oldLogicalBottom));
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000264 } else {
265 if (logicalBottom != oldLogicalBottom) {
andersca@apple.com86298632013-11-10 19:32:33 +0000266 changeLogicalTop = std::min(changeLogicalTop, std::min(logicalBottom, oldLogicalBottom));
267 changeLogicalBottom = std::max(changeLogicalBottom, std::max(logicalBottom, oldLogicalBottom));
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000268 }
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000269 LayoutUnit logicalTop = logicalTopForFloat(floatingObject);
zalan@apple.com84ccfa12015-10-17 03:36:56 +0000270 LayoutUnit oldLogicalTop = logicalTopForFloat(*oldFloatingObject);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000271 if (logicalTop != oldLogicalTop) {
andersca@apple.com86298632013-11-10 19:32:33 +0000272 changeLogicalTop = std::min(changeLogicalTop, std::min(logicalTop, oldLogicalTop));
273 changeLogicalBottom = std::max(changeLogicalBottom, std::max(logicalTop, oldLogicalTop));
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000274 }
275 }
276
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000277 if (oldFloatingObject->originatingLine() && !selfNeedsLayout()) {
278 ASSERT(&oldFloatingObject->originatingLine()->renderer() == this);
279 oldFloatingObject->originatingLine()->markDirty();
280 }
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000281 } else {
282 changeLogicalTop = 0;
andersca@apple.com86298632013-11-10 19:32:33 +0000283 changeLogicalBottom = std::max(changeLogicalBottom, logicalBottom);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000284 }
285 }
286 }
287
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000288 auto end = floatMap.end();
289 for (auto it = floatMap.begin(); it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +0000290 const auto& floatingObject = *it->value.get();
291 if (!floatingObject.isDescendant()) {
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000292 changeLogicalTop = 0;
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000293 changeLogicalBottom = std::max(changeLogicalBottom, logicalBottomForFloat(floatingObject));
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000294 }
295 }
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000296
297 markLinesDirtyInBlockRange(changeLogicalTop, changeLogicalBottom);
298 } else if (!oldIntrudingFloatSet.isEmpty()) {
299 // If there are previously intruding floats that no longer intrude, then children with floats
300 // should also get layout because they might need their floating object lists cleared.
301 if (m_floatingObjects->set().size() < oldIntrudingFloatSet.size())
302 markAllDescendantsWithFloatsForLayout();
303 else {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +0000304 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
305 auto end = floatingObjectSet.end();
306 for (auto it = floatingObjectSet.begin(); it != end && !oldIntrudingFloatSet.isEmpty(); ++it)
307 oldIntrudingFloatSet.remove(&(*it)->renderer());
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000308 if (!oldIntrudingFloatSet.isEmpty())
309 markAllDescendantsWithFloatsForLayout();
310 }
311 }
312}
313
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000314void RenderBlockFlow::adjustIntrinsicLogicalWidthsForColumns(LayoutUnit& minLogicalWidth, LayoutUnit& maxLogicalWidth) const
315{
316 if (!style().hasAutoColumnCount() || !style().hasAutoColumnWidth()) {
317 // The min/max intrinsic widths calculated really tell how much space elements need when
318 // laid out inside the columns. In order to eventually end up with the desired column width,
319 // we need to convert them to values pertaining to the multicol container.
320 int columnCount = style().hasAutoColumnCount() ? 1 : style().columnCount();
321 LayoutUnit columnWidth;
322 LayoutUnit colGap = columnGap();
323 LayoutUnit gapExtra = (columnCount - 1) * colGap;
324 if (style().hasAutoColumnWidth())
325 minLogicalWidth = minLogicalWidth * columnCount + gapExtra;
326 else {
327 columnWidth = style().columnWidth();
328 minLogicalWidth = std::min(minLogicalWidth, columnWidth);
329 }
330 // FIXME: If column-count is auto here, we should resolve it to calculate the maximum
331 // intrinsic width, instead of pretending that it's 1. The only way to do that is by
332 // performing a layout pass, but this is not an appropriate time or place for layout. The
333 // good news is that if height is unconstrained and there are no explicit breaks, the
334 // resolved column-count really should be 1.
335 maxLogicalWidth = std::max(maxLogicalWidth, columnWidth) * columnCount + gapExtra;
336 }
337}
338
339void RenderBlockFlow::computeIntrinsicLogicalWidths(LayoutUnit& minLogicalWidth, LayoutUnit& maxLogicalWidth) const
340{
zalan@apple.comac6956c2014-09-05 14:18:06 +0000341 if (childrenInline())
342 computeInlinePreferredLogicalWidths(minLogicalWidth, maxLogicalWidth);
343 else
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000344 computeBlockPreferredLogicalWidths(minLogicalWidth, maxLogicalWidth);
345
346 maxLogicalWidth = std::max(minLogicalWidth, maxLogicalWidth);
347
348 adjustIntrinsicLogicalWidthsForColumns(minLogicalWidth, maxLogicalWidth);
349
350 if (!style().autoWrap() && childrenInline()) {
351 // A horizontal marquee with inline children has no minimum width.
352 if (layer() && layer()->marquee() && layer()->marquee()->isHorizontal())
353 minLogicalWidth = 0;
354 }
355
cdumez@apple.com8faf7722014-10-13 18:21:11 +0000356 if (is<RenderTableCell>(*this)) {
357 Length tableCellWidth = downcast<RenderTableCell>(*this).styleOrColLogicalWidth();
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000358 if (tableCellWidth.isFixed() && tableCellWidth.value() > 0)
359 maxLogicalWidth = std::max(minLogicalWidth, adjustContentBoxLogicalWidthForBoxSizing(tableCellWidth.value()));
360 }
361
rego@igalia.com76760c72015-05-14 16:37:27 +0000362 int scrollbarWidth = intrinsicScrollbarLogicalWidth();
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000363 maxLogicalWidth += scrollbarWidth;
364 minLogicalWidth += scrollbarWidth;
365}
366
367bool RenderBlockFlow::recomputeLogicalWidthAndColumnWidth()
368{
369 bool changed = recomputeLogicalWidth();
370
371 LayoutUnit oldColumnWidth = computedColumnWidth();
372 computeColumnCountAndWidth();
373
374 return changed || oldColumnWidth != computedColumnWidth();
375}
376
377LayoutUnit RenderBlockFlow::columnGap() const
378{
rego@igalia.comfcd8dd02018-01-26 14:57:47 +0000379 if (style().columnGap().isNormal())
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000380 return style().fontDescription().computedPixelSize(); // "1em" is recommended as the normal gap setting. Matches <p> margins.
rego@igalia.comfcd8dd02018-01-26 14:57:47 +0000381 return valueForLength(style().columnGap().length(), availableLogicalWidth());
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000382}
383
384void RenderBlockFlow::computeColumnCountAndWidth()
zalan@apple.com748d3822016-12-02 21:25:15 +0000385{
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000386 // Calculate our column width and column count.
387 // FIXME: Can overflow on fast/block/float/float-not-removed-from-next-sibling4.html, see https://bugs.webkit.org/show_bug.cgi?id=68744
388 unsigned desiredColumnCount = 1;
389 LayoutUnit desiredColumnWidth = contentLogicalWidth();
zalan@apple.com08aaba72016-12-02 23:05:48 +0000390
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000391 // For now, we don't support multi-column layouts when printing, since we have to do a lot of work for proper pagination.
392 if (document().paginated() || (style().hasAutoColumnCount() && style().hasAutoColumnWidth()) || !style().hasInlineColumnAxis()) {
393 setComputedColumnCountAndWidth(desiredColumnCount, desiredColumnWidth);
394 return;
395 }
zalan@apple.com08aaba72016-12-02 23:05:48 +0000396
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000397 LayoutUnit availWidth = desiredColumnWidth;
398 LayoutUnit colGap = columnGap();
ross.kirsling@sony.com80414652019-05-21 01:36:11 +0000399 LayoutUnit colWidth = std::max(1_lu, LayoutUnit(style().columnWidth()));
zalan@apple.com585798f2016-05-18 23:25:47 +0000400 unsigned colCount = std::max<unsigned>(1, style().columnCount());
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000401
402 if (style().hasAutoColumnWidth() && !style().hasAutoColumnCount()) {
403 desiredColumnCount = colCount;
404 desiredColumnWidth = std::max<LayoutUnit>(0, (availWidth - ((desiredColumnCount - 1) * colGap)) / desiredColumnCount);
405 } else if (!style().hasAutoColumnWidth() && style().hasAutoColumnCount()) {
zalan@apple.com08aaba72016-12-02 23:05:48 +0000406 desiredColumnCount = std::max<unsigned>(1, ((availWidth + colGap) / (colWidth + colGap)).toUnsigned());
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000407 desiredColumnWidth = ((availWidth + colGap) / desiredColumnCount) - colGap;
408 } else {
zalan@apple.com08aaba72016-12-02 23:05:48 +0000409 desiredColumnCount = std::max<unsigned>(std::min(colCount, ((availWidth + colGap) / (colWidth + colGap)).toUnsigned()), 1);
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000410 desiredColumnWidth = ((availWidth + colGap) / desiredColumnCount) - colGap;
411 }
412 setComputedColumnCountAndWidth(desiredColumnCount, desiredColumnWidth);
413}
414
cdumez@apple.com8b7a0222018-12-20 04:41:11 +0000415bool RenderBlockFlow::willCreateColumns(Optional<unsigned> desiredColumnCount) const
zalan@apple.com748d3822016-12-02 21:25:15 +0000416{
zalan@apple.com809f4872016-12-08 18:20:01 +0000417 // The following types are not supposed to create multicol context.
hyatt@apple.com80914862017-03-06 18:00:35 +0000418 if (isFileUploadControl() || isTextControl() || isListBox())
zalan@apple.com809f4872016-12-08 18:20:01 +0000419 return false;
commit-queue@webkit.org875cdb32017-10-02 17:59:36 +0000420 if (isRenderSVGBlock() || isRubyRun())
antti@apple.com411949d2017-08-30 17:28:10 +0000421 return false;
commit-queue@webkit.org875cdb32017-10-02 17:59:36 +0000422#if ENABLE(MATHML)
423 if (isRenderMathMLBlock())
424 return false;
425#endif // ENABLE(MATHML)
zalan@apple.com809f4872016-12-08 18:20:01 +0000426
zalan@apple.com748d3822016-12-02 21:25:15 +0000427 if (!firstChild())
428 return false;
429
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +0000430 if (style().styleType() != PseudoId::None)
antti@apple.com411949d2017-08-30 17:28:10 +0000431 return false;
432
zalan@apple.com809f4872016-12-08 18:20:01 +0000433 // If overflow-y is set to paged-x or paged-y on the body or html element, we'll handle the paginating in the RenderView instead.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +0000434 if ((style().overflowY() == Overflow::PagedX || style().overflowY() == Overflow::PagedY) && !(isDocumentElementRenderer() || isBody()))
zalan@apple.com809f4872016-12-08 18:20:01 +0000435 return true;
zalan@apple.com2b9b6102018-07-25 22:44:48 +0000436
zalan@apple.com748d3822016-12-02 21:25:15 +0000437 if (!style().specifiesColumns())
438 return false;
439
hyatt@apple.com4e0bf862017-09-27 20:54:17 +0000440 // column-axis with opposite writing direction initiates MultiColumnFlow.
zalan@apple.com748d3822016-12-02 21:25:15 +0000441 if (!style().hasInlineColumnAxis())
442 return true;
443
hyatt@apple.com4e0bf862017-09-27 20:54:17 +0000444 // Non-auto column-width always initiates MultiColumnFlow.
zalan@apple.com748d3822016-12-02 21:25:15 +0000445 if (!style().hasAutoColumnWidth())
446 return true;
447
zalan@apple.com809f4872016-12-08 18:20:01 +0000448 if (desiredColumnCount)
449 return desiredColumnCount.value() > 1;
450
hyatt@apple.com4e0bf862017-09-27 20:54:17 +0000451 // column-count > 1 always initiates MultiColumnFlow.
zalan@apple.com748d3822016-12-02 21:25:15 +0000452 if (!style().hasAutoColumnCount())
453 return style().columnCount() > 1;
454
455 ASSERT_NOT_REACHED();
456 return false;
457}
458
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000459void RenderBlockFlow::layoutBlock(bool relayoutChildren, LayoutUnit pageLogicalHeight)
460{
461 ASSERT(needsLayout());
462
463 if (!relayoutChildren && simplifiedLayout())
464 return;
465
466 LayoutRepainter repainter(*this, checkForRepaintDuringLayout());
467
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000468 if (recomputeLogicalWidthAndColumnWidth())
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000469 relayoutChildren = true;
470
bjonesbe@adobe.comf9f10402014-02-20 19:40:28 +0000471 rebuildFloatingObjectSetFromIntrudingFloats();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000472
473 LayoutUnit previousHeight = logicalHeight();
474 // FIXME: should this start out as borderAndPaddingLogicalHeight() + scrollbarLogicalHeight(),
475 // for consistency with other render classes?
zalan@apple.comad3d6b72019-11-20 23:52:40 +0000476 resetLogicalHeightBeforeLayoutIfNeeded();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000477
478 bool pageLogicalHeightChanged = false;
hyatt@apple.com73715ca2014-05-06 21:35:52 +0000479 checkForPaginationLogicalHeightChange(relayoutChildren, pageLogicalHeight, pageLogicalHeightChanged);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000480
ross.kirsling@sony.combd744282018-11-18 03:14:31 +0000481 LayoutUnit repaintLogicalTop;
482 LayoutUnit repaintLogicalBottom;
483 LayoutUnit maxFloatLogicalBottom;
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000484 const RenderStyle& styleToUse = style();
485 {
486 LayoutStateMaintainer statePusher(*this, locationOffset(), hasTransform() || hasReflection() || styleToUse.isFlippedBlocksWritingMode(), pageLogicalHeight, pageLogicalHeightChanged);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000487
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000488 preparePaginationBeforeBlockLayout(relayoutChildren);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000489
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000490 // We use four values, maxTopPos, maxTopNeg, maxBottomPos, and maxBottomNeg, to track
491 // our current maximal positive and negative margins. These values are used when we
492 // are collapsed with adjacent blocks, so for example, if you have block A and B
493 // collapsing together, then you'd take the maximal positive margin from both A and B
494 // and subtract it from the maximal negative margin from both A and B to get the
495 // true collapsed margin. This algorithm is recursive, so when we finish layout()
496 // our block knows its current maximal positive/negative values.
497 //
498 // Start out by setting our margin values to our current margins. Table cells have
499 // no margins, so we don't fill in the values for table cells.
500 bool isCell = isTableCell();
501 if (!isCell) {
502 initMaxMarginValues();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000503
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000504 setHasMarginBeforeQuirk(styleToUse.hasMarginBeforeQuirk());
505 setHasMarginAfterQuirk(styleToUse.hasMarginAfterQuirk());
506 setPaginationStrut(0);
507 }
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000508 if (!firstChild() && !isAnonymousBlock())
509 setChildrenInline(true);
510 if (childrenInline())
511 layoutInlineChildren(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
512 else
513 layoutBlockChildren(relayoutChildren, maxFloatLogicalBottom);
zalan@apple.com4a440322017-11-10 00:31:24 +0000514 }
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000515
zalan@apple.com4a440322017-11-10 00:31:24 +0000516 // Expand our intrinsic height to encompass floats.
517 LayoutUnit toAdd = borderAndPaddingAfter() + scrollbarLogicalHeight();
518 if (lowestFloatLogicalBottom() > (logicalHeight() - toAdd) && createsNewFormattingContext())
519 setLogicalHeight(lowestFloatLogicalBottom() + toAdd);
520 if (relayoutForPagination() || relayoutToAvoidWidows()) {
521 ASSERT(!shouldBreakAtLineToAvoidWidow());
522 return;
523 }
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000524
zalan@apple.com4a440322017-11-10 00:31:24 +0000525 // Calculate our new height.
526 LayoutUnit oldHeight = logicalHeight();
527 LayoutUnit oldClientAfterEdge = clientLogicalBottom();
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000528
zalan@apple.com4a440322017-11-10 00:31:24 +0000529 // Before updating the final size of the flow thread make sure a forced break is applied after the content.
530 // This ensures the size information is correctly computed for the last auto-height fragment receiving content.
531 if (is<RenderFragmentedFlow>(*this))
532 downcast<RenderFragmentedFlow>(*this).applyBreakAfterContent(oldClientAfterEdge);
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000533
zalan@apple.com4a440322017-11-10 00:31:24 +0000534 updateLogicalHeight();
535 LayoutUnit newHeight = logicalHeight();
536 {
537 // FIXME: This could be removed once relayoutForPagination()/relayoutToAvoidWidows() either stop recursing or we manage to
538 // re-order them.
539 LayoutStateMaintainer statePusher(*this, locationOffset(), hasTransform() || hasReflection() || styleToUse.isFlippedBlocksWritingMode(), pageLogicalHeight, pageLogicalHeightChanged);
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000540
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000541 if (oldHeight != newHeight) {
542 if (oldHeight > newHeight && maxFloatLogicalBottom > newHeight && !childrenInline()) {
543 // One of our children's floats may have become an overhanging float for us. We need to look for it.
544 for (auto& blockFlow : childrenOfType<RenderBlockFlow>(*this)) {
545 if (blockFlow.isFloatingOrOutOfFlowPositioned())
546 continue;
547 if (blockFlow.lowestFloatLogicalBottom() + blockFlow.logicalTop() > newHeight)
548 addOverhangingFloats(blockFlow, false);
549 }
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000550 }
551 }
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000552
553 bool heightChanged = (previousHeight != newHeight);
554 if (heightChanged)
555 relayoutChildren = true;
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000556 layoutPositionedObjects(relayoutChildren || isDocumentElementRenderer());
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000557 }
zalan@apple.com4a440322017-11-10 00:31:24 +0000558 // Add overflow from children (unless we're multi-column, since in that case all our child overflow is clipped anyway).
559 computeOverflow(oldClientAfterEdge);
560
561 fitBorderToLinesIfNeeded();
562
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000563 auto* state = view().frameView().layoutContext().layoutState();
zalan@apple.com7fd79ce2017-11-08 15:02:10 +0000564 if (state && state->pageLogicalHeight())
zalan@apple.com85ab1ec2017-11-07 21:36:07 +0000565 setPageLogicalOffset(state->pageLogicalOffset(this, logicalTop()));
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000566
567 updateLayerTransform();
568
569 // Update our scroll information if we're overflow:auto/scroll/hidden now that we know if
570 // we overflow or not.
571 updateScrollInfoAfterLayout();
572
573 // FIXME: This repaint logic should be moved into a separate helper function!
574 // Repaint with our new bounds if they are different from our old bounds.
575 bool didFullRepaint = repainter.repaintAfterLayout();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +0000576 if (!didFullRepaint && repaintLogicalTop != repaintLogicalBottom && (styleToUse.visibility() == Visibility::Visible || enclosingLayer()->hasVisibleContent())) {
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000577 // FIXME: We could tighten up the left and right invalidation points if we let layoutInlineChildren fill them in based off the particular lines
578 // it had to lay out. We wouldn't need the hasOverflowClip() hack in that case either.
579 LayoutUnit repaintLogicalLeft = logicalLeftVisualOverflow();
580 LayoutUnit repaintLogicalRight = logicalRightVisualOverflow();
581 if (hasOverflowClip()) {
582 // 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.
583 // 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.
584 // layoutInlineChildren should be patched to compute the entire repaint rect.
andersca@apple.com86298632013-11-10 19:32:33 +0000585 repaintLogicalLeft = std::min(repaintLogicalLeft, logicalLeftLayoutOverflow());
586 repaintLogicalRight = std::max(repaintLogicalRight, logicalRightLayoutOverflow());
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000587 }
588
589 LayoutRect repaintRect;
590 if (isHorizontalWritingMode())
591 repaintRect = LayoutRect(repaintLogicalLeft, repaintLogicalTop, repaintLogicalRight - repaintLogicalLeft, repaintLogicalBottom - repaintLogicalTop);
592 else
593 repaintRect = LayoutRect(repaintLogicalTop, repaintLogicalLeft, repaintLogicalBottom - repaintLogicalTop, repaintLogicalRight - repaintLogicalLeft);
594
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000595 if (hasOverflowClip()) {
596 // Adjust repaint rect for scroll offset
simon.fraser@apple.com24e03ba2016-05-11 04:48:08 +0000597 repaintRect.moveBy(-scrollPosition());
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000598
599 // Don't allow this rect to spill out of our overflow box.
600 repaintRect.intersect(LayoutRect(LayoutPoint(), size()));
601 }
602
603 // Make sure the rect is still non-empty after intersecting for overflow above
604 if (!repaintRect.isEmpty()) {
605 repaintRectangle(repaintRect); // We need to do a partial repaint of our content.
606 if (hasReflection())
607 repaintRectangle(reflectedRect(repaintRect));
608 }
609 }
610
antti@apple.comca2a8ff2013-10-04 04:04:35 +0000611 clearNeedsLayout();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000612}
613
614void RenderBlockFlow::layoutBlockChildren(bool relayoutChildren, LayoutUnit& maxFloatLogicalBottom)
615{
616 dirtyForLayoutFromPercentageHeightDescendants();
617
618 LayoutUnit beforeEdge = borderAndPaddingBefore();
619 LayoutUnit afterEdge = borderAndPaddingAfter() + scrollbarLogicalHeight();
620
621 setLogicalHeight(beforeEdge);
622
623 // Lay out our hypothetical grid line as though it occurs at the top of the block.
zalan@apple.com4de96c52017-11-07 04:09:59 +0000624 if (view().frameView().layoutContext().layoutState()->lineGrid() == this)
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000625 layoutLineGridBox();
626
627 // The margin struct caches all our current margin collapsing state.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000628 MarginInfo marginInfo(*this, beforeEdge, afterEdge);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000629
630 // Fieldsets need to find their legend and position it inside the border of the object.
631 // The legend then gets skipped during normal layout. The same is true for ruby text.
632 // It doesn't get included in the normal layout process but is instead skipped.
hyatt@apple.com80914862017-03-06 18:00:35 +0000633 layoutExcludedChildren(relayoutChildren);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000634
ross.kirsling@sony.combd744282018-11-18 03:14:31 +0000635 LayoutUnit previousFloatLogicalBottom;
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000636 maxFloatLogicalBottom = 0;
637
638 RenderBox* next = firstChildBox();
639
640 while (next) {
weinig@apple.com12840dc2013-10-22 23:59:08 +0000641 RenderBox& child = *next;
642 next = child.nextSiblingBox();
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000643
hyatt@apple.com80914862017-03-06 18:00:35 +0000644 if (child.isExcludedFromNormalLayout())
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000645 continue; // Skip this child, since it will be positioned by the specialized subclass (fieldsets and ruby runs).
646
647 updateBlockChildDirtyBitsBeforeLayout(relayoutChildren, child);
648
weinig@apple.com12840dc2013-10-22 23:59:08 +0000649 if (child.isOutOfFlowPositioned()) {
650 child.containingBlock()->insertPositionedObject(child);
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000651 adjustPositionedBlock(child, marginInfo);
652 continue;
653 }
weinig@apple.com12840dc2013-10-22 23:59:08 +0000654 if (child.isFloating()) {
hyatt@apple.comc9b1aef2013-09-09 20:23:21 +0000655 insertFloatingObject(child);
656 adjustFloatingBlock(marginInfo);
657 continue;
658 }
659
660 // Lay out the child.
661 layoutBlockChild(child, marginInfo, previousFloatLogicalBottom, maxFloatLogicalBottom);
662 }
663
664 // Now do the handling of the bottom of the block, adding in our bottom border/padding and
665 // determining the correct collapsed bottom margin information.
666 handleAfterSideOfBlock(beforeEdge, afterEdge, marginInfo);
667}
668
antti@apple.com940f5872013-10-24 20:31:11 +0000669void RenderBlockFlow::layoutInlineChildren(bool relayoutChildren, LayoutUnit& repaintLogicalTop, LayoutUnit& repaintLogicalBottom)
670{
antti@apple.com1880e712019-11-26 20:15:39 +0000671 auto computeLineLayoutPath = [&] {
antti@apple.coma6e9e512020-01-15 15:46:09 +0000672 bool canUseSimpleLines = SimpleLineLayout::canUseFor(*this);
antti@apple.com1880e712019-11-26 20:15:39 +0000673#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
antti@apple.coma6e9e512020-01-15 15:46:09 +0000674 if (LayoutIntegration::LineLayout::canUseFor(*this, canUseSimpleLines))
antti@apple.comd4dda742019-12-03 15:41:43 +0000675 return LayoutFormattingContextPath;
antti@apple.com1880e712019-11-26 20:15:39 +0000676#endif
antti@apple.coma6e9e512020-01-15 15:46:09 +0000677 if (canUseSimpleLines)
antti@apple.com1880e712019-11-26 20:15:39 +0000678 return SimpleLinesPath;
antti@apple.coma6e9e512020-01-15 15:46:09 +0000679
antti@apple.com1880e712019-11-26 20:15:39 +0000680 return LineBoxesPath;
681 };
682
akling@apple.coma12fee22015-02-01 02:58:13 +0000683 if (lineLayoutPath() == UndeterminedPath)
antti@apple.com1880e712019-11-26 20:15:39 +0000684 setLineLayoutPath(computeLineLayoutPath());
antti@apple.com42fb53d2013-10-25 02:33:11 +0000685
akling@apple.coma12fee22015-02-01 02:58:13 +0000686 if (lineLayoutPath() == SimpleLinesPath) {
zalan@apple.come37da962014-12-11 03:29:29 +0000687 layoutSimpleLines(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
antti@apple.com940f5872013-10-24 20:31:11 +0000688 return;
689 }
690
antti@apple.com1880e712019-11-26 20:15:39 +0000691#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
antti@apple.comd4dda742019-12-03 15:41:43 +0000692 if (lineLayoutPath() == LayoutFormattingContextPath) {
antti@apple.com1880e712019-11-26 20:15:39 +0000693 layoutLFCLines(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
694 return;
695 }
696#endif
697
antti@apple.com8aa23022019-11-25 23:18:05 +0000698 if (!complexLineLayout())
699 m_lineLayout = makeUnique<ComplexLineLayout>(*this);
antti@apple.comc6a9c732019-08-12 15:31:34 +0000700
antti@apple.com8aa23022019-11-25 23:18:05 +0000701 complexLineLayout()->layoutLineBoxes(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
antti@apple.com940f5872013-10-24 20:31:11 +0000702}
703
weinig@apple.com12840dc2013-10-22 23:59:08 +0000704void RenderBlockFlow::layoutBlockChild(RenderBox& child, MarginInfo& marginInfo, LayoutUnit& previousFloatLogicalBottom, LayoutUnit& maxFloatLogicalBottom)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000705{
706 LayoutUnit oldPosMarginBefore = maxPositiveMarginBefore();
707 LayoutUnit oldNegMarginBefore = maxNegativeMarginBefore();
708
709 // The child is a normal flow object. Compute the margins we will use for collapsing now.
mmaxfield@apple.com09804f42016-03-23 00:58:34 +0000710 child.computeAndSetBlockDirectionMargins(*this);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000711
712 // Try to guess our correct logical top position. In most cases this guess will
713 // be correct. Only if we're wrong (when we compute the real logical top position)
714 // will we have to potentially relayout.
715 LayoutUnit estimateWithoutPagination;
716 LayoutUnit logicalTopEstimate = estimateLogicalTopPosition(child, marginInfo, estimateWithoutPagination);
717
718 // Cache our old rect so that we can dirty the proper repaint rects if the child moves.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000719 LayoutRect oldRect = child.frameRect();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000720 LayoutUnit oldLogicalTop = logicalTopForChild(child);
721
mark.lam@apple.com65724362020-01-06 22:24:50 +0000722#if ASSERT_ENABLED
zalan@apple.com4de96c52017-11-07 04:09:59 +0000723 LayoutSize oldLayoutDelta = view().frameView().layoutContext().layoutDelta();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000724#endif
simon.fraser@apple.com03e61032015-04-05 20:17:11 +0000725 // Position the child as though it didn't collapse with the top.
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000726 setLogicalTopForChild(child, logicalTopEstimate, ApplyLayoutDelta);
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +0000727 estimateFragmentRangeForBoxChild(child);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000728
cdumez@apple.com34e77ab2014-10-09 16:17:06 +0000729 RenderBlockFlow* childBlockFlow = is<RenderBlockFlow>(child) ? &downcast<RenderBlockFlow>(child) : nullptr;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000730 bool markDescendantsWithFloats = false;
weinig@apple.com12840dc2013-10-22 23:59:08 +0000731 if (logicalTopEstimate != oldLogicalTop && !child.avoidsFloats() && childBlockFlow && childBlockFlow->containsFloats())
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000732 markDescendantsWithFloats = true;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000733 else if (UNLIKELY(logicalTopEstimate.mightBeSaturated()))
734 // logicalTopEstimate, returned by estimateLogicalTopPosition, might be saturated for
735 // very large elements. If it does the comparison with oldLogicalTop might yield a
736 // false negative as adding and removing margins, borders etc from a saturated number
737 // might yield incorrect results. If this is the case always mark for layout.
738 markDescendantsWithFloats = true;
weinig@apple.com12840dc2013-10-22 23:59:08 +0000739 else if (!child.avoidsFloats() || child.shrinkToAvoidFloats()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000740 // If an element might be affected by the presence of floats, then always mark it for
741 // layout.
andersca@apple.com86298632013-11-10 19:32:33 +0000742 LayoutUnit fb = std::max(previousFloatLogicalBottom, lowestFloatLogicalBottom());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000743 if (fb > logicalTopEstimate)
744 markDescendantsWithFloats = true;
745 }
746
hyatt@apple.com2ea59882013-09-17 16:41:42 +0000747 if (childBlockFlow) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000748 if (markDescendantsWithFloats)
hyatt@apple.com2ea59882013-09-17 16:41:42 +0000749 childBlockFlow->markAllDescendantsWithFloatsForLayout();
weinig@apple.com12840dc2013-10-22 23:59:08 +0000750 if (!child.isWritingModeRoot())
andersca@apple.com86298632013-11-10 19:32:33 +0000751 previousFloatLogicalBottom = std::max(previousFloatLogicalBottom, oldLogicalTop + childBlockFlow->lowestFloatLogicalBottom());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000752 }
753
hyatt@apple.comccad3742015-02-04 21:39:00 +0000754 child.markForPaginationRelayoutIfNeeded();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000755
weinig@apple.com12840dc2013-10-22 23:59:08 +0000756 bool childHadLayout = child.everHadLayout();
757 bool childNeededLayout = child.needsLayout();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000758 if (childNeededLayout)
weinig@apple.com12840dc2013-10-22 23:59:08 +0000759 child.layout();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000760
761 // Cache if we are at the top of the block right now.
762 bool atBeforeSideOfBlock = marginInfo.atBeforeSideOfBlock();
763
764 // Now determine the correct ypos based off examination of collapsing margin
765 // values.
766 LayoutUnit logicalTopBeforeClear = collapseMargins(child, marginInfo);
767
768 // Now check for clear.
769 LayoutUnit logicalTopAfterClear = clearFloatsIfNeeded(child, marginInfo, oldPosMarginBefore, oldNegMarginBefore, logicalTopBeforeClear);
770
zalan@apple.com4de96c52017-11-07 04:09:59 +0000771 bool paginated = view().frameView().layoutContext().layoutState()->isPaginated();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000772 if (paginated)
weinig@apple.com12840dc2013-10-22 23:59:08 +0000773 logicalTopAfterClear = adjustBlockChildForPagination(logicalTopAfterClear, estimateWithoutPagination, child, atBeforeSideOfBlock && logicalTopBeforeClear == logicalTopAfterClear);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000774
775 setLogicalTopForChild(child, logicalTopAfterClear, ApplyLayoutDelta);
776
777 // Now we have a final top position. See if it really does end up being different from our estimate.
778 // clearFloatsIfNeeded can also mark the child as needing a layout even though we didn't move. This happens
779 // when collapseMargins dynamically adds overhanging floats because of a child with negative margins.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000780 if (logicalTopAfterClear != logicalTopEstimate || child.needsLayout() || (paginated && childBlockFlow && childBlockFlow->shouldBreakAtLineToAvoidWidow())) {
781 if (child.shrinkToAvoidFloats()) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +0000782 // The child's width depends on the line width. When the child shifts to clear an item, its width can
783 // change (because it has more available line width). So mark the item as dirty.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000784 child.setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000785 }
786
hyatt@apple.com2ea59882013-09-17 16:41:42 +0000787 if (childBlockFlow) {
weinig@apple.com12840dc2013-10-22 23:59:08 +0000788 if (!child.avoidsFloats() && childBlockFlow->containsFloats())
hyatt@apple.com2ea59882013-09-17 16:41:42 +0000789 childBlockFlow->markAllDescendantsWithFloatsForLayout();
hyatt@apple.comccad3742015-02-04 21:39:00 +0000790 child.markForPaginationRelayoutIfNeeded();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000791 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000792 }
793
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +0000794 if (updateFragmentRangeForBoxChild(child))
weinig@apple.com12840dc2013-10-22 23:59:08 +0000795 child.setNeedsLayout(MarkOnlyThis);
abucur@adobe.comeaf5e222014-05-14 14:35:07 +0000796
797 // In case our guess was wrong, relayout the child.
798 child.layoutIfNeeded();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000799
800 // We are no longer at the top of the block if we encounter a non-empty child.
801 // This has to be done after checking for clear, so that margins can be reset if a clear occurred.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000802 if (marginInfo.atBeforeSideOfBlock() && !child.isSelfCollapsingBlock())
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000803 marginInfo.setAtBeforeSideOfBlock(false);
804
805 // Now place the child in the correct left position
806 determineLogicalLeftPositionForChild(child, ApplyLayoutDelta);
807
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000808 // Update our height now that the child has been placed in the correct position.
stavila@adobe.comb0d86c42014-04-09 17:07:50 +0000809 setLogicalHeight(logicalHeight() + logicalHeightForChildForFragmentation(child));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000810 if (mustSeparateMarginAfterForChild(child)) {
811 setLogicalHeight(logicalHeight() + marginAfterForChild(child));
812 marginInfo.clearMargin();
813 }
814 // If the child has overhanging floats that intrude into following siblings (or possibly out
815 // of this block), then the parent gets notified of the floats now.
hyatt@apple.com2ea59882013-09-17 16:41:42 +0000816 if (childBlockFlow && childBlockFlow->containsFloats())
andersca@apple.com86298632013-11-10 19:32:33 +0000817 maxFloatLogicalBottom = std::max(maxFloatLogicalBottom, addOverhangingFloats(*childBlockFlow, !childNeededLayout));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000818
zoltan@webkit.org7d4f8cc2014-03-26 18:20:15 +0000819 LayoutSize childOffset = child.location() - oldRect.location();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000820 if (childOffset.width() || childOffset.height()) {
zalan@apple.com4de96c52017-11-07 04:09:59 +0000821 view().frameView().layoutContext().addLayoutDelta(childOffset);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000822
823 // If the child moved, we have to repaint it as well as any floating/positioned
824 // descendants. An exception is if we need a layout. In this case, we know we're going to
825 // repaint ourselves (and the child) anyway.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000826 if (childHadLayout && !selfNeedsLayout() && child.checkForRepaintDuringLayout())
827 child.repaintDuringLayoutIfMoved(oldRect);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000828 }
829
weinig@apple.com12840dc2013-10-22 23:59:08 +0000830 if (!childHadLayout && child.checkForRepaintDuringLayout()) {
831 child.repaint();
832 child.repaintOverhangingFloats(true);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000833 }
834
835 if (paginated) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +0000836 if (RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow())
837 fragmentedFlow->fragmentedFlowDescendantBoxLaidOut(&child);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000838 // Check for an after page/column break.
839 LayoutUnit newHeight = applyAfterBreak(child, logicalHeight(), marginInfo);
840 if (newHeight != height())
841 setLogicalHeight(newHeight);
842 }
843
zalan@apple.com4de96c52017-11-07 04:09:59 +0000844 ASSERT(view().frameView().layoutContext().layoutDeltaMatches(oldLayoutDelta));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000845}
846
weinig@apple.com12840dc2013-10-22 23:59:08 +0000847void RenderBlockFlow::adjustPositionedBlock(RenderBox& child, const MarginInfo& marginInfo)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000848{
849 bool isHorizontal = isHorizontalWritingMode();
akling@apple.com827be9c2013-10-29 02:58:43 +0000850 bool hasStaticBlockPosition = child.style().hasStaticBlockPosition(isHorizontal);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000851
852 LayoutUnit logicalTop = logicalHeight();
zalan@apple.com4d97a002016-02-24 17:13:33 +0000853 updateStaticInlinePositionForChild(child, logicalTop, DoNotIndentText);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000854
855 if (!marginInfo.canCollapseWithMarginBefore()) {
856 // Positioned blocks don't collapse margins, so add the margin provided by
857 // the container now. The child's own margin is added later when calculating its logical top.
858 LayoutUnit collapsedBeforePos = marginInfo.positiveMargin();
859 LayoutUnit collapsedBeforeNeg = marginInfo.negativeMargin();
860 logicalTop += collapsedBeforePos - collapsedBeforeNeg;
861 }
862
weinig@apple.com12840dc2013-10-22 23:59:08 +0000863 RenderLayer* childLayer = child.layer();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000864 if (childLayer->staticBlockPosition() != logicalTop) {
865 childLayer->setStaticBlockPosition(logicalTop);
866 if (hasStaticBlockPosition)
weinig@apple.com12840dc2013-10-22 23:59:08 +0000867 child.setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000868 }
869}
870
robert@webkit.org97037ef2013-11-20 19:26:10 +0000871LayoutUnit RenderBlockFlow::marginOffsetForSelfCollapsingBlock()
872{
873 ASSERT(isSelfCollapsingBlock());
cdumez@apple.com34e77ab2014-10-09 16:17:06 +0000874 RenderBlockFlow* parentBlock = downcast<RenderBlockFlow>(parent());
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +0000875 if (parentBlock && style().clear() != Clear::None && parentBlock->getClearDelta(*this, logicalHeight()))
robert@webkit.org97037ef2013-11-20 19:26:10 +0000876 return marginValuesForChild(*this).positiveMarginBefore();
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +0000877 return 0_lu;
robert@webkit.org97037ef2013-11-20 19:26:10 +0000878}
879
hyatt@apple.com31a5daa2014-01-28 01:26:37 +0000880void RenderBlockFlow::determineLogicalLeftPositionForChild(RenderBox& child, ApplyLayoutDeltaMode applyDelta)
881{
882 LayoutUnit startPosition = borderStart() + paddingStart();
mmaxfield@apple.com4195a702016-04-27 01:25:26 +0000883 if (shouldPlaceBlockDirectionScrollbarOnLeft())
mmaxfield@apple.comaf573be2016-03-12 21:18:25 +0000884 startPosition += (style().isLeftToRightDirection() ? 1 : -1) * verticalScrollbarWidth();
hyatt@apple.com31a5daa2014-01-28 01:26:37 +0000885 LayoutUnit totalAvailableLogicalWidth = borderAndPaddingLogicalWidth() + availableLogicalWidth();
886
887 // Add in our start margin.
888 LayoutUnit childMarginStart = marginStartForChild(child);
889 LayoutUnit newPosition = startPosition + childMarginStart;
890
891 // Some objects (e.g., tables, horizontal rules, overflow:auto blocks) avoid floats. They need
892 // to shift over as necessary to dodge any floats that might get in the way.
antti@apple.comeae76122017-09-20 15:37:23 +0000893 if (child.avoidsFloats() && containsFloats())
hyatt@apple.com31a5daa2014-01-28 01:26:37 +0000894 newPosition += computeStartPositionDeltaForChildAvoidingFloats(child, marginStartForChild(child));
895
896 setLogicalLeftForChild(child, style().isLeftToRightDirection() ? newPosition : totalAvailableLogicalWidth - newPosition - logicalWidthForChild(child), applyDelta);
897}
robert@webkit.org97037ef2013-11-20 19:26:10 +0000898
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000899void RenderBlockFlow::adjustFloatingBlock(const MarginInfo& marginInfo)
900{
901 // The float should be positioned taking into account the bottom margin
902 // of the previous flow. We add that margin into the height, get the
903 // float positioned properly, and then subtract the margin out of the
904 // height again. In the case of self-collapsing blocks, we always just
905 // use the top margins, since the self-collapsing block collapsed its
906 // own bottom margin into its top margin.
907 //
908 // Note also that the previous flow may collapse its margin into the top of
909 // our block. If this is the case, then we do not add the margin in to our
910 // height when computing the position of the float. This condition can be tested
911 // for by simply calling canCollapseWithMarginBefore. See
912 // http://www.hixie.ch/tests/adhoc/css/box/block/margin-collapse/046.html for
913 // an example of this scenario.
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +0000914 LayoutUnit marginOffset = marginInfo.canCollapseWithMarginBefore() ? 0_lu : marginInfo.margin();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000915 setLogicalHeight(logicalHeight() + marginOffset);
916 positionNewFloats();
917 setLogicalHeight(logicalHeight() - marginOffset);
918}
919
zalan@apple.com4d97a002016-02-24 17:13:33 +0000920void RenderBlockFlow::updateStaticInlinePositionForChild(RenderBox& child, LayoutUnit logicalTop, IndentTextOrNot shouldIndentText)
weinig@apple.com12840dc2013-10-22 23:59:08 +0000921{
akling@apple.com827be9c2013-10-29 02:58:43 +0000922 if (child.style().isOriginalDisplayInlineType())
antti@apple.comc6a9c732019-08-12 15:31:34 +0000923 setStaticInlinePositionForChild(child, logicalTop, startAlignedOffsetForLine(logicalTop, shouldIndentText));
weinig@apple.com12840dc2013-10-22 23:59:08 +0000924 else
925 setStaticInlinePositionForChild(child, logicalTop, startOffsetForContent(logicalTop));
926}
927
928void RenderBlockFlow::setStaticInlinePositionForChild(RenderBox& child, LayoutUnit blockOffset, LayoutUnit inlinePosition)
929{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +0000930 if (enclosingFragmentedFlow()) {
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +0000931 // Shift the inline position to exclude the fragment offset.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000932 inlinePosition += startOffsetForContent() - startOffsetForContent(blockOffset);
933 }
934 child.layer()->setStaticInlinePosition(inlinePosition);
935}
936
antti@apple.comc6a9c732019-08-12 15:31:34 +0000937LayoutUnit RenderBlockFlow::startAlignedOffsetForLine(LayoutUnit position, IndentTextOrNot shouldIndentText)
938{
939 TextAlignMode textAlign = style().textAlign();
940 bool shouldApplyIndentText = false;
941 switch (textAlign) {
942 case TextAlignMode::Left:
943 case TextAlignMode::WebKitLeft:
944 shouldApplyIndentText = style().isLeftToRightDirection();
945 break;
946 case TextAlignMode::Right:
947 case TextAlignMode::WebKitRight:
948 shouldApplyIndentText = !style().isLeftToRightDirection();
949 break;
950 case TextAlignMode::Start:
951 shouldApplyIndentText = true;
952 break;
953 default:
954 shouldApplyIndentText = false;
955 }
956 // <rdar://problem/15427571>
957 // https://bugs.webkit.org/show_bug.cgi?id=124522
958 // This quirk is for legacy content that doesn't work properly with the center positioning scheme
959 // being honored (e.g., epubs).
960 if (shouldApplyIndentText || settings().useLegacyTextAlignPositionedElementBehavior()) // FIXME: Handle TextAlignMode::End here
961 return startOffsetForLine(position, shouldIndentText);
962
963 // updateLogicalWidthForAlignment() handles the direction of the block so no need to consider it here
964 float totalLogicalWidth = 0;
965 float logicalLeft = logicalLeftOffsetForLine(logicalHeight(), DoNotIndentText);
966 float availableLogicalWidth = logicalRightOffsetForLine(logicalHeight(), DoNotIndentText) - logicalLeft;
967
968 ComplexLineLayout::updateLogicalWidthForAlignment(*this, textAlign, nullptr, nullptr, logicalLeft, totalLogicalWidth, availableLogicalWidth, 0);
969
970 if (!style().isLeftToRightDirection())
971 return LayoutUnit(logicalWidth() - logicalLeft);
972
973 return LayoutUnit(logicalLeft);
974}
975
weinig@apple.com12840dc2013-10-22 23:59:08 +0000976RenderBlockFlow::MarginValues RenderBlockFlow::marginValuesForChild(RenderBox& child) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000977{
ross.kirsling@sony.combd744282018-11-18 03:14:31 +0000978 LayoutUnit childBeforePositive;
979 LayoutUnit childBeforeNegative;
980 LayoutUnit childAfterPositive;
981 LayoutUnit childAfterNegative;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000982
ross.kirsling@sony.combd744282018-11-18 03:14:31 +0000983 LayoutUnit beforeMargin;
984 LayoutUnit afterMargin;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000985
cdumez@apple.com34e77ab2014-10-09 16:17:06 +0000986 RenderBlockFlow* childRenderBlock = is<RenderBlockFlow>(child) ? &downcast<RenderBlockFlow>(child) : nullptr;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000987
988 // If the child has the same directionality as we do, then we can just return its
989 // margins in the same direction.
weinig@apple.com12840dc2013-10-22 23:59:08 +0000990 if (!child.isWritingModeRoot()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000991 if (childRenderBlock) {
992 childBeforePositive = childRenderBlock->maxPositiveMarginBefore();
993 childBeforeNegative = childRenderBlock->maxNegativeMarginBefore();
994 childAfterPositive = childRenderBlock->maxPositiveMarginAfter();
995 childAfterNegative = childRenderBlock->maxNegativeMarginAfter();
996 } else {
weinig@apple.com12840dc2013-10-22 23:59:08 +0000997 beforeMargin = child.marginBefore();
998 afterMargin = child.marginAfter();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +0000999 }
weinig@apple.com12840dc2013-10-22 23:59:08 +00001000 } else if (child.isHorizontalWritingMode() == isHorizontalWritingMode()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001001 // The child has a different directionality. If the child is parallel, then it's just
1002 // flipped relative to us. We can use the margins for the opposite edges.
1003 if (childRenderBlock) {
1004 childBeforePositive = childRenderBlock->maxPositiveMarginAfter();
1005 childBeforeNegative = childRenderBlock->maxNegativeMarginAfter();
1006 childAfterPositive = childRenderBlock->maxPositiveMarginBefore();
1007 childAfterNegative = childRenderBlock->maxNegativeMarginBefore();
1008 } else {
weinig@apple.com12840dc2013-10-22 23:59:08 +00001009 beforeMargin = child.marginAfter();
1010 afterMargin = child.marginBefore();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001011 }
1012 } else {
1013 // The child is perpendicular to us, which means its margins don't collapse but are on the
1014 // "logical left/right" sides of the child box. We can just return the raw margin in this case.
1015 beforeMargin = marginBeforeForChild(child);
1016 afterMargin = marginAfterForChild(child);
1017 }
1018
1019 // Resolve uncollapsing margins into their positive/negative buckets.
1020 if (beforeMargin) {
1021 if (beforeMargin > 0)
1022 childBeforePositive = beforeMargin;
1023 else
1024 childBeforeNegative = -beforeMargin;
1025 }
1026 if (afterMargin) {
1027 if (afterMargin > 0)
1028 childAfterPositive = afterMargin;
1029 else
1030 childAfterNegative = -afterMargin;
1031 }
1032
1033 return MarginValues(childBeforePositive, childBeforeNegative, childAfterPositive, childAfterNegative);
1034}
1035
hyatt@apple.com72311dc2015-09-10 22:15:46 +00001036bool RenderBlockFlow::childrenPreventSelfCollapsing() const
1037{
1038 if (!childrenInline())
1039 return RenderBlock::childrenPreventSelfCollapsing();
1040
antti@apple.comae85e112017-08-31 23:27:02 +00001041 return hasLines();
hyatt@apple.com72311dc2015-09-10 22:15:46 +00001042}
1043
weinig@apple.com12840dc2013-10-22 23:59:08 +00001044LayoutUnit RenderBlockFlow::collapseMargins(RenderBox& child, MarginInfo& marginInfo)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001045{
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001046 return collapseMarginsWithChildInfo(&child, child.previousSibling(), marginInfo);
1047}
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001048
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001049LayoutUnit RenderBlockFlow::collapseMarginsWithChildInfo(RenderBox* child, RenderObject* prevSibling, MarginInfo& marginInfo)
1050{
1051 bool childDiscardMarginBefore = child ? mustDiscardMarginBeforeForChild(*child) : false;
1052 bool childDiscardMarginAfter = child ? mustDiscardMarginAfterForChild(*child) : false;
1053 bool childIsSelfCollapsing = child ? child->isSelfCollapsingBlock() : false;
1054 bool beforeQuirk = child ? hasMarginBeforeQuirk(*child) : false;
1055 bool afterQuirk = child ? hasMarginAfterQuirk(*child) : false;
1056
rmorisset@apple.comab64b6c2017-10-09 16:49:13 +00001057 // The child discards the before margin when the after margin has discarded in the case of a self collapsing block.
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001058 childDiscardMarginBefore = childDiscardMarginBefore || (childDiscardMarginAfter && childIsSelfCollapsing);
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001059
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001060 // Get the four margin values for the child and cache them.
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001061 const MarginValues childMargins = child ? marginValuesForChild(*child) : MarginValues(0, 0, 0, 0);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001062
1063 // Get our max pos and neg top margins.
1064 LayoutUnit posTop = childMargins.positiveMarginBefore();
1065 LayoutUnit negTop = childMargins.negativeMarginBefore();
1066
1067 // For self-collapsing blocks, collapse our bottom margins into our
1068 // top to get new posTop and negTop values.
1069 if (childIsSelfCollapsing) {
andersca@apple.com86298632013-11-10 19:32:33 +00001070 posTop = std::max(posTop, childMargins.positiveMarginAfter());
1071 negTop = std::max(negTop, childMargins.negativeMarginAfter());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001072 }
1073
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001074 if (marginInfo.canCollapseWithMarginBefore()) {
1075 if (!childDiscardMarginBefore && !marginInfo.discardMargin()) {
1076 // This child is collapsing with the top of the
1077 // block. If it has larger margin values, then we need to update
1078 // our own maximal values.
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001079 if (!document().inQuirksMode() || !marginInfo.quirkContainer() || !beforeQuirk)
andersca@apple.com86298632013-11-10 19:32:33 +00001080 setMaxMarginBeforeValues(std::max(posTop, maxPositiveMarginBefore()), std::max(negTop, maxNegativeMarginBefore()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001081
1082 // The minute any of the margins involved isn't a quirk, don't
1083 // collapse it away, even if the margin is smaller (www.webreference.com
1084 // has an example of this, a <dt> with 0.8em author-specified inside
1085 // a <dl> inside a <td>.
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001086 if (!marginInfo.determinedMarginBeforeQuirk() && !beforeQuirk && (posTop - negTop)) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001087 setHasMarginBeforeQuirk(false);
1088 marginInfo.setDeterminedMarginBeforeQuirk(true);
1089 }
1090
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001091 if (!marginInfo.determinedMarginBeforeQuirk() && beforeQuirk && !marginBefore()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001092 // We have no top margin and our top child has a quirky margin.
1093 // We will pick up this quirky margin and pass it through.
1094 // This deals with the <td><div><p> case.
1095 // Don't do this for a block that split two inlines though. You do
1096 // still apply margins in this case.
1097 setHasMarginBeforeQuirk(true);
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001098 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001099 } else
1100 // The before margin of the container will also discard all the margins it is collapsing with.
1101 setMustDiscardMarginBefore();
1102 }
1103
1104 // Once we find a child with discardMarginBefore all the margins collapsing with us must also discard.
1105 if (childDiscardMarginBefore) {
1106 marginInfo.setDiscardMargin(true);
1107 marginInfo.clearMargin();
1108 }
1109
1110 if (marginInfo.quirkContainer() && marginInfo.atBeforeSideOfBlock() && (posTop - negTop))
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001111 marginInfo.setHasMarginBeforeQuirk(beforeQuirk);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001112
1113 LayoutUnit beforeCollapseLogicalTop = logicalHeight();
1114 LayoutUnit logicalTop = beforeCollapseLogicalTop;
robert@webkit.org97037ef2013-11-20 19:26:10 +00001115
1116 LayoutUnit clearanceForSelfCollapsingBlock;
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001117
robert@webkit.org97037ef2013-11-20 19:26:10 +00001118 // If the child's previous sibling is a self-collapsing block that cleared a float then its top border edge has been set at the bottom border edge
1119 // of the float. Since we want to collapse the child's top margin with the self-collapsing block's top and bottom margins we need to adjust our parent's height to match the
1120 // margin top of the self-collapsing block. If the resulting collapsed margin leaves the child still intruding into the float then we will want to clear it.
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001121 if (!marginInfo.canCollapseWithMarginBefore() && is<RenderBlockFlow>(prevSibling) && downcast<RenderBlockFlow>(*prevSibling).isSelfCollapsingBlock()) {
1122 clearanceForSelfCollapsingBlock = downcast<RenderBlockFlow>(*prevSibling).marginOffsetForSelfCollapsingBlock();
robert@webkit.org97037ef2013-11-20 19:26:10 +00001123 setLogicalHeight(logicalHeight() - clearanceForSelfCollapsingBlock);
1124 }
1125
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001126 if (childIsSelfCollapsing) {
1127 // For a self collapsing block both the before and after margins get discarded. The block doesn't contribute anything to the height of the block.
1128 // Also, the child's top position equals the logical height of the container.
1129 if (!childDiscardMarginBefore && !marginInfo.discardMargin()) {
1130 // This child has no height. We need to compute our
1131 // position before we collapse the child's margins together,
1132 // so that we can get an accurate position for the zero-height block.
andersca@apple.com86298632013-11-10 19:32:33 +00001133 LayoutUnit collapsedBeforePos = std::max(marginInfo.positiveMargin(), childMargins.positiveMarginBefore());
1134 LayoutUnit collapsedBeforeNeg = std::max(marginInfo.negativeMargin(), childMargins.negativeMarginBefore());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001135 marginInfo.setMargin(collapsedBeforePos, collapsedBeforeNeg);
1136
1137 // Now collapse the child's margins together, which means examining our
1138 // bottom margin values as well.
1139 marginInfo.setPositiveMarginIfLarger(childMargins.positiveMarginAfter());
1140 marginInfo.setNegativeMarginIfLarger(childMargins.negativeMarginAfter());
1141
1142 if (!marginInfo.canCollapseWithMarginBefore())
1143 // We need to make sure that the position of the self-collapsing block
1144 // is correct, since it could have overflowing content
1145 // that needs to be positioned correctly (e.g., a block that
1146 // had a specified height of 0 but that actually had subcontent).
1147 logicalTop = logicalHeight() + collapsedBeforePos - collapsedBeforeNeg;
1148 }
1149 } else {
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001150 if (child && mustSeparateMarginBeforeForChild(*child)) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001151 ASSERT(!marginInfo.discardMargin() || (marginInfo.discardMargin() && !marginInfo.margin()));
1152 // If we are at the before side of the block and we collapse, ignore the computed margin
1153 // and just add the child margin to the container height. This will correctly position
1154 // the child inside the container.
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00001155 LayoutUnit separateMargin = !marginInfo.canCollapseWithMarginBefore() ? marginInfo.margin() : 0_lu;
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001156 setLogicalHeight(logicalHeight() + separateMargin + marginBeforeForChild(*child));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001157 logicalTop = logicalHeight();
1158 } else if (!marginInfo.discardMargin() && (!marginInfo.atBeforeSideOfBlock()
1159 || (!marginInfo.canCollapseMarginBeforeWithChildren()
1160 && (!document().inQuirksMode() || !marginInfo.quirkContainer() || !marginInfo.hasMarginBeforeQuirk())))) {
1161 // We're collapsing with a previous sibling's margins and not
1162 // with the top of the block.
andersca@apple.com86298632013-11-10 19:32:33 +00001163 setLogicalHeight(logicalHeight() + std::max(marginInfo.positiveMargin(), posTop) - std::max(marginInfo.negativeMargin(), negTop));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001164 logicalTop = logicalHeight();
1165 }
1166
1167 marginInfo.setDiscardMargin(childDiscardMarginAfter);
1168
1169 if (!marginInfo.discardMargin()) {
1170 marginInfo.setPositiveMargin(childMargins.positiveMarginAfter());
1171 marginInfo.setNegativeMargin(childMargins.negativeMarginAfter());
1172 } else
1173 marginInfo.clearMargin();
1174
1175 if (marginInfo.margin())
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001176 marginInfo.setHasMarginAfterQuirk(afterQuirk);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001177 }
1178
1179 // If margins would pull us past the top of the next page, then we need to pull back and pretend like the margins
1180 // collapsed into the page edge.
zalan@apple.com4de96c52017-11-07 04:09:59 +00001181 auto* layoutState = view().frameView().layoutContext().layoutState();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001182 if (layoutState->isPaginated() && layoutState->pageLogicalHeight() && logicalTop > beforeCollapseLogicalTop
1183 && hasNextPage(beforeCollapseLogicalTop)) {
1184 LayoutUnit oldLogicalTop = logicalTop;
andersca@apple.com86298632013-11-10 19:32:33 +00001185 logicalTop = std::min(logicalTop, nextPageLogicalTop(beforeCollapseLogicalTop));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001186 setLogicalHeight(logicalHeight() + (logicalTop - oldLogicalTop));
1187 }
1188
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001189 if (is<RenderBlockFlow>(prevSibling) && !prevSibling->isFloatingOrOutOfFlowPositioned()) {
robert@webkit.org97037ef2013-11-20 19:26:10 +00001190 // If |child| is a self-collapsing block it may have collapsed into a previous sibling and although it hasn't reduced the height of the parent yet
1191 // any floats from the parent will now overhang.
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001192 RenderBlockFlow& block = downcast<RenderBlockFlow>(*prevSibling);
robert@webkit.org97037ef2013-11-20 19:26:10 +00001193 LayoutUnit oldLogicalHeight = logicalHeight();
1194 setLogicalHeight(logicalTop);
weinig@apple.com12840dc2013-10-22 23:59:08 +00001195 if (block.containsFloats() && !block.avoidsFloats() && (block.logicalTop() + block.lowestFloatLogicalBottom()) > logicalTop)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001196 addOverhangingFloats(block, false);
robert@webkit.org97037ef2013-11-20 19:26:10 +00001197 setLogicalHeight(oldLogicalHeight);
1198
1199 // If |child|'s previous sibling is a self-collapsing block that cleared a float and margin collapsing resulted in |child| moving up
1200 // into the margin area of the self-collapsing block then the float it clears is now intruding into |child|. Layout again so that we can look for
1201 // floats in the parent that overhang |child|'s new logical top.
1202 bool logicalTopIntrudesIntoFloat = clearanceForSelfCollapsingBlock > 0 && logicalTop < beforeCollapseLogicalTop;
hyatt@apple.com9a79c622015-09-15 18:38:18 +00001203 if (child && logicalTopIntrudesIntoFloat && containsFloats() && !child->avoidsFloats() && lowestFloatLogicalBottom() > logicalTop)
1204 child->setNeedsLayout();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001205 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001206
1207 return logicalTop;
1208}
1209
weinig@apple.com12840dc2013-10-22 23:59:08 +00001210LayoutUnit RenderBlockFlow::clearFloatsIfNeeded(RenderBox& child, MarginInfo& marginInfo, LayoutUnit oldTopPosMargin, LayoutUnit oldTopNegMargin, LayoutUnit yPos)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001211{
1212 LayoutUnit heightIncrease = getClearDelta(child, yPos);
1213 if (!heightIncrease)
1214 return yPos;
1215
weinig@apple.com12840dc2013-10-22 23:59:08 +00001216 if (child.isSelfCollapsingBlock()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001217 bool childDiscardMargin = mustDiscardMarginBeforeForChild(child) || mustDiscardMarginAfterForChild(child);
1218
1219 // For self-collapsing blocks that clear, they can still collapse their
1220 // margins with following siblings. Reset the current margins to represent
1221 // the self-collapsing block's margins only.
1222 // If DISCARD is specified for -webkit-margin-collapse, reset the margin values.
robert@webkit.org97037ef2013-11-20 19:26:10 +00001223 MarginValues childMargins = marginValuesForChild(child);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001224 if (!childDiscardMargin) {
andersca@apple.com86298632013-11-10 19:32:33 +00001225 marginInfo.setPositiveMargin(std::max(childMargins.positiveMarginBefore(), childMargins.positiveMarginAfter()));
1226 marginInfo.setNegativeMargin(std::max(childMargins.negativeMarginBefore(), childMargins.negativeMarginAfter()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001227 } else
1228 marginInfo.clearMargin();
1229 marginInfo.setDiscardMargin(childDiscardMargin);
1230
1231 // CSS2.1 states:
1232 // "If the top and bottom margins of an element with clearance are adjoining, its margins collapse with
1233 // the adjoining margins of following siblings but that resulting margin does not collapse with the bottom margin of the parent block."
1234 // So the parent's bottom margin cannot collapse through this block or any subsequent self-collapsing blocks. Check subsequent siblings
1235 // for a block with height - if none is found then don't allow the margins to collapse with the parent.
1236 bool wouldCollapseMarginsWithParent = marginInfo.canCollapseMarginAfterWithChildren();
weinig@apple.com12840dc2013-10-22 23:59:08 +00001237 for (RenderBox* curr = child.nextSiblingBox(); curr && wouldCollapseMarginsWithParent; curr = curr->nextSiblingBox()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001238 if (!curr->isFloatingOrOutOfFlowPositioned() && !curr->isSelfCollapsingBlock())
1239 wouldCollapseMarginsWithParent = false;
1240 }
1241 if (wouldCollapseMarginsWithParent)
1242 marginInfo.setCanCollapseMarginAfterWithChildren(false);
1243
robert@webkit.org97037ef2013-11-20 19:26:10 +00001244 // For now set the border-top of |child| flush with the bottom border-edge of the float so it can layout any floating or positioned children of
1245 // its own at the correct vertical position. If subsequent siblings attempt to collapse with |child|'s margins in |collapseMargins| we will
1246 // adjust the height of the parent to |child|'s margin top (which if it is positive sits up 'inside' the float it's clearing) so that all three
1247 // margins can collapse at the correct vertical position.
1248 // Per CSS2.1 we need to ensure that any negative margin-top clears |child| beyond the bottom border-edge of the float so that the top border edge of the child
1249 // (i.e. its clearance) is at a position that satisfies the equation: "the amount of clearance is set so that clearance + margin-top = [height of float],
1250 // i.e., clearance = [height of float] - margin-top".
1251 setLogicalHeight(child.logicalTop() + childMargins.negativeMarginBefore());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001252 } else
1253 // Increase our height by the amount we had to clear.
1254 setLogicalHeight(logicalHeight() + heightIncrease);
1255
1256 if (marginInfo.canCollapseWithMarginBefore()) {
1257 // We can no longer collapse with the top of the block since a clear
1258 // occurred. The empty blocks collapse into the cleared block.
1259 // FIXME: This isn't quite correct. Need clarification for what to do
1260 // if the height the cleared block is offset by is smaller than the
1261 // margins involved.
1262 setMaxMarginBeforeValues(oldTopPosMargin, oldTopNegMargin);
1263 marginInfo.setAtBeforeSideOfBlock(false);
1264
1265 // In case the child discarded the before margin of the block we need to reset the mustDiscardMarginBefore flag to the initial value.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001266 setMustDiscardMarginBefore(style().marginBeforeCollapse() == MarginCollapse::Discard);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001267 }
1268
robert@webkit.org97037ef2013-11-20 19:26:10 +00001269 return yPos + heightIncrease;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001270}
1271
weinig@apple.com12840dc2013-10-22 23:59:08 +00001272void RenderBlockFlow::marginBeforeEstimateForChild(RenderBox& child, LayoutUnit& positiveMarginBefore, LayoutUnit& negativeMarginBefore, bool& discardMarginBefore) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001273{
1274 // Give up if in quirks mode and we're a body/table cell and the top margin of the child box is quirky.
1275 // Give up if the child specified -webkit-margin-collapse: separate that prevents collapsing.
1276 // FIXME: Use writing mode independent accessor for marginBeforeCollapse.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001277 if ((document().inQuirksMode() && hasMarginAfterQuirk(child) && (isTableCell() || isBody())) || child.style().marginBeforeCollapse() == MarginCollapse::Separate)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001278 return;
1279
1280 // The margins are discarded by a child that specified -webkit-margin-collapse: discard.
1281 // FIXME: Use writing mode independent accessor for marginBeforeCollapse.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001282 if (child.style().marginBeforeCollapse() == MarginCollapse::Discard) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001283 positiveMarginBefore = 0;
1284 negativeMarginBefore = 0;
1285 discardMarginBefore = true;
1286 return;
1287 }
1288
1289 LayoutUnit beforeChildMargin = marginBeforeForChild(child);
andersca@apple.com86298632013-11-10 19:32:33 +00001290 positiveMarginBefore = std::max(positiveMarginBefore, beforeChildMargin);
1291 negativeMarginBefore = std::max(negativeMarginBefore, -beforeChildMargin);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001292
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00001293 if (!is<RenderBlockFlow>(child))
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001294 return;
1295
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00001296 RenderBlockFlow& childBlock = downcast<RenderBlockFlow>(child);
weinig@apple.com12840dc2013-10-22 23:59:08 +00001297 if (childBlock.childrenInline() || childBlock.isWritingModeRoot())
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001298 return;
1299
weinig@apple.com12840dc2013-10-22 23:59:08 +00001300 MarginInfo childMarginInfo(childBlock, childBlock.borderAndPaddingBefore(), childBlock.borderAndPaddingAfter());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001301 if (!childMarginInfo.canCollapseMarginBeforeWithChildren())
1302 return;
1303
weinig@apple.com12840dc2013-10-22 23:59:08 +00001304 RenderBox* grandchildBox = childBlock.firstChildBox();
1305 for (; grandchildBox; grandchildBox = grandchildBox->nextSiblingBox()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001306 if (!grandchildBox->isFloatingOrOutOfFlowPositioned())
1307 break;
1308 }
1309
1310 // Give up if there is clearance on the box, since it probably won't collapse into us.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001311 if (!grandchildBox || grandchildBox->style().clear() != Clear::None)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001312 return;
1313
1314 // Make sure to update the block margins now for the grandchild box so that we're looking at current values.
1315 if (grandchildBox->needsLayout()) {
mmaxfield@apple.com09804f42016-03-23 00:58:34 +00001316 grandchildBox->computeAndSetBlockDirectionMargins(*this);
cdumez@apple.come9437792014-10-08 23:33:43 +00001317 if (is<RenderBlock>(*grandchildBox)) {
1318 RenderBlock& grandchildBlock = downcast<RenderBlock>(*grandchildBox);
1319 grandchildBlock.setHasMarginBeforeQuirk(grandchildBox->style().hasMarginBeforeQuirk());
1320 grandchildBlock.setHasMarginAfterQuirk(grandchildBox->style().hasMarginAfterQuirk());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001321 }
1322 }
1323
1324 // Collapse the margin of the grandchild box with our own to produce an estimate.
weinig@apple.com12840dc2013-10-22 23:59:08 +00001325 childBlock.marginBeforeEstimateForChild(*grandchildBox, positiveMarginBefore, negativeMarginBefore, discardMarginBefore);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001326}
1327
weinig@apple.com12840dc2013-10-22 23:59:08 +00001328LayoutUnit RenderBlockFlow::estimateLogicalTopPosition(RenderBox& child, const MarginInfo& marginInfo, LayoutUnit& estimateWithoutPagination)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001329{
1330 // FIXME: We need to eliminate the estimation of vertical position, because when it's wrong we sometimes trigger a pathological
1331 // relayout if there are intruding floats.
1332 LayoutUnit logicalTopEstimate = logicalHeight();
1333 if (!marginInfo.canCollapseWithMarginBefore()) {
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00001334 LayoutUnit positiveMarginBefore;
1335 LayoutUnit negativeMarginBefore;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001336 bool discardMarginBefore = false;
weinig@apple.com12840dc2013-10-22 23:59:08 +00001337 if (child.selfNeedsLayout()) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001338 // Try to do a basic estimation of how the collapse is going to go.
1339 marginBeforeEstimateForChild(child, positiveMarginBefore, negativeMarginBefore, discardMarginBefore);
1340 } else {
1341 // Use the cached collapsed margin values from a previous layout. Most of the time they
1342 // will be right.
1343 MarginValues marginValues = marginValuesForChild(child);
andersca@apple.com86298632013-11-10 19:32:33 +00001344 positiveMarginBefore = std::max(positiveMarginBefore, marginValues.positiveMarginBefore());
1345 negativeMarginBefore = std::max(negativeMarginBefore, marginValues.negativeMarginBefore());
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001346 discardMarginBefore = mustDiscardMarginBeforeForChild(child);
1347 }
1348
1349 // Collapse the result with our current margins.
1350 if (!discardMarginBefore)
andersca@apple.com86298632013-11-10 19:32:33 +00001351 logicalTopEstimate += std::max(marginInfo.positiveMargin(), positiveMarginBefore) - std::max(marginInfo.negativeMargin(), negativeMarginBefore);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001352 }
1353
1354 // Adjust logicalTopEstimate down to the next page if the margins are so large that we don't fit on the current
1355 // page.
zalan@apple.com4de96c52017-11-07 04:09:59 +00001356 auto* layoutState = view().frameView().layoutContext().layoutState();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001357 if (layoutState->isPaginated() && layoutState->pageLogicalHeight() && logicalTopEstimate > logicalHeight()
1358 && hasNextPage(logicalHeight()))
andersca@apple.com86298632013-11-10 19:32:33 +00001359 logicalTopEstimate = std::min(logicalTopEstimate, nextPageLogicalTop(logicalHeight()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001360
1361 logicalTopEstimate += getClearDelta(child, logicalTopEstimate);
1362
1363 estimateWithoutPagination = logicalTopEstimate;
1364
1365 if (layoutState->isPaginated()) {
1366 // If the object has a page or column break value of "before", then we should shift to the top of the next page.
1367 logicalTopEstimate = applyBeforeBreak(child, logicalTopEstimate);
1368
1369 // For replaced elements and scrolled elements, we want to shift them to the next page if they don't fit on the current one.
1370 logicalTopEstimate = adjustForUnsplittableChild(child, logicalTopEstimate);
1371
cdumez@apple.come9437792014-10-08 23:33:43 +00001372 if (!child.selfNeedsLayout() && is<RenderBlock>(child))
1373 logicalTopEstimate += downcast<RenderBlock>(child).paginationStrut();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001374 }
1375
1376 return logicalTopEstimate;
1377}
1378
1379void RenderBlockFlow::setCollapsedBottomMargin(const MarginInfo& marginInfo)
1380{
1381 if (marginInfo.canCollapseWithMarginAfter() && !marginInfo.canCollapseWithMarginBefore()) {
1382 // Update the after side margin of the container to discard if the after margin of the last child also discards and we collapse with it.
1383 // Don't update the max margin values because we won't need them anyway.
1384 if (marginInfo.discardMargin()) {
1385 setMustDiscardMarginAfter();
1386 return;
1387 }
1388
1389 // Update our max pos/neg bottom margins, since we collapsed our bottom margins
1390 // with our children.
andersca@apple.com86298632013-11-10 19:32:33 +00001391 setMaxMarginAfterValues(std::max(maxPositiveMarginAfter(), marginInfo.positiveMargin()), std::max(maxNegativeMarginAfter(), marginInfo.negativeMargin()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001392
1393 if (!marginInfo.hasMarginAfterQuirk())
1394 setHasMarginAfterQuirk(false);
1395
1396 if (marginInfo.hasMarginAfterQuirk() && !marginAfter())
1397 // We have no bottom margin and our last child has a quirky margin.
1398 // We will pick up this quirky margin and pass it through.
1399 // This deals with the <td><div><p> case.
1400 setHasMarginAfterQuirk(true);
1401 }
1402}
1403
1404void RenderBlockFlow::handleAfterSideOfBlock(LayoutUnit beforeSide, LayoutUnit afterSide, MarginInfo& marginInfo)
1405{
1406 marginInfo.setAtAfterSideOfBlock(true);
1407
robert@webkit.org97037ef2013-11-20 19:26:10 +00001408 // If our last child was a self-collapsing block with clearance then our logical height is flush with the
1409 // bottom edge of the float that the child clears. The correct vertical position for the margin-collapsing we want
1410 // to perform now is at the child's margin-top - so adjust our height to that position.
1411 RenderObject* lastBlock = lastChild();
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00001412 if (is<RenderBlockFlow>(lastBlock) && downcast<RenderBlockFlow>(*lastBlock).isSelfCollapsingBlock())
1413 setLogicalHeight(logicalHeight() - downcast<RenderBlockFlow>(*lastBlock).marginOffsetForSelfCollapsingBlock());
robert@webkit.org97037ef2013-11-20 19:26:10 +00001414
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00001415 // If we can't collapse with children then add in the bottom margin.
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001416 if (!marginInfo.discardMargin() && (!marginInfo.canCollapseWithMarginAfter() && !marginInfo.canCollapseWithMarginBefore()
1417 && (!document().inQuirksMode() || !marginInfo.quirkContainer() || !marginInfo.hasMarginAfterQuirk())))
1418 setLogicalHeight(logicalHeight() + marginInfo.margin());
1419
1420 // Now add in our bottom border/padding.
1421 setLogicalHeight(logicalHeight() + afterSide);
1422
1423 // Negative margins can cause our height to shrink below our minimal height (border/padding).
1424 // If this happens, ensure that the computed height is increased to the minimal height.
andersca@apple.com86298632013-11-10 19:32:33 +00001425 setLogicalHeight(std::max(logicalHeight(), beforeSide + afterSide));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001426
1427 // Update our bottom collapsed margin info.
1428 setCollapsedBottomMargin(marginInfo);
1429}
1430
1431void RenderBlockFlow::setMaxMarginBeforeValues(LayoutUnit pos, LayoutUnit neg)
1432{
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001433 if (!hasRareBlockFlowData()) {
weinig@apple.com12840dc2013-10-22 23:59:08 +00001434 if (pos == RenderBlockFlowRareData::positiveMarginBeforeDefault(*this) && neg == RenderBlockFlowRareData::negativeMarginBeforeDefault(*this))
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001435 return;
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001436 materializeRareBlockFlowData();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001437 }
weinig@apple.com924a77a2013-11-11 00:22:38 +00001438
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001439 rareBlockFlowData()->m_margins.setPositiveMarginBefore(pos);
1440 rareBlockFlowData()->m_margins.setNegativeMarginBefore(neg);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001441}
1442
1443void RenderBlockFlow::setMaxMarginAfterValues(LayoutUnit pos, LayoutUnit neg)
1444{
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001445 if (!hasRareBlockFlowData()) {
weinig@apple.com12840dc2013-10-22 23:59:08 +00001446 if (pos == RenderBlockFlowRareData::positiveMarginAfterDefault(*this) && neg == RenderBlockFlowRareData::negativeMarginAfterDefault(*this))
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001447 return;
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001448 materializeRareBlockFlowData();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001449 }
weinig@apple.com924a77a2013-11-11 00:22:38 +00001450
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001451 rareBlockFlowData()->m_margins.setPositiveMarginAfter(pos);
1452 rareBlockFlowData()->m_margins.setNegativeMarginAfter(neg);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001453}
1454
1455void RenderBlockFlow::setMustDiscardMarginBefore(bool value)
1456{
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001457 if (style().marginBeforeCollapse() == MarginCollapse::Discard) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001458 ASSERT(value);
1459 return;
1460 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001461
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001462 if (!hasRareBlockFlowData()) {
weinig@apple.com924a77a2013-11-11 00:22:38 +00001463 if (!value)
1464 return;
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001465 materializeRareBlockFlowData();
weinig@apple.com924a77a2013-11-11 00:22:38 +00001466 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001467
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001468 rareBlockFlowData()->m_discardMarginBefore = value;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001469}
1470
1471void RenderBlockFlow::setMustDiscardMarginAfter(bool value)
1472{
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001473 if (style().marginAfterCollapse() == MarginCollapse::Discard) {
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001474 ASSERT(value);
1475 return;
1476 }
1477
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001478 if (!hasRareBlockFlowData()) {
weinig@apple.com924a77a2013-11-11 00:22:38 +00001479 if (!value)
1480 return;
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001481 materializeRareBlockFlowData();
weinig@apple.com924a77a2013-11-11 00:22:38 +00001482 }
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001483
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001484 rareBlockFlowData()->m_discardMarginAfter = value;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001485}
1486
1487bool RenderBlockFlow::mustDiscardMarginBefore() const
1488{
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001489 return style().marginBeforeCollapse() == MarginCollapse::Discard || (hasRareBlockFlowData() && rareBlockFlowData()->m_discardMarginBefore);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001490}
1491
1492bool RenderBlockFlow::mustDiscardMarginAfter() const
1493{
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001494 return style().marginAfterCollapse() == MarginCollapse::Discard || (hasRareBlockFlowData() && rareBlockFlowData()->m_discardMarginAfter);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001495}
1496
weinig@apple.com12840dc2013-10-22 23:59:08 +00001497bool RenderBlockFlow::mustDiscardMarginBeforeForChild(const RenderBox& child) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001498{
weinig@apple.com12840dc2013-10-22 23:59:08 +00001499 ASSERT(!child.selfNeedsLayout());
1500 if (!child.isWritingModeRoot())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001501 return is<RenderBlockFlow>(child) ? downcast<RenderBlockFlow>(child).mustDiscardMarginBefore() : (child.style().marginBeforeCollapse() == MarginCollapse::Discard);
weinig@apple.com12840dc2013-10-22 23:59:08 +00001502 if (child.isHorizontalWritingMode() == isHorizontalWritingMode())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001503 return is<RenderBlockFlow>(child) ? downcast<RenderBlockFlow>(child).mustDiscardMarginAfter() : (child.style().marginAfterCollapse() == MarginCollapse::Discard);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001504
1505 // FIXME: We return false here because the implementation is not geometrically complete. We have values only for before/after, not start/end.
1506 // In case the boxes are perpendicular we assume the property is not specified.
1507 return false;
1508}
1509
weinig@apple.com12840dc2013-10-22 23:59:08 +00001510bool RenderBlockFlow::mustDiscardMarginAfterForChild(const RenderBox& child) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001511{
weinig@apple.com12840dc2013-10-22 23:59:08 +00001512 ASSERT(!child.selfNeedsLayout());
1513 if (!child.isWritingModeRoot())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001514 return is<RenderBlockFlow>(child) ? downcast<RenderBlockFlow>(child).mustDiscardMarginAfter() : (child.style().marginAfterCollapse() == MarginCollapse::Discard);
weinig@apple.com12840dc2013-10-22 23:59:08 +00001515 if (child.isHorizontalWritingMode() == isHorizontalWritingMode())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001516 return is<RenderBlockFlow>(child) ? downcast<RenderBlockFlow>(child).mustDiscardMarginBefore() : (child.style().marginBeforeCollapse() == MarginCollapse::Discard);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001517
1518 // FIXME: See |mustDiscardMarginBeforeForChild| above.
1519 return false;
1520}
1521
weinig@apple.com12840dc2013-10-22 23:59:08 +00001522bool RenderBlockFlow::mustSeparateMarginBeforeForChild(const RenderBox& child) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001523{
weinig@apple.com12840dc2013-10-22 23:59:08 +00001524 ASSERT(!child.selfNeedsLayout());
akling@apple.com827be9c2013-10-29 02:58:43 +00001525 const RenderStyle& childStyle = child.style();
weinig@apple.com12840dc2013-10-22 23:59:08 +00001526 if (!child.isWritingModeRoot())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001527 return childStyle.marginBeforeCollapse() == MarginCollapse::Separate;
weinig@apple.com12840dc2013-10-22 23:59:08 +00001528 if (child.isHorizontalWritingMode() == isHorizontalWritingMode())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001529 return childStyle.marginAfterCollapse() == MarginCollapse::Separate;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001530
1531 // FIXME: See |mustDiscardMarginBeforeForChild| above.
1532 return false;
1533}
1534
weinig@apple.com12840dc2013-10-22 23:59:08 +00001535bool RenderBlockFlow::mustSeparateMarginAfterForChild(const RenderBox& child) const
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001536{
weinig@apple.com12840dc2013-10-22 23:59:08 +00001537 ASSERT(!child.selfNeedsLayout());
akling@apple.com827be9c2013-10-29 02:58:43 +00001538 const RenderStyle& childStyle = child.style();
weinig@apple.com12840dc2013-10-22 23:59:08 +00001539 if (!child.isWritingModeRoot())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001540 return childStyle.marginAfterCollapse() == MarginCollapse::Separate;
weinig@apple.com12840dc2013-10-22 23:59:08 +00001541 if (child.isHorizontalWritingMode() == isHorizontalWritingMode())
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00001542 return childStyle.marginBeforeCollapse() == MarginCollapse::Separate;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001543
1544 // FIXME: See |mustDiscardMarginBeforeForChild| above.
1545 return false;
1546}
1547
weinig@apple.com12840dc2013-10-22 23:59:08 +00001548static bool inNormalFlow(RenderBox& child)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001549{
weinig@apple.com12840dc2013-10-22 23:59:08 +00001550 RenderBlock* curr = child.containingBlock();
1551 while (curr && curr != &child.view()) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001552 if (curr->isRenderFragmentedFlow())
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001553 return true;
1554 if (curr->isFloatingOrOutOfFlowPositioned())
1555 return false;
1556 curr = curr->containingBlock();
1557 }
1558 return true;
1559}
1560
weinig@apple.com12840dc2013-10-22 23:59:08 +00001561LayoutUnit RenderBlockFlow::applyBeforeBreak(RenderBox& child, LayoutUnit logicalOffset)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001562{
1563 // FIXME: Add page break checking here when we support printing.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001564 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
1565 bool isInsideMulticolFlow = fragmentedFlow;
1566 bool checkColumnBreaks = fragmentedFlow && fragmentedFlow->shouldCheckColumnBreaks();
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00001567 bool checkPageBreaks = !checkColumnBreaks && view().frameView().layoutContext().layoutState()->pageLogicalHeight(); // FIXME: Once columns can print we have to check this.
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001568 bool checkFragmentBreaks = false;
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00001569 bool checkBeforeAlways = (checkColumnBreaks && child.style().breakBefore() == BreakBetween::Column)
antti@apple.com52d83832017-09-20 12:58:45 +00001570 || (checkPageBreaks && alwaysPageBreak(child.style().breakBefore()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001571 if (checkBeforeAlways && inNormalFlow(child) && hasNextPage(logicalOffset, IncludePageBoundary)) {
commit-queue@webkit.orgfd8f1a22014-01-20 19:55:59 +00001572 if (checkColumnBreaks) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001573 if (isInsideMulticolFlow)
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001574 checkFragmentBreaks = true;
commit-queue@webkit.orgfd8f1a22014-01-20 19:55:59 +00001575 }
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001576 if (checkFragmentBreaks) {
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00001577 LayoutUnit offsetBreakAdjustment;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001578 if (fragmentedFlow->addForcedFragmentBreak(this, offsetFromLogicalTopOfFirstPage() + logicalOffset, &child, true, &offsetBreakAdjustment))
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001579 return logicalOffset + offsetBreakAdjustment;
1580 }
1581 return nextPageLogicalTop(logicalOffset, IncludePageBoundary);
1582 }
1583 return logicalOffset;
1584}
1585
weinig@apple.com12840dc2013-10-22 23:59:08 +00001586LayoutUnit RenderBlockFlow::applyAfterBreak(RenderBox& child, LayoutUnit logicalOffset, MarginInfo& marginInfo)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001587{
1588 // FIXME: Add page break checking here when we support printing.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001589 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
1590 bool isInsideMulticolFlow = fragmentedFlow;
1591 bool checkColumnBreaks = fragmentedFlow && fragmentedFlow->shouldCheckColumnBreaks();
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00001592 bool checkPageBreaks = !checkColumnBreaks && view().frameView().layoutContext().layoutState()->pageLogicalHeight(); // FIXME: Once columns can print we have to check this.
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001593 bool checkFragmentBreaks = false;
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00001594 bool checkAfterAlways = (checkColumnBreaks && child.style().breakAfter() == BreakBetween::Column)
antti@apple.com52d83832017-09-20 12:58:45 +00001595 || (checkPageBreaks && alwaysPageBreak(child.style().breakAfter()));
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001596 if (checkAfterAlways && inNormalFlow(child) && hasNextPage(logicalOffset, IncludePageBoundary)) {
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00001597 LayoutUnit marginOffset = marginInfo.canCollapseWithMarginBefore() ? 0_lu : marginInfo.margin();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001598
1599 // So our margin doesn't participate in the next collapsing steps.
1600 marginInfo.clearMargin();
1601
commit-queue@webkit.orgfd8f1a22014-01-20 19:55:59 +00001602 if (checkColumnBreaks) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001603 if (isInsideMulticolFlow)
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001604 checkFragmentBreaks = true;
commit-queue@webkit.orgfd8f1a22014-01-20 19:55:59 +00001605 }
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001606 if (checkFragmentBreaks) {
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00001607 LayoutUnit offsetBreakAdjustment;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001608 if (fragmentedFlow->addForcedFragmentBreak(this, offsetFromLogicalTopOfFirstPage() + logicalOffset + marginOffset, &child, false, &offsetBreakAdjustment))
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001609 return logicalOffset + marginOffset + offsetBreakAdjustment;
1610 }
1611 return nextPageLogicalTop(logicalOffset, IncludePageBoundary);
1612 }
1613 return logicalOffset;
1614}
1615
weinig@apple.com12840dc2013-10-22 23:59:08 +00001616LayoutUnit RenderBlockFlow::adjustBlockChildForPagination(LayoutUnit logicalTopAfterClear, LayoutUnit estimateWithoutPagination, RenderBox& child, bool atBeforeSideOfBlock)
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001617{
cdumez@apple.come9437792014-10-08 23:33:43 +00001618 RenderBlock* childRenderBlock = is<RenderBlock>(child) ? &downcast<RenderBlock>(child) : nullptr;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001619
1620 if (estimateWithoutPagination != logicalTopAfterClear) {
1621 // Our guess prior to pagination movement was wrong. Before we attempt to paginate, let's try again at the new
1622 // position.
1623 setLogicalHeight(logicalTopAfterClear);
1624 setLogicalTopForChild(child, logicalTopAfterClear, ApplyLayoutDelta);
1625
weinig@apple.com12840dc2013-10-22 23:59:08 +00001626 if (child.shrinkToAvoidFloats()) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00001627 // The child's width depends on the line width. When the child shifts to clear an item, its width can
1628 // change (because it has more available line width). So mark the item as dirty.
weinig@apple.com12840dc2013-10-22 23:59:08 +00001629 child.setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001630 }
1631
1632 if (childRenderBlock) {
weinig@apple.com12840dc2013-10-22 23:59:08 +00001633 if (!child.avoidsFloats() && childRenderBlock->containsFloats())
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00001634 downcast<RenderBlockFlow>(*childRenderBlock).markAllDescendantsWithFloatsForLayout();
hyatt@apple.comccad3742015-02-04 21:39:00 +00001635 child.markForPaginationRelayoutIfNeeded();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001636 }
1637
1638 // Our guess was wrong. Make the child lay itself out again.
weinig@apple.com12840dc2013-10-22 23:59:08 +00001639 child.layoutIfNeeded();
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001640 }
1641
1642 LayoutUnit oldTop = logicalTopAfterClear;
1643
1644 // If the object has a page or column break value of "before", then we should shift to the top of the next page.
1645 LayoutUnit result = applyBeforeBreak(child, logicalTopAfterClear);
1646
1647 if (pageLogicalHeightForOffset(result)) {
1648 LayoutUnit remainingLogicalHeight = pageRemainingLogicalHeightForOffset(result, ExcludePageBoundary);
weinig@apple.com12840dc2013-10-22 23:59:08 +00001649 LayoutUnit spaceShortage = child.logicalHeight() - remainingLogicalHeight;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001650 if (spaceShortage > 0) {
1651 // If the child crosses a column boundary, report a break, in case nothing inside it has already
1652 // done so. The column balancer needs to know how much it has to stretch the columns to make more
1653 // content fit. If no breaks are reported (but do occur), the balancer will have no clue. FIXME:
1654 // This should be improved, though, because here we just pretend that the child is
1655 // unsplittable. A splittable child, on the other hand, has break opportunities at every position
1656 // where there's no child content, border or padding. In other words, we risk stretching more
1657 // than necessary.
1658 setPageBreak(result, spaceShortage);
1659 }
1660 }
1661
1662 // For replaced elements and scrolled elements, we want to shift them to the next page if they don't fit on the current one.
1663 LayoutUnit logicalTopBeforeUnsplittableAdjustment = result;
1664 LayoutUnit logicalTopAfterUnsplittableAdjustment = adjustForUnsplittableChild(child, result);
1665
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00001666 LayoutUnit paginationStrut;
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001667 LayoutUnit unsplittableAdjustmentDelta = logicalTopAfterUnsplittableAdjustment - logicalTopBeforeUnsplittableAdjustment;
1668 if (unsplittableAdjustmentDelta)
1669 paginationStrut = unsplittableAdjustmentDelta;
1670 else if (childRenderBlock && childRenderBlock->paginationStrut())
1671 paginationStrut = childRenderBlock->paginationStrut();
1672
1673 if (paginationStrut) {
1674 // We are willing to propagate out to our parent block as long as we were at the top of the block prior
1675 // to collapsing our margins, and as long as we didn't clear or move as a result of other pagination.
1676 if (atBeforeSideOfBlock && oldTop == result && !isOutOfFlowPositioned() && !isTableCell()) {
1677 // FIXME: Should really check if we're exceeding the page height before propagating the strut, but we don't
1678 // have all the information to do so (the strut only has the remaining amount to push). Gecko gets this wrong too
1679 // and pushes to the next page anyway, so not too concerned about it.
1680 setPaginationStrut(result + paginationStrut);
1681 if (childRenderBlock)
1682 childRenderBlock->setPaginationStrut(0);
1683 } else
1684 result += paginationStrut;
1685 }
1686
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00001687 // Similar to how we apply clearance. Boost height() to be the place where we're going to position the child.
hyatt@apple.com1807b5b2013-09-11 19:50:03 +00001688 setLogicalHeight(logicalHeight() + (result - oldTop));
1689
1690 // Return the final adjusted logical top.
1691 return result;
1692}
1693
antti@apple.com5c4302b2016-04-26 18:20:09 +00001694static inline LayoutUnit calculateMinimumPageHeight(const RenderStyle& renderStyle, RootInlineBox& lastLine, LayoutUnit lineTop, LayoutUnit lineBottom)
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001695{
1696 // We may require a certain minimum number of lines per page in order to satisfy
1697 // orphans and widows, and that may affect the minimum page height.
mmaxfield@apple.comf8e26e72014-10-30 21:39:27 +00001698 unsigned lineCount = std::max<unsigned>(renderStyle.hasAutoOrphans() ? 1 : renderStyle.orphans(), renderStyle.hasAutoWidows() ? 1 : renderStyle.widows());
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001699 if (lineCount > 1) {
mmaxfield@apple.comf8e26e72014-10-30 21:39:27 +00001700 RootInlineBox* line = &lastLine;
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001701 for (unsigned i = 1; i < lineCount && line->prevRootBox(); i++)
1702 line = line->prevRootBox();
1703
1704 // FIXME: Paginating using line overflow isn't all fine. See FIXME in
1705 // adjustLinePositionForPagination() for more details.
1706 LayoutRect overflow = line->logicalVisualOverflowRect(line->lineTop(), line->lineBottom());
andersca@apple.com86298632013-11-10 19:32:33 +00001707 lineTop = std::min(line->lineTopWithLeading(), overflow.y());
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001708 }
1709 return lineBottom - lineTop;
1710}
1711
bfulgham@apple.comb5953432015-02-13 21:56:01 +00001712static inline bool needsAppleMailPaginationQuirk(RootInlineBox& lineBox)
1713{
zalan@apple.coma1cff3a2017-01-14 18:45:20 +00001714 auto& renderer = lineBox.renderer();
bfulgham@apple.com62729772015-04-29 02:26:07 +00001715
zalan@apple.coma1cff3a2017-01-14 18:45:20 +00001716 if (!renderer.settings().appleMailPaginationQuirkEnabled())
bfulgham@apple.com62729772015-04-29 02:26:07 +00001717 return false;
1718
cdumez@apple.combee81ac2016-04-21 23:28:48 +00001719 if (renderer.element() && renderer.element()->idForStyleResolution() == "messageContentContainer")
bfulgham@apple.comb5953432015-02-13 21:56:01 +00001720 return true;
1721
1722 return false;
1723}
zalan@apple.come031dfb2016-08-25 18:41:22 +00001724
1725static void clearShouldBreakAtLineToAvoidWidowIfNeeded(RenderBlockFlow& blockFlow)
1726{
1727 if (!blockFlow.shouldBreakAtLineToAvoidWidow())
1728 return;
1729 blockFlow.clearShouldBreakAtLineToAvoidWidow();
1730 blockFlow.setDidBreakAtLineToAvoidWidow();
1731}
1732
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001733void RenderBlockFlow::adjustLinePositionForPagination(RootInlineBox* lineBox, LayoutUnit& delta, bool& overflowsFragment, RenderFragmentedFlow* fragmentedFlow)
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001734{
1735 // FIXME: For now we paginate using line overflow. This ensures that lines don't overlap at all when we
1736 // put a strut between them for pagination purposes. However, this really isn't the desired rendering, since
1737 // the line on the top of the next page will appear too far down relative to the same kind of line at the top
1738 // of the first column.
1739 //
1740 // The rendering we would like to see is one where the lineTopWithLeading is at the top of the column, and any line overflow
1741 // simply spills out above the top of the column. This effect would match what happens at the top of the first column.
1742 // We can't achieve this rendering, however, until we stop columns from clipping to the column bounds (thus allowing
1743 // for overflow to occur), and then cache visible overflow for each column rect.
1744 //
1745 // Furthermore, the paint we have to do when a column has overflow has to be special. We need to exclude
1746 // content that paints in a previous column (and content that paints in the following column).
1747 //
1748 // For now we'll at least honor the lineTopWithLeading when paginating if it is above the logical top overflow. This will
1749 // at least make positive leading work in typical cases.
1750 //
1751 // FIXME: Another problem with simply moving lines is that the available line width may change (because of floats).
1752 // Technically if the location we move the line to has a different line width than our old position, then we need to dirty the
1753 // line and all following lines.
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001754 overflowsFragment = false;
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001755 LayoutRect logicalVisualOverflow = lineBox->logicalVisualOverflowRect(lineBox->lineTop(), lineBox->lineBottom());
andersca@apple.com86298632013-11-10 19:32:33 +00001756 LayoutUnit logicalOffset = std::min(lineBox->lineTopWithLeading(), logicalVisualOverflow.y());
1757 LayoutUnit logicalBottom = std::max(lineBox->lineBottomWithLeading(), logicalVisualOverflow.maxY());
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001758 LayoutUnit lineHeight = logicalBottom - logicalOffset;
mmaxfield@apple.comf8e26e72014-10-30 21:39:27 +00001759 updateMinimumPageHeight(logicalOffset, calculateMinimumPageHeight(style(), *lineBox, logicalOffset, logicalBottom));
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001760 logicalOffset += delta;
1761 lineBox->setPaginationStrut(0);
1762 lineBox->setIsFirstAfterPageBreak(false);
1763 LayoutUnit pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001764 bool hasUniformPageLogicalHeight = !fragmentedFlow || fragmentedFlow->fragmentsHaveUniformLogicalHeight();
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001765 // If lineHeight is greater than pageLogicalHeight, but logicalVisualOverflow.height() still fits, we are
1766 // still going to add a strut, so that the visible overflow fits on a single page.
hyatt@apple.comcb5ab702014-11-19 23:40:23 +00001767 if (!pageLogicalHeight || !hasNextPage(logicalOffset)) {
abucur@adobe.comd40287b2013-10-08 17:33:05 +00001768 // FIXME: In case the line aligns with the top of the page (or it's slightly shifted downwards) it will not be marked as the first line in the page.
1769 // From here, the fix is not straightforward because it's not easy to always determine when the current line is the first in the page.
zalan@apple.com3b1271d2020-03-25 01:46:09 +00001770 // With no valid page height, we can't possibly accommodate the widow rules.
1771 clearShouldBreakAtLineToAvoidWidowIfNeeded(*this);
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001772 return;
hyatt@apple.comcb5ab702014-11-19 23:40:23 +00001773 }
1774
1775 if (hasUniformPageLogicalHeight && logicalVisualOverflow.height() > pageLogicalHeight) {
1776 // We are so tall that we are bigger than a page. Before we give up and just leave the line where it is, try drilling into the
1777 // line and computing a new height that excludes anything we consider "blank space". We will discard margins, descent, and even overflow. If we are
1778 // able to fit with the blank space and overflow excluded, we will give the line its own page with the highest non-blank element being aligned with the
1779 // top of the page.
1780 // FIXME: We are still honoring gigantic margins, which does leave open the possibility of blank pages caused by this heuristic. It remains to be seen whether or not
1781 // this will be a real-world issue. For now we don't try to deal with this problem.
1782 logicalOffset = intMaxForLayoutUnit;
1783 logicalBottom = intMinForLayoutUnit;
1784 lineBox->computeReplacedAndTextLineTopAndBottom(logicalOffset, logicalBottom);
1785 lineHeight = logicalBottom - logicalOffset;
zalan@apple.come031dfb2016-08-25 18:41:22 +00001786 if (logicalOffset == intMaxForLayoutUnit || lineHeight > pageLogicalHeight) {
1787 // Give up. We're genuinely too big even after excluding blank space and overflow.
1788 clearShouldBreakAtLineToAvoidWidowIfNeeded(*this);
1789 return;
1790 }
hyatt@apple.comcb5ab702014-11-19 23:40:23 +00001791 pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset);
1792 }
1793
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001794 LayoutUnit remainingLogicalHeight = pageRemainingLogicalHeightForOffset(logicalOffset, ExcludePageBoundary);
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001795 overflowsFragment = (lineHeight > remainingLogicalHeight);
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001796
antti@apple.com702ef7d2019-12-06 14:37:43 +00001797 int lineIndex = complexLineLayout()->lineCountUntil(lineBox);
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001798 if (remainingLogicalHeight < lineHeight || (shouldBreakAtLineToAvoidWidow() && lineBreakToAvoidWidow() == lineIndex)) {
zalan@apple.come031dfb2016-08-25 18:41:22 +00001799 if (lineBreakToAvoidWidow() == lineIndex)
1800 clearShouldBreakAtLineToAvoidWidowIfNeeded(*this);
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001801 // If we have a non-uniform page height, then we have to shift further possibly.
1802 if (!hasUniformPageLogicalHeight && !pushToNextPageWithMinimumLogicalHeight(remainingLogicalHeight, logicalOffset, lineHeight))
1803 return;
1804 if (lineHeight > pageLogicalHeight) {
1805 // Split the top margin in order to avoid splitting the visible part of the line.
andersca@apple.com86298632013-11-10 19:32:33 +00001806 remainingLogicalHeight -= std::min(lineHeight - pageLogicalHeight, std::max<LayoutUnit>(0, logicalVisualOverflow.y() - lineBox->lineTopWithLeading()));
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001807 }
stavila@adobe.come1efa7f2014-05-20 14:34:56 +00001808 LayoutUnit remainingLogicalHeightAtNewOffset = pageRemainingLogicalHeightForOffset(logicalOffset + remainingLogicalHeight, ExcludePageBoundary);
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001809 overflowsFragment = (lineHeight > remainingLogicalHeightAtNewOffset);
andersca@apple.com86298632013-11-10 19:32:33 +00001810 LayoutUnit totalLogicalHeight = lineHeight + std::max<LayoutUnit>(0, logicalOffset);
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001811 LayoutUnit pageLogicalHeightAtNewOffset = hasUniformPageLogicalHeight ? pageLogicalHeight : pageLogicalHeightForOffset(logicalOffset + remainingLogicalHeight);
1812 setPageBreak(logicalOffset, lineHeight - remainingLogicalHeight);
akling@apple.com827be9c2013-10-29 02:58:43 +00001813 if (((lineBox == firstRootBox() && totalLogicalHeight < pageLogicalHeightAtNewOffset) || (!style().hasAutoOrphans() && style().orphans() >= lineIndex))
mmaxfield@apple.com4d7e9a22014-11-18 22:40:29 +00001814 && !isOutOfFlowPositioned() && !isTableCell()) {
1815 auto firstRootBox = this->firstRootBox();
commit-queue@webkit.orge2d17aa2020-01-30 18:07:10 +00001816 if (!firstRootBox) {
1817 setPaginationStrut(remainingLogicalHeight + logicalOffset);
1818 return;
1819 }
mmaxfield@apple.com4d7e9a22014-11-18 22:40:29 +00001820 auto firstRootBoxOverflowRect = firstRootBox->logicalVisualOverflowRect(firstRootBox->lineTop(), firstRootBox->lineBottom());
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00001821 auto firstLineUpperOverhang = std::max(-firstRootBoxOverflowRect.y(), 0_lu);
bfulgham@apple.comb5953432015-02-13 21:56:01 +00001822 if (needsAppleMailPaginationQuirk(*lineBox))
1823 return;
mmaxfield@apple.com4d7e9a22014-11-18 22:40:29 +00001824 setPaginationStrut(remainingLogicalHeight + logicalOffset + firstLineUpperOverhang);
1825 } else {
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001826 delta += remainingLogicalHeight;
1827 lineBox->setPaginationStrut(remainingLogicalHeight);
1828 lineBox->setIsFirstAfterPageBreak(true);
1829 }
commit-queue@webkit.org883b01c2014-01-20 08:58:36 +00001830 } else if (remainingLogicalHeight == pageLogicalHeight) {
1831 // We're at the very top of a page or column.
1832 if (lineBox != firstRootBox())
1833 lineBox->setIsFirstAfterPageBreak(true);
1834 if (lineBox != firstRootBox() || offsetFromLogicalTopOfFirstPage())
1835 setPageBreak(logicalOffset, lineHeight);
1836 }
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001837}
1838
1839void RenderBlockFlow::setBreakAtLineToAvoidWidow(int lineToBreak)
1840{
abucur@adobe.comfc497132013-10-04 08:49:21 +00001841 ASSERT(lineToBreak >= 0);
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001842 ASSERT(!ensureRareBlockFlowData().m_didBreakAtLineToAvoidWidow);
1843 ensureRareBlockFlowData().m_lineBreakToAvoidWidow = lineToBreak;
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001844}
1845
abucur@adobe.comfc497132013-10-04 08:49:21 +00001846void RenderBlockFlow::setDidBreakAtLineToAvoidWidow()
1847{
1848 ASSERT(!shouldBreakAtLineToAvoidWidow());
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001849 if (!hasRareBlockFlowData())
abucur@adobe.comfc497132013-10-04 08:49:21 +00001850 return;
1851
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001852 rareBlockFlowData()->m_didBreakAtLineToAvoidWidow = true;
abucur@adobe.comfc497132013-10-04 08:49:21 +00001853}
1854
1855void RenderBlockFlow::clearDidBreakAtLineToAvoidWidow()
1856{
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001857 if (!hasRareBlockFlowData())
abucur@adobe.comfc497132013-10-04 08:49:21 +00001858 return;
1859
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001860 rareBlockFlowData()->m_didBreakAtLineToAvoidWidow = false;
abucur@adobe.comfc497132013-10-04 08:49:21 +00001861}
1862
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001863void RenderBlockFlow::clearShouldBreakAtLineToAvoidWidow() const
1864{
abucur@adobe.comfc497132013-10-04 08:49:21 +00001865 ASSERT(shouldBreakAtLineToAvoidWidow());
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001866 if (!hasRareBlockFlowData())
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001867 return;
abucur@adobe.comfc497132013-10-04 08:49:21 +00001868
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00001869 rareBlockFlowData()->m_lineBreakToAvoidWidow = -1;
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001870}
1871
zalan@apple.com4a440322017-11-10 00:31:24 +00001872bool RenderBlockFlow::relayoutToAvoidWidows()
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001873{
1874 if (!shouldBreakAtLineToAvoidWidow())
1875 return false;
1876
hyatt@apple.com2ea59882013-09-17 16:41:42 +00001877 setEverHadLayout(true);
1878 layoutBlock(false);
1879 return true;
1880}
1881
weinig@apple.com31324fd2013-10-28 19:22:51 +00001882bool RenderBlockFlow::hasNextPage(LayoutUnit logicalOffset, PageBoundaryRule pageBoundaryRule) const
1883{
zalan@apple.com4de96c52017-11-07 04:09:59 +00001884 ASSERT(view().frameView().layoutContext().layoutState() && view().frameView().layoutContext().layoutState()->isPaginated());
weinig@apple.com31324fd2013-10-28 19:22:51 +00001885
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001886 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
1887 if (!fragmentedFlow)
weinig@apple.com31324fd2013-10-28 19:22:51 +00001888 return true; // Printing and multi-column both make new pages to accommodate content.
1889
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001890 // See if we're in the last fragment.
weinig@apple.com31324fd2013-10-28 19:22:51 +00001891 LayoutUnit pageOffset = offsetFromLogicalTopOfFirstPage() + logicalOffset;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001892 RenderFragmentContainer* fragment = fragmentedFlow->fragmentAtBlockOffset(this, pageOffset, true);
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001893 if (!fragment)
weinig@apple.com31324fd2013-10-28 19:22:51 +00001894 return false;
mihnea@adobe.comc191b0a2014-03-19 12:38:51 +00001895
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001896 if (fragment->isLastFragment())
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001897 return fragment->isRenderFragmentContainerSet() || (pageBoundaryRule == IncludePageBoundary && pageOffset == fragment->logicalTopForFragmentedFlowContent());
stavila@adobe.com6cb976d2013-11-21 06:57:19 +00001898
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001899 RenderFragmentContainer* startFragment = nullptr;
1900 RenderFragmentContainer* endFragment = nullptr;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001901 fragmentedFlow->getFragmentRangeForBox(this, startFragment, endFragment);
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001902 return (endFragment && fragment != endFragment);
weinig@apple.com31324fd2013-10-28 19:22:51 +00001903}
1904
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00001905LayoutUnit RenderBlockFlow::adjustForUnsplittableChild(RenderBox& child, LayoutUnit logicalOffset, LayoutUnit childBeforeMargin, LayoutUnit childAfterMargin)
weinig@apple.com31324fd2013-10-28 19:22:51 +00001906{
hyatt@apple.com531e35d2017-04-13 16:37:00 +00001907 // When flexboxes are embedded inside a block flow, they don't perform any adjustments for unsplittable
1908 // children. We'll treat flexboxes themselves as unsplittable just to get them to paginate properly inside
1909 // a block flow.
1910 bool isUnsplittable = childBoxIsUnsplittableForFragmentation(child);
1911 if (!isUnsplittable && !(child.isFlexibleBox() && !downcast<RenderFlexibleBox>(child).isFlexibleBoxImpl()))
weinig@apple.com31324fd2013-10-28 19:22:51 +00001912 return logicalOffset;
hyatt@apple.com531e35d2017-04-13 16:37:00 +00001913
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001914 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00001915 LayoutUnit childLogicalHeight = logicalHeightForChild(child) + childBeforeMargin + childAfterMargin;
weinig@apple.com31324fd2013-10-28 19:22:51 +00001916 LayoutUnit pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001917 bool hasUniformPageLogicalHeight = !fragmentedFlow || fragmentedFlow->fragmentsHaveUniformLogicalHeight();
hyatt@apple.com531e35d2017-04-13 16:37:00 +00001918 if (isUnsplittable)
1919 updateMinimumPageHeight(logicalOffset, childLogicalHeight);
weinig@apple.com31324fd2013-10-28 19:22:51 +00001920 if (!pageLogicalHeight || (hasUniformPageLogicalHeight && childLogicalHeight > pageLogicalHeight)
1921 || !hasNextPage(logicalOffset))
1922 return logicalOffset;
1923 LayoutUnit remainingLogicalHeight = pageRemainingLogicalHeightForOffset(logicalOffset, ExcludePageBoundary);
1924 if (remainingLogicalHeight < childLogicalHeight) {
1925 if (!hasUniformPageLogicalHeight && !pushToNextPageWithMinimumLogicalHeight(remainingLogicalHeight, logicalOffset, childLogicalHeight))
1926 return logicalOffset;
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00001927 auto result = logicalOffset + remainingLogicalHeight;
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00001928 bool isInitialLetter = child.isFloating() && child.style().styleType() == PseudoId::FirstLetter && child.style().initialLetterDrop() > 0;
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00001929 if (isInitialLetter) {
1930 // Increase our logical height to ensure that lines all get pushed along with the letter.
1931 setLogicalHeight(logicalOffset + remainingLogicalHeight);
1932 }
1933 return result;
weinig@apple.com31324fd2013-10-28 19:22:51 +00001934 }
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00001935
weinig@apple.com31324fd2013-10-28 19:22:51 +00001936 return logicalOffset;
1937}
1938
1939bool RenderBlockFlow::pushToNextPageWithMinimumLogicalHeight(LayoutUnit& adjustment, LayoutUnit logicalOffset, LayoutUnit minimumLogicalHeight) const
1940{
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001941 bool checkFragment = false;
weinig@apple.com31324fd2013-10-28 19:22:51 +00001942 for (LayoutUnit pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset + adjustment); pageLogicalHeight;
1943 pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset + adjustment)) {
1944 if (minimumLogicalHeight <= pageLogicalHeight)
1945 return true;
1946 if (!hasNextPage(logicalOffset + adjustment))
1947 return false;
1948 adjustment += pageLogicalHeight;
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001949 checkFragment = true;
weinig@apple.com31324fd2013-10-28 19:22:51 +00001950 }
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00001951 return !checkFragment;
weinig@apple.com31324fd2013-10-28 19:22:51 +00001952}
1953
1954void RenderBlockFlow::setPageBreak(LayoutUnit offset, LayoutUnit spaceShortage)
1955{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001956 if (RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow())
1957 fragmentedFlow->setPageBreak(this, offsetFromLogicalTopOfFirstPage() + offset, spaceShortage);
weinig@apple.com31324fd2013-10-28 19:22:51 +00001958}
1959
1960void RenderBlockFlow::updateMinimumPageHeight(LayoutUnit offset, LayoutUnit minHeight)
1961{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001962 if (RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow())
1963 fragmentedFlow->updateMinimumPageHeight(this, offsetFromLogicalTopOfFirstPage() + offset, minHeight);
weinig@apple.com31324fd2013-10-28 19:22:51 +00001964}
1965
1966LayoutUnit RenderBlockFlow::nextPageLogicalTop(LayoutUnit logicalOffset, PageBoundaryRule pageBoundaryRule) const
1967{
1968 LayoutUnit pageLogicalHeight = pageLogicalHeightForOffset(logicalOffset);
1969 if (!pageLogicalHeight)
1970 return logicalOffset;
1971
1972 // The logicalOffset is in our coordinate space. We can add in our pushed offset.
1973 LayoutUnit remainingLogicalHeight = pageRemainingLogicalHeightForOffset(logicalOffset);
1974 if (pageBoundaryRule == ExcludePageBoundary)
1975 return logicalOffset + (remainingLogicalHeight ? remainingLogicalHeight : pageLogicalHeight);
1976 return logicalOffset + remainingLogicalHeight;
1977}
1978
1979LayoutUnit RenderBlockFlow::pageLogicalTopForOffset(LayoutUnit offset) const
1980{
hyatt@apple.com6c9d5d32015-02-19 21:42:21 +00001981 // Unsplittable objects clear out the pageLogicalHeight in the layout state as a way of signaling that no
1982 // pagination should occur. Therefore we have to check this first and bail if the value has been set to 0.
zalan@apple.com4de96c52017-11-07 04:09:59 +00001983 auto* layoutState = view().frameView().layoutContext().layoutState();
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00001984 LayoutUnit pageLogicalHeight = layoutState->pageLogicalHeight();
hyatt@apple.com6c9d5d32015-02-19 21:42:21 +00001985 if (!pageLogicalHeight)
1986 return 0;
1987
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00001988 LayoutUnit firstPageLogicalTop = isHorizontalWritingMode() ? layoutState->pageOffset().height() : layoutState->pageOffset().width();
1989 LayoutUnit blockLogicalTop = isHorizontalWritingMode() ? layoutState->layoutOffset().height() : layoutState->layoutOffset().width();
weinig@apple.com31324fd2013-10-28 19:22:51 +00001990
1991 LayoutUnit cumulativeOffset = offset + blockLogicalTop;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001992 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
1993 if (!fragmentedFlow)
weinig@apple.com31324fd2013-10-28 19:22:51 +00001994 return cumulativeOffset - roundToInt(cumulativeOffset - firstPageLogicalTop) % roundToInt(pageLogicalHeight);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00001995 return firstPageLogicalTop + fragmentedFlow->pageLogicalTopForOffset(cumulativeOffset - firstPageLogicalTop);
weinig@apple.com31324fd2013-10-28 19:22:51 +00001996}
1997
1998LayoutUnit RenderBlockFlow::pageLogicalHeightForOffset(LayoutUnit offset) const
1999{
hyatt@apple.com6c9d5d32015-02-19 21:42:21 +00002000 // Unsplittable objects clear out the pageLogicalHeight in the layout state as a way of signaling that no
2001 // pagination should occur. Therefore we have to check this first and bail if the value has been set to 0.
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00002002 LayoutUnit pageLogicalHeight = view().frameView().layoutContext().layoutState()->pageLogicalHeight();
hyatt@apple.com6c9d5d32015-02-19 21:42:21 +00002003 if (!pageLogicalHeight)
2004 return 0;
2005
2006 // Now check for a flow thread.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002007 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
2008 if (!fragmentedFlow)
hyatt@apple.com6c9d5d32015-02-19 21:42:21 +00002009 return pageLogicalHeight;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002010 return fragmentedFlow->pageLogicalHeightForOffset(offset + offsetFromLogicalTopOfFirstPage());
weinig@apple.com31324fd2013-10-28 19:22:51 +00002011}
2012
2013LayoutUnit RenderBlockFlow::pageRemainingLogicalHeightForOffset(LayoutUnit offset, PageBoundaryRule pageBoundaryRule) const
2014{
2015 offset += offsetFromLogicalTopOfFirstPage();
2016
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002017 RenderFragmentedFlow* fragmentedFlow = enclosingFragmentedFlow();
2018 if (!fragmentedFlow) {
zalan@apple.com7fd79ce2017-11-08 15:02:10 +00002019 LayoutUnit pageLogicalHeight = view().frameView().layoutContext().layoutState()->pageLogicalHeight();
weinig@apple.com31324fd2013-10-28 19:22:51 +00002020 LayoutUnit remainingHeight = pageLogicalHeight - intMod(offset, pageLogicalHeight);
2021 if (pageBoundaryRule == IncludePageBoundary) {
2022 // If includeBoundaryPoint is true the line exactly on the top edge of a
2023 // column will act as being part of the previous column.
2024 remainingHeight = intMod(remainingHeight, pageLogicalHeight);
2025 }
2026 return remainingHeight;
2027 }
2028
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002029 return fragmentedFlow->pageRemainingLogicalHeightForOffset(offset, pageBoundaryRule);
weinig@apple.com31324fd2013-10-28 19:22:51 +00002030}
2031
stavila@adobe.comb0d86c42014-04-09 17:07:50 +00002032LayoutUnit RenderBlockFlow::logicalHeightForChildForFragmentation(const RenderBox& child) const
2033{
antti@apple.comeae76122017-09-20 15:37:23 +00002034 return logicalHeightForChild(child);
stavila@adobe.comb0d86c42014-04-09 17:07:50 +00002035}
weinig@apple.com31324fd2013-10-28 19:22:51 +00002036
hyatt@apple.com3cd5c772013-09-27 18:22:50 +00002037void RenderBlockFlow::layoutLineGridBox()
2038{
akling@apple.com827be9c2013-10-29 02:58:43 +00002039 if (style().lineGrid() == RenderStyle::initialLineGrid()) {
hyatt@apple.com3cd5c772013-09-27 18:22:50 +00002040 setLineGridBox(0);
2041 return;
2042 }
2043
2044 setLineGridBox(0);
2045
ysuzuki@apple.com1d8e24d2019-08-19 06:59:40 +00002046 auto lineGridBox = makeUnique<RootInlineBox>(*this);
hyatt@apple.com3cd5c772013-09-27 18:22:50 +00002047 lineGridBox->setHasTextChildren(); // Needed to make the line ascent/descent actually be honored in quirks mode.
2048 lineGridBox->setConstructed();
2049 GlyphOverflowAndFallbackFontsMap textBoxDataMap;
2050 VerticalPositionCache verticalPositionCache;
2051 lineGridBox->alignBoxesInBlockDirection(logicalHeight(), textBoxDataMap, verticalPositionCache);
2052
aestes@apple.com13aae082016-01-02 08:03:08 +00002053 setLineGridBox(WTFMove(lineGridBox));
akling@apple.com1aa97b02013-10-31 21:59:49 +00002054
hyatt@apple.com3cd5c772013-09-27 18:22:50 +00002055 // FIXME: If any of the characteristics of the box change compared to the old one, then we need to do a deep dirtying
2056 // (similar to what happens when the page height changes). Ideally, though, we only do this if someone is actually snapping
2057 // to this grid.
2058}
2059
weinig@apple.com12840dc2013-10-22 23:59:08 +00002060bool RenderBlockFlow::containsFloat(RenderBox& renderer) const
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002061{
commit-queue@webkit.orgc27f5ca2018-11-26 19:29:23 +00002062 return m_floatingObjects && m_floatingObjects->set().contains<FloatingObjectHashTranslator>(renderer);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002063}
2064
2065void RenderBlockFlow::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
2066{
2067 RenderBlock::styleDidChange(diff, oldStyle);
2068
2069 // After our style changed, if we lose our ability to propagate floats into next sibling
2070 // blocks, then we need to find the top most parent containing that overhanging float and
2071 // then mark its descendants with floats for layout and clear all floats from its next
2072 // sibling blocks that exist in our floating objects list. See bug 56299 and 62875.
2073 bool canPropagateFloatIntoSibling = !isFloatingOrOutOfFlowPositioned() && !avoidsFloats();
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002074 if (diff == StyleDifference::Layout && s_canPropagateFloatIntoSibling && !canPropagateFloatIntoSibling && hasOverhangingFloats()) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002075 RenderBlockFlow* parentBlock = this;
2076 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002077
weinig@apple.comc77041e2013-12-14 18:05:45 +00002078 for (auto& ancestor : ancestorsOfType<RenderBlockFlow>(*this)) {
2079 if (ancestor.isRenderView())
akling@apple.comf3028052013-11-04 08:46:06 +00002080 break;
weinig@apple.comc77041e2013-12-14 18:05:45 +00002081 if (ancestor.hasOverhangingFloats()) {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002082 for (auto it = floatingObjectSet.begin(), end = floatingObjectSet.end(); it != end; ++it) {
2083 RenderBox& renderer = (*it)->renderer();
weinig@apple.comc77041e2013-12-14 18:05:45 +00002084 if (ancestor.hasOverhangingFloat(renderer)) {
2085 parentBlock = &ancestor;
akling@apple.comf3028052013-11-04 08:46:06 +00002086 break;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002087 }
2088 }
2089 }
2090 }
2091
2092 parentBlock->markAllDescendantsWithFloatsForLayout();
2093 parentBlock->markSiblingsWithFloatsForLayout();
2094 }
mihnea@adobe.combe79cf12013-10-17 09:02:19 +00002095
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002096 if (diff >= StyleDifference::Repaint) {
zalan@apple.comd4880dc2020-01-05 15:13:08 +00002097 auto shouldInvalidateLineLayoutPath = [&] {
2098 if (selfNeedsLayout() || complexLineLayout())
2099 return true;
2100 // FIXME: This could use a cheaper style-only test instead of SimpleLineLayout::canUseFor.
2101 if (simpleLineLayout() && !SimpleLineLayout::canUseFor(*this))
2102 return true;
2103#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
2104 if (layoutFormattingContextLineLayout() && !LayoutIntegration::LineLayout::canUseFor(*this))
2105 return true;
2106#endif
2107 return false;
2108 };
2109 if (shouldInvalidateLineLayoutPath())
antti@apple.com9e891c82014-05-22 06:12:34 +00002110 invalidateLineLayoutPath();
2111 }
2112
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002113 if (multiColumnFlow())
hyatt@apple.comfdb12812014-06-23 18:56:52 +00002114 updateStylesForColumnChildren();
antti@apple.com7a503402020-01-11 00:13:12 +00002115
2116#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
2117 if (layoutFormattingContextLineLayout())
2118 layoutFormattingContextLineLayout()->updateStyle();
2119#endif
hyatt@apple.comfdb12812014-06-23 18:56:52 +00002120}
2121
2122void RenderBlockFlow::updateStylesForColumnChildren()
2123{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002124 for (auto* child = firstChildBox(); child && (child->isInFlowRenderFragmentedFlow() || child->isRenderMultiColumnSet()); child = child->nextSiblingBox())
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00002125 child->setStyle(RenderStyle::createAnonymousStyleWithDisplay(style(), DisplayType::Block));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002126}
2127
akling@apple.combdae43242013-10-25 12:00:20 +00002128void RenderBlockFlow::styleWillChange(StyleDifference diff, const RenderStyle& newStyle)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002129{
akling@apple.com827be9c2013-10-29 02:58:43 +00002130 const RenderStyle* oldStyle = hasInitializedStyle() ? &style() : nullptr;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002131 s_canPropagateFloatIntoSibling = oldStyle ? !isFloatingOrOutOfFlowPositioned() && !avoidsFloats() : false;
2132
stavila@adobe.comd40a2dc2014-06-23 14:59:48 +00002133 if (oldStyle) {
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002134 auto oldPosition = oldStyle->position();
2135 auto newPosition = newStyle.position();
abucur@adobe.comc0a88a62014-10-16 06:50:30 +00002136
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002137 if (parent() && diff == StyleDifference::Layout && oldPosition != newPosition) {
stavila@adobe.comd40a2dc2014-06-23 14:59:48 +00002138 if (containsFloats() && !isFloating() && !isOutOfFlowPositioned() && newStyle.hasOutOfFlowPosition())
2139 markAllDescendantsWithFloatsForLayout();
stavila@adobe.comd40a2dc2014-06-23 14:59:48 +00002140 }
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002141 }
2142
2143 RenderBlock::styleWillChange(diff, newStyle);
2144}
2145
antti@apple.coma2c7f242013-10-22 22:37:25 +00002146void RenderBlockFlow::deleteLines()
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002147{
antti@apple.com620243d2019-11-27 17:01:11 +00002148 m_lineLayout = WTF::Monostate();
weinig@apple.com611b9292013-10-20 22:57:54 +00002149
antti@apple.coma2c7f242013-10-22 22:37:25 +00002150 RenderBlock::deleteLines();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002151}
2152
zalan@apple.com232b0932016-12-24 18:00:00 +00002153void RenderBlockFlow::addFloatsToNewParent(RenderBlockFlow& toBlockFlow) const
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002154{
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002155 // When a portion of the render tree is being detached, anonymous blocks
2156 // will be combined as their children are deleted. In this process, the
2157 // anonymous block later in the tree is merged into the one preceeding it.
2158 // It can happen that the later block (this) contains floats that the
2159 // previous block (toBlockFlow) did not contain, and thus are not in the
2160 // floating objects list for toBlockFlow. This can result in toBlockFlow
2161 // containing floats that are not in it's floating objects list, but are in
2162 // the floating objects lists of siblings and parents. This can cause
2163 // problems when the float itself is deleted, since the deletion code
2164 // assumes that if a float is not in it's containing block's floating
2165 // objects list, it isn't in any floating objects list. In order to
2166 // preserve this condition (removing it has serious performance
2167 // implications), we need to copy the floating objects from the old block
2168 // (this) to the new block (toBlockFlow). The float's metrics will likely
2169 // all be wrong, but since toBlockFlow is already marked for layout, this
2170 // will get fixed before anything gets displayed.
2171 // See bug https://bugs.webkit.org/show_bug.cgi?id=115566
zalan@apple.com232b0932016-12-24 18:00:00 +00002172 if (!m_floatingObjects)
2173 return;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002174
zalan@apple.com232b0932016-12-24 18:00:00 +00002175 if (!toBlockFlow.m_floatingObjects)
2176 toBlockFlow.createFloatingObjects();
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002177
zalan@apple.com26018542017-03-30 01:25:00 +00002178 for (auto& floatingObject : m_floatingObjects->set()) {
2179 if (toBlockFlow.containsFloat(floatingObject->renderer()))
2180 continue;
zalan@apple.com232b0932016-12-24 18:00:00 +00002181 toBlockFlow.m_floatingObjects->add(floatingObject->cloneForNewParent());
zalan@apple.com26018542017-03-30 01:25:00 +00002182 }
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002183}
2184
2185void RenderBlockFlow::addOverflowFromFloats()
2186{
2187 if (!m_floatingObjects)
2188 return;
2189
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002190 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2191 auto end = floatingObjectSet.end();
2192 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com6816b132015-10-17 19:14:53 +00002193 const auto& floatingObject = *it->get();
2194 if (floatingObject.isDescendant())
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002195 addOverflowFromChild(&floatingObject.renderer(), floatingObject.locationOffsetOfBorderBox());
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002196 }
2197}
2198
2199void RenderBlockFlow::computeOverflow(LayoutUnit oldClientAfterEdge, bool recomputeFloats)
2200{
2201 RenderBlock::computeOverflow(oldClientAfterEdge, recomputeFloats);
2202
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002203 if (!multiColumnFlow() && (recomputeFloats || createsNewFormattingContext() || hasSelfPaintingLayer()))
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002204 addOverflowFromFloats();
2205}
2206
2207void RenderBlockFlow::repaintOverhangingFloats(bool paintAllDescendants)
2208{
2209 // Repaint any overhanging floats (if we know we're the one to paint them).
2210 // Otherwise, bail out.
2211 if (!hasOverhangingFloats())
2212 return;
2213
2214 // FIXME: Avoid disabling LayoutState. At the very least, don't disable it for floats originating
2215 // in this block. Better yet would be to push extra state for the containers of other floats.
zalan@apple.com4de96c52017-11-07 04:09:59 +00002216 LayoutStateDisabler layoutStateDisabler(view().frameView().layoutContext());
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002217 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2218 auto end = floatingObjectSet.end();
2219 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002220 const auto& floatingObject = *it->get();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002221 // Only repaint the object if it is overhanging, is not in its own layer, and
2222 // is our responsibility to paint (m_shouldPaint is set). When paintAllDescendants is true, the latter
2223 // condition is replaced with being a descendant of us.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002224 auto& renderer = floatingObject.renderer();
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002225 if (logicalBottomForFloat(floatingObject) > logicalHeight()
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002226 && !renderer.hasSelfPaintingLayer()
2227 && (floatingObject.shouldPaint() || (paintAllDescendants && renderer.isDescendantOf(this)))) {
2228 renderer.repaint();
2229 renderer.repaintOverhangingFloats(false);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002230 }
2231 }
2232}
2233
hyatt@apple.comc9021b72014-04-25 21:05:59 +00002234void RenderBlockFlow::paintColumnRules(PaintInfo& paintInfo, const LayoutPoint& point)
hyatt@apple.com58b5ecc2014-04-17 23:06:02 +00002235{
hyatt@apple.comc9021b72014-04-25 21:05:59 +00002236 RenderBlock::paintColumnRules(paintInfo, point);
hyatt@apple.com58b5ecc2014-04-17 23:06:02 +00002237
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002238 if (!multiColumnFlow() || paintInfo.context().paintingDisabled())
hyatt@apple.com58b5ecc2014-04-17 23:06:02 +00002239 return;
hyatt@apple.comc9021b72014-04-25 21:05:59 +00002240
hyatt@apple.com58b5ecc2014-04-17 23:06:02 +00002241 // Iterate over our children and paint the column rules as needed.
2242 for (auto& columnSet : childrenOfType<RenderMultiColumnSet>(*this)) {
2243 LayoutPoint childPoint = columnSet.location() + flipForWritingModeForChild(&columnSet, point);
2244 columnSet.paintColumnRules(paintInfo, childPoint);
2245 }
2246}
2247
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002248void RenderBlockFlow::paintFloats(PaintInfo& paintInfo, const LayoutPoint& paintOffset, bool preservePhase)
2249{
2250 if (!m_floatingObjects)
2251 return;
2252
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002253 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2254 auto end = floatingObjectSet.end();
2255 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com6816b132015-10-17 19:14:53 +00002256 const auto& floatingObject = *it->get();
2257 auto& renderer = floatingObject.renderer();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002258 // Only paint the object if our m_shouldPaint flag is set.
zalan@apple.com6816b132015-10-17 19:14:53 +00002259 if (floatingObject.shouldPaint() && !renderer.hasSelfPaintingLayer()) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002260 PaintInfo currentPaintInfo(paintInfo);
achristensen@apple.com9a618a92018-08-06 20:41:05 +00002261 currentPaintInfo.phase = preservePhase ? paintInfo.phase : PaintPhase::BlockBackground;
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002262 LayoutPoint childPoint = flipFloatForWritingModeForChild(floatingObject, paintOffset + floatingObject.translationOffsetToAncestor());
zalan@apple.com6816b132015-10-17 19:14:53 +00002263 renderer.paint(currentPaintInfo, childPoint);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002264 if (!preservePhase) {
achristensen@apple.com9a618a92018-08-06 20:41:05 +00002265 currentPaintInfo.phase = PaintPhase::ChildBlockBackgrounds;
zalan@apple.com6816b132015-10-17 19:14:53 +00002266 renderer.paint(currentPaintInfo, childPoint);
achristensen@apple.com9a618a92018-08-06 20:41:05 +00002267 currentPaintInfo.phase = PaintPhase::Float;
zalan@apple.com6816b132015-10-17 19:14:53 +00002268 renderer.paint(currentPaintInfo, childPoint);
achristensen@apple.com9a618a92018-08-06 20:41:05 +00002269 currentPaintInfo.phase = PaintPhase::Foreground;
zalan@apple.com6816b132015-10-17 19:14:53 +00002270 renderer.paint(currentPaintInfo, childPoint);
achristensen@apple.com9a618a92018-08-06 20:41:05 +00002271 currentPaintInfo.phase = PaintPhase::Outline;
zalan@apple.com6816b132015-10-17 19:14:53 +00002272 renderer.paint(currentPaintInfo, childPoint);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002273 }
2274 }
2275 }
2276}
2277
weinig@apple.com12840dc2013-10-22 23:59:08 +00002278void RenderBlockFlow::clipOutFloatingObjects(RenderBlock& rootBlock, const PaintInfo* paintInfo, const LayoutPoint& rootBlockPhysicalPosition, const LayoutSize& offsetFromRootBlock)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002279{
2280 if (m_floatingObjects) {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002281 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2282 auto end = floatingObjectSet.end();
2283 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com6816b132015-10-17 19:14:53 +00002284 const auto& floatingObject = *it->get();
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002285 LayoutRect floatBox(offsetFromRootBlock.width(), offsetFromRootBlock.height(), floatingObject.renderer().width(), floatingObject.renderer().height());
2286 floatBox.move(floatingObject.locationOffsetOfBorderBox());
weinig@apple.com12840dc2013-10-22 23:59:08 +00002287 rootBlock.flipForWritingMode(floatBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002288 floatBox.move(rootBlockPhysicalPosition.x(), rootBlockPhysicalPosition.y());
mmaxfield@apple.coma93d7ef2015-08-29 06:15:28 +00002289 paintInfo->context().clipOut(snappedIntRect(floatBox));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002290 }
2291 }
2292}
2293
2294void RenderBlockFlow::createFloatingObjects()
2295{
ysuzuki@apple.com1d8e24d2019-08-19 06:59:40 +00002296 m_floatingObjects = makeUnique<FloatingObjects>(*this);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002297}
2298
2299void RenderBlockFlow::removeFloatingObjects()
2300{
2301 if (!m_floatingObjects)
2302 return;
2303
bjonesbe@adobe.com0b2195a2014-04-11 22:46:02 +00002304 markSiblingsWithFloatsForLayout();
2305
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002306 m_floatingObjects->clear();
2307}
2308
weinig@apple.com12840dc2013-10-22 23:59:08 +00002309FloatingObject* RenderBlockFlow::insertFloatingObject(RenderBox& floatBox)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002310{
weinig@apple.com12840dc2013-10-22 23:59:08 +00002311 ASSERT(floatBox.isFloating());
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002312
2313 // Create the list of special objects if we don't aleady have one
2314 if (!m_floatingObjects)
2315 createFloatingObjects();
2316 else {
2317 // Don't insert the floatingObject again if it's already in the list
2318 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
commit-queue@webkit.orgc27f5ca2018-11-26 19:29:23 +00002319 auto it = floatingObjectSet.find<FloatingObjectHashTranslator>(floatBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002320 if (it != floatingObjectSet.end())
2321 return it->get();
2322 }
2323
2324 // Create the special floatingObject entry & append it to the list
2325
weinig@apple.com12840dc2013-10-22 23:59:08 +00002326 std::unique_ptr<FloatingObject> floatingObject = FloatingObject::create(floatBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002327
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00002328 // Our location is irrelevant if we're unsplittable or no pagination is in effect. Just lay out the float.
weinig@apple.com12840dc2013-10-22 23:59:08 +00002329 bool isChildRenderBlock = floatBox.isRenderBlock();
zalan@apple.com4de96c52017-11-07 04:09:59 +00002330 if (isChildRenderBlock && !floatBox.needsLayout() && view().frameView().layoutContext().layoutState()->pageLogicalHeightChanged())
weinig@apple.com12840dc2013-10-22 23:59:08 +00002331 floatBox.setChildNeedsLayout(MarkOnlyThis);
zalan@apple.com4de96c52017-11-07 04:09:59 +00002332
2333 bool needsBlockDirectionLocationSetBeforeLayout = isChildRenderBlock && view().frameView().layoutContext().layoutState()->needsBlockDirectionLocationSetBeforeLayout();
bjonesbe@adobe.com9c29e692014-12-10 00:57:10 +00002334 if (!needsBlockDirectionLocationSetBeforeLayout || isWritingModeRoot()) {
2335 // We are unsplittable if we're a block flow root.
weinig@apple.com12840dc2013-10-22 23:59:08 +00002336 floatBox.layoutIfNeeded();
bjonesbe@adobe.com9c29e692014-12-10 00:57:10 +00002337 floatingObject->setShouldPaint(!floatBox.hasSelfPaintingLayer());
2338 }
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002339 else {
weinig@apple.com12840dc2013-10-22 23:59:08 +00002340 floatBox.updateLogicalWidth();
mmaxfield@apple.com09804f42016-03-23 00:58:34 +00002341 floatBox.computeAndSetBlockDirectionMargins(*this);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002342 }
2343
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002344 setLogicalWidthForFloat(*floatingObject, logicalWidthForChild(floatBox) + marginStartForChild(floatBox) + marginEndForChild(floatBox));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002345
aestes@apple.com13aae082016-01-02 08:03:08 +00002346 return m_floatingObjects->add(WTFMove(floatingObject));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002347}
2348
weinig@apple.com12840dc2013-10-22 23:59:08 +00002349void RenderBlockFlow::removeFloatingObject(RenderBox& floatBox)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002350{
2351 if (m_floatingObjects) {
2352 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
commit-queue@webkit.orgc27f5ca2018-11-26 19:29:23 +00002353 auto it = floatingObjectSet.find<FloatingObjectHashTranslator>(floatBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002354 if (it != floatingObjectSet.end()) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002355 auto& floatingObject = *it->get();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002356 if (childrenInline()) {
bjonesbe@adobe.com1ccd3a12013-10-10 00:35:38 +00002357 LayoutUnit logicalTop = logicalTopForFloat(floatingObject);
2358 LayoutUnit logicalBottom = logicalBottomForFloat(floatingObject);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002359
2360 // Fix for https://bugs.webkit.org/show_bug.cgi?id=54995.
2361 if (logicalBottom < 0 || logicalBottom < logicalTop || logicalTop == LayoutUnit::max())
2362 logicalBottom = LayoutUnit::max();
2363 else {
2364 // Special-case zero- and less-than-zero-height floats: those don't touch
2365 // the line that they're on, but it still needs to be dirtied. This is
2366 // accomplished by pretending they have a height of 1.
andersca@apple.com86298632013-11-10 19:32:33 +00002367 logicalBottom = std::max(logicalBottom, logicalTop + 1);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002368 }
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002369 if (floatingObject.originatingLine()) {
2370 floatingObject.originatingLine()->removeFloat(floatBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002371 if (!selfNeedsLayout()) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002372 ASSERT(&floatingObject.originatingLine()->renderer() == this);
2373 floatingObject.originatingLine()->markDirty();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002374 }
mark.lam@apple.com65724362020-01-06 22:24:50 +00002375#if ASSERT_ENABLED
zalan@apple.comd2acead2017-09-20 22:03:06 +00002376 floatingObject.clearOriginatingLine();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002377#endif
2378 }
2379 markLinesDirtyInBlockRange(0, logicalBottom);
2380 }
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002381 m_floatingObjects->remove(&floatingObject);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002382 }
2383 }
2384}
2385
2386void RenderBlockFlow::removeFloatingObjectsBelow(FloatingObject* lastFloat, int logicalOffset)
2387{
2388 if (!containsFloats())
2389 return;
2390
2391 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2392 FloatingObject* curr = floatingObjectSet.last().get();
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002393 while (curr != lastFloat && (!curr->isPlaced() || logicalTopForFloat(*curr) >= logicalOffset)) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002394 m_floatingObjects->remove(curr);
2395 if (floatingObjectSet.isEmpty())
2396 break;
2397 curr = floatingObjectSet.last().get();
2398 }
2399}
2400
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002401LayoutUnit RenderBlockFlow::logicalLeftOffsetForPositioningFloat(LayoutUnit logicalTop, LayoutUnit fixedOffset, bool applyTextIndent, LayoutUnit* heightRemaining) const
2402{
2403 LayoutUnit offset = fixedOffset;
2404 if (m_floatingObjects && m_floatingObjects->hasLeftObjects())
2405 offset = m_floatingObjects->logicalLeftOffsetForPositioningFloat(fixedOffset, logicalTop, heightRemaining);
2406 return adjustLogicalLeftOffsetForLine(offset, applyTextIndent);
2407}
2408
2409LayoutUnit RenderBlockFlow::logicalRightOffsetForPositioningFloat(LayoutUnit logicalTop, LayoutUnit fixedOffset, bool applyTextIndent, LayoutUnit* heightRemaining) const
2410{
2411 LayoutUnit offset = fixedOffset;
2412 if (m_floatingObjects && m_floatingObjects->hasRightObjects())
2413 offset = m_floatingObjects->logicalRightOffsetForPositioningFloat(fixedOffset, logicalTop, heightRemaining);
2414 return adjustLogicalRightOffsetForLine(offset, applyTextIndent);
2415}
2416
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002417void RenderBlockFlow::computeLogicalLocationForFloat(FloatingObject& floatingObject, LayoutUnit& logicalTopOffset)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002418{
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002419 auto& childBox = floatingObject.renderer();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002420 LayoutUnit logicalLeftOffset = logicalLeftOffsetForContent(logicalTopOffset); // Constant part of left offset.
zoltan@webkit.org7d4f8cc2014-03-26 18:20:15 +00002421 LayoutUnit logicalRightOffset = logicalRightOffsetForContent(logicalTopOffset); // Constant part of right offset.
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002422
andersca@apple.com86298632013-11-10 19:32:33 +00002423 LayoutUnit floatLogicalWidth = std::min(logicalWidthForFloat(floatingObject), logicalRightOffset - logicalLeftOffset); // The width we look for.
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002424
2425 LayoutUnit floatLogicalLeft;
2426
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002427 bool insideFragmentedFlow = enclosingFragmentedFlow();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00002428 bool isInitialLetter = childBox.style().styleType() == PseudoId::FirstLetter && childBox.style().initialLetterDrop() > 0;
hyatt@apple.com87515262014-09-04 21:20:12 +00002429
2430 if (isInitialLetter) {
2431 int letterClearance = lowestInitialLetterLogicalBottom() - logicalTopOffset;
2432 if (letterClearance > 0) {
2433 logicalTopOffset += letterClearance;
2434 setLogicalHeight(logicalHeight() + letterClearance);
2435 }
2436 }
2437
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002438 if (childBox.style().floating() == Float::Left) {
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00002439 LayoutUnit heightRemainingLeft = 1_lu;
2440 LayoutUnit heightRemainingRight = 1_lu;
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002441 floatLogicalLeft = logicalLeftOffsetForPositioningFloat(logicalTopOffset, logicalLeftOffset, false, &heightRemainingLeft);
2442 while (logicalRightOffsetForPositioningFloat(logicalTopOffset, logicalRightOffset, false, &heightRemainingRight) - floatLogicalLeft < floatLogicalWidth) {
andersca@apple.com86298632013-11-10 19:32:33 +00002443 logicalTopOffset += std::min(heightRemainingLeft, heightRemainingRight);
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002444 floatLogicalLeft = logicalLeftOffsetForPositioningFloat(logicalTopOffset, logicalLeftOffset, false, &heightRemainingLeft);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002445 if (insideFragmentedFlow) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002446 // Have to re-evaluate all of our offsets, since they may have changed.
2447 logicalRightOffset = logicalRightOffsetForContent(logicalTopOffset); // Constant part of right offset.
2448 logicalLeftOffset = logicalLeftOffsetForContent(logicalTopOffset); // Constant part of left offset.
andersca@apple.com86298632013-11-10 19:32:33 +00002449 floatLogicalWidth = std::min(logicalWidthForFloat(floatingObject), logicalRightOffset - logicalLeftOffset);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002450 }
2451 }
andersca@apple.com86298632013-11-10 19:32:33 +00002452 floatLogicalLeft = std::max(logicalLeftOffset - borderAndPaddingLogicalLeft(), floatLogicalLeft);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002453 } else {
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00002454 LayoutUnit heightRemainingLeft = 1_lu;
2455 LayoutUnit heightRemainingRight = 1_lu;
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002456 floatLogicalLeft = logicalRightOffsetForPositioningFloat(logicalTopOffset, logicalRightOffset, false, &heightRemainingRight);
2457 while (floatLogicalLeft - logicalLeftOffsetForPositioningFloat(logicalTopOffset, logicalLeftOffset, false, &heightRemainingLeft) < floatLogicalWidth) {
andersca@apple.com86298632013-11-10 19:32:33 +00002458 logicalTopOffset += std::min(heightRemainingLeft, heightRemainingRight);
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002459 floatLogicalLeft = logicalRightOffsetForPositioningFloat(logicalTopOffset, logicalRightOffset, false, &heightRemainingRight);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00002460 if (insideFragmentedFlow) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002461 // Have to re-evaluate all of our offsets, since they may have changed.
2462 logicalRightOffset = logicalRightOffsetForContent(logicalTopOffset); // Constant part of right offset.
2463 logicalLeftOffset = logicalLeftOffsetForContent(logicalTopOffset); // Constant part of left offset.
andersca@apple.com86298632013-11-10 19:32:33 +00002464 floatLogicalWidth = std::min(logicalWidthForFloat(floatingObject), logicalRightOffset - logicalLeftOffset);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002465 }
2466 }
2467 // Use the original width of the float here, since the local variable
2468 // |floatLogicalWidth| was capped to the available line width. See
2469 // fast/block/float/clamped-right-float.html.
bjonesbe@adobe.com1ccd3a12013-10-10 00:35:38 +00002470 floatLogicalLeft -= logicalWidthForFloat(floatingObject);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002471 }
2472
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002473 LayoutUnit childLogicalLeftMargin = style().isLeftToRightDirection() ? marginStartForChild(childBox) : marginEndForChild(childBox);
2474 LayoutUnit childBeforeMargin = marginBeforeForChild(childBox);
hyatt@apple.comc2e15522014-09-03 19:26:38 +00002475
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002476 if (isInitialLetter)
2477 adjustInitialLetterPosition(childBox, logicalTopOffset, childBeforeMargin);
2478
2479 setLogicalLeftForFloat(floatingObject, floatLogicalLeft);
2480 setLogicalLeftForChild(childBox, floatLogicalLeft + childLogicalLeftMargin);
2481
2482 setLogicalTopForFloat(floatingObject, logicalTopOffset);
2483 setLogicalTopForChild(childBox, logicalTopOffset + childBeforeMargin);
2484
2485 setLogicalMarginsForFloat(floatingObject, childLogicalLeftMargin, childBeforeMargin);
2486}
2487
2488void RenderBlockFlow::adjustInitialLetterPosition(RenderBox& childBox, LayoutUnit& logicalTopOffset, LayoutUnit& marginBeforeOffset)
2489{
2490 const RenderStyle& style = firstLineStyle();
2491 const FontMetrics& fontMetrics = style.fontMetrics();
2492 if (!fontMetrics.hasCapHeight())
2493 return;
2494
2495 LayoutUnit heightOfLine = lineHeight(true, isHorizontalWritingMode() ? HorizontalLine : VerticalLine, PositionOfInteriorLineBoxes);
2496 LayoutUnit beforeMarginBorderPadding = childBox.borderAndPaddingBefore() + childBox.marginBefore();
2497
2498 // Make an adjustment to align with the cap height of a theoretical block line.
2499 LayoutUnit adjustment = fontMetrics.ascent() + (heightOfLine - fontMetrics.height()) / 2 - fontMetrics.capHeight() - beforeMarginBorderPadding;
2500 logicalTopOffset += adjustment;
2501
2502 // For sunken and raised caps, we have to make some adjustments. Test if we're sunken or raised (dropHeightDelta will be
2503 // positive for raised and negative for sunken).
2504 int dropHeightDelta = childBox.style().initialLetterHeight() - childBox.style().initialLetterDrop();
2505
2506 // If we're sunken, the float needs to shift down but lines still need to avoid it. In order to do that we increase the float's margin.
2507 if (dropHeightDelta < 0)
2508 marginBeforeOffset += -dropHeightDelta * heightOfLine;
2509
2510 // If we're raised, then we actually have to grow the height of the block, since the lines have to be pushed down as though we're placing
2511 // empty lines beside the first letter.
2512 if (dropHeightDelta > 0)
2513 setLogicalHeight(logicalHeight() + dropHeightDelta * heightOfLine);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002514}
2515
2516bool RenderBlockFlow::positionNewFloats()
2517{
2518 if (!m_floatingObjects)
2519 return false;
2520
2521 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2522 if (floatingObjectSet.isEmpty())
2523 return false;
2524
2525 // If all floats have already been positioned, then we have no work to do.
2526 if (floatingObjectSet.last()->isPlaced())
2527 return false;
2528
2529 // Move backwards through our floating object list until we find a float that has
2530 // already been positioned. Then we'll be able to move forward, positioning all of
2531 // the new floats that need it.
2532 auto it = floatingObjectSet.end();
2533 --it; // Go to last item.
2534 auto begin = floatingObjectSet.begin();
2535 FloatingObject* lastPlacedFloatingObject = 0;
2536 while (it != begin) {
2537 --it;
2538 if ((*it)->isPlaced()) {
2539 lastPlacedFloatingObject = it->get();
2540 ++it;
2541 break;
2542 }
2543 }
2544
2545 LayoutUnit logicalTop = logicalHeight();
2546
2547 // The float cannot start above the top position of the last positioned float.
2548 if (lastPlacedFloatingObject)
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002549 logicalTop = std::max(logicalTopForFloat(*lastPlacedFloatingObject), logicalTop);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002550
2551 auto end = floatingObjectSet.end();
2552 // Now walk through the set of unpositioned floats and place them.
2553 for (; it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002554 auto& floatingObject = *it->get();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002555 // The containing block is responsible for positioning floats, so if we have floats in our
2556 // list that come from somewhere else, do not attempt to position them.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002557 auto& childBox = floatingObject.renderer();
2558 if (childBox.containingBlock() != this)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002559 continue;
2560
weinig@apple.com12840dc2013-10-22 23:59:08 +00002561 LayoutRect oldRect = childBox.frameRect();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002562
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002563 if (childBox.style().clear() == Clear::Left || childBox.style().clear() == Clear::Both)
andersca@apple.com86298632013-11-10 19:32:33 +00002564 logicalTop = std::max(lowestFloatLogicalBottom(FloatingObject::FloatLeft), logicalTop);
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002565 if (childBox.style().clear() == Clear::Right || childBox.style().clear() == Clear::Both)
andersca@apple.com86298632013-11-10 19:32:33 +00002566 logicalTop = std::max(lowestFloatLogicalBottom(FloatingObject::FloatRight), logicalTop);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002567
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002568 computeLogicalLocationForFloat(floatingObject, logicalTop);
2569 LayoutUnit childLogicalTop = logicalTopForChild(childBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002570
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00002571 estimateFragmentRangeForBoxChild(childBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002572
hyatt@apple.comccad3742015-02-04 21:39:00 +00002573 childBox.markForPaginationRelayoutIfNeeded();
2574 childBox.layoutIfNeeded();
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002575
zalan@apple.com4de96c52017-11-07 04:09:59 +00002576 auto* layoutState = view().frameView().layoutContext().layoutState();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002577 bool isPaginated = layoutState->isPaginated();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002578 if (isPaginated) {
2579 // If we are unsplittable and don't fit, then we need to move down.
2580 // We include our margins as part of the unsplittable area.
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002581 LayoutUnit newLogicalTop = adjustForUnsplittableChild(childBox, logicalTop, childLogicalTop - logicalTop, marginAfterForChild(childBox));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002582
2583 // See if we have a pagination strut that is making us move down further.
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002584 // Note that an unsplittable child can't also have a pagination strut, so this
2585 // is exclusive with the case above.
cdumez@apple.come9437792014-10-08 23:33:43 +00002586 RenderBlock* childBlock = is<RenderBlock>(childBox) ? &downcast<RenderBlock>(childBox) : nullptr;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002587 if (childBlock && childBlock->paginationStrut()) {
2588 newLogicalTop += childBlock->paginationStrut();
2589 childBlock->setPaginationStrut(0);
2590 }
2591
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002592 if (newLogicalTop != logicalTop) {
2593 floatingObject.setPaginationStrut(newLogicalTop - logicalTop);
2594 computeLogicalLocationForFloat(floatingObject, newLogicalTop);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002595 if (childBlock)
2596 childBlock->setChildNeedsLayout(MarkOnlyThis);
weinig@apple.com12840dc2013-10-22 23:59:08 +00002597 childBox.layoutIfNeeded();
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002598 logicalTop = newLogicalTop;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002599 }
2600
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00002601 if (updateFragmentRangeForBoxChild(childBox)) {
weinig@apple.com12840dc2013-10-22 23:59:08 +00002602 childBox.setNeedsLayout(MarkOnlyThis);
2603 childBox.layoutIfNeeded();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002604 }
2605 }
2606
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002607 setLogicalHeightForFloat(floatingObject, logicalHeightForChildForFragmentation(childBox) + (logicalTopForChild(childBox) - logicalTop) + marginAfterForChild(childBox));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002608
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002609 m_floatingObjects->addPlacedObject(&floatingObject);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002610
zoltan@webkit.org0faf5722013-11-05 02:34:16 +00002611 if (ShapeOutsideInfo* shapeOutside = childBox.shapeOutsideInfo())
bjonesbe@adobe.com029f74e2014-02-13 03:02:53 +00002612 shapeOutside->setReferenceBoxLogicalSize(logicalSizeForChild(childBox));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002613 // If the child moved, we have to repaint it.
weinig@apple.com12840dc2013-10-22 23:59:08 +00002614 if (childBox.checkForRepaintDuringLayout())
2615 childBox.repaintDuringLayoutIfMoved(oldRect);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002616 }
2617 return true;
2618}
2619
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002620void RenderBlockFlow::clearFloats(Clear clear)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002621{
2622 positionNewFloats();
2623 // set y position
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00002624 LayoutUnit newY;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002625 switch (clear) {
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002626 case Clear::Left:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002627 newY = lowestFloatLogicalBottom(FloatingObject::FloatLeft);
2628 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002629 case Clear::Right:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002630 newY = lowestFloatLogicalBottom(FloatingObject::FloatRight);
2631 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002632 case Clear::Both:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002633 newY = lowestFloatLogicalBottom();
joepeck@webkit.orgaa676ee52014-01-28 04:04:52 +00002634 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002635 case Clear::None:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002636 break;
2637 }
2638 if (height() < newY)
2639 setLogicalHeight(newY);
2640}
2641
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002642LayoutUnit RenderBlockFlow::logicalLeftFloatOffsetForLine(LayoutUnit logicalTop, LayoutUnit fixedOffset, LayoutUnit logicalHeight) const
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002643{
2644 if (m_floatingObjects && m_floatingObjects->hasLeftObjects())
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002645 return m_floatingObjects->logicalLeftOffset(fixedOffset, logicalTop, logicalHeight);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002646
2647 return fixedOffset;
2648}
2649
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002650LayoutUnit RenderBlockFlow::logicalRightFloatOffsetForLine(LayoutUnit logicalTop, LayoutUnit fixedOffset, LayoutUnit logicalHeight) const
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002651{
2652 if (m_floatingObjects && m_floatingObjects->hasRightObjects())
bjonesbe@adobe.com98b899b2013-11-07 18:11:43 +00002653 return m_floatingObjects->logicalRightOffset(fixedOffset, logicalTop, logicalHeight);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002654
2655 return fixedOffset;
2656}
2657
bjonesbe@adobe.comedea3422013-11-08 22:01:33 +00002658LayoutUnit RenderBlockFlow::nextFloatLogicalBottomBelow(LayoutUnit logicalHeight) const
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002659{
2660 if (!m_floatingObjects)
2661 return logicalHeight;
2662
bjonesbe@adobe.comedea3422013-11-08 22:01:33 +00002663 return m_floatingObjects->findNextFloatLogicalBottomBelow(logicalHeight);
2664}
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002665
bjonesbe@adobe.comedea3422013-11-08 22:01:33 +00002666LayoutUnit RenderBlockFlow::nextFloatLogicalBottomBelowForBlock(LayoutUnit logicalHeight) const
2667{
2668 if (!m_floatingObjects)
2669 return logicalHeight;
2670
2671 return m_floatingObjects->findNextFloatLogicalBottomBelowForBlock(logicalHeight);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002672}
2673
2674LayoutUnit RenderBlockFlow::lowestFloatLogicalBottom(FloatingObject::Type floatType) const
2675{
2676 if (!m_floatingObjects)
2677 return 0;
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00002678 LayoutUnit lowestFloatBottom;
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002679 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2680 auto end = floatingObjectSet.end();
2681 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002682 const auto& floatingObject = *it->get();
2683 if (floatingObject.isPlaced() && floatingObject.type() & floatType)
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002684 lowestFloatBottom = std::max(lowestFloatBottom, logicalBottomForFloat(floatingObject));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002685 }
2686 return lowestFloatBottom;
2687}
2688
hyatt@apple.com87515262014-09-04 21:20:12 +00002689LayoutUnit RenderBlockFlow::lowestInitialLetterLogicalBottom() const
2690{
2691 if (!m_floatingObjects)
2692 return 0;
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00002693 LayoutUnit lowestFloatBottom;
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002694 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2695 auto end = floatingObjectSet.end();
2696 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002697 const auto& floatingObject = *it->get();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00002698 if (floatingObject.isPlaced() && floatingObject.renderer().style().styleType() == PseudoId::FirstLetter && floatingObject.renderer().style().initialLetterDrop() > 0)
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002699 lowestFloatBottom = std::max(lowestFloatBottom, logicalBottomForFloat(floatingObject));
hyatt@apple.com87515262014-09-04 21:20:12 +00002700 }
2701 return lowestFloatBottom;
2702}
2703
weinig@apple.com12840dc2013-10-22 23:59:08 +00002704LayoutUnit RenderBlockFlow::addOverhangingFloats(RenderBlockFlow& child, bool makeChildPaintOtherFloats)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002705{
2706 // Prevent floats from being added to the canvas by the root element, e.g., <html>.
jfernandez@igalia.com136f1702014-12-08 19:13:16 +00002707 if (!child.containsFloats() || child.createsNewFormattingContext())
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002708 return 0;
2709
weinig@apple.com12840dc2013-10-22 23:59:08 +00002710 LayoutUnit childLogicalTop = child.logicalTop();
2711 LayoutUnit childLogicalLeft = child.logicalLeft();
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00002712 LayoutUnit lowestFloatLogicalBottom;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002713
2714 // Floats that will remain the child's responsibility to paint should factor into its
2715 // overflow.
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002716 auto childEnd = child.m_floatingObjects->set().end();
2717 for (auto childIt = child.m_floatingObjects->set().begin(); childIt != childEnd; ++childIt) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002718 auto& floatingObject = *childIt->get();
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002719 LayoutUnit floatLogicalBottom = std::min(logicalBottomForFloat(floatingObject), LayoutUnit::max() - childLogicalTop);
bjonesbe@adobe.com1ccd3a12013-10-10 00:35:38 +00002720 LayoutUnit logicalBottom = childLogicalTop + floatLogicalBottom;
andersca@apple.com86298632013-11-10 19:32:33 +00002721 lowestFloatLogicalBottom = std::max(lowestFloatLogicalBottom, logicalBottom);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002722
2723 if (logicalBottom > logicalHeight()) {
2724 // If the object is not in the list, we add it now.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002725 if (!containsFloat(floatingObject.renderer())) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002726 LayoutSize offset = isHorizontalWritingMode() ? LayoutSize(-childLogicalLeft, -childLogicalTop) : LayoutSize(-childLogicalTop, -childLogicalLeft);
2727 bool shouldPaint = false;
2728
2729 // The nearest enclosing layer always paints the float (so that zindex and stacking
2730 // behaves properly). We always want to propagate the desire to paint the float as
2731 // far out as we can, to the outermost block that overlaps the float, stopping only
2732 // if we hit a self-painting layer boundary.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002733 if (floatingObject.renderer().enclosingFloatPaintingLayer() == enclosingFloatPaintingLayer()) {
2734 floatingObject.setShouldPaint(false);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002735 shouldPaint = true;
2736 }
2737 // We create the floating object list lazily.
2738 if (!m_floatingObjects)
2739 createFloatingObjects();
2740
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002741 m_floatingObjects->add(floatingObject.copyToNewContainer(offset, shouldPaint, true));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002742 }
2743 } else {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002744 const auto& renderer = floatingObject.renderer();
2745 if (makeChildPaintOtherFloats && !floatingObject.shouldPaint() && !renderer.hasSelfPaintingLayer()
2746 && renderer.isDescendantOf(&child) && renderer.enclosingFloatPaintingLayer() == child.enclosingFloatPaintingLayer()) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002747 // The float is not overhanging from this block, so if it is a descendant of the child, the child should
2748 // paint it (the other case is that it is intruding into the child), unless it has its own layer or enclosing
2749 // layer.
2750 // If makeChildPaintOtherFloats is false, it means that the child must already know about all the floats
2751 // it should paint.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002752 floatingObject.setShouldPaint(true);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002753 }
2754
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00002755 // Since the float doesn't overhang, it didn't get put into our list. We need to add its overflow in to the child now.
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002756 if (floatingObject.isDescendant())
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002757 child.addOverflowFromChild(&renderer, floatingObject.locationOffsetOfBorderBox());
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002758 }
2759 }
2760 return lowestFloatLogicalBottom;
2761}
2762
weinig@apple.com12840dc2013-10-22 23:59:08 +00002763bool RenderBlockFlow::hasOverhangingFloat(RenderBox& renderer)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002764{
hyatt@apple.com73715ca2014-05-06 21:35:52 +00002765 if (!m_floatingObjects || !parent())
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002766 return false;
2767
2768 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
commit-queue@webkit.orgc27f5ca2018-11-26 19:29:23 +00002769 const auto it = floatingObjectSet.find<FloatingObjectHashTranslator>(renderer);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002770 if (it == floatingObjectSet.end())
2771 return false;
2772
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002773 return logicalBottomForFloat(*it->get()) > logicalHeight();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002774}
2775
hyatt@apple.com21c60802015-04-01 18:10:32 +00002776void RenderBlockFlow::addIntrudingFloats(RenderBlockFlow* prev, RenderBlockFlow* container, LayoutUnit logicalLeftOffset, LayoutUnit logicalTopOffset)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002777{
2778 ASSERT(!avoidsFloats());
2779
jfernandez@igalia.com70658682014-12-15 21:07:30 +00002780 // If we create our own block formatting context then our contents don't interact with floats outside it, even those from our parent.
2781 if (createsNewFormattingContext())
2782 return;
2783
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002784 // If the parent or previous sibling doesn't have any floats to add, don't bother.
2785 if (!prev->m_floatingObjects)
2786 return;
2787
2788 logicalLeftOffset += marginLogicalLeft();
2789
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002790 const FloatingObjectSet& prevSet = prev->m_floatingObjects->set();
2791 auto prevEnd = prevSet.end();
2792 for (auto prevIt = prevSet.begin(); prevIt != prevEnd; ++prevIt) {
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002793 auto& floatingObject = *prevIt->get();
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002794 if (logicalBottomForFloat(floatingObject) > logicalTopOffset) {
commit-queue@webkit.orgc27f5ca2018-11-26 19:29:23 +00002795 if (!m_floatingObjects || !m_floatingObjects->set().contains(&floatingObject)) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002796 // We create the floating object list lazily.
2797 if (!m_floatingObjects)
2798 createFloatingObjects();
2799
2800 // Applying the child's margin makes no sense in the case where the child was passed in.
2801 // since this margin was added already through the modification of the |logicalLeftOffset| variable
2802 // above. |logicalLeftOffset| will equal the margin in this case, so it's already been taken
2803 // into account. Only apply this code if prev is the parent, since otherwise the left margin
2804 // will get applied twice.
2805 LayoutSize offset = isHorizontalWritingMode()
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00002806 ? LayoutSize(logicalLeftOffset - (prev != container ? prev->marginLeft() : 0_lu), logicalTopOffset)
2807 : LayoutSize(logicalTopOffset, logicalLeftOffset - (prev != container ? prev->marginTop() : 0_lu));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002808
zalan@apple.com84ccfa12015-10-17 03:36:56 +00002809 m_floatingObjects->add(floatingObject.copyToNewContainer(offset));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002810 }
2811 }
2812 }
2813}
2814
2815void RenderBlockFlow::markAllDescendantsWithFloatsForLayout(RenderBox* floatToRemove, bool inLayout)
2816{
2817 if (!everHadLayout() && !containsFloats())
2818 return;
2819
2820 MarkingBehavior markParents = inLayout ? MarkOnlyThis : MarkContainingBlockChain;
2821 setChildNeedsLayout(markParents);
2822
2823 if (floatToRemove)
weinig@apple.com12840dc2013-10-22 23:59:08 +00002824 removeFloatingObject(*floatToRemove);
hyatt@apple.com11a5e5c2016-05-19 21:51:31 +00002825 else if (childrenInline())
hyatt@apple.com00e93142016-05-18 18:59:40 +00002826 return;
2827
zalan@apple.com5d7ffdf2014-10-29 21:13:12 +00002828 // Iterate over our block children and mark them as needed.
akling@apple.com525dae62014-01-03 20:22:09 +00002829 for (auto& block : childrenOfType<RenderBlock>(*this)) {
2830 if (!floatToRemove && block.isFloatingOrOutOfFlowPositioned())
2831 continue;
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00002832 if (!is<RenderBlockFlow>(block)) {
akling@apple.com525dae62014-01-03 20:22:09 +00002833 if (block.shrinkToAvoidFloats() && block.everHadLayout())
2834 block.setChildNeedsLayout(markParents);
2835 continue;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002836 }
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00002837 auto& blockFlow = downcast<RenderBlockFlow>(block);
akling@apple.com525dae62014-01-03 20:22:09 +00002838 if ((floatToRemove ? blockFlow.containsFloat(*floatToRemove) : blockFlow.containsFloats()) || blockFlow.shrinkToAvoidFloats())
2839 blockFlow.markAllDescendantsWithFloatsForLayout(floatToRemove, inLayout);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002840 }
2841}
2842
2843void RenderBlockFlow::markSiblingsWithFloatsForLayout(RenderBox* floatToRemove)
2844{
2845 if (!m_floatingObjects)
2846 return;
2847
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002848 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2849 auto end = floatingObjectSet.end();
2850
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002851 for (RenderObject* next = nextSibling(); next; next = next->nextSibling()) {
zalan@apple.comc2472ea2015-05-26 22:59:40 +00002852 if (!is<RenderBlockFlow>(*next) || next->isFloatingOrOutOfFlowPositioned())
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002853 continue;
2854
cdumez@apple.come9437792014-10-08 23:33:43 +00002855 RenderBlockFlow& nextBlock = downcast<RenderBlockFlow>(*next);
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00002856 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
2857 RenderBox& floatingBox = (*it)->renderer();
weinig@apple.com12840dc2013-10-22 23:59:08 +00002858 if (floatToRemove && &floatingBox != floatToRemove)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002859 continue;
cdumez@apple.come9437792014-10-08 23:33:43 +00002860 if (nextBlock.containsFloat(floatingBox))
2861 nextBlock.markAllDescendantsWithFloatsForLayout(&floatingBox);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002862 }
2863 }
2864}
2865
zalan@apple.com6816b132015-10-17 19:14:53 +00002866LayoutPoint RenderBlockFlow::flipFloatForWritingModeForChild(const FloatingObject& child, const LayoutPoint& point) const
weinig@apple.com31324fd2013-10-28 19:22:51 +00002867{
akling@apple.com827be9c2013-10-29 02:58:43 +00002868 if (!style().isFlippedBlocksWritingMode())
weinig@apple.com31324fd2013-10-28 19:22:51 +00002869 return point;
2870
2871 // This is similar to RenderBox::flipForWritingModeForChild. We have to subtract out our left/top offsets twice, since
2872 // it's going to get added back in. We hide this complication here so that the calling code looks normal for the unflipped
2873 // case.
2874 if (isHorizontalWritingMode())
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002875 return LayoutPoint(point.x(), point.y() + height() - child.renderer().height() - 2 * child.locationOffsetOfBorderBox().height());
2876 return LayoutPoint(point.x() + width() - child.renderer().width() - 2 * child.locationOffsetOfBorderBox().width(), point.y());
weinig@apple.com31324fd2013-10-28 19:22:51 +00002877}
2878
weinig@apple.com12840dc2013-10-22 23:59:08 +00002879LayoutUnit RenderBlockFlow::getClearDelta(RenderBox& child, LayoutUnit logicalTop)
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002880{
2881 // There is no need to compute clearance if we have no floats.
2882 if (!containsFloats())
2883 return 0;
2884
2885 // At least one float is present. We need to perform the clearance computation.
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002886 bool clearSet = child.style().clear() != Clear::None;
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00002887 LayoutUnit logicalBottom;
akling@apple.com827be9c2013-10-29 02:58:43 +00002888 switch (child.style().clear()) {
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002889 case Clear::None:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002890 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002891 case Clear::Left:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002892 logicalBottom = lowestFloatLogicalBottom(FloatingObject::FloatLeft);
2893 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002894 case Clear::Right:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002895 logicalBottom = lowestFloatLogicalBottom(FloatingObject::FloatRight);
2896 break;
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00002897 case Clear::Both:
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002898 logicalBottom = lowestFloatLogicalBottom();
2899 break;
2900 }
2901
2902 // 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).
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00002903 LayoutUnit result = clearSet ? std::max<LayoutUnit>(0, logicalBottom - logicalTop) : 0_lu;
weinig@apple.com12840dc2013-10-22 23:59:08 +00002904 if (!result && child.avoidsFloats()) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002905 LayoutUnit newLogicalTop = logicalTop;
2906 while (true) {
zalan@apple.com64761fe2016-03-02 21:42:22 +00002907 LayoutUnit availableLogicalWidthAtNewLogicalTopOffset = availableLogicalWidthForLine(newLogicalTop, DoNotIndentText, logicalHeightForChild(child));
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002908 if (availableLogicalWidthAtNewLogicalTopOffset == availableLogicalWidthForContent(newLogicalTop))
2909 return newLogicalTop - logicalTop;
2910
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00002911 RenderFragmentContainer* fragment = fragmentAtBlockOffset(logicalTopForChild(child));
2912 LayoutRect borderBox = child.borderBoxRectInFragment(fragment, DoNotCacheRenderBoxFragmentInfo);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002913 LayoutUnit childLogicalWidthAtOldLogicalTopOffset = isHorizontalWritingMode() ? borderBox.width() : borderBox.height();
2914
2915 // FIXME: None of this is right for perpendicular writing-mode children.
weinig@apple.com12840dc2013-10-22 23:59:08 +00002916 LayoutUnit childOldLogicalWidth = child.logicalWidth();
2917 LayoutUnit childOldMarginLeft = child.marginLeft();
2918 LayoutUnit childOldMarginRight = child.marginRight();
2919 LayoutUnit childOldLogicalTop = child.logicalTop();
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002920
weinig@apple.com12840dc2013-10-22 23:59:08 +00002921 child.setLogicalTop(newLogicalTop);
2922 child.updateLogicalWidth();
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00002923 fragment = fragmentAtBlockOffset(logicalTopForChild(child));
2924 borderBox = child.borderBoxRectInFragment(fragment, DoNotCacheRenderBoxFragmentInfo);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002925 LayoutUnit childLogicalWidthAtNewLogicalTopOffset = isHorizontalWritingMode() ? borderBox.width() : borderBox.height();
2926
weinig@apple.com12840dc2013-10-22 23:59:08 +00002927 child.setLogicalTop(childOldLogicalTop);
2928 child.setLogicalWidth(childOldLogicalWidth);
2929 child.setMarginLeft(childOldMarginLeft);
2930 child.setMarginRight(childOldMarginRight);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002931
2932 if (childLogicalWidthAtNewLogicalTopOffset <= availableLogicalWidthAtNewLogicalTopOffset) {
2933 // Even though we may not be moving, if the logical width did shrink because of the presence of new floats, then
2934 // we need to force a relayout as though we shifted. This happens because of the dynamic addition of overhanging floats
2935 // from previous siblings when negative margins exist on a child (see the addOverhangingFloats call at the end of collapseMargins).
2936 if (childLogicalWidthAtOldLogicalTopOffset != childLogicalWidthAtNewLogicalTopOffset)
weinig@apple.com12840dc2013-10-22 23:59:08 +00002937 child.setChildNeedsLayout(MarkOnlyThis);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002938 return newLogicalTop - logicalTop;
2939 }
2940
bjonesbe@adobe.comedea3422013-11-08 22:01:33 +00002941 newLogicalTop = nextFloatLogicalBottomBelowForBlock(newLogicalTop);
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002942 ASSERT(newLogicalTop >= logicalTop);
2943 if (newLogicalTop < logicalTop)
2944 break;
2945 }
2946 ASSERT_NOT_REACHED();
2947 }
2948 return result;
2949}
2950
2951bool RenderBlockFlow::hitTestFloats(const HitTestRequest& request, HitTestResult& result, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset)
2952{
2953 if (!m_floatingObjects)
2954 return false;
2955
2956 LayoutPoint adjustedLocation = accumulatedOffset;
cdumez@apple.com3abcc792014-10-20 03:42:03 +00002957 if (is<RenderView>(*this))
2958 adjustedLocation += toLayoutSize(downcast<RenderView>(*this).frameView().scrollPosition());
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002959
2960 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
2961 auto begin = floatingObjectSet.begin();
2962 for (auto it = floatingObjectSet.end(); it != begin;) {
2963 --it;
zalan@apple.com6816b132015-10-17 19:14:53 +00002964 const auto& floatingObject = *it->get();
2965 auto& renderer = floatingObject.renderer();
2966 if (floatingObject.shouldPaint() && !renderer.hasSelfPaintingLayer()) {
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00002967 LayoutPoint childPoint = flipFloatForWritingModeForChild(floatingObject, adjustedLocation + floatingObject.translationOffsetToAncestor());
zalan@apple.com6816b132015-10-17 19:14:53 +00002968 if (renderer.hitTest(request, result, locationInContainer, childPoint)) {
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00002969 updateHitTestResult(result, locationInContainer.point() - toLayoutSize(childPoint));
2970 return true;
2971 }
2972 }
2973 }
2974
2975 return false;
2976}
2977
weinig@apple.com611b9292013-10-20 22:57:54 +00002978bool RenderBlockFlow::hitTestInlineChildren(const HitTestRequest& request, HitTestResult& result, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset, HitTestAction hitTestAction)
2979{
2980 ASSERT(childrenInline());
antti@apple.com940f5872013-10-24 20:31:11 +00002981
antti@apple.com04bca452019-12-09 15:28:48 +00002982 if (simpleLineLayout())
2983 return SimpleLineLayout::hitTestFlow(*this, *simpleLineLayout(), request, result, locationInContainer, accumulatedOffset, hitTestAction);
2984
2985#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
2986 if (layoutFormattingContextLineLayout())
2987 return layoutFormattingContextLineLayout()->hitTest(request, result, locationInContainer, accumulatedOffset, hitTestAction);
2988#endif
antti@apple.com940f5872013-10-24 20:31:11 +00002989
antti@apple.com8aa23022019-11-25 23:18:05 +00002990 return complexLineLayout() && complexLineLayout()->lineBoxes().hitTest(this, request, result, locationInContainer, accumulatedOffset, hitTestAction);
antti@apple.com50b36fde2019-08-11 11:02:01 +00002991}
2992
2993void RenderBlockFlow::addOverflowFromInlineChildren()
2994{
2995 if (auto simpleLineLayout = this->simpleLineLayout()) {
2996 ASSERT(!hasOverflowClip());
2997 SimpleLineLayout::collectFlowOverflow(*this, *simpleLineLayout);
2998 return;
2999 }
3000
antti@apple.com4789a082019-12-08 15:30:50 +00003001#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3002 if (layoutFormattingContextLineLayout()) {
3003 layoutFormattingContextLineLayout()->collectOverflow(*this);
3004 return;
3005 }
3006#endif
3007
antti@apple.comebf3d5a2019-11-26 18:28:13 +00003008 if (complexLineLayout())
3009 complexLineLayout()->addOverflowFromInlineChildren();
weinig@apple.com611b9292013-10-20 22:57:54 +00003010}
3011
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00003012void RenderBlockFlow::adjustForBorderFit(LayoutUnit x, LayoutUnit& left, LayoutUnit& right) const
3013{
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003014 if (style().visibility() != Visibility::Visible)
weinig@apple.com611b9292013-10-20 22:57:54 +00003015 return;
3016
3017 // We don't deal with relative positioning. Our assumption is that you shrink to fit the lines without accounting
3018 // for either overflow or translations via relative positioning.
3019 if (childrenInline()) {
antti@apple.com940f5872013-10-24 20:31:11 +00003020 const_cast<RenderBlockFlow&>(*this).ensureLineBoxes();
3021
cdumez@apple.comc1d54fa2015-10-13 19:15:55 +00003022 for (auto* box = firstRootBox(); box; box = box->nextRootBox()) {
weinig@apple.com611b9292013-10-20 22:57:54 +00003023 if (box->firstChild())
zalan@apple.com390064f2014-02-26 06:23:03 +00003024 left = std::min(left, x + LayoutUnit(box->firstChild()->x()));
weinig@apple.com611b9292013-10-20 22:57:54 +00003025 if (box->lastChild())
zalan@apple.com390064f2014-02-26 06:23:03 +00003026 right = std::max(right, x + LayoutUnit(ceilf(box->lastChild()->logicalRight())));
weinig@apple.com611b9292013-10-20 22:57:54 +00003027 }
3028 } else {
3029 for (RenderBox* obj = firstChildBox(); obj; obj = obj->nextSiblingBox()) {
3030 if (!obj->isFloatingOrOutOfFlowPositioned()) {
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00003031 if (is<RenderBlockFlow>(*obj) && !obj->hasOverflowClip())
3032 downcast<RenderBlockFlow>(*obj).adjustForBorderFit(x + obj->x(), left, right);
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003033 else if (obj->style().visibility() == Visibility::Visible) {
weinig@apple.com611b9292013-10-20 22:57:54 +00003034 // We are a replaced element or some kind of non-block-flow object.
andersca@apple.com86298632013-11-10 19:32:33 +00003035 left = std::min(left, x + obj->x());
3036 right = std::max(right, x + obj->x() + obj->width());
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00003037 }
3038 }
3039 }
3040 }
weinig@apple.com611b9292013-10-20 22:57:54 +00003041
3042 if (m_floatingObjects) {
commit-queue@webkit.org6ad70c22015-06-12 18:43:01 +00003043 const FloatingObjectSet& floatingObjectSet = m_floatingObjects->set();
3044 auto end = floatingObjectSet.end();
3045 for (auto it = floatingObjectSet.begin(); it != end; ++it) {
zalan@apple.com6816b132015-10-17 19:14:53 +00003046 const auto& floatingObject = *it->get();
weinig@apple.com611b9292013-10-20 22:57:54 +00003047 // Only examine the object if our m_shouldPaint flag is set.
zalan@apple.com6816b132015-10-17 19:14:53 +00003048 if (floatingObject.shouldPaint()) {
hyatt@apple.comb618b2a2017-03-17 18:54:47 +00003049 LayoutUnit floatLeft = floatingObject.translationOffsetToAncestor().width();
zalan@apple.com6816b132015-10-17 19:14:53 +00003050 LayoutUnit floatRight = floatLeft + floatingObject.renderer().width();
andersca@apple.com86298632013-11-10 19:32:33 +00003051 left = std::min(left, floatLeft);
3052 right = std::max(right, floatRight);
weinig@apple.com611b9292013-10-20 22:57:54 +00003053 }
3054 }
3055 }
3056}
3057
3058void RenderBlockFlow::fitBorderToLinesIfNeeded()
3059{
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003060 if (style().borderFit() == BorderFit::Border || hasOverrideContentLogicalWidth())
weinig@apple.com611b9292013-10-20 22:57:54 +00003061 return;
3062
3063 // Walk any normal flow lines to snugly fit.
3064 LayoutUnit left = LayoutUnit::max();
3065 LayoutUnit right = LayoutUnit::min();
3066 LayoutUnit oldWidth = contentWidth();
3067 adjustForBorderFit(0, left, right);
3068
3069 // Clamp to our existing edges. We can never grow. We only shrink.
3070 LayoutUnit leftEdge = borderLeft() + paddingLeft();
3071 LayoutUnit rightEdge = leftEdge + oldWidth;
andersca@apple.com86298632013-11-10 19:32:33 +00003072 left = std::min(rightEdge, std::max(leftEdge, left));
3073 right = std::max(leftEdge, std::min(rightEdge, right));
weinig@apple.com611b9292013-10-20 22:57:54 +00003074
3075 LayoutUnit newContentWidth = right - left;
3076 if (newContentWidth == oldWidth)
3077 return;
3078
rego@igalia.com8bac95d2018-05-14 15:20:47 +00003079 setOverrideContentLogicalWidth(newContentWidth);
weinig@apple.com611b9292013-10-20 22:57:54 +00003080 layoutBlock(false);
rego@igalia.com8bac95d2018-05-14 15:20:47 +00003081 clearOverrideContentLogicalWidth();
weinig@apple.com611b9292013-10-20 22:57:54 +00003082}
3083
3084void RenderBlockFlow::markLinesDirtyInBlockRange(LayoutUnit logicalTop, LayoutUnit logicalBottom, RootInlineBox* highest)
3085{
3086 if (logicalTop >= logicalBottom)
3087 return;
3088
antti@apple.comd685c412019-12-07 13:56:51 +00003089 // Floats currently affect the choice of layout path.
antti@apple.com8aa23022019-11-25 23:18:05 +00003090 if (simpleLineLayout()) {
antti@apple.combe9d3e12014-05-11 09:42:47 +00003091 invalidateLineLayoutPath();
3092 return;
3093 }
3094
antti@apple.comd685c412019-12-07 13:56:51 +00003095#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3096 if (layoutFormattingContextLineLayout()) {
3097 invalidateLineLayoutPath();
3098 return;
3099 }
3100#endif
3101
weinig@apple.com611b9292013-10-20 22:57:54 +00003102 RootInlineBox* lowestDirtyLine = lastRootBox();
3103 RootInlineBox* afterLowest = lowestDirtyLine;
3104 while (lowestDirtyLine && lowestDirtyLine->lineBottomWithLeading() >= logicalBottom && logicalBottom < LayoutUnit::max()) {
3105 afterLowest = lowestDirtyLine;
3106 lowestDirtyLine = lowestDirtyLine->prevRootBox();
3107 }
3108
3109 while (afterLowest && afterLowest != highest && (afterLowest->lineBottomWithLeading() >= logicalTop || afterLowest->lineBottomWithLeading() < 0)) {
3110 afterLowest->markDirty();
3111 afterLowest = afterLowest->prevRootBox();
3112 }
3113}
3114
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003115Optional<int> RenderBlockFlow::firstLineBaseline() const
weinig@apple.com611b9292013-10-20 22:57:54 +00003116{
jfernandez@igalia.comf119ec12018-11-23 11:04:23 +00003117 if (isWritingModeRoot() && !isRubyRun() && !isGridItem())
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003118 return WTF::nullopt;
weinig@apple.com611b9292013-10-20 22:57:54 +00003119
3120 if (!childrenInline())
antti@apple.com0e632aa2013-10-22 21:03:38 +00003121 return RenderBlock::firstLineBaseline();
weinig@apple.com611b9292013-10-20 22:57:54 +00003122
antti@apple.com940f5872013-10-24 20:31:11 +00003123 if (!hasLines())
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003124 return WTF::nullopt;
weinig@apple.com611b9292013-10-20 22:57:54 +00003125
antti@apple.comd685c412019-12-07 13:56:51 +00003126#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3127 if (layoutFormattingContextLineLayout())
3128 return floorToInt(layoutFormattingContextLineLayout()->firstLineBaseline());
3129#endif
3130
3131 if (simpleLineLayout())
3132 return { SimpleLineLayout::computeFlowFirstLineBaseline(*this, *simpleLineLayout()) };
antti@apple.com940f5872013-10-24 20:31:11 +00003133
akling@apple.comee3c8df2013-11-06 08:09:44 +00003134 ASSERT(firstRootBox());
jfernandez@igalia.combe5a04c2018-11-22 08:45:48 +00003135 if (style().isFlippedLinesWritingMode())
3136 return firstRootBox()->logicalTop() + firstLineStyle().fontMetrics().descent(firstRootBox()->baselineType());
akling@apple.comee3c8df2013-11-06 08:09:44 +00003137 return firstRootBox()->logicalTop() + firstLineStyle().fontMetrics().ascent(firstRootBox()->baselineType());
weinig@apple.com611b9292013-10-20 22:57:54 +00003138}
3139
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003140Optional<int> RenderBlockFlow::inlineBlockBaseline(LineDirectionMode lineDirection) const
weinig@apple.com611b9292013-10-20 22:57:54 +00003141{
3142 if (isWritingModeRoot() && !isRubyRun())
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003143 return WTF::nullopt;
weinig@apple.com611b9292013-10-20 22:57:54 +00003144
mmaxfield@apple.com9f4af632015-03-09 23:43:34 +00003145 // Note that here we only take the left and bottom into consideration. Our caller takes the right and top into consideration.
3146 float boxHeight = lineDirection == HorizontalLine ? height() + m_marginBox.bottom() : width() + m_marginBox.left();
3147 float lastBaseline;
mmaxfield@apple.com5f907742015-03-11 18:22:06 +00003148 if (!childrenInline()) {
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003149 Optional<int> inlineBlockBaseline = RenderBlock::inlineBlockBaseline(lineDirection);
mmaxfield@apple.com5f907742015-03-11 18:22:06 +00003150 if (!inlineBlockBaseline)
3151 return inlineBlockBaseline;
3152 lastBaseline = inlineBlockBaseline.value();
3153 } else {
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003154 if (!hasLines()) {
3155 if (!hasLineIfEmpty())
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003156 return WTF::nullopt;
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003157 const auto& fontMetrics = firstLineStyle().fontMetrics();
cdumez@apple.com8b7a0222018-12-20 04:41:11 +00003158 return Optional<int>(fontMetrics.ascent()
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003159 + (lineHeight(true, lineDirection, PositionOfInteriorLineBoxes) - fontMetrics.height()) / 2
mmaxfield@apple.com5f907742015-03-11 18:22:06 +00003160 + (lineDirection == HorizontalLine ? borderTop() + paddingTop() : borderRight() + paddingRight()));
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003161 }
3162
antti@apple.comd685c412019-12-07 13:56:51 +00003163 if (simpleLineLayout())
3164 lastBaseline = SimpleLineLayout::computeFlowLastLineBaseline(*this, *simpleLineLayout());
3165#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3166 else if (layoutFormattingContextLineLayout())
3167 lastBaseline = floorToInt(layoutFormattingContextLineLayout()->lastLineBaseline());
3168#endif
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003169 else {
3170 bool isFirstLine = lastRootBox() == firstRootBox();
3171 const auto& style = isFirstLine ? firstLineStyle() : this->style();
jfernandez@igalia.com7e696b92018-02-01 01:56:52 +00003172 // InlineFlowBox::placeBoxesInBlockDirection will flip lines in case of verticalLR mode, so we can assume verticalRL for now.
3173 lastBaseline = style.fontMetrics().ascent(lastRootBox()->baselineType())
3174 + (style.isFlippedLinesWritingMode() ? logicalHeight() - lastRootBox()->logicalBottom() : lastRootBox()->logicalTop());
mmaxfield@apple.coma52ab462015-03-11 14:41:01 +00003175 }
mmaxfield@apple.com9f4af632015-03-09 23:43:34 +00003176 }
3177 // According to the CSS spec http://www.w3.org/TR/CSS21/visudet.html, we shouldn't be performing this min, but should
3178 // instead be returning boxHeight directly. However, we feel that a min here is better behavior (and is consistent
3179 // enough with the spec to not cause tons of breakages).
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00003180 return style().overflowY() == Overflow::Visible ? lastBaseline : std::min(boxHeight, lastBaseline);
weinig@apple.com611b9292013-10-20 22:57:54 +00003181}
3182
megan_gardner@apple.comc66f2b82020-02-10 19:32:19 +00003183void RenderBlockFlow::setSelectionState(HighlightState state)
zalan@apple.com8bf2a912015-04-10 03:15:50 +00003184{
megan_gardner@apple.comc66f2b82020-02-10 19:32:19 +00003185 if (state != HighlightState::None)
zalan@apple.com8bf2a912015-04-10 03:15:50 +00003186 ensureLineBoxes();
3187 RenderBoxModelObject::setSelectionState(state);
3188}
3189
weinig@apple.com12840dc2013-10-22 23:59:08 +00003190GapRects RenderBlockFlow::inlineSelectionGaps(RenderBlock& rootBlock, const LayoutPoint& rootBlockPhysicalPosition, const LayoutSize& offsetFromRootBlock,
weinig@apple.com611b9292013-10-20 22:57:54 +00003191 LayoutUnit& lastLogicalTop, LayoutUnit& lastLogicalLeft, LayoutUnit& lastLogicalRight, const LogicalSelectionOffsetCaches& cache, const PaintInfo* paintInfo)
3192{
antti@apple.com8aa23022019-11-25 23:18:05 +00003193 ASSERT(!simpleLineLayout());
antti@apple.com940f5872013-10-24 20:31:11 +00003194
weinig@apple.com611b9292013-10-20 22:57:54 +00003195 GapRects result;
3196
megan_gardner@apple.comc66f2b82020-02-10 19:32:19 +00003197 bool containsStart = selectionState() == HighlightState::Start || selectionState() == HighlightState::Both;
weinig@apple.com611b9292013-10-20 22:57:54 +00003198
antti@apple.com0e632aa2013-10-22 21:03:38 +00003199 if (!hasLines()) {
weinig@apple.com611b9292013-10-20 22:57:54 +00003200 if (containsStart) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00003201 // Update our lastLogicalTop to be the bottom of the block. <hr>s or empty blocks with height can trip this case.
weinig@apple.com611b9292013-10-20 22:57:54 +00003202 lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + logicalHeight();
3203 lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, logicalHeight(), cache);
3204 lastLogicalRight = logicalRightSelectionOffset(rootBlock, logicalHeight(), cache);
3205 }
3206 return result;
3207 }
3208
3209 RootInlineBox* lastSelectedLine = 0;
3210 RootInlineBox* curr;
3211 for (curr = firstRootBox(); curr && !curr->hasSelectedChildren(); curr = curr->nextRootBox()) { }
3212
3213 // Now paint the gaps for the lines.
3214 for (; curr && curr->hasSelectedChildren(); curr = curr->nextRootBox()) {
3215 LayoutUnit selTop = curr->selectionTopAdjustedForPrecedingBlock();
3216 LayoutUnit selHeight = curr->selectionHeightAdjustedForPrecedingBlock();
3217
3218 if (!containsStart && !lastSelectedLine &&
megan_gardner@apple.comc66f2b82020-02-10 19:32:19 +00003219 selectionState() != HighlightState::Start && selectionState() != HighlightState::Both && !isRubyBase())
weinig@apple.com611b9292013-10-20 22:57:54 +00003220 result.uniteCenter(blockSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, lastLogicalTop, lastLogicalLeft, lastLogicalRight, selTop, cache, paintInfo));
3221
ross.kirsling@sony.com80414652019-05-21 01:36:11 +00003222 LayoutRect logicalRect { LayoutUnit(curr->logicalLeft()), selTop, LayoutUnit(curr->logicalWidth()), selTop + selHeight };
weinig@apple.com611b9292013-10-20 22:57:54 +00003223 logicalRect.move(isHorizontalWritingMode() ? offsetFromRootBlock : offsetFromRootBlock.transposedSize());
weinig@apple.com12840dc2013-10-22 23:59:08 +00003224 LayoutRect physicalRect = rootBlock.logicalRectToPhysicalRect(rootBlockPhysicalPosition, logicalRect);
weinig@apple.com611b9292013-10-20 22:57:54 +00003225 if (!paintInfo || (isHorizontalWritingMode() && physicalRect.y() < paintInfo->rect.maxY() && physicalRect.maxY() > paintInfo->rect.y())
3226 || (!isHorizontalWritingMode() && physicalRect.x() < paintInfo->rect.maxX() && physicalRect.maxX() > paintInfo->rect.x()))
3227 result.unite(curr->lineSelectionGap(rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock, selTop, selHeight, cache, paintInfo));
3228
3229 lastSelectedLine = curr;
3230 }
3231
3232 if (containsStart && !lastSelectedLine)
3233 // VisibleSelection must start just after our last line.
3234 lastSelectedLine = lastRootBox();
3235
megan_gardner@apple.comc66f2b82020-02-10 19:32:19 +00003236 if (lastSelectedLine && selectionState() != HighlightState::End && selectionState() != HighlightState::Both) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00003237 // Update our lastY to be the bottom of the last selected line.
weinig@apple.com611b9292013-10-20 22:57:54 +00003238 lastLogicalTop = blockDirectionOffset(rootBlock, offsetFromRootBlock) + lastSelectedLine->selectionBottom();
3239 lastLogicalLeft = logicalLeftSelectionOffset(rootBlock, lastSelectedLine->selectionBottom(), cache);
3240 lastLogicalRight = logicalRightSelectionOffset(rootBlock, lastSelectedLine->selectionBottom(), cache);
3241 }
3242 return result;
bjonesbe@adobe.com24199752013-10-08 23:20:42 +00003243}
3244
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003245bool RenderBlockFlow::needsLayoutAfterFragmentRangeChange() const
abucur@adobe.comeaf5e222014-05-14 14:35:07 +00003246{
3247 // A block without floats or that expands to enclose them won't need a relayout
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003248 // after a fragment range change. There is no overflow content needing relayout
3249 // in the fragment chain because the fragment range can only shrink after the estimation.
jfernandez@igalia.com136f1702014-12-08 19:13:16 +00003250 if (!containsFloats() || createsNewFormattingContext())
abucur@adobe.comeaf5e222014-05-14 14:35:07 +00003251 return false;
3252
3253 return true;
3254}
3255
zalan@apple.com624c2642017-10-05 14:04:27 +00003256void RenderBlockFlow::setMultiColumnFlow(RenderMultiColumnFlow& fragmentedFlow)
hyatt@apple.come9fe3d32014-01-24 17:14:22 +00003257{
zalan@apple.com624c2642017-10-05 14:04:27 +00003258 ASSERT(!hasRareBlockFlowData() || !rareBlockFlowData()->m_multiColumnFlow);
3259 ensureRareBlockFlowData().m_multiColumnFlow = makeWeakPtr(fragmentedFlow);
3260}
3261
3262void RenderBlockFlow::clearMultiColumnFlow()
3263{
3264 ASSERT(hasRareBlockFlowData());
3265 ASSERT(rareBlockFlowData()->m_multiColumnFlow);
3266 rareBlockFlowData()->m_multiColumnFlow.clear();
hyatt@apple.come9fe3d32014-01-24 17:14:22 +00003267}
3268
akling@apple.com525dae62014-01-03 20:22:09 +00003269static bool shouldCheckLines(const RenderBlockFlow& blockFlow)
weinig@apple.com17140912013-10-19 19:55:40 +00003270{
akling@apple.com38f0a652014-02-06 21:24:17 +00003271 return !blockFlow.isFloatingOrOutOfFlowPositioned() && blockFlow.style().height().isAuto();
weinig@apple.com17140912013-10-19 19:55:40 +00003272}
3273
3274RootInlineBox* RenderBlockFlow::lineAtIndex(int i) const
3275{
3276 ASSERT(i >= 0);
3277
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003278 if (style().visibility() != Visibility::Visible)
weinig@apple.com17140912013-10-19 19:55:40 +00003279 return nullptr;
3280
3281 if (childrenInline()) {
cdumez@apple.comc1d54fa2015-10-13 19:15:55 +00003282 for (auto* box = firstRootBox(); box; box = box->nextRootBox()) {
weinig@apple.com17140912013-10-19 19:55:40 +00003283 if (!i--)
3284 return box;
3285 }
akling@apple.com525dae62014-01-03 20:22:09 +00003286 return nullptr;
3287 }
3288
3289 for (auto& blockFlow : childrenOfType<RenderBlockFlow>(*this)) {
3290 if (!shouldCheckLines(blockFlow))
3291 continue;
3292 if (RootInlineBox* box = blockFlow.lineAtIndex(i))
3293 return box;
weinig@apple.com17140912013-10-19 19:55:40 +00003294 }
3295
3296 return nullptr;
3297}
3298
antti@apple.com702ef7d2019-12-06 14:37:43 +00003299int RenderBlockFlow::lineCount() const
weinig@apple.com17140912013-10-19 19:55:40 +00003300{
antti@apple.com702ef7d2019-12-06 14:37:43 +00003301 // FIXME: This should be tested by clients.
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003302 if (style().visibility() != Visibility::Visible)
weinig@apple.com17140912013-10-19 19:55:40 +00003303 return 0;
3304
weinig@apple.com17140912013-10-19 19:55:40 +00003305 if (childrenInline()) {
antti@apple.com702ef7d2019-12-06 14:37:43 +00003306#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3307 if (layoutFormattingContextLineLayout())
3308 return layoutFormattingContextLineLayout()->lineCount();
3309#endif
3310 if (simpleLineLayout())
3311 return simpleLineLayout()->lineCount();
3312
3313 if (complexLineLayout())
3314 return complexLineLayout()->lineCount();
3315
3316 return 0;
akling@apple.com525dae62014-01-03 20:22:09 +00003317 }
3318
antti@apple.com702ef7d2019-12-06 14:37:43 +00003319 int count = 0;
akling@apple.com525dae62014-01-03 20:22:09 +00003320 for (auto& blockFlow : childrenOfType<RenderBlockFlow>(*this)) {
3321 if (!shouldCheckLines(blockFlow))
3322 continue;
antti@apple.com702ef7d2019-12-06 14:37:43 +00003323 count += blockFlow.lineCount();
weinig@apple.com17140912013-10-19 19:55:40 +00003324 }
3325
3326 return count;
3327}
3328
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003329static int getHeightForLineCount(const RenderBlockFlow& block, int lineCount, bool includeBottom, int& count)
weinig@apple.com17140912013-10-19 19:55:40 +00003330{
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003331 if (block.style().visibility() != Visibility::Visible)
weinig@apple.com17140912013-10-19 19:55:40 +00003332 return -1;
3333
3334 if (block.childrenInline()) {
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003335 for (auto* box = block.firstRootBox(); box; box = box->nextRootBox()) {
3336 if (++count == lineCount)
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00003337 return box->lineBottom() + (includeBottom ? (block.borderBottom() + block.paddingBottom()) : 0_lu);
weinig@apple.com17140912013-10-19 19:55:40 +00003338 }
3339 } else {
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00003340 RenderBox* normalFlowChildWithoutLines = nullptr;
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003341 for (auto* obj = block.firstChildBox(); obj; obj = obj->nextSiblingBox()) {
cdumez@apple.com34e77ab2014-10-09 16:17:06 +00003342 if (is<RenderBlockFlow>(*obj) && shouldCheckLines(downcast<RenderBlockFlow>(*obj))) {
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003343 int result = getHeightForLineCount(downcast<RenderBlockFlow>(*obj), lineCount, false, count);
3344 if (result != -1)
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00003345 return result + obj->y() + (includeBottom ? (block.borderBottom() + block.paddingBottom()) : 0_lu);
akling@apple.com38f0a652014-02-06 21:24:17 +00003346 } else if (!obj->isFloatingOrOutOfFlowPositioned())
weinig@apple.com17140912013-10-19 19:55:40 +00003347 normalFlowChildWithoutLines = obj;
3348 }
3349 if (normalFlowChildWithoutLines && !lineCount)
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003350 return normalFlowChildWithoutLines->y() + normalFlowChildWithoutLines->height();
weinig@apple.com17140912013-10-19 19:55:40 +00003351 }
3352
3353 return -1;
3354}
3355
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003356int RenderBlockFlow::heightForLineCount(int lineCount)
weinig@apple.com17140912013-10-19 19:55:40 +00003357{
3358 int count = 0;
zalan@apple.com2b9b6102018-07-25 22:44:48 +00003359 return getHeightForLineCount(*this, lineCount, true, count);
weinig@apple.com17140912013-10-19 19:55:40 +00003360}
3361
3362void RenderBlockFlow::clearTruncation()
3363{
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003364 if (style().visibility() != Visibility::Visible)
weinig@apple.com17140912013-10-19 19:55:40 +00003365 return;
3366
3367 if (childrenInline() && hasMarkupTruncation()) {
antti@apple.com940f5872013-10-24 20:31:11 +00003368 ensureLineBoxes();
3369
weinig@apple.com17140912013-10-19 19:55:40 +00003370 setHasMarkupTruncation(false);
cdumez@apple.comc1d54fa2015-10-13 19:15:55 +00003371 for (auto* box = firstRootBox(); box; box = box->nextRootBox())
weinig@apple.com17140912013-10-19 19:55:40 +00003372 box->clearTruncation();
akling@apple.com525dae62014-01-03 20:22:09 +00003373 return;
3374 }
3375
3376 for (auto& blockFlow : childrenOfType<RenderBlockFlow>(*this)) {
3377 if (shouldCheckLines(blockFlow))
3378 blockFlow.clearTruncation();
weinig@apple.com17140912013-10-19 19:55:40 +00003379 }
3380}
3381
weinig@apple.com3f23b382013-10-19 20:26:58 +00003382bool RenderBlockFlow::containsNonZeroBidiLevel() const
3383{
cdumez@apple.comc1d54fa2015-10-13 19:15:55 +00003384 for (auto* root = firstRootBox(); root; root = root->nextRootBox()) {
antti@apple.com7ac7f602019-09-25 14:23:03 +00003385 for (auto* box = root->firstLeafDescendant(); box; box = box->nextLeafOnLine()) {
weinig@apple.com3f23b382013-10-19 20:26:58 +00003386 if (box->bidiLevel())
3387 return true;
3388 }
3389 }
3390 return false;
3391}
3392
weinig@apple.com611b9292013-10-20 22:57:54 +00003393Position RenderBlockFlow::positionForBox(InlineBox *box, bool start) const
3394{
3395 if (!box)
3396 return Position();
3397
3398 if (!box->renderer().nonPseudoNode())
3399 return createLegacyEditingPosition(nonPseudoElement(), start ? caretMinOffset() : caretMaxOffset());
3400
cdumez@apple.com57d544c2014-10-16 00:05:37 +00003401 if (!is<InlineTextBox>(*box))
weinig@apple.com611b9292013-10-20 22:57:54 +00003402 return createLegacyEditingPosition(box->renderer().nonPseudoNode(), start ? box->renderer().caretMinOffset() : box->renderer().caretMaxOffset());
3403
cdumez@apple.com57d544c2014-10-16 00:05:37 +00003404 auto& textBox = downcast<InlineTextBox>(*box);
3405 return createLegacyEditingPosition(textBox.renderer().nonPseudoNode(), start ? textBox.start() : textBox.start() + textBox.len());
weinig@apple.com611b9292013-10-20 22:57:54 +00003406}
3407
enrica@apple.com0db51a62016-04-27 23:53:08 +00003408RenderText* RenderBlockFlow::findClosestTextAtAbsolutePoint(const FloatPoint& point)
3409{
3410 // A light, non-recursive version of RenderBlock::positionForCoordinates that looks at
3411 // whether a point lies within the gaps between its root line boxes, to be called against
3412 // a node returned from elementAtPoint. We make the assumption that either the node or one
3413 // of its immediate children contains the root line boxes in question.
3414 // See <rdar://problem/6824650> for context.
3415
3416 RenderBlock* block = this;
3417
3418 FloatPoint localPoint = block->absoluteToLocal(point);
3419
3420 if (!block->childrenInline()) {
3421 // Look among our immediate children for an alternate box that contains the point.
3422 for (RenderBox* child = block->firstChildBox(); child; child = child->nextSiblingBox()) {
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003423 if (!child->height() || child->style().visibility() != WebCore::Visibility::Visible || child->isFloatingOrOutOfFlowPositioned())
enrica@apple.com0db51a62016-04-27 23:53:08 +00003424 continue;
3425 float top = child->y();
3426
3427 RenderBox* nextChild = child->nextSiblingBox();
3428 while (nextChild && nextChild->isFloatingOrOutOfFlowPositioned())
3429 nextChild = nextChild->nextSiblingBox();
3430 if (!nextChild) {
3431 if (localPoint.y() >= top) {
3432 block = downcast<RenderBlock>(child);
3433 break;
3434 }
3435 continue;
3436 }
3437
3438 float bottom = nextChild->y();
3439
3440 if (localPoint.y() >= top && localPoint.y() < bottom && is<RenderBlock>(*child)) {
3441 block = downcast<RenderBlock>(child);
3442 break;
3443 }
3444 }
3445
3446 if (!block->childrenInline())
3447 return nullptr;
3448
3449 localPoint = block->absoluteToLocal(point);
3450 }
3451
3452 RenderBlockFlow& blockFlow = downcast<RenderBlockFlow>(*block);
3453
3454 // Only check the gaps between the root line boxes. We deliberately ignore overflow because
3455 // experience has shown that hit tests on an exploded text node can fail when within the
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003456 // overflow fragment.
enrica@apple.com0db51a62016-04-27 23:53:08 +00003457 for (RootInlineBox* current = blockFlow.firstRootBox(); current && current != blockFlow.lastRootBox(); current = current->nextRootBox()) {
3458 float currentBottom = current->y() + current->logicalHeight();
3459 if (localPoint.y() < currentBottom)
3460 return nullptr;
3461
3462 RootInlineBox* next = current->nextRootBox();
3463 float nextTop = next->y();
3464 if (localPoint.y() < nextTop) {
3465 InlineBox* inlineBox = current->closestLeafChildForLogicalLeftPosition(localPoint.x());
3466 if (inlineBox && inlineBox->behavesLikeText() && is<RenderText>(inlineBox->renderer()))
3467 return &downcast<RenderText>(inlineBox->renderer());
3468 }
3469 }
3470 return nullptr;
3471}
3472
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003473VisiblePosition RenderBlockFlow::positionForPointWithInlineChildren(const LayoutPoint& pointInLogicalContents, const RenderFragmentContainer* fragment)
weinig@apple.com611b9292013-10-20 22:57:54 +00003474{
3475 ASSERT(childrenInline());
3476
antti@apple.com940f5872013-10-24 20:31:11 +00003477 ensureLineBoxes();
3478
weinig@apple.com611b9292013-10-20 22:57:54 +00003479 if (!firstRootBox())
3480 return createVisiblePosition(0, DOWNSTREAM);
3481
akling@apple.com827be9c2013-10-29 02:58:43 +00003482 bool linesAreFlipped = style().isFlippedLinesWritingMode();
3483 bool blocksAreFlipped = style().isFlippedBlocksWritingMode();
weinig@apple.com611b9292013-10-20 22:57:54 +00003484
3485 // look for the closest line box in the root box which is at the passed-in y coordinate
3486 InlineBox* closestBox = 0;
3487 RootInlineBox* firstRootBoxWithChildren = 0;
3488 RootInlineBox* lastRootBoxWithChildren = 0;
3489 for (RootInlineBox* root = firstRootBox(); root; root = root->nextRootBox()) {
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003490 if (fragment && root->containingFragment() != fragment)
stavila@adobe.com4ce2fff2014-04-25 13:56:12 +00003491 continue;
3492
antti@apple.com7ac7f602019-09-25 14:23:03 +00003493 if (!root->firstLeafDescendant())
weinig@apple.com611b9292013-10-20 22:57:54 +00003494 continue;
3495 if (!firstRootBoxWithChildren)
3496 firstRootBoxWithChildren = root;
3497
3498 if (!linesAreFlipped && root->isFirstAfterPageBreak() && (pointInLogicalContents.y() < root->lineTopWithLeading()
3499 || (blocksAreFlipped && pointInLogicalContents.y() == root->lineTopWithLeading())))
3500 break;
3501
3502 lastRootBoxWithChildren = root;
3503
3504 // check if this root line box is located at this y coordinate
3505 if (pointInLogicalContents.y() < root->selectionBottom() || (blocksAreFlipped && pointInLogicalContents.y() == root->selectionBottom())) {
3506 if (linesAreFlipped) {
3507 RootInlineBox* nextRootBoxWithChildren = root->nextRootBox();
antti@apple.com7ac7f602019-09-25 14:23:03 +00003508 while (nextRootBoxWithChildren && !nextRootBoxWithChildren->firstLeafDescendant())
weinig@apple.com611b9292013-10-20 22:57:54 +00003509 nextRootBoxWithChildren = nextRootBoxWithChildren->nextRootBox();
3510
3511 if (nextRootBoxWithChildren && nextRootBoxWithChildren->isFirstAfterPageBreak() && (pointInLogicalContents.y() > nextRootBoxWithChildren->lineTopWithLeading()
3512 || (!blocksAreFlipped && pointInLogicalContents.y() == nextRootBoxWithChildren->lineTopWithLeading())))
3513 continue;
3514 }
3515 closestBox = root->closestLeafChildForLogicalLeftPosition(pointInLogicalContents.x());
3516 if (closestBox)
3517 break;
3518 }
3519 }
3520
3521 bool moveCaretToBoundary = frame().editor().behavior().shouldMoveCaretToHorizontalBoundaryWhenPastTopOrBottom();
3522
3523 if (!moveCaretToBoundary && !closestBox && lastRootBoxWithChildren) {
3524 // y coordinate is below last root line box, pretend we hit it
3525 closestBox = lastRootBoxWithChildren->closestLeafChildForLogicalLeftPosition(pointInLogicalContents.x());
3526 }
3527
3528 if (closestBox) {
3529 if (moveCaretToBoundary) {
ross.kirsling@sony.com80414652019-05-21 01:36:11 +00003530 LayoutUnit firstRootBoxWithChildrenTop = std::min(firstRootBoxWithChildren->selectionTop(), LayoutUnit(firstRootBoxWithChildren->logicalTop()));
weinig@apple.com611b9292013-10-20 22:57:54 +00003531 if (pointInLogicalContents.y() < firstRootBoxWithChildrenTop
3532 || (blocksAreFlipped && pointInLogicalContents.y() == firstRootBoxWithChildrenTop)) {
antti@apple.com7ac7f602019-09-25 14:23:03 +00003533 InlineBox* box = firstRootBoxWithChildren->firstLeafDescendant();
weinig@apple.com611b9292013-10-20 22:57:54 +00003534 if (box->isLineBreak()) {
antti@apple.com7ac7f602019-09-25 14:23:03 +00003535 if (InlineBox* newBox = box->nextLeafOnLineIgnoringLineBreak())
weinig@apple.com611b9292013-10-20 22:57:54 +00003536 box = newBox;
3537 }
3538 // y coordinate is above first root line box, so return the start of the first
3539 return VisiblePosition(positionForBox(box, true), DOWNSTREAM);
3540 }
3541 }
3542
3543 // pass the box a top position that is inside it
3544 LayoutPoint point(pointInLogicalContents.x(), closestBox->root().blockDirectionPointInLine());
3545 if (!isHorizontalWritingMode())
3546 point = point.transposedPoint();
3547 if (closestBox->renderer().isReplaced())
cdumez@apple.com0abff8b2014-10-17 21:25:10 +00003548 return positionForPointRespectingEditingBoundaries(*this, downcast<RenderBox>(closestBox->renderer()), point);
stavila@adobe.com4ce2fff2014-04-25 13:56:12 +00003549 return closestBox->renderer().positionForPoint(point, nullptr);
weinig@apple.com611b9292013-10-20 22:57:54 +00003550 }
3551
3552 if (lastRootBoxWithChildren) {
3553 // We hit this case for Mac behavior when the Y coordinate is below the last box.
3554 ASSERT(moveCaretToBoundary);
3555 InlineBox* logicallyLastBox;
3556 if (lastRootBoxWithChildren->getLogicalEndBoxWithNode(logicallyLastBox))
3557 return VisiblePosition(positionForBox(logicallyLastBox, false), DOWNSTREAM);
3558 }
3559
3560 // Can't reach this. We have a root line box, but it has no kids.
3561 // FIXME: This should ASSERT_NOT_REACHED(), but clicking on placeholder text
3562 // seems to hit this code path.
3563 return createVisiblePosition(0, DOWNSTREAM);
3564}
3565
zalan@apple.com0d5951b2017-02-19 16:24:20 +00003566Position RenderBlockFlow::positionForPoint(const LayoutPoint& point)
3567{
3568 // FIXME: It supports single text child only (which is the majority of simple line layout supported content at this point).
3569 if (!simpleLineLayout() || firstChild() != lastChild() || !is<RenderText>(firstChild()))
3570 return positionForPoint(point, nullptr).deepEquivalent();
3571 return downcast<RenderText>(*firstChild()).positionForPoint(point);
3572}
3573
hyatt@apple.come0d2e0f2017-09-27 16:19:08 +00003574VisiblePosition RenderBlockFlow::positionForPoint(const LayoutPoint& point, const RenderFragmentContainer*)
commit-queue@webkit.org5ce6c902013-11-11 18:21:05 +00003575{
antti@apple.come0e8a562017-09-21 18:29:47 +00003576 return RenderBlock::positionForPoint(point, nullptr);
commit-queue@webkit.org5ce6c902013-11-11 18:21:05 +00003577}
3578
zalan@apple.com8ee1af52016-02-11 22:15:45 +00003579void RenderBlockFlow::addFocusRingRectsForInlineChildren(Vector<LayoutRect>& rects, const LayoutPoint& additionalOffset, const RenderLayerModelObject*)
weinig@apple.com611b9292013-10-20 22:57:54 +00003580{
antti@apple.com940f5872013-10-24 20:31:11 +00003581 ASSERT(childrenInline());
weinig@apple.com611b9292013-10-20 22:57:54 +00003582 for (RootInlineBox* curr = firstRootBox(); curr; curr = curr->nextRootBox()) {
ross.kirsling@sony.com80414652019-05-21 01:36:11 +00003583 LayoutUnit top = std::max(curr->lineTop(), LayoutUnit(curr->top()));
3584 LayoutUnit bottom = std::min(curr->lineBottom(), LayoutUnit(curr->top() + curr->height()));
3585 LayoutRect rect { LayoutUnit(additionalOffset.x() + curr->x()), additionalOffset.y() + top, LayoutUnit(curr->width()), bottom - top };
weinig@apple.com611b9292013-10-20 22:57:54 +00003586 if (!rect.isEmpty())
zalan@apple.com8ee1af52016-02-11 22:15:45 +00003587 rects.append(rect);
weinig@apple.com611b9292013-10-20 22:57:54 +00003588 }
3589}
3590
3591void RenderBlockFlow::paintInlineChildren(PaintInfo& paintInfo, const LayoutPoint& paintOffset)
3592{
3593 ASSERT(childrenInline());
antti@apple.com940f5872013-10-24 20:31:11 +00003594
antti@apple.com1880e712019-11-26 20:15:39 +00003595#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
antti@apple.comd4dda742019-12-03 15:41:43 +00003596 if (layoutFormattingContextLineLayout()) {
3597 layoutFormattingContextLineLayout()->paint(paintInfo, paintOffset);
antti@apple.com1880e712019-11-26 20:15:39 +00003598 return;
3599 }
3600#endif
3601
darin@apple.come1be6ca2014-04-28 04:19:10 +00003602 if (auto simpleLineLayout = this->simpleLineLayout()) {
3603 SimpleLineLayout::paintFlow(*this, *simpleLineLayout, paintInfo, paintOffset);
antti@apple.com940f5872013-10-24 20:31:11 +00003604 return;
3605 }
antti@apple.comc6a9c732019-08-12 15:31:34 +00003606
antti@apple.com8aa23022019-11-25 23:18:05 +00003607 if (complexLineLayout())
3608 complexLineLayout()->lineBoxes().paint(this, paintInfo, paintOffset);
weinig@apple.com611b9292013-10-20 22:57:54 +00003609}
3610
zalan@apple.com4a440322017-11-10 00:31:24 +00003611bool RenderBlockFlow::relayoutForPagination()
weinig@apple.com611b9292013-10-20 22:57:54 +00003612{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003613 if (!multiColumnFlow() || !multiColumnFlow()->shouldRelayoutForPagination())
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003614 return false;
3615
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003616 multiColumnFlow()->setNeedsHeightsRecalculation(false);
3617 multiColumnFlow()->setInBalancingPass(true); // Prevent re-entering this method (and recursion into layout).
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003618
3619 bool needsRelayout;
3620 bool neededRelayout = false;
3621 bool firstPass = true;
3622 do {
3623 // Column heights may change here because of balancing. We may have to do multiple layout
3624 // passes, depending on how the contents is fitted to the changed column heights. In most
3625 // cases, laying out again twice or even just once will suffice. Sometimes we need more
3626 // passes than that, though, but the number of retries should not exceed the number of
3627 // columns, unless we have a bug.
3628 needsRelayout = false;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003629 for (RenderMultiColumnSet* multicolSet = multiColumnFlow()->firstMultiColumnSet(); multicolSet; multicolSet = multicolSet->nextSiblingMultiColumnSet()) {
hyatt@apple.comc1c39032014-04-15 23:25:58 +00003630 if (multicolSet->recalculateColumnHeight(firstPass))
3631 needsRelayout = true;
3632 if (needsRelayout) {
3633 // Once a column set gets a new column height, that column set and all successive column
3634 // sets need to be laid out over again, since their logical top will be affected by
3635 // this, and therefore their column heights may change as well, at least if the multicol
3636 // height is constrained.
3637 multicolSet->setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003638 }
hyatt@apple.comc1c39032014-04-15 23:25:58 +00003639 }
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003640 if (needsRelayout) {
3641 // Layout again. Column balancing resulted in a new height.
3642 neededRelayout = true;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003643 multiColumnFlow()->setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003644 setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003645 layoutBlock(false);
3646 }
3647 firstPass = false;
3648 } while (needsRelayout);
3649
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003650 multiColumnFlow()->setInBalancingPass(false);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003651
3652 return neededRelayout;
weinig@apple.com611b9292013-10-20 22:57:54 +00003653}
3654
antti@apple.com940f5872013-10-24 20:31:11 +00003655bool RenderBlockFlow::hasLines() const
3656{
zalan@apple.coma2fe1762016-08-24 18:33:43 +00003657 if (!childrenInline())
3658 return false;
antti@apple.com940f5872013-10-24 20:31:11 +00003659
antti@apple.com702ef7d2019-12-06 14:37:43 +00003660#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3661 if (layoutFormattingContextLineLayout())
3662 return layoutFormattingContextLineLayout()->lineCount();
3663#endif
3664 if (simpleLineLayout())
3665 return simpleLineLayout()->lineCount();
antti@apple.com940f5872013-10-24 20:31:11 +00003666
antti@apple.com8aa23022019-11-25 23:18:05 +00003667 return complexLineLayout() && complexLineLayout()->lineBoxes().firstLineBox();
antti@apple.com940f5872013-10-24 20:31:11 +00003668}
3669
antti@apple.com9e891c82014-05-22 06:12:34 +00003670void RenderBlockFlow::invalidateLineLayoutPath()
3671{
akling@apple.coma12fee22015-02-01 02:58:13 +00003672 switch (lineLayoutPath()) {
antti@apple.com9e891c82014-05-22 06:12:34 +00003673 case UndeterminedPath:
3674 case ForceLineBoxesPath:
antti@apple.com8aa23022019-11-25 23:18:05 +00003675 ASSERT(!simpleLineLayout());
antti@apple.com9e891c82014-05-22 06:12:34 +00003676 return;
3677 case LineBoxesPath:
antti@apple.com8aa23022019-11-25 23:18:05 +00003678 ASSERT(!simpleLineLayout());
akling@apple.coma12fee22015-02-01 02:58:13 +00003679 setLineLayoutPath(UndeterminedPath);
antti@apple.com9e891c82014-05-22 06:12:34 +00003680 return;
antti@apple.comd4dda742019-12-03 15:41:43 +00003681 case LayoutFormattingContextPath: // FIXME: Not all clients of invalidateLineLayoutPath() actually need to wipe the layout.
antti@apple.com9e891c82014-05-22 06:12:34 +00003682 case SimpleLinesPath:
3683 // The simple line layout may have become invalid.
antti@apple.com620243d2019-11-27 17:01:11 +00003684 m_lineLayout = WTF::Monostate();
akling@apple.coma12fee22015-02-01 02:58:13 +00003685 setLineLayoutPath(UndeterminedPath);
zalan@apple.comde191042017-06-06 19:35:56 +00003686 if (needsLayout())
3687 return;
3688 // FIXME: We should just kick off a subtree layout here (if needed at all) see webkit.org/b/172947.
3689 setNeedsLayout();
antti@apple.com9e891c82014-05-22 06:12:34 +00003690 return;
3691 }
3692 ASSERT_NOT_REACHED();
3693}
3694
zalan@apple.come37da962014-12-11 03:29:29 +00003695void RenderBlockFlow::layoutSimpleLines(bool relayoutChildren, LayoutUnit& repaintLogicalTop, LayoutUnit& repaintLogicalBottom)
antti@apple.com940f5872013-10-24 20:31:11 +00003696{
antti@apple.com8aa23022019-11-25 23:18:05 +00003697 bool needsLayout = selfNeedsLayout() || relayoutChildren || !simpleLineLayout();
antti@apple.comebf3d5a2019-11-26 18:28:13 +00003698 if (needsLayout)
antti@apple.com8aa23022019-11-25 23:18:05 +00003699 m_lineLayout = SimpleLineLayout::create(*this);
antti@apple.comebf3d5a2019-11-26 18:28:13 +00003700
antti@apple.com8aa23022019-11-25 23:18:05 +00003701 auto& simpleLineLayout = *this->simpleLineLayout();
3702
zalan@apple.com4de96c52017-11-07 04:09:59 +00003703 if (view().frameView().layoutContext().layoutState() && view().frameView().layoutContext().layoutState()->isPaginated()) {
antti@apple.com8aa23022019-11-25 23:18:05 +00003704 simpleLineLayout.setIsPaginated();
3705 SimpleLineLayout::adjustLinePositionsForPagination(simpleLineLayout, *this);
zalan@apple.com5cd09e52017-02-25 02:14:40 +00003706 }
antti@apple.com8aa23022019-11-25 23:18:05 +00003707
zalan@apple.comd6ab7b52016-11-01 05:35:48 +00003708 for (auto& renderer : childrenOfType<RenderObject>(*this))
3709 renderer.clearNeedsLayout();
antti@apple.com8aa23022019-11-25 23:18:05 +00003710
3711 LayoutUnit lineLayoutHeight = SimpleLineLayout::computeFlowHeight(*this, simpleLineLayout);
antti@apple.com940f5872013-10-24 20:31:11 +00003712 LayoutUnit lineLayoutTop = borderAndPaddingBefore();
antti@apple.com940f5872013-10-24 20:31:11 +00003713 repaintLogicalTop = lineLayoutTop;
zalan@apple.comd120edd2017-12-01 02:54:29 +00003714 repaintLogicalBottom = needsLayout ? repaintLogicalTop + lineLayoutHeight + borderAndPaddingAfter() : repaintLogicalTop;
antti@apple.com940f5872013-10-24 20:31:11 +00003715 setLogicalHeight(lineLayoutTop + lineLayoutHeight + borderAndPaddingAfter());
3716}
3717
antti@apple.com1880e712019-11-26 20:15:39 +00003718#if ENABLE(LAYOUT_FORMATTING_CONTEXT)
3719void RenderBlockFlow::layoutLFCLines(bool, LayoutUnit& repaintLogicalTop, LayoutUnit& repaintLogicalBottom)
3720{
antti@apple.comd4dda742019-12-03 15:41:43 +00003721 if (!layoutFormattingContextLineLayout())
3722 m_lineLayout = makeUnique<LayoutIntegration::LineLayout>(*this);
antti@apple.com1880e712019-11-26 20:15:39 +00003723
antti@apple.comd4dda742019-12-03 15:41:43 +00003724 auto& layoutFormattingContextLineLayout = *this->layoutFormattingContextLineLayout();
antti@apple.com1880e712019-11-26 20:15:39 +00003725
3726 for (auto& renderer : childrenOfType<RenderObject>(*this))
3727 renderer.clearNeedsLayout();
3728
antti@apple.comd4dda742019-12-03 15:41:43 +00003729 layoutFormattingContextLineLayout.layout();
antti@apple.com1880e712019-11-26 20:15:39 +00003730
antti@apple.comd4dda742019-12-03 15:41:43 +00003731 auto contentHeight = layoutFormattingContextLineLayout.contentLogicalHeight();
zalan@apple.com7b374f82020-01-05 14:50:08 +00003732 auto contentBoxTop = borderAndPaddingBefore();
3733 auto contentBoxBottom = contentBoxTop + contentHeight;
3734 auto borderBoxBottom = contentBoxBottom + borderAndPaddingAfter();
antti@apple.com1ebf7e12019-12-03 10:51:46 +00003735
zalan@apple.com7b374f82020-01-05 14:50:08 +00003736 repaintLogicalTop = contentBoxTop;
3737 repaintLogicalBottom = borderBoxBottom;
3738 setLogicalHeight(borderBoxBottom);
antti@apple.com1880e712019-11-26 20:15:39 +00003739}
3740#endif
3741
antti@apple.com940f5872013-10-24 20:31:11 +00003742void RenderBlockFlow::ensureLineBoxes()
3743{
antti@apple.comc6a9c732019-08-12 15:31:34 +00003744 if (!childrenInline())
3745 return;
3746
akling@apple.coma12fee22015-02-01 02:58:13 +00003747 setLineLayoutPath(ForceLineBoxesPath);
antti@apple.comc6a9c732019-08-12 15:31:34 +00003748
antti@apple.com620243d2019-11-27 17:01:11 +00003749 if (complexLineLayout() || !hasLineLayout())
antti@apple.com940f5872013-10-24 20:31:11 +00003750 return;
zalan@apple.com95cae4f2018-04-23 14:47:00 +00003751
antti@apple.com620243d2019-11-27 17:01:11 +00003752 auto simpleLineLayout = makeRefPtr(this->simpleLineLayout());
antti@apple.comc6a9c732019-08-12 15:31:34 +00003753
antti@apple.com8aa23022019-11-25 23:18:05 +00003754 m_lineLayout = makeUnique<ComplexLineLayout>(*this);
3755
antti@apple.com620243d2019-11-27 17:01:11 +00003756 if (simpleLineLayout) {
3757 if (SimpleLineLayout::canUseForLineBoxTree(*this, *simpleLineLayout)) {
3758 SimpleLineLayout::generateLineBoxTree(*this, *simpleLineLayout);
3759 return;
3760 }
zalan@apple.com95cae4f2018-04-23 14:47:00 +00003761 }
antti@apple.com8aa23022019-11-25 23:18:05 +00003762
3763 auto& complexLineLayout = *this->complexLineLayout();
antti@apple.com940f5872013-10-24 20:31:11 +00003764
mark.lam@apple.com65724362020-01-06 22:24:50 +00003765#if ASSERT_ENABLED
antti@apple.com940f5872013-10-24 20:31:11 +00003766 LayoutUnit oldHeight = logicalHeight();
3767#endif
3768 bool didNeedLayout = needsLayout();
3769
3770 bool relayoutChildren = false;
3771 LayoutUnit repaintLogicalTop;
3772 LayoutUnit repaintLogicalBottom;
antti@apple.com620243d2019-11-27 17:01:11 +00003773 if (simpleLineLayout && simpleLineLayout->isPaginated()) {
zalan@apple.com6c04c202017-05-01 00:15:38 +00003774 PaginatedLayoutStateMaintainer state(*this);
antti@apple.com8aa23022019-11-25 23:18:05 +00003775 complexLineLayout.layoutLineBoxes(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
zalan@apple.come9f08212017-04-27 11:02:09 +00003776 // This matches relayoutToAvoidWidows.
3777 if (shouldBreakAtLineToAvoidWidow())
antti@apple.com8aa23022019-11-25 23:18:05 +00003778 complexLineLayout.layoutLineBoxes(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
zalan@apple.comb16bcdc2017-07-21 21:04:43 +00003779 // FIXME: This is needed as long as simple and normal line layout produce different line breakings.
3780 repaint();
zalan@apple.comfa5add62017-02-22 19:33:02 +00003781 } else
antti@apple.com8aa23022019-11-25 23:18:05 +00003782 complexLineLayout.layoutLineBoxes(relayoutChildren, repaintLogicalTop, repaintLogicalBottom);
antti@apple.com940f5872013-10-24 20:31:11 +00003783
3784 updateLogicalHeight();
3785 ASSERT(didNeedLayout || logicalHeight() == oldHeight);
3786
3787 if (!didNeedLayout)
3788 clearNeedsLayout();
3789}
3790
simon.fraser@apple.comc9f96132015-03-06 18:20:40 +00003791#if ENABLE(TREE_DEBUGGING)
don.olmstead@sony.comd57eb472017-08-10 01:15:14 +00003792void RenderBlockFlow::outputLineTreeAndMark(WTF::TextStream& stream, const InlineBox* markedBox, int depth) const
weinig@apple.com611b9292013-10-20 22:57:54 +00003793{
weinig@apple.com611b9292013-10-20 22:57:54 +00003794 for (const RootInlineBox* root = firstRootBox(); root; root = root->nextRootBox())
zalan@apple.com360de752017-07-06 21:13:24 +00003795 root->outputLineTreeAndMark(stream, markedBox, depth);
simon.fraser@apple.com3518b142014-09-03 21:18:05 +00003796
3797 if (auto simpleLineLayout = this->simpleLineLayout())
zalan@apple.com360de752017-07-06 21:13:24 +00003798 SimpleLineLayout::outputLineLayoutForFlow(stream, *this, *simpleLineLayout, depth);
weinig@apple.com611b9292013-10-20 22:57:54 +00003799}
3800#endif
3801
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00003802RenderBlockFlow::RenderBlockFlowRareData& RenderBlockFlow::ensureRareBlockFlowData()
3803{
3804 if (hasRareBlockFlowData())
3805 return *m_rareBlockFlowData;
3806 materializeRareBlockFlowData();
3807 return *m_rareBlockFlowData;
3808}
3809
3810void RenderBlockFlow::materializeRareBlockFlowData()
3811{
3812 ASSERT(!hasRareBlockFlowData());
ysuzuki@apple.com1d8e24d2019-08-19 06:59:40 +00003813 m_rareBlockFlowData = makeUnique<RenderBlockFlow::RenderBlockFlowRareData>(*this);
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00003814}
3815
dbates@webkit.org102013c2016-09-26 21:51:25 +00003816#if ENABLE(TEXT_AUTOSIZING)
darin@apple.com5917cb12017-11-23 17:32:42 +00003817
cdumez@apple.com83102df2016-05-19 19:09:41 +00003818static inline bool isVisibleRenderText(const RenderObject& renderer)
aestes@apple.com6751d842014-01-12 02:51:25 +00003819{
cdumez@apple.com83102df2016-05-19 19:09:41 +00003820 if (!is<RenderText>(renderer))
aestes@apple.com6751d842014-01-12 02:51:25 +00003821 return false;
cdumez@apple.com83102df2016-05-19 19:09:41 +00003822
3823 auto& renderText = downcast<RenderText>(renderer);
darin@apple.com5917cb12017-11-23 17:32:42 +00003824 return !renderText.linesBoundingBox().isEmpty() && !renderText.text().isAllSpecialCharacters<isHTMLSpace>();
aestes@apple.com6751d842014-01-12 02:51:25 +00003825}
3826
cdumez@apple.com83102df2016-05-19 19:09:41 +00003827static inline bool resizeTextPermitted(const RenderObject& renderer)
aestes@apple.com6751d842014-01-12 02:51:25 +00003828{
3829 // We disallow resizing for text input fields and textarea to address <rdar://problem/5792987> and <rdar://problem/8021123>
cdumez@apple.com83102df2016-05-19 19:09:41 +00003830 for (auto* ancestor = renderer.parent(); ancestor; ancestor = ancestor->parent()) {
aestes@apple.com6751d842014-01-12 02:51:25 +00003831 // Get the first non-shadow HTMLElement and see if it's an input.
cdumez@apple.com83102df2016-05-19 19:09:41 +00003832 if (is<HTMLElement>(ancestor->element()) && !ancestor->element()->isInShadowTree()) {
3833 auto& element = downcast<HTMLElement>(*ancestor->element());
cdumez@apple.com59fdc8a2014-09-24 21:25:22 +00003834 return !is<HTMLInputElement>(element) && !is<HTMLTextAreaElement>(element);
aestes@apple.com6751d842014-01-12 02:51:25 +00003835 }
aestes@apple.com6751d842014-01-12 02:51:25 +00003836 }
3837 return true;
3838}
3839
antti@apple.com0b3dffe2014-03-24 16:30:52 +00003840int RenderBlockFlow::lineCountForTextAutosizing()
aestes@apple.com6751d842014-01-12 02:51:25 +00003841{
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00003842 if (style().visibility() != Visibility::Visible)
antti@apple.com0b3dffe2014-03-24 16:30:52 +00003843 return 0;
3844 if (childrenInline())
3845 return lineCount();
aestes@apple.com6751d842014-01-12 02:51:25 +00003846 // Only descend into list items.
3847 int count = 0;
antti@apple.com0b3dffe2014-03-24 16:30:52 +00003848 for (auto& listItem : childrenOfType<RenderListItem>(*this))
3849 count += listItem.lineCount();
aestes@apple.com6751d842014-01-12 02:51:25 +00003850 return count;
3851}
3852
cdumez@apple.com83102df2016-05-19 19:09:41 +00003853static bool isNonBlocksOrNonFixedHeightListItems(const RenderObject& renderer)
aestes@apple.com6751d842014-01-12 02:51:25 +00003854{
cdumez@apple.com83102df2016-05-19 19:09:41 +00003855 if (!renderer.isRenderBlock())
aestes@apple.com6751d842014-01-12 02:51:25 +00003856 return true;
cdumez@apple.com83102df2016-05-19 19:09:41 +00003857 if (renderer.isListItem())
3858 return renderer.style().height().type() != Fixed;
aestes@apple.com6751d842014-01-12 02:51:25 +00003859 return false;
3860}
3861
timothy_horton@apple.com117a5f82018-02-19 18:47:30 +00003862// For now, we auto size single lines of text the same as multiple lines.
3863// We've been experimenting with low values for single lines of text.
3864static inline float oneLineTextMultiplier(RenderObject& renderer, float specifiedSize)
aestes@apple.com6751d842014-01-12 02:51:25 +00003865{
timothy_horton@apple.com117a5f82018-02-19 18:47:30 +00003866 const float coefficient = renderer.settings().oneLineTextMultiplierCoefficient();
3867 return std::max((1.0f / log10f(specifiedSize) * coefficient), 1.0f);
aestes@apple.com6751d842014-01-12 02:51:25 +00003868}
3869
timothy_horton@apple.com117a5f82018-02-19 18:47:30 +00003870static inline float textMultiplier(RenderObject& renderer, float specifiedSize)
aestes@apple.com6751d842014-01-12 02:51:25 +00003871{
timothy_horton@apple.com117a5f82018-02-19 18:47:30 +00003872 const float coefficient = renderer.settings().multiLineTextMultiplierCoefficient();
3873 return std::max((1.0f / log10f(specifiedSize) * coefficient), 1.0f);
aestes@apple.com6751d842014-01-12 02:51:25 +00003874}
3875
mmaxfield@apple.com59146a72019-05-29 02:27:20 +00003876void RenderBlockFlow::adjustComputedFontSizes(float size, float visibleWidth)
aestes@apple.com6751d842014-01-12 02:51:25 +00003877{
simon.fraser@apple.com36676e52016-05-07 00:05:58 +00003878 LOG(TextAutosizing, "RenderBlockFlow %p adjustComputedFontSizes, size=%f visibleWidth=%f, width()=%f. Bailing: %d", this, size, visibleWidth, width().toFloat(), visibleWidth >= width());
3879
aestes@apple.com6751d842014-01-12 02:51:25 +00003880 // Don't do any work if the block is smaller than the visible area.
mmaxfield@apple.com59146a72019-05-29 02:27:20 +00003881 if (visibleWidth >= width())
aestes@apple.com6751d842014-01-12 02:51:25 +00003882 return;
3883
3884 unsigned lineCount;
3885 if (m_lineCountForTextAutosizing == NOT_SET) {
antti@apple.com0b3dffe2014-03-24 16:30:52 +00003886 int count = lineCountForTextAutosizing();
aestes@apple.com6751d842014-01-12 02:51:25 +00003887 if (!count)
3888 lineCount = NO_LINE;
3889 else if (count == 1)
3890 lineCount = ONE_LINE;
3891 else
3892 lineCount = MULTI_LINE;
3893 } else
3894 lineCount = m_lineCountForTextAutosizing;
3895
3896 ASSERT(lineCount != NOT_SET);
3897 if (lineCount == NO_LINE)
3898 return;
3899
3900 float actualWidth = m_widthForTextAutosizing != -1 ? static_cast<float>(m_widthForTextAutosizing) : static_cast<float>(width());
3901 float scale = visibleWidth / actualWidth;
3902 float minFontSize = roundf(size / scale);
cdumez@apple.com83102df2016-05-19 19:09:41 +00003903
3904 for (auto* descendant = RenderObjectTraversal::firstChild(*this); descendant; ) {
3905 if (!isNonBlocksOrNonFixedHeightListItems(*descendant)) {
3906 descendant = RenderObjectTraversal::nextSkippingChildren(*descendant, this);
3907 continue;
aestes@apple.com6751d842014-01-12 02:51:25 +00003908 }
cdumez@apple.com83102df2016-05-19 19:09:41 +00003909 if (!isVisibleRenderText(*descendant) || !resizeTextPermitted(*descendant)) {
3910 descendant = RenderObjectTraversal::next(*descendant, this);
3911 continue;
3912 }
3913
3914 auto& text = downcast<RenderText>(*descendant);
3915 auto& oldStyle = text.style();
3916 auto& fontDescription = oldStyle.fontDescription();
3917 float specifiedSize = fontDescription.specifiedSize();
3918 float scaledSize = roundf(specifiedSize * scale);
mmaxfield@apple.com59146a72019-05-29 02:27:20 +00003919 if (scaledSize > 0 && scaledSize < minFontSize) {
cdumez@apple.com83102df2016-05-19 19:09:41 +00003920 // Record the width of the block and the line count the first time we resize text and use it from then on for text resizing.
3921 // This makes text resizing consistent even if the block's width or line count changes (which can be caused by text resizing itself 5159915).
3922 if (m_lineCountForTextAutosizing == NOT_SET)
3923 m_lineCountForTextAutosizing = lineCount;
3924 if (m_widthForTextAutosizing == -1)
3925 m_widthForTextAutosizing = actualWidth;
3926
mmaxfield@apple.com59146a72019-05-29 02:27:20 +00003927 float lineTextMultiplier = lineCount == ONE_LINE ? oneLineTextMultiplier(text, specifiedSize) : textMultiplier(text, specifiedSize);
3928 float candidateNewSize = roundf(std::min(minFontSize, specifiedSize * lineTextMultiplier));
mmaxfield@apple.comac71afa2019-04-26 06:33:56 +00003929
commit-queue@webkit.orgf2872372019-04-26 18:32:41 +00003930 if (candidateNewSize > specifiedSize && candidateNewSize != fontDescription.computedSize() && text.textNode() && oldStyle.textSizeAdjust().isAuto())
antti@apple.com4918b4d2017-08-14 13:20:47 +00003931 document().textAutoSizing().addTextNode(*text.textNode(), candidateNewSize);
cdumez@apple.com83102df2016-05-19 19:09:41 +00003932 }
3933
3934 descendant = RenderObjectTraversal::nextSkippingChildren(text, this);
aestes@apple.com6751d842014-01-12 02:51:25 +00003935 }
3936}
darin@apple.com5917cb12017-11-23 17:32:42 +00003937
dbates@webkit.org102013c2016-09-26 21:51:25 +00003938#endif // ENABLE(TEXT_AUTOSIZING)
aestes@apple.com6751d842014-01-12 02:51:25 +00003939
hyatt@apple.com80914862017-03-06 18:00:35 +00003940void RenderBlockFlow::layoutExcludedChildren(bool relayoutChildren)
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003941{
hyatt@apple.com80914862017-03-06 18:00:35 +00003942 RenderBlock::layoutExcludedChildren(relayoutChildren);
3943
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003944 auto* fragmentedFlow = multiColumnFlow();
3945 if (!fragmentedFlow)
hyatt@apple.com80914862017-03-06 18:00:35 +00003946 return;
3947
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003948 fragmentedFlow->setIsExcludedFromNormalLayout(true);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003949
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003950 setLogicalTopForChild(*fragmentedFlow, borderAndPaddingBefore());
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003951
3952 if (relayoutChildren)
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003953 fragmentedFlow->setChildNeedsLayout(MarkOnlyThis);
hyatt@apple.comc1c39032014-04-15 23:25:58 +00003954
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003955 if (fragmentedFlow->needsLayout()) {
3956 for (RenderMultiColumnSet* columnSet = fragmentedFlow->firstMultiColumnSet(); columnSet; columnSet = columnSet->nextSiblingMultiColumnSet())
3957 columnSet->prepareForLayout(!fragmentedFlow->inBalancingPass());
hyatt@apple.comc1c39032014-04-15 23:25:58 +00003958
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003959 fragmentedFlow->invalidateFragments(MarkOnlyThis);
3960 fragmentedFlow->setNeedsHeightsRecalculation(true);
3961 fragmentedFlow->layout();
hyatt@apple.comc1c39032014-04-15 23:25:58 +00003962 } else {
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003963 // At the end of multicol layout, relayoutForPagination() is called unconditionally, but if
3964 // no children are to be laid out (e.g. fixed width with layout already being up-to-date),
3965 // we want to prevent it from doing any work, so that the column balancing machinery doesn't
3966 // kick in and trigger additional unnecessary layout passes. Actually, it's not just a good
3967 // idea in general to not waste time on balancing content that hasn't been re-laid out; we
3968 // are actually required to guarantee this. The calculation of implicit breaks needs to be
3969 // preceded by a proper layout pass, since it's layout that sets up content runs, and the
3970 // runs get deleted right after every pass.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003971 fragmentedFlow->setNeedsHeightsRecalculation(false);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003972 }
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003973 determineLogicalLeftPositionForChild(*fragmentedFlow);
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003974}
3975
hyatt@apple.com73715ca2014-05-06 21:35:52 +00003976void RenderBlockFlow::checkForPaginationLogicalHeightChange(bool& relayoutChildren, LayoutUnit& pageLogicalHeight, bool& pageLogicalHeightChanged)
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003977{
hyatt@apple.com73715ca2014-05-06 21:35:52 +00003978 // If we don't use columns or flow threads, then bail.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003979 if (!isRenderFragmentedFlow() && !multiColumnFlow())
hyatt@apple.comd4be3772014-01-24 19:55:33 +00003980 return;
3981
3982 // We don't actually update any of the variables. We just subclassed to adjust our column height.
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003983 if (RenderMultiColumnFlow* fragmentedFlow = multiColumnFlow()) {
zalan@apple.com53e79072016-12-12 20:17:50 +00003984 LayoutUnit newColumnHeight;
3985 if (hasDefiniteLogicalHeight() || view().frameView().pagination().mode != Pagination::Unpaginated) {
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00003986 auto computedValues = computeLogicalHeight(0_lu, logicalTop());
zalan@apple.com53e79072016-12-12 20:17:50 +00003987 newColumnHeight = std::max<LayoutUnit>(computedValues.m_extent - borderAndPaddingLogicalHeight() - scrollbarLogicalHeight(), 0);
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003988 if (fragmentedFlow->columnHeightAvailable() != newColumnHeight)
zalan@apple.com53e79072016-12-12 20:17:50 +00003989 relayoutChildren = true;
3990 }
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00003991 fragmentedFlow->setColumnHeightAvailable(newColumnHeight);
3992 } else if (is<RenderFragmentedFlow>(*this)) {
3993 RenderFragmentedFlow& fragmentedFlow = downcast<RenderFragmentedFlow>(*this);
commit-queue@webkit.org3d0f60b2014-04-08 18:19:47 +00003994
3995 // FIXME: This is a hack to always make sure we have a page logical height, if said height
antti@apple.comf41183e2018-12-07 20:10:14 +00003996 // is known. The page logical height thing in RenderLayoutState is meaningless for flow
commit-queue@webkit.org3d0f60b2014-04-08 18:19:47 +00003997 // thread-based pagination (page height isn't necessarily uniform throughout the flow
3998 // thread), but as long as it is used universally as a means to determine whether page
3999 // height is known or not, we need this. Page height is unknown when column balancing is
4000 // enabled and flow thread height is still unknown (i.e. during the first layout pass). When
4001 // it's unknown, we need to prevent the pagination code from assuming page breaks everywhere
4002 // and thereby eating every top margin. It should be trivial to clean up and get rid of this
zalan@apple.comfa5add62017-02-22 19:33:02 +00004003 // hack once the old multicol implementation is gone (see also RenderView::pushLayoutStateForPagination).
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00004004 pageLogicalHeight = fragmentedFlow.isPageLogicalHeightKnown() ? 1_lu : 0_lu;
commit-queue@webkit.org3d0f60b2014-04-08 18:19:47 +00004005
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004006 pageLogicalHeightChanged = fragmentedFlow.pageLogicalSizeChanged();
hyatt@apple.comd4be3772014-01-24 19:55:33 +00004007 }
4008}
4009
hyatt@apple.com73715ca2014-05-06 21:35:52 +00004010bool RenderBlockFlow::requiresColumns(int desiredColumnCount) const
zalan@apple.com809f4872016-12-08 18:20:01 +00004011{
4012 return willCreateColumns(desiredColumnCount);
hyatt@apple.com73715ca2014-05-06 21:35:52 +00004013}
4014
hyatt@apple.com39746fd2014-01-24 22:52:41 +00004015void RenderBlockFlow::setComputedColumnCountAndWidth(int count, LayoutUnit width)
4016{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004017 ASSERT(!!multiColumnFlow() == requiresColumns(count));
4018 if (!multiColumnFlow())
antti@apple.com411949d2017-08-30 17:28:10 +00004019 return;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004020 multiColumnFlow()->setColumnCountAndWidth(count, width);
4021 multiColumnFlow()->setProgressionIsInline(style().hasInlineColumnAxis());
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00004022 multiColumnFlow()->setProgressionIsReversed(style().columnProgression() == ColumnProgression::Reverse);
hyatt@apple.com39746fd2014-01-24 22:52:41 +00004023}
4024
cdumez@apple.com78141732014-11-04 23:00:48 +00004025void RenderBlockFlow::updateColumnProgressionFromStyle(RenderStyle& style)
hyatt@apple.com86919862014-01-27 16:27:45 +00004026{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004027 if (!multiColumnFlow())
hyatt@apple.com86919862014-01-27 16:27:45 +00004028 return;
4029
4030 bool needsLayout = false;
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004031 bool oldProgressionIsInline = multiColumnFlow()->progressionIsInline();
cdumez@apple.com78141732014-11-04 23:00:48 +00004032 bool newProgressionIsInline = style.hasInlineColumnAxis();
hyatt@apple.com86919862014-01-27 16:27:45 +00004033 if (oldProgressionIsInline != newProgressionIsInline) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004034 multiColumnFlow()->setProgressionIsInline(newProgressionIsInline);
hyatt@apple.com86919862014-01-27 16:27:45 +00004035 needsLayout = true;
4036 }
4037
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004038 bool oldProgressionIsReversed = multiColumnFlow()->progressionIsReversed();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00004039 bool newProgressionIsReversed = style.columnProgression() == ColumnProgression::Reverse;
hyatt@apple.com86919862014-01-27 16:27:45 +00004040 if (oldProgressionIsReversed != newProgressionIsReversed) {
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004041 multiColumnFlow()->setProgressionIsReversed(newProgressionIsReversed);
hyatt@apple.com86919862014-01-27 16:27:45 +00004042 needsLayout = true;
4043 }
4044
4045 if (needsLayout)
4046 setNeedsLayoutAndPrefWidthsRecalc();
4047}
4048
hyatt@apple.com39746fd2014-01-24 22:52:41 +00004049LayoutUnit RenderBlockFlow::computedColumnWidth() const
4050{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004051 if (multiColumnFlow())
4052 return multiColumnFlow()->computedColumnWidth();
hyatt@apple.com39746fd2014-01-24 22:52:41 +00004053 return contentLogicalWidth();
4054}
4055
4056unsigned RenderBlockFlow::computedColumnCount() const
4057{
hyatt@apple.com4e0bf862017-09-27 20:54:17 +00004058 if (multiColumnFlow())
4059 return multiColumnFlow()->computedColumnCount();
hyatt@apple.com39746fd2014-01-24 22:52:41 +00004060
4061 return 1;
4062}
4063
hyatt@apple.com31a5daa2014-01-28 01:26:37 +00004064bool RenderBlockFlow::isTopLayoutOverflowAllowed() const
4065{
4066 bool hasTopOverflow = RenderBlock::isTopLayoutOverflowAllowed();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00004067 if (!multiColumnFlow() || style().columnProgression() == ColumnProgression::Normal)
hyatt@apple.com31a5daa2014-01-28 01:26:37 +00004068 return hasTopOverflow;
4069
4070 if (!(isHorizontalWritingMode() ^ !style().hasInlineColumnAxis()))
4071 hasTopOverflow = !hasTopOverflow;
4072
4073 return hasTopOverflow;
4074}
4075
4076bool RenderBlockFlow::isLeftLayoutOverflowAllowed() const
4077{
4078 bool hasLeftOverflow = RenderBlock::isLeftLayoutOverflowAllowed();
commit-queue@webkit.orgeea2d6a2018-05-25 01:42:36 +00004079 if (!multiColumnFlow() || style().columnProgression() == ColumnProgression::Normal)
hyatt@apple.com31a5daa2014-01-28 01:26:37 +00004080 return hasLeftOverflow;
4081
4082 if (isHorizontalWritingMode() ^ !style().hasInlineColumnAxis())
4083 hasLeftOverflow = !hasLeftOverflow;
4084
4085 return hasLeftOverflow;
4086}
4087
zalan@apple.comac6956c2014-09-05 14:18:06 +00004088struct InlineMinMaxIterator {
4089/* InlineMinMaxIterator is a class that will iterate over all render objects that contribute to
4090 inline min/max width calculations. Note the following about the way it walks:
4091 (1) Positioned content is skipped (since it does not contribute to min/max width of a block)
4092 (2) We do not drill into the children of floats or replaced elements, since you can't break
4093 in the middle of such an element.
4094 (3) Inline flows (e.g., <a>, <span>, <i>) are walked twice, since each side can have
4095 distinct borders/margin/padding that contribute to the min/max width.
4096*/
4097 const RenderBlockFlow& parent;
4098 RenderObject* current;
4099 bool endOfInline;
4100 bool initial;
4101
4102 InlineMinMaxIterator(const RenderBlockFlow& p)
4103 : parent(p)
4104 , current(nullptr)
4105 , endOfInline(false)
4106 , initial(true)
4107 { }
4108
4109 RenderObject* next();
4110};
4111
4112RenderObject* InlineMinMaxIterator::next()
4113{
4114 RenderObject* result = nullptr;
4115 bool oldEndOfInline = endOfInline;
4116 endOfInline = false;
4117 do {
4118 if (!oldEndOfInline && (current && !current->isFloating() && !current->isReplaced() && !current->isOutOfFlowPositioned()))
4119 result = current->firstChildSlow();
4120 else if (initial) {
4121 result = parent.firstChild();
4122 initial = false;
4123 }
4124
4125 if (!result) {
4126 // We hit the end of our inline. (It was empty, e.g., <span></span>.)
4127 if (!oldEndOfInline && current && current->isRenderInline()) {
4128 result = current;
4129 endOfInline = true;
4130 break;
4131 }
4132
4133 while (current && current != &parent) {
4134 result = current->nextSibling();
4135 if (result)
4136 break;
4137 current = current->parent();
4138 if (current && current != &parent && current->isRenderInline()) {
4139 result = current;
4140 endOfInline = true;
4141 break;
4142 }
4143 }
4144 }
4145
4146 if (!result)
4147 break;
4148
4149 if (!result->isOutOfFlowPositioned() && (result->isTextOrLineBreak() || result->isFloating() || result->isReplaced() || result->isRenderInline()))
4150 break;
4151
4152 current = result;
4153 result = nullptr;
4154 } while (current || current == &parent);
4155 // Update our position.
4156 current = result;
4157 return result;
4158}
4159
4160static LayoutUnit getBPMWidth(LayoutUnit childValue, Length cssUnit)
4161{
4162 if (cssUnit.type() != Auto)
4163 return (cssUnit.isFixed() ? LayoutUnit(cssUnit.value()) : childValue);
4164 return 0;
4165}
4166
4167static LayoutUnit getBorderPaddingMargin(const RenderBoxModelObject& child, bool endOfInline)
4168{
4169 const RenderStyle& childStyle = child.style();
4170 if (endOfInline) {
4171 return getBPMWidth(child.marginEnd(), childStyle.marginEnd()) +
4172 getBPMWidth(child.paddingEnd(), childStyle.paddingEnd()) +
4173 child.borderEnd();
4174 }
4175 return getBPMWidth(child.marginStart(), childStyle.marginStart()) +
4176 getBPMWidth(child.paddingStart(), childStyle.paddingStart()) +
4177 child.borderStart();
4178}
4179
4180static inline void stripTrailingSpace(float& inlineMax, float& inlineMin, RenderObject* trailingSpaceChild)
4181{
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004182 if (is<RenderText>(trailingSpaceChild)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004183 // Collapse away the trailing space at the end of a block.
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004184 RenderText& renderText = downcast<RenderText>(*trailingSpaceChild);
zalan@apple.comac6956c2014-09-05 14:18:06 +00004185 const UChar space = ' ';
antti@apple.comc54cbc92015-01-15 14:19:56 +00004186 const FontCascade& font = renderText.style().fontCascade(); // FIXME: This ignores first-line.
mmaxfield@apple.com21a4dcb2016-03-13 00:36:59 +00004187 float spaceWidth = font.width(RenderBlock::constructTextRun(&space, 1, renderText.style()));
zalan@apple.comac6956c2014-09-05 14:18:06 +00004188 inlineMax -= spaceWidth + font.wordSpacing();
4189 if (inlineMin > inlineMax)
4190 inlineMin = inlineMax;
4191 }
4192}
4193
4194static inline LayoutUnit preferredWidth(LayoutUnit preferredWidth, float result)
4195{
4196 return std::max(preferredWidth, LayoutUnit::fromFloatCeil(result));
4197}
4198
4199void RenderBlockFlow::computeInlinePreferredLogicalWidths(LayoutUnit& minLogicalWidth, LayoutUnit& maxLogicalWidth) const
4200{
4201 float inlineMax = 0;
4202 float inlineMin = 0;
4203
4204 const RenderStyle& styleToUse = style();
4205 RenderBlock* containingBlock = this->containingBlock();
ross.kirsling@sony.coma10d10c2018-11-23 20:47:11 +00004206 LayoutUnit cw = containingBlock ? containingBlock->contentLogicalWidth() : 0_lu;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004207
4208 // If we are at the start of a line, we want to ignore all white-space.
4209 // Also strip spaces if we previously had text that ended in a trailing space.
4210 bool stripFrontSpaces = true;
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004211 RenderObject* trailingSpaceChild = nullptr;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004212
4213 // Firefox and Opera will allow a table cell to grow to fit an image inside it under
4214 // very specific cirucumstances (in order to match common WinIE renderings).
4215 // Not supporting the quirk has caused us to mis-render some real sites. (See Bugzilla 10517.)
4216 bool allowImagesToBreak = !document().inQuirksMode() || !isTableCell() || !styleToUse.logicalWidth().isIntrinsicOrAuto();
4217
cdumez@apple.comc28103d52014-10-31 23:25:05 +00004218 bool oldAutoWrap = styleToUse.autoWrap();
zalan@apple.comac6956c2014-09-05 14:18:06 +00004219
4220 InlineMinMaxIterator childIterator(*this);
4221
4222 // Only gets added to the max preffered width once.
4223 bool addedTextIndent = false;
4224 // Signals the text indent was more negative than the min preferred width
4225 bool hasRemainingNegativeTextIndent = false;
4226
4227 LayoutUnit textIndent = minimumValueForLength(styleToUse.textIndent(), cw);
4228 RenderObject* prevFloat = 0;
4229 bool isPrevChildInlineFlow = false;
4230 bool shouldBreakLineAfterText = false;
commit-queue@webkit.org02cb0002018-05-28 00:49:21 +00004231 bool canHangPunctuationAtStart = styleToUse.hangingPunctuation().contains(HangingPunctuation::First);
4232 bool canHangPunctuationAtEnd = styleToUse.hangingPunctuation().contains(HangingPunctuation::Last);
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004233 RenderText* lastText = nullptr;
4234
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004235 bool addedStartPunctuationHang = false;
4236
zalan@apple.comac6956c2014-09-05 14:18:06 +00004237 while (RenderObject* child = childIterator.next()) {
cdumez@apple.comc28103d52014-10-31 23:25:05 +00004238 bool autoWrap = child->isReplaced() ? child->parent()->style().autoWrap() :
zalan@apple.comac6956c2014-09-05 14:18:06 +00004239 child->style().autoWrap();
zalan@apple.comac6956c2014-09-05 14:18:06 +00004240 if (!child->isBR()) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004241 // Step One: determine whether or not we need to terminate our current line.
4242 // Each discrete chunk can become the new min-width, if it is the widest chunk
4243 // seen so far, and it can also become the max-width.
zalan@apple.comac6956c2014-09-05 14:18:06 +00004244
4245 // Children fall into three categories:
4246 // (1) An inline flow object. These objects always have a min/max of 0,
4247 // and are included in the iteration solely so that their margins can
4248 // be added in.
4249 //
4250 // (2) An inline non-text non-flow object, e.g., an inline replaced element.
4251 // These objects can always be on a line by themselves, so in this situation
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004252 // we need to break the current line, and then add in our own margins and min/max
4253 // width on its own line, and then terminate the line.
zalan@apple.comac6956c2014-09-05 14:18:06 +00004254 //
4255 // (3) A text object. Text runs can have breakable characters at the start,
4256 // the middle or the end. They may also lose whitespace off the front if
4257 // we're already ignoring whitespace. In order to compute accurate min-width
4258 // information, we need three pieces of information.
4259 // (a) the min-width of the first non-breakable run. Should be 0 if the text string
4260 // starts with whitespace.
4261 // (b) the min-width of the last non-breakable run. Should be 0 if the text string
4262 // ends with whitespace.
4263 // (c) the min/max width of the string (trimmed for whitespace).
4264 //
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004265 // If the text string starts with whitespace, then we need to terminate our current line
4266 // (unless we're already in a whitespace stripping mode.
zalan@apple.comac6956c2014-09-05 14:18:06 +00004267 //
4268 // If the text string has a breakable character in the middle, but didn't start
4269 // with whitespace, then we add the width of the first non-breakable run and
4270 // then end the current line. We then need to use the intermediate min/max width
4271 // values (if any of them are larger than our current min/max). We then look at
4272 // the width of the last non-breakable run and use that to start a new line
4273 // (unless we end in whitespace).
4274 const RenderStyle& childStyle = child->style();
4275 float childMin = 0;
4276 float childMax = 0;
4277
4278 if (!child->isText()) {
4279 if (child->isLineBreakOpportunity()) {
4280 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4281 inlineMin = 0;
4282 continue;
4283 }
4284 // Case (1) and (2). Inline replaced and inline flow elements.
cdumez@apple.comf8022152014-10-15 00:29:51 +00004285 if (is<RenderInline>(*child)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004286 // Add in padding/border/margin from the appropriate side of
4287 // the element.
cdumez@apple.comf8022152014-10-15 00:29:51 +00004288 float bpm = getBorderPaddingMargin(downcast<RenderInline>(*child), childIterator.endOfInline);
zalan@apple.comac6956c2014-09-05 14:18:06 +00004289 childMin += bpm;
4290 childMax += bpm;
4291
4292 inlineMin += childMin;
4293 inlineMax += childMax;
4294
4295 child->setPreferredLogicalWidthsDirty(false);
4296 } else {
4297 // Inline replaced elts add in their margins to their min/max values.
hyatt@apple.com14520e42016-04-20 18:01:40 +00004298 if (!child->isFloating())
4299 lastText = nullptr;
ross.kirsling@sony.combd744282018-11-18 03:14:31 +00004300 LayoutUnit margins;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004301 Length startMargin = childStyle.marginStart();
4302 Length endMargin = childStyle.marginEnd();
4303 if (startMargin.isFixed())
4304 margins += LayoutUnit::fromFloatCeil(startMargin.value());
4305 if (endMargin.isFixed())
4306 margins += LayoutUnit::fromFloatCeil(endMargin.value());
4307 childMin += margins.ceilToFloat();
4308 childMax += margins.ceilToFloat();
4309 }
4310 }
4311
cdumez@apple.comf8022152014-10-15 00:29:51 +00004312 if (!is<RenderInline>(*child) && !is<RenderText>(*child)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004313 // Case (2). Inline replaced elements and floats.
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004314 // Terminate the current line as far as minwidth is concerned.
hyatt@apple.com247170f2017-02-28 16:23:15 +00004315 LayoutUnit childMinPreferredLogicalWidth, childMaxPreferredLogicalWidth;
4316 computeChildPreferredLogicalWidths(*child, childMinPreferredLogicalWidth, childMaxPreferredLogicalWidth);
4317 childMin += childMinPreferredLogicalWidth.ceilToFloat();
4318 childMax += childMaxPreferredLogicalWidth.ceilToFloat();
zalan@apple.comac6956c2014-09-05 14:18:06 +00004319
4320 bool clearPreviousFloat;
4321 if (child->isFloating()) {
4322 clearPreviousFloat = (prevFloat
commit-queue@webkit.org1a4e6672018-05-21 16:55:45 +00004323 && ((prevFloat->style().floating() == Float::Left && (childStyle.clear() == Clear::Left || childStyle.clear() == Clear::Both))
4324 || (prevFloat->style().floating() == Float::Right && (childStyle.clear() == Clear::Right || childStyle.clear() == Clear::Both))));
zalan@apple.comac6956c2014-09-05 14:18:06 +00004325 prevFloat = child;
4326 } else
4327 clearPreviousFloat = false;
4328
4329 bool canBreakReplacedElement = !child->isImage() || allowImagesToBreak;
antti@apple.comae85e112017-08-31 23:27:02 +00004330 if (((canBreakReplacedElement && (autoWrap || oldAutoWrap) && (!isPrevChildInlineFlow || shouldBreakLineAfterText)) || clearPreviousFloat)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004331 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4332 inlineMin = 0;
4333 }
4334
4335 // If we're supposed to clear the previous float, then terminate maxwidth as well.
antti@apple.comae85e112017-08-31 23:27:02 +00004336 if (clearPreviousFloat) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004337 maxLogicalWidth = preferredWidth(maxLogicalWidth, inlineMax);
4338 inlineMax = 0;
4339 }
4340
4341 // Add in text-indent. This is added in only once.
antti@apple.comae85e112017-08-31 23:27:02 +00004342 if (!addedTextIndent && !child->isFloating()) {
ross.kirsling@sony.com80414652019-05-21 01:36:11 +00004343 LayoutUnit ceiledIndent { textIndent.ceilToFloat() };
zalan@apple.comac6956c2014-09-05 14:18:06 +00004344 childMin += ceiledIndent;
4345 childMax += ceiledIndent;
4346
4347 if (childMin < 0)
4348 textIndent = LayoutUnit::fromFloatCeil(childMin);
4349 else
4350 addedTextIndent = true;
4351 }
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004352
antti@apple.comae85e112017-08-31 23:27:02 +00004353 if (canHangPunctuationAtStart && !addedStartPunctuationHang && !child->isFloating())
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004354 addedStartPunctuationHang = true;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004355
4356 // Add our width to the max.
4357 inlineMax += std::max<float>(0, childMax);
4358
antti@apple.comae85e112017-08-31 23:27:02 +00004359 if ((!autoWrap || !canBreakReplacedElement || (isPrevChildInlineFlow && !shouldBreakLineAfterText))) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004360 if (child->isFloating())
4361 minLogicalWidth = preferredWidth(minLogicalWidth, childMin);
4362 else
4363 inlineMin += childMin;
4364 } else {
4365 // Now check our line.
4366 minLogicalWidth = preferredWidth(minLogicalWidth, childMin);
4367
4368 // Now start a new line.
antti@apple.comae85e112017-08-31 23:27:02 +00004369 inlineMin = 0;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004370 }
4371
4372 if (autoWrap && canBreakReplacedElement && isPrevChildInlineFlow) {
4373 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4374 inlineMin = 0;
4375 }
4376
4377 // We are no longer stripping whitespace at the start of a line.
4378 if (!child->isFloating()) {
4379 stripFrontSpaces = false;
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004380 trailingSpaceChild = nullptr;
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004381 lastText = nullptr;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004382 }
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004383 } else if (is<RenderText>(*child)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004384 // Case (3). Text.
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004385 RenderText& renderText = downcast<RenderText>(*child);
zalan@apple.comac6956c2014-09-05 14:18:06 +00004386
cdumez@apple.com35094bd2014-10-07 19:33:53 +00004387 if (renderText.style().hasTextCombine() && renderText.isCombineText())
zalan@apple.com6cad78f2017-09-19 20:28:29 +00004388 downcast<RenderCombineText>(renderText).combineTextIfNeeded();
zalan@apple.comac6956c2014-09-05 14:18:06 +00004389
4390 // Determine if we have a breakable character. Pass in
4391 // whether or not we should ignore any spaces at the front
4392 // of the string. If those are going to be stripped out,
4393 // then they shouldn't be considered in the breakable char
4394 // check.
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004395 bool strippingBeginWS = stripFrontSpaces;
darin@apple.com5917cb12017-11-23 17:32:42 +00004396 auto widths = renderText.trimmedPreferredWidths(inlineMax, stripFrontSpaces);
4397
4398 childMin = widths.min;
4399 childMax = widths.max;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004400
4401 // This text object will not be rendered, but it may still provide a breaking opportunity.
darin@apple.com5917cb12017-11-23 17:32:42 +00004402 if (!widths.hasBreak && !childMax) {
4403 if (autoWrap && (widths.beginWS || widths.endWS)) {
zalan@apple.comac6956c2014-09-05 14:18:06 +00004404 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4405 inlineMin = 0;
4406 }
4407 continue;
4408 }
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004409
4410 lastText = &renderText;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004411
4412 if (stripFrontSpaces)
4413 trailingSpaceChild = child;
4414 else
4415 trailingSpaceChild = 0;
4416
4417 // Add in text-indent. This is added in only once.
4418 float ti = 0;
4419 if (!addedTextIndent || hasRemainingNegativeTextIndent) {
4420 ti = textIndent.ceilToFloat();
4421 childMin += ti;
darin@apple.com5917cb12017-11-23 17:32:42 +00004422 widths.beginMin += ti;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004423
4424 // It the text indent negative and larger than the child minimum, we re-use the remainder
4425 // in future minimum calculations, but using the negative value again on the maximum
4426 // will lead to under-counting the max pref width.
4427 if (!addedTextIndent) {
4428 childMax += ti;
darin@apple.com5917cb12017-11-23 17:32:42 +00004429 widths.beginMax += ti;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004430 addedTextIndent = true;
4431 }
4432
4433 if (childMin < 0) {
4434 textIndent = childMin;
4435 hasRemainingNegativeTextIndent = true;
4436 }
4437 }
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004438
4439 // See if we have a hanging punctuation situation at the start.
4440 if (canHangPunctuationAtStart && !addedStartPunctuationHang) {
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004441 unsigned startIndex = strippingBeginWS ? renderText.firstCharacterIndexStrippingSpaces() : 0;
4442 float hangStartWidth = renderText.hangablePunctuationStartWidth(startIndex);
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004443 childMin -= hangStartWidth;
darin@apple.com5917cb12017-11-23 17:32:42 +00004444 widths.beginMin -= hangStartWidth;
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004445 childMax -= hangStartWidth;
darin@apple.com5917cb12017-11-23 17:32:42 +00004446 widths.beginMax -= hangStartWidth;
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004447 addedStartPunctuationHang = true;
4448 }
4449
zalan@apple.comac6956c2014-09-05 14:18:06 +00004450 // If we have no breakable characters at all,
4451 // then this is the easy case. We add ourselves to the current
4452 // min and max and continue.
darin@apple.com5917cb12017-11-23 17:32:42 +00004453 if (!widths.hasBreakableChar)
zalan@apple.comac6956c2014-09-05 14:18:06 +00004454 inlineMin += childMin;
4455 else {
4456 // We have a breakable character. Now we need to know if
4457 // we start and end with whitespace.
darin@apple.com5917cb12017-11-23 17:32:42 +00004458 if (widths.beginWS) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004459 // End the current line.
zalan@apple.comac6956c2014-09-05 14:18:06 +00004460 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4461 } else {
darin@apple.com5917cb12017-11-23 17:32:42 +00004462 inlineMin += widths.beginMin;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004463 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4464 childMin -= ti;
4465 }
4466
4467 inlineMin = childMin;
4468
darin@apple.com5917cb12017-11-23 17:32:42 +00004469 if (widths.endWS) {
simon.fraser@apple.com03e61032015-04-05 20:17:11 +00004470 // We end in whitespace, which means we can end our current line.
zalan@apple.comac6956c2014-09-05 14:18:06 +00004471 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4472 inlineMin = 0;
4473 shouldBreakLineAfterText = false;
4474 } else {
4475 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
darin@apple.com5917cb12017-11-23 17:32:42 +00004476 inlineMin = widths.endMin;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004477 shouldBreakLineAfterText = true;
4478 }
4479 }
4480
darin@apple.com5917cb12017-11-23 17:32:42 +00004481 if (widths.hasBreak) {
4482 inlineMax += widths.beginMax;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004483 maxLogicalWidth = preferredWidth(maxLogicalWidth, inlineMax);
4484 maxLogicalWidth = preferredWidth(maxLogicalWidth, childMax);
darin@apple.com5917cb12017-11-23 17:32:42 +00004485 inlineMax = widths.endMax;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004486 addedTextIndent = true;
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004487 addedStartPunctuationHang = true;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004488 } else
4489 inlineMax += std::max<float>(0, childMax);
4490 }
4491
antti@apple.comae85e112017-08-31 23:27:02 +00004492 // Ignore spaces after a list marker.
4493 if (child->isListMarker())
zalan@apple.comac6956c2014-09-05 14:18:06 +00004494 stripFrontSpaces = true;
4495 } else {
4496 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4497 maxLogicalWidth = preferredWidth(maxLogicalWidth, inlineMax);
4498 inlineMin = inlineMax = 0;
4499 stripFrontSpaces = true;
4500 trailingSpaceChild = 0;
4501 addedTextIndent = true;
hyatt@apple.com41c12c92016-03-02 22:29:26 +00004502 addedStartPunctuationHang = true;
zalan@apple.comac6956c2014-09-05 14:18:06 +00004503 }
4504
4505 if (!child->isText() && child->isRenderInline())
4506 isPrevChildInlineFlow = true;
4507 else
4508 isPrevChildInlineFlow = false;
4509
4510 oldAutoWrap = autoWrap;
4511 }
4512
4513 if (styleToUse.collapseWhiteSpace())
4514 stripTrailingSpace(inlineMax, inlineMin, trailingSpaceChild);
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004515
darin@apple.com5917cb12017-11-23 17:32:42 +00004516 if (canHangPunctuationAtEnd && lastText && lastText->text().length() > 0) {
4517 unsigned endIndex = trailingSpaceChild == lastText ? lastText->lastCharacterIndexStrippingSpaces() : lastText->text().length() - 1;
hyatt@apple.com6b422a72016-03-03 21:49:32 +00004518 float endHangWidth = lastText->hangablePunctuationEndWidth(endIndex);
4519 inlineMin -= endHangWidth;
4520 inlineMax -= endHangWidth;
4521 }
zalan@apple.comac6956c2014-09-05 14:18:06 +00004522
4523 minLogicalWidth = preferredWidth(minLogicalWidth, inlineMin);
4524 maxLogicalWidth = preferredWidth(maxLogicalWidth, inlineMax);
4525}
4526
hyatt@apple.come96ebaf2013-11-12 22:51:12 +00004527}
4528// namespace WebCore