mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
805 lines
30 KiB
C++
805 lines
30 KiB
C++
/*
|
|
* Copyright (C) 2003, 2006, 2008 Apple Inc. All rights reserved.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public License
|
|
* along with this library; see the file COPYING.LIB. If not, write to
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "flutter/sky/engine/core/rendering/RootInlineBox.h"
|
|
|
|
#include "flutter/sky/engine/core/rendering/HitTestResult.h"
|
|
#include "flutter/sky/engine/core/rendering/InlineTextBox.h"
|
|
#include "flutter/sky/engine/core/rendering/PaintInfo.h"
|
|
#include "flutter/sky/engine/core/rendering/RenderInline.h"
|
|
#include "flutter/sky/engine/core/rendering/RenderParagraph.h"
|
|
#include "flutter/sky/engine/core/rendering/RenderView.h"
|
|
#include "flutter/sky/engine/core/rendering/VerticalPositionCache.h"
|
|
#include "flutter/sky/engine/platform/text/BidiResolver.h"
|
|
#include "flutter/sky/engine/wtf/unicode/Unicode.h"
|
|
|
|
namespace blink {
|
|
|
|
struct SameSizeAsRootInlineBox : public InlineFlowBox {
|
|
unsigned unsignedVariable;
|
|
void* pointers[2];
|
|
LayoutUnit layoutVariables[5];
|
|
};
|
|
|
|
COMPILE_ASSERT(sizeof(RootInlineBox) == sizeof(SameSizeAsRootInlineBox),
|
|
RootInlineBox_should_stay_small);
|
|
|
|
RootInlineBox::RootInlineBox(RenderParagraph& block)
|
|
: InlineFlowBox(block),
|
|
m_lineBreakPos(0),
|
|
m_lineBreakObj(0),
|
|
m_lineTop(0),
|
|
m_lineBottom(0),
|
|
m_lineTopWithLeading(0),
|
|
m_lineBottomWithLeading(0),
|
|
m_selectionBottom(0) {}
|
|
|
|
void RootInlineBox::destroy() {
|
|
InlineFlowBox::destroy();
|
|
}
|
|
|
|
RenderLineBoxList* RootInlineBox::rendererLineBoxes() const {
|
|
return block().lineBoxes();
|
|
}
|
|
|
|
void RootInlineBox::clearTruncation() {}
|
|
|
|
int RootInlineBox::baselinePosition(FontBaseline baselineType) const {
|
|
return boxModelObject()->baselinePosition(baselineType, isFirstLineStyle(),
|
|
HorizontalLine,
|
|
PositionOfInteriorLineBoxes);
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::lineHeight() const {
|
|
return boxModelObject()->lineHeight(isFirstLineStyle(), HorizontalLine,
|
|
PositionOfInteriorLineBoxes);
|
|
}
|
|
|
|
void RootInlineBox::paint(PaintInfo& paintInfo,
|
|
const LayoutPoint& paintOffset,
|
|
LayoutUnit lineTop,
|
|
LayoutUnit lineBottom,
|
|
Vector<RenderBox*>& layers) {
|
|
InlineFlowBox::paint(paintInfo, paintOffset, lineTop, lineBottom, layers);
|
|
}
|
|
|
|
bool RootInlineBox::nodeAtPoint(const HitTestRequest& request,
|
|
HitTestResult& result,
|
|
const HitTestLocation& locationInContainer,
|
|
const LayoutPoint& accumulatedOffset,
|
|
LayoutUnit lineTop,
|
|
LayoutUnit lineBottom) {
|
|
return InlineFlowBox::nodeAtPoint(request, result, locationInContainer,
|
|
accumulatedOffset, lineTop, lineBottom);
|
|
}
|
|
|
|
void RootInlineBox::adjustPosition(float dx, float dy) {
|
|
InlineFlowBox::adjustPosition(dx, dy);
|
|
LayoutUnit blockDirectionDelta =
|
|
dy; // The block direction delta is a LayoutUnit.
|
|
m_lineTop += blockDirectionDelta;
|
|
m_lineBottom += blockDirectionDelta;
|
|
m_lineTopWithLeading += blockDirectionDelta;
|
|
m_lineBottomWithLeading += blockDirectionDelta;
|
|
m_selectionBottom += blockDirectionDelta;
|
|
}
|
|
|
|
void RootInlineBox::childRemoved(InlineBox* box) {
|
|
if (&box->renderer() == m_lineBreakObj)
|
|
setLineBreakInfo(0, 0, BidiStatus());
|
|
|
|
for (RootInlineBox* prev = prevRootBox();
|
|
prev && prev->lineBreakObj() == &box->renderer();
|
|
prev = prev->prevRootBox()) {
|
|
prev->setLineBreakInfo(0, 0, BidiStatus());
|
|
prev->markDirty();
|
|
}
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::alignBoxesInBlockDirection(
|
|
LayoutUnit heightOfBlock,
|
|
GlyphOverflowAndFallbackFontsMap& textBoxDataMap,
|
|
VerticalPositionCache& verticalPositionCache) {
|
|
// SVG will handle vertical alignment on its own.
|
|
if (isSVGRootInlineBox())
|
|
return 0;
|
|
|
|
LayoutUnit maxPositionTop = 0;
|
|
LayoutUnit maxPositionBottom = 0;
|
|
int maxAscent = 0;
|
|
int maxDescent = 0;
|
|
bool setMaxAscent = false;
|
|
bool setMaxDescent = false;
|
|
|
|
m_baselineType = AlphabeticBaseline;
|
|
|
|
computeLogicalBoxHeights(this, maxPositionTop, maxPositionBottom, maxAscent,
|
|
maxDescent, setMaxAscent, setMaxDescent, true,
|
|
textBoxDataMap, baselineType(),
|
|
verticalPositionCache);
|
|
|
|
if (maxAscent + maxDescent < std::max(maxPositionTop, maxPositionBottom))
|
|
adjustMaxAscentAndDescent(maxAscent, maxDescent, maxPositionTop,
|
|
maxPositionBottom);
|
|
|
|
LayoutUnit maxHeight = maxAscent + maxDescent;
|
|
LayoutUnit lineTop = heightOfBlock;
|
|
LayoutUnit lineBottom = heightOfBlock;
|
|
LayoutUnit lineTopIncludingMargins = heightOfBlock;
|
|
LayoutUnit lineBottomIncludingMargins = heightOfBlock;
|
|
LayoutUnit selectionBottom = heightOfBlock;
|
|
bool setLineTop = false;
|
|
bool hasAnnotationsBefore = false;
|
|
bool hasAnnotationsAfter = false;
|
|
placeBoxesInBlockDirection(heightOfBlock, maxHeight, maxAscent, true, lineTop,
|
|
lineBottom, selectionBottom, setLineTop,
|
|
lineTopIncludingMargins,
|
|
lineBottomIncludingMargins, hasAnnotationsBefore,
|
|
hasAnnotationsAfter, baselineType());
|
|
m_hasAnnotationsBefore = hasAnnotationsBefore;
|
|
m_hasAnnotationsAfter = hasAnnotationsAfter;
|
|
|
|
maxHeight =
|
|
std::max<LayoutUnit>(0, maxHeight); // FIXME: Is this really necessary?
|
|
|
|
setLineTopBottomPositions(lineTop, lineBottom, heightOfBlock,
|
|
heightOfBlock + maxHeight, selectionBottom);
|
|
|
|
LayoutUnit annotationsAdjustment = beforeAnnotationsAdjustment();
|
|
if (annotationsAdjustment) {
|
|
// FIXME: Need to handle pagination here. We might have to move to the next
|
|
// page/column as a result of the ruby expansion.
|
|
adjustBlockDirectionPosition(annotationsAdjustment.toFloat());
|
|
heightOfBlock += annotationsAdjustment;
|
|
}
|
|
|
|
return heightOfBlock + maxHeight;
|
|
}
|
|
|
|
float RootInlineBox::maxLogicalTop() const {
|
|
float maxLogicalTop = 0;
|
|
computeMaxLogicalTop(maxLogicalTop);
|
|
return maxLogicalTop;
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::beforeAnnotationsAdjustment() const {
|
|
LayoutUnit result = 0;
|
|
|
|
// Annotations under the previous line may push us down.
|
|
if (prevRootBox() && prevRootBox()->hasAnnotationsAfter())
|
|
result = prevRootBox()->computeUnderAnnotationAdjustment(lineTop());
|
|
|
|
if (!hasAnnotationsBefore())
|
|
return result;
|
|
|
|
// Annotations over this line may push us further down.
|
|
LayoutUnit highestAllowedPosition =
|
|
prevRootBox() ? std::min(prevRootBox()->lineBottom(), lineTop()) + result
|
|
: static_cast<LayoutUnit>(block().borderBefore());
|
|
result = computeOverAnnotationAdjustment(highestAllowedPosition);
|
|
|
|
return result;
|
|
}
|
|
|
|
GapRects RootInlineBox::lineSelectionGap(
|
|
RenderBlock* rootBlock,
|
|
const LayoutPoint& rootBlockPhysicalPosition,
|
|
const LayoutSize& offsetFromRootBlock,
|
|
LayoutUnit selTop,
|
|
LayoutUnit selHeight,
|
|
const PaintInfo* paintInfo) {
|
|
RenderObject::SelectionState lineState = selectionState();
|
|
|
|
bool leftGap, rightGap;
|
|
block().getSelectionGapInfo(lineState, leftGap, rightGap);
|
|
|
|
GapRects result;
|
|
|
|
InlineBox* firstBox = firstSelectedBox();
|
|
InlineBox* lastBox = lastSelectedBox();
|
|
if (leftGap) {
|
|
result.uniteLeft(block().logicalLeftSelectionGap(
|
|
rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock,
|
|
&firstBox->parent()->renderer(), firstBox->logicalLeft(), selTop,
|
|
selHeight, paintInfo));
|
|
}
|
|
if (rightGap) {
|
|
result.uniteRight(block().logicalRightSelectionGap(
|
|
rootBlock, rootBlockPhysicalPosition, offsetFromRootBlock,
|
|
&lastBox->parent()->renderer(), lastBox->logicalRight(), selTop,
|
|
selHeight, paintInfo));
|
|
}
|
|
|
|
// When dealing with bidi text, a non-contiguous selection region is possible.
|
|
// e.g. The logical text aaaAAAbbb (capitals denote RTL text and non-capitals
|
|
// LTR) is layed out visually as 3 text runs |aaa|bbb|AAA| if we select 4
|
|
// characters from the start of the text the selection will look like
|
|
// (underline denotes selection): |aaa|bbb|AAA|
|
|
// ___ _
|
|
// We can see that the |bbb| run is not part of the selection while the runs
|
|
// around it are.
|
|
if (firstBox && firstBox != lastBox) {
|
|
// Now fill in any gaps on the line that occurred between two selected
|
|
// elements.
|
|
LayoutUnit lastLogicalLeft = firstBox->logicalRight();
|
|
bool isPreviousBoxSelected =
|
|
firstBox->selectionState() != RenderObject::SelectionNone;
|
|
for (InlineBox* box = firstBox->nextLeafChild(); box;
|
|
box = box->nextLeafChild()) {
|
|
if (box->selectionState() != RenderObject::SelectionNone) {
|
|
LayoutRect logicalRect(lastLogicalLeft, selTop,
|
|
box->logicalLeft() - lastLogicalLeft, selHeight);
|
|
logicalRect.move(offsetFromRootBlock);
|
|
LayoutRect gapRect = rootBlock->logicalRectToPhysicalRect(
|
|
rootBlockPhysicalPosition, logicalRect);
|
|
if (isPreviousBoxSelected && gapRect.width() > 0 &&
|
|
gapRect.height() > 0) {
|
|
if (paintInfo)
|
|
paintInfo->context->fillRect(
|
|
gapRect, box->parent()->renderer().selectionBackgroundColor());
|
|
// VisibleSelection may be non-contiguous, see comment above.
|
|
result.uniteCenter(gapRect);
|
|
}
|
|
lastLogicalLeft = box->logicalRight();
|
|
}
|
|
if (box == lastBox)
|
|
break;
|
|
isPreviousBoxSelected =
|
|
box->selectionState() != RenderObject::SelectionNone;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
RenderObject::SelectionState RootInlineBox::selectionState() {
|
|
// Walk over all of the selected boxes.
|
|
RenderObject::SelectionState state = RenderObject::SelectionNone;
|
|
for (InlineBox* box = firstLeafChild(); box; box = box->nextLeafChild()) {
|
|
RenderObject::SelectionState boxState = box->selectionState();
|
|
if ((boxState == RenderObject::SelectionStart &&
|
|
state == RenderObject::SelectionEnd) ||
|
|
(boxState == RenderObject::SelectionEnd &&
|
|
state == RenderObject::SelectionStart))
|
|
state = RenderObject::SelectionBoth;
|
|
else if (state == RenderObject::SelectionNone ||
|
|
((boxState == RenderObject::SelectionStart ||
|
|
boxState == RenderObject::SelectionEnd) &&
|
|
(state == RenderObject::SelectionNone ||
|
|
state == RenderObject::SelectionInside)))
|
|
state = boxState;
|
|
else if (boxState == RenderObject::SelectionNone &&
|
|
state == RenderObject::SelectionStart) {
|
|
// We are past the end of the selection.
|
|
state = RenderObject::SelectionBoth;
|
|
}
|
|
if (state == RenderObject::SelectionBoth)
|
|
break;
|
|
}
|
|
|
|
return state;
|
|
}
|
|
|
|
InlineBox* RootInlineBox::firstSelectedBox() {
|
|
for (InlineBox* box = firstLeafChild(); box; box = box->nextLeafChild()) {
|
|
if (box->selectionState() != RenderObject::SelectionNone)
|
|
return box;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
InlineBox* RootInlineBox::lastSelectedBox() {
|
|
for (InlineBox* box = lastLeafChild(); box; box = box->prevLeafChild()) {
|
|
if (box->selectionState() != RenderObject::SelectionNone)
|
|
return box;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::selectionTop() const {
|
|
LayoutUnit selectionTop = m_lineTop;
|
|
|
|
if (m_hasAnnotationsBefore)
|
|
selectionTop -= computeOverAnnotationAdjustment(m_lineTop);
|
|
|
|
if (!prevRootBox())
|
|
return selectionTop;
|
|
|
|
return prevRootBox()->selectionBottom();
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::selectionTopAdjustedForPrecedingBlock() const {
|
|
LayoutUnit top = selectionTop();
|
|
|
|
RenderObject::SelectionState blockSelectionState =
|
|
root().block().selectionState();
|
|
if (blockSelectionState != RenderObject::SelectionInside &&
|
|
blockSelectionState != RenderObject::SelectionEnd)
|
|
return top;
|
|
|
|
LayoutSize offsetToBlockBefore;
|
|
if (RenderBlock* block =
|
|
root().block().blockBeforeWithinSelectionRoot(offsetToBlockBefore)) {
|
|
if (block->isRenderParagraph()) {
|
|
if (RootInlineBox* lastLine = toRenderParagraph(block)->lastRootBox()) {
|
|
RenderObject::SelectionState lastLineSelectionState =
|
|
lastLine->selectionState();
|
|
if (lastLineSelectionState != RenderObject::SelectionInside &&
|
|
lastLineSelectionState != RenderObject::SelectionStart)
|
|
return top;
|
|
|
|
LayoutUnit lastLineSelectionBottom =
|
|
lastLine->selectionBottom() + offsetToBlockBefore.height();
|
|
top = std::max(top, lastLineSelectionBottom);
|
|
}
|
|
}
|
|
}
|
|
|
|
return top;
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::selectionBottom() const {
|
|
LayoutUnit selectionBottom = m_selectionBottom;
|
|
if (m_hasAnnotationsAfter)
|
|
selectionBottom += computeUnderAnnotationAdjustment(m_lineBottom);
|
|
return selectionBottom;
|
|
}
|
|
|
|
int RootInlineBox::blockDirectionPointInLine() const {
|
|
return std::max(lineTop(), selectionTop());
|
|
}
|
|
|
|
RenderParagraph& RootInlineBox::block() const {
|
|
return toRenderParagraph(renderer());
|
|
}
|
|
|
|
static bool isEditableLeaf(InlineBox* leaf) {
|
|
return false;
|
|
}
|
|
|
|
InlineBox* RootInlineBox::closestLeafChildForPoint(
|
|
const IntPoint& pointInContents,
|
|
bool onlyEditableLeaves) {
|
|
return closestLeafChildForLogicalLeftPosition(pointInContents.x(),
|
|
onlyEditableLeaves);
|
|
}
|
|
|
|
InlineBox* RootInlineBox::closestLeafChildForLogicalLeftPosition(
|
|
int leftPosition,
|
|
bool onlyEditableLeaves) {
|
|
InlineBox* firstLeaf = firstLeafChild();
|
|
InlineBox* lastLeaf = lastLeafChild();
|
|
|
|
if (firstLeaf != lastLeaf) {
|
|
if (firstLeaf->isLineBreak())
|
|
firstLeaf = firstLeaf->nextLeafChildIgnoringLineBreak();
|
|
else if (lastLeaf->isLineBreak())
|
|
lastLeaf = lastLeaf->prevLeafChildIgnoringLineBreak();
|
|
}
|
|
|
|
if (firstLeaf == lastLeaf &&
|
|
(!onlyEditableLeaves || isEditableLeaf(firstLeaf)))
|
|
return firstLeaf;
|
|
|
|
// Avoid returning a list marker when possible.
|
|
if (leftPosition <= firstLeaf->logicalLeft() &&
|
|
(!onlyEditableLeaves || isEditableLeaf(firstLeaf)))
|
|
// The leftPosition coordinate is less or equal to left edge of the
|
|
// firstLeaf. Return it.
|
|
return firstLeaf;
|
|
|
|
if (leftPosition >= lastLeaf->logicalRight() &&
|
|
(!onlyEditableLeaves || isEditableLeaf(lastLeaf)))
|
|
// The leftPosition coordinate is greater or equal to right edge of the
|
|
// lastLeaf. Return it.
|
|
return lastLeaf;
|
|
|
|
InlineBox* closestLeaf = 0;
|
|
for (InlineBox* leaf = firstLeaf; leaf;
|
|
leaf = leaf->nextLeafChildIgnoringLineBreak()) {
|
|
if (!onlyEditableLeaves || isEditableLeaf(leaf)) {
|
|
closestLeaf = leaf;
|
|
if (leftPosition < leaf->logicalRight())
|
|
// The x coordinate is less than the right edge of the box.
|
|
// Return it.
|
|
return leaf;
|
|
}
|
|
}
|
|
|
|
return closestLeaf ? closestLeaf : lastLeaf;
|
|
}
|
|
|
|
BidiStatus RootInlineBox::lineBreakBidiStatus() const {
|
|
return BidiStatus(
|
|
static_cast<WTF::Unicode::Direction>(m_lineBreakBidiStatusEor),
|
|
static_cast<WTF::Unicode::Direction>(m_lineBreakBidiStatusLastStrong),
|
|
static_cast<WTF::Unicode::Direction>(m_lineBreakBidiStatusLast),
|
|
m_lineBreakContext);
|
|
}
|
|
|
|
void RootInlineBox::setLineBreakInfo(RenderObject* obj,
|
|
unsigned breakPos,
|
|
const BidiStatus& status) {
|
|
// When setting lineBreakObj, the RenderObject must not be a RenderInline
|
|
// with no line boxes, otherwise all sorts of invariants are broken later.
|
|
// This has security implications because if the RenderObject does not
|
|
// point to at least one line box, then that RenderInline can be deleted
|
|
// later without resetting the lineBreakObj, leading to use-after-free.
|
|
ASSERT_WITH_SECURITY_IMPLICATION(!obj || obj->isText() ||
|
|
!(obj->isRenderInline() && obj->isBox() &&
|
|
!toRenderBox(obj)->inlineBoxWrapper()));
|
|
|
|
m_lineBreakObj = obj;
|
|
m_lineBreakPos = breakPos;
|
|
m_lineBreakBidiStatusEor = status.eor;
|
|
m_lineBreakBidiStatusLastStrong = status.lastStrong;
|
|
m_lineBreakBidiStatusLast = status.last;
|
|
m_lineBreakContext = status.context;
|
|
}
|
|
|
|
void RootInlineBox::removeLineBoxFromRenderObject() {
|
|
block().lineBoxes()->removeLineBox(this);
|
|
}
|
|
|
|
void RootInlineBox::extractLineBoxFromRenderObject() {
|
|
block().lineBoxes()->extractLineBox(this);
|
|
}
|
|
|
|
void RootInlineBox::attachLineBoxToRenderObject() {
|
|
block().lineBoxes()->attachLineBox(this);
|
|
}
|
|
|
|
LayoutRect RootInlineBox::paddedLayoutOverflowRect(
|
|
LayoutUnit endPadding) const {
|
|
LayoutRect lineLayoutOverflow = layoutOverflowRect(lineTop(), lineBottom());
|
|
if (!endPadding)
|
|
return lineLayoutOverflow;
|
|
|
|
if (isLeftToRightDirection())
|
|
lineLayoutOverflow.shiftMaxXEdgeTo(std::max<LayoutUnit>(
|
|
lineLayoutOverflow.maxX(), logicalRight() + endPadding));
|
|
else
|
|
lineLayoutOverflow.shiftXEdgeTo(std::min<LayoutUnit>(
|
|
lineLayoutOverflow.x(), logicalLeft() - endPadding));
|
|
|
|
return lineLayoutOverflow;
|
|
}
|
|
|
|
static void setAscentAndDescent(int& ascent,
|
|
int& descent,
|
|
int newAscent,
|
|
int newDescent,
|
|
bool& ascentDescentSet) {
|
|
if (!ascentDescentSet) {
|
|
ascentDescentSet = true;
|
|
ascent = newAscent;
|
|
descent = newDescent;
|
|
} else {
|
|
ascent = std::max(ascent, newAscent);
|
|
descent = std::max(descent, newDescent);
|
|
}
|
|
}
|
|
|
|
void RootInlineBox::ascentAndDescentForBox(
|
|
InlineBox* box,
|
|
GlyphOverflowAndFallbackFontsMap& textBoxDataMap,
|
|
int& ascent,
|
|
int& descent,
|
|
bool& affectsAscent,
|
|
bool& affectsDescent) const {
|
|
bool ascentDescentSet = false;
|
|
|
|
// Replaced boxes will return 0 for the line-height if line-box-contain says
|
|
// they are not to be included.
|
|
if (box->renderer().isReplaced()) {
|
|
if (renderer().style(isFirstLineStyle())->lineBoxContain() &
|
|
LineBoxContainReplaced) {
|
|
ascent = box->baselinePosition(baselineType());
|
|
descent = box->lineHeight() - ascent;
|
|
|
|
// Replaced elements always affect both the ascent and descent.
|
|
affectsAscent = true;
|
|
affectsDescent = true;
|
|
}
|
|
return;
|
|
}
|
|
|
|
Vector<const SimpleFontData*>* usedFonts = 0;
|
|
GlyphOverflow* glyphOverflow = 0;
|
|
if (box->isText()) {
|
|
GlyphOverflowAndFallbackFontsMap::iterator it =
|
|
textBoxDataMap.find(toInlineTextBox(box));
|
|
usedFonts = it == textBoxDataMap.end() ? 0 : &it->value.first;
|
|
glyphOverflow = it == textBoxDataMap.end() ? 0 : &it->value.second;
|
|
}
|
|
|
|
bool includeLeading = includeLeadingForBox(box);
|
|
bool includeFont = includeFontForBox(box);
|
|
|
|
bool setUsedFont = false;
|
|
bool setUsedFontWithLeading = false;
|
|
|
|
if (usedFonts && !usedFonts->isEmpty() &&
|
|
(includeFont ||
|
|
(box->renderer().style(isFirstLineStyle())->lineHeight().isNegative() &&
|
|
includeLeading))) {
|
|
usedFonts->append(
|
|
box->renderer().style(isFirstLineStyle())->font().primaryFont());
|
|
for (size_t i = 0; i < usedFonts->size(); ++i) {
|
|
const FontMetrics& fontMetrics = usedFonts->at(i)->fontMetrics();
|
|
int usedFontAscent = fontMetrics.ascent(baselineType());
|
|
int usedFontDescent = fontMetrics.descent(baselineType());
|
|
int halfLeading = (fontMetrics.lineSpacing() - fontMetrics.height()) / 2;
|
|
int usedFontAscentAndLeading = usedFontAscent + halfLeading;
|
|
int usedFontDescentAndLeading =
|
|
fontMetrics.lineSpacing() - usedFontAscentAndLeading;
|
|
if (includeFont) {
|
|
setAscentAndDescent(ascent, descent, usedFontAscent, usedFontDescent,
|
|
ascentDescentSet);
|
|
setUsedFont = true;
|
|
}
|
|
if (includeLeading) {
|
|
setAscentAndDescent(ascent, descent, usedFontAscentAndLeading,
|
|
usedFontDescentAndLeading, ascentDescentSet);
|
|
setUsedFontWithLeading = true;
|
|
}
|
|
if (!affectsAscent)
|
|
affectsAscent = usedFontAscent - box->logicalTop() > 0;
|
|
if (!affectsDescent)
|
|
affectsDescent = usedFontDescent + box->logicalTop() > 0;
|
|
}
|
|
}
|
|
|
|
// If leading is included for the box, then we compute that box.
|
|
if (includeLeading && !setUsedFontWithLeading) {
|
|
int ascentWithLeading = box->baselinePosition(baselineType());
|
|
int descentWithLeading = box->lineHeight() - ascentWithLeading;
|
|
setAscentAndDescent(ascent, descent, ascentWithLeading, descentWithLeading,
|
|
ascentDescentSet);
|
|
|
|
// Examine the font box for inline flows and text boxes to see if any part
|
|
// of it is above the baseline. If the top of our font box relative to the
|
|
// root box baseline is above the root box baseline, then we are
|
|
// contributing to the maxAscent value. Descent is similar. If any part of
|
|
// our font box is below the root box's baseline, then we contribute to the
|
|
// maxDescent value.
|
|
affectsAscent = ascentWithLeading - box->logicalTop() > 0;
|
|
affectsDescent = descentWithLeading + box->logicalTop() > 0;
|
|
}
|
|
|
|
if (includeFontForBox(box) && !setUsedFont) {
|
|
int fontAscent = box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.ascent(baselineType());
|
|
int fontDescent = box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.descent(baselineType());
|
|
setAscentAndDescent(ascent, descent, fontAscent, fontDescent,
|
|
ascentDescentSet);
|
|
affectsAscent = fontAscent - box->logicalTop() > 0;
|
|
affectsDescent = fontDescent + box->logicalTop() > 0;
|
|
}
|
|
|
|
if (includeGlyphsForBox(box) && glyphOverflow &&
|
|
glyphOverflow->computeBounds) {
|
|
setAscentAndDescent(ascent, descent, glyphOverflow->top,
|
|
glyphOverflow->bottom, ascentDescentSet);
|
|
affectsAscent = glyphOverflow->top - box->logicalTop() > 0;
|
|
affectsDescent = glyphOverflow->bottom + box->logicalTop() > 0;
|
|
glyphOverflow->top =
|
|
std::min(glyphOverflow->top,
|
|
std::max(0, glyphOverflow->top - box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.ascent(baselineType())));
|
|
glyphOverflow->bottom = std::min(
|
|
glyphOverflow->bottom,
|
|
std::max(0, glyphOverflow->bottom - box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.descent(baselineType())));
|
|
}
|
|
|
|
if (includeMarginForBox(box)) {
|
|
LayoutUnit ascentWithMargin = box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.ascent(baselineType());
|
|
LayoutUnit descentWithMargin = box->renderer()
|
|
.style(isFirstLineStyle())
|
|
->fontMetrics()
|
|
.descent(baselineType());
|
|
if (box->parent() && !box->renderer().isText()) {
|
|
ascentWithMargin += box->boxModelObject()->borderBefore() +
|
|
box->boxModelObject()->paddingBefore() +
|
|
box->boxModelObject()->marginBefore();
|
|
descentWithMargin += box->boxModelObject()->borderAfter() +
|
|
box->boxModelObject()->paddingAfter() +
|
|
box->boxModelObject()->marginAfter();
|
|
}
|
|
setAscentAndDescent(ascent, descent, ascentWithMargin, descentWithMargin,
|
|
ascentDescentSet);
|
|
|
|
// Treat like a replaced element, since we're using the margin box.
|
|
affectsAscent = true;
|
|
affectsDescent = true;
|
|
}
|
|
}
|
|
|
|
LayoutUnit RootInlineBox::verticalPositionForBox(
|
|
InlineBox* box,
|
|
VerticalPositionCache& verticalPositionCache) {
|
|
if (box->renderer().isText())
|
|
return box->parent()->logicalTop();
|
|
|
|
RenderBoxModelObject* renderer = box->boxModelObject();
|
|
ASSERT(renderer->isInline());
|
|
if (!renderer->isInline())
|
|
return 0;
|
|
|
|
bool firstLine = false;
|
|
|
|
// Check the cache.
|
|
bool isRenderInline = renderer->isRenderInline();
|
|
if (isRenderInline && !firstLine) {
|
|
LayoutUnit verticalPosition =
|
|
verticalPositionCache.get(renderer, baselineType());
|
|
if (verticalPosition != PositionUndefined)
|
|
return verticalPosition;
|
|
}
|
|
|
|
LayoutUnit verticalPosition = 0;
|
|
EVerticalAlign verticalAlign = renderer->style()->verticalAlign();
|
|
if (verticalAlign == TOP || verticalAlign == BOTTOM)
|
|
return 0;
|
|
|
|
RenderObject* parent = renderer->parent();
|
|
if (parent->isRenderInline() && parent->style()->verticalAlign() != TOP &&
|
|
parent->style()->verticalAlign() != BOTTOM)
|
|
verticalPosition = box->parent()->logicalTop();
|
|
|
|
if (verticalAlign != BASELINE) {
|
|
const Font& font = parent->style(firstLine)->font();
|
|
const FontMetrics& fontMetrics = font.fontMetrics();
|
|
int fontSize = font.fontDescription().computedPixelSize();
|
|
|
|
LineDirectionMode lineDirection = HorizontalLine;
|
|
|
|
if (verticalAlign == SUB)
|
|
verticalPosition += fontSize / 5 + 1;
|
|
else if (verticalAlign == SUPER)
|
|
verticalPosition -= fontSize / 3 + 1;
|
|
else if (verticalAlign == TEXT_TOP)
|
|
verticalPosition +=
|
|
renderer->baselinePosition(baselineType(), firstLine, lineDirection) -
|
|
fontMetrics.ascent(baselineType());
|
|
else if (verticalAlign == MIDDLE)
|
|
verticalPosition =
|
|
(verticalPosition -
|
|
static_cast<LayoutUnit>(fontMetrics.xHeight() / 2) -
|
|
renderer->lineHeight(firstLine, lineDirection) / 2 +
|
|
renderer->baselinePosition(baselineType(), firstLine, lineDirection))
|
|
.round();
|
|
else if (verticalAlign == TEXT_BOTTOM) {
|
|
verticalPosition += fontMetrics.descent(baselineType());
|
|
// lineHeight - baselinePosition is always 0 for replaced elements (except
|
|
// inline blocks), so don't bother wasting time in that case.
|
|
if (!renderer->isReplaced() || renderer->isInlineBlock())
|
|
verticalPosition -= (renderer->lineHeight(firstLine, lineDirection) -
|
|
renderer->baselinePosition(
|
|
baselineType(), firstLine, lineDirection));
|
|
} else if (verticalAlign == BASELINE_MIDDLE)
|
|
verticalPosition +=
|
|
-renderer->lineHeight(firstLine, lineDirection) / 2 +
|
|
renderer->baselinePosition(baselineType(), firstLine, lineDirection);
|
|
else if (verticalAlign == LENGTH) {
|
|
LayoutUnit lineHeight;
|
|
// Per http://www.w3.org/TR/CSS21/visudet.html#propdef-vertical-align:
|
|
// 'Percentages: refer to the 'line-height' of the element itself'.
|
|
if (renderer->style()->verticalAlignLength().isPercent())
|
|
lineHeight = renderer->style()->computedLineHeight();
|
|
else
|
|
lineHeight = renderer->lineHeight(firstLine, lineDirection);
|
|
verticalPosition -=
|
|
valueForLength(renderer->style()->verticalAlignLength(), lineHeight);
|
|
}
|
|
}
|
|
|
|
// Store the cached value.
|
|
if (isRenderInline && !firstLine)
|
|
verticalPositionCache.set(renderer, baselineType(), verticalPosition);
|
|
|
|
return verticalPosition;
|
|
}
|
|
|
|
bool RootInlineBox::includeLeadingForBox(InlineBox* box) const {
|
|
if (box->renderer().isReplaced() ||
|
|
(box->renderer().isText() && !box->isText()))
|
|
return false;
|
|
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return (lineBoxContain & LineBoxContainInline) ||
|
|
(box == this && (lineBoxContain & LineBoxContainBlock));
|
|
}
|
|
|
|
bool RootInlineBox::includeFontForBox(InlineBox* box) const {
|
|
if (box->renderer().isReplaced() ||
|
|
(box->renderer().isText() && !box->isText()))
|
|
return false;
|
|
|
|
if (!box->isText() && box->isInlineFlowBox() &&
|
|
!toInlineFlowBox(box)->hasTextChildren())
|
|
return false;
|
|
|
|
// For now map "glyphs" to "font" in vertical text mode until the bounds
|
|
// returned by glyphs aren't garbage.
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return lineBoxContain & LineBoxContainFont;
|
|
}
|
|
|
|
bool RootInlineBox::includeGlyphsForBox(InlineBox* box) const {
|
|
if (box->renderer().isReplaced() ||
|
|
(box->renderer().isText() && !box->isText()))
|
|
return false;
|
|
|
|
if (!box->isText() && box->isInlineFlowBox() &&
|
|
!toInlineFlowBox(box)->hasTextChildren())
|
|
return false;
|
|
|
|
// FIXME: We can't fit to glyphs yet for vertical text, since the bounds
|
|
// returned are garbage.
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return lineBoxContain & LineBoxContainGlyphs;
|
|
}
|
|
|
|
bool RootInlineBox::includeMarginForBox(InlineBox* box) const {
|
|
if (box->renderer().isReplaced() ||
|
|
(box->renderer().isText() && !box->isText()))
|
|
return false;
|
|
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return lineBoxContain & LineBoxContainInlineBox;
|
|
}
|
|
|
|
bool RootInlineBox::fitsToGlyphs() const {
|
|
// FIXME: We can't fit to glyphs yet for vertical text, since the bounds
|
|
// returned are garbage.
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return lineBoxContain & LineBoxContainGlyphs;
|
|
}
|
|
|
|
bool RootInlineBox::includesRootLineBoxFontOrLeading() const {
|
|
LineBoxContain lineBoxContain = renderer().style()->lineBoxContain();
|
|
return (lineBoxContain & LineBoxContainBlock) ||
|
|
(lineBoxContain & LineBoxContainInline) ||
|
|
(lineBoxContain & LineBoxContainFont);
|
|
}
|
|
|
|
#ifndef NDEBUG
|
|
const char* RootInlineBox::boxName() const {
|
|
return "RootInlineBox";
|
|
}
|
|
#endif
|
|
|
|
} // namespace blink
|