Factor complex line layout path out from RenderBlockFlow
[WebKit-https.git] / Source / WebCore / rendering / line / LineInlineHeaders.h
1 /*
2  * Copyright (C) 2000 Lars Knoll (knoll@kde.org)
3  * Copyright (C) 2003, 2004, 2006, 2007, 2008, 2009, 2010, 2011 Apple Inc. All right reserved.
4  * Copyright (C) 2010 Google Inc. All rights reserved.
5  * Copyright (C) 2013 ChangSeok Oh <shivamidow@gmail.com>
6  * Copyright (C) 2013 Adobe Systems Inc. All right 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
25 #pragma once
26
27 #include "LineInfo.h"
28 #include "RenderLayer.h"
29
30 namespace WebCore {
31
32 enum WhitespacePosition { LeadingWhitespace, TrailingWhitespace };
33
34 inline bool hasInlineDirectionBordersPaddingOrMargin(const RenderInline& flow)
35 {
36     // Where an empty inline is split across anonymous blocks we should only give lineboxes to the 'sides' of the
37     // inline that have borders, padding or margin.
38     bool shouldApplyStartBorderPaddingOrMargin = !flow.parent()->isAnonymousBlock() || !flow.isContinuation();
39     if (shouldApplyStartBorderPaddingOrMargin && (flow.borderStart() || flow.marginStart() || flow.paddingStart()))
40         return true;
41
42     bool shouldApplyEndBorderPaddingOrMargin = !flow.parent()->isAnonymousBlock() || flow.isContinuation() || !flow.inlineContinuation();
43     return shouldApplyEndBorderPaddingOrMargin && (flow.borderEnd() || flow.marginEnd() || flow.paddingEnd());
44 }
45
46 inline const RenderStyle& lineStyle(const RenderObject& renderer, const LineInfo& lineInfo)
47 {
48     return lineInfo.isFirstLine() ? renderer.firstLineStyle() : renderer.style();
49 }
50
51 inline bool requiresLineBoxForContent(const RenderInline& flow, const LineInfo& lineInfo)
52 {
53     RenderElement* parent = flow.parent();
54     if (flow.document().inNoQuirksMode()) {
55         const RenderStyle& flowStyle = lineStyle(flow, lineInfo);
56         const RenderStyle& parentStyle = lineStyle(*parent, lineInfo);
57         if (flowStyle.lineHeight() != parentStyle.lineHeight()
58             || flowStyle.verticalAlign() != parentStyle.verticalAlign()
59             || !parentStyle.fontCascade().fontMetrics().hasIdenticalAscentDescentAndLineGap(flowStyle.fontCascade().fontMetrics()))
60         return true;
61     }
62     return false;
63 }
64
65 inline bool shouldCollapseWhiteSpace(const RenderStyle* style, const LineInfo& lineInfo, WhitespacePosition whitespacePosition)
66 {
67     // CSS2 16.6.1
68     // If a space (U+0020) at the beginning of a line has 'white-space' set to 'normal', 'nowrap', or 'pre-line', it is removed.
69     // If a space (U+0020) at the end of a line has 'white-space' set to 'normal', 'nowrap', or 'pre-line', it is also removed.
70     // If spaces (U+0020) or tabs (U+0009) at the end of a line have 'white-space' set to 'pre-wrap', UAs may visually collapse them.
71     return style->collapseWhiteSpace()
72         || (whitespacePosition == TrailingWhitespace && style->whiteSpace() == WhiteSpace::PreWrap && (!lineInfo.isEmpty() || !lineInfo.previousLineBrokeCleanly()));
73 }
74
75 inline bool skipNonBreakingSpace(const InlineIterator& it, const LineInfo& lineInfo)
76 {
77     if (it.renderer()->style().nbspMode() != NBSPMode::Space || it.current() != noBreakSpace)
78         return false;
79
80     // FIXME: This is bad. It makes nbsp inconsistent with space and won't work correctly
81     // with m_minWidth/m_maxWidth.
82     // Do not skip a non-breaking space if it is the first character
83     // on a line after a clean line break (or on the first line, since previousLineBrokeCleanly starts off
84     // |true|).
85     if (lineInfo.isEmpty() && lineInfo.previousLineBrokeCleanly())
86         return false;
87
88     return true;
89 }
90
91 inline bool alwaysRequiresLineBox(const RenderInline& flow)
92 {
93     // FIXME: Right now, we only allow line boxes for inlines that are truly empty.
94     // We need to fix this, though, because at the very least, inlines containing only
95     // ignorable whitespace should should also have line boxes.
96     return isEmptyInline(flow) && hasInlineDirectionBordersPaddingOrMargin(flow);
97 }
98
99 inline bool requiresLineBox(const InlineIterator& it, const LineInfo& lineInfo = LineInfo(), WhitespacePosition whitespacePosition = LeadingWhitespace)
100 {
101     if (it.renderer()->isFloatingOrOutOfFlowPositioned())
102         return false;
103
104     if (it.renderer()->isBR())
105         return true;
106
107     bool rendererIsEmptyInline = false;
108     if (is<RenderInline>(*it.renderer())) {
109         const auto& inlineRenderer = downcast<RenderInline>(*it.renderer());
110         if (!alwaysRequiresLineBox(inlineRenderer) && !requiresLineBoxForContent(inlineRenderer, lineInfo))
111             return false;
112         rendererIsEmptyInline = isEmptyInline(inlineRenderer);
113     }
114
115     if (!shouldCollapseWhiteSpace(&it.renderer()->style(), lineInfo, whitespacePosition))
116         return true;
117
118     UChar current = it.current();
119     bool notJustWhitespace = current != ' ' && current != '\t' && current != softHyphen && (current != '\n' || it.renderer()->preservesNewline()) && !skipNonBreakingSpace(it, lineInfo);
120     return notJustWhitespace || rendererIsEmptyInline;
121 }
122
123 inline void setStaticPositions(RenderBlockFlow& block, RenderBox& child, IndentTextOrNot shouldIndentText)
124 {
125     // FIXME: The math here is actually not really right. It's a best-guess approximation that
126     // will work for the common cases
127     RenderElement* containerBlock = child.container();
128     LayoutUnit blockHeight = block.logicalHeight();
129     if (is<RenderInline>(*containerBlock)) {
130         // A relative positioned inline encloses us. In this case, we also have to determine our
131         // position as though we were an inline. Set |staticInlinePosition| and |staticBlockPosition| on the relative positioned
132         // inline so that we can obtain the value later.
133         downcast<RenderInline>(*containerBlock).layer()->setStaticInlinePosition(block.complexLineLayout().startAlignedOffsetForLine(blockHeight, DoNotIndentText));
134         downcast<RenderInline>(*containerBlock).layer()->setStaticBlockPosition(blockHeight);
135     }
136     block.updateStaticInlinePositionForChild(child, blockHeight, shouldIndentText);
137     child.layer()->setStaticBlockPosition(blockHeight);
138 }
139
140 } // namespace WebCore