This source file includes following definitions.
- writeDebugInfo
- formInputTags
- getAncestorListItem
- getAncestorList
- getGeneratingElementNode
- hashMemory
- computeLocalHash
- m_previouslyAutosized
- getCachedHash
- isApplicable
- recalculateMultipliers
- processSubtree
- clusterMultiplier
- processClusterInternal
- computeCompositeClusterHash
- addNonAutosizedCluster
- computeMultiplier
- processCluster
- processCompositeCluster
- secondPassProcessStaleNonAutosizedClusters
- processStaleContainer
- processContainer
- setMultiplier
- setMultiplierForList
- computeAutosizedFontSize
- isAutosizingContainer
- isNarrowDescendant
- isWiderDescendant
- isIndependentDescendant
- isAutosizingCluster
- containerShouldBeAutosized
- containerContainsOneOfTags
- containerIsRowOfLinks
- contentHeightIsConstrained
- compositeClusterShouldBeAutosized
- measureDescendantTextWidth
- nextInPreOrderSkippingDescendantsOfContainers
- findDeepestBlockContainingAllText
- findFirstTextLeafNotInCluster
- clusterWiderThanComparisonFn
- getNarrowDescendantsGroupedByWidth
#include "config.h"
#include "core/rendering/TextAutosizer.h"
#include <algorithm>
#include "core/dom/Document.h"
#include "core/frame/Settings.h"
#include "core/frame/UseCounter.h"
#include "core/html/HTMLElement.h"
#include "core/inspector/InspectorInstrumentation.h"
#include "core/rendering/RenderListItem.h"
#include "core/rendering/RenderObject.h"
#include "core/rendering/RenderText.h"
#include "core/rendering/RenderView.h"
#include "core/rendering/style/RenderStyle.h"
#include "core/rendering/style/StyleInheritedData.h"
#include "platform/TraceEvent.h"
#include "platform/geometry/IntSize.h"
#include "wtf/StdLibExtras.h"
namespace WebCore {
#define AUTOSIZING_CLUSTER_HASH
using namespace HTMLNames;
struct TextAutosizingWindowInfo {
IntSize windowSize;
IntSize minLayoutSize;
};
struct RenderObjectPodForHash {
RenderObjectPodForHash()
: qualifiedNameHash(0)
, packedStyleProperties(0)
, width(0)
{
}
~RenderObjectPodForHash() { }
unsigned qualifiedNameHash;
unsigned packedStyleProperties;
float width;
};
COMPILE_ASSERT(!(sizeof(RenderObjectPodForHash) % sizeof(UChar)), RenderObjectPodForHashMultipleOfUchar);
#ifdef AUTOSIZING_DOM_DEBUG_INFO
static void writeDebugInfo(RenderObject* renderObject, const AtomicString& output)
{
Node* node = renderObject->node();
if (node && node->isElementNode())
toElement(node)->setAttribute("data-autosizing", output, ASSERT_NO_EXCEPTION);
}
#endif
static const Vector<QualifiedName>& formInputTags()
{
DEFINE_STATIC_LOCAL(Vector<QualifiedName>, formInputTags, ());
if (formInputTags.isEmpty()) {
formInputTags.append(inputTag);
formInputTags.append(buttonTag);
formInputTags.append(selectTag);
}
return formInputTags;
}
static RenderListItem* getAncestorListItem(const RenderObject* renderer)
{
RenderObject* ancestor = renderer->parent();
while (ancestor && (ancestor->isRenderInline() || ancestor->isAnonymousBlock()))
ancestor = ancestor->parent();
return (ancestor && ancestor->isListItem()) ? toRenderListItem(ancestor) : 0;
}
static RenderObject* getAncestorList(const RenderObject* renderer)
{
for (RenderObject* ancestor = renderer->parent(); ancestor; ancestor = ancestor->parent()) {
Node* parentNode = ancestor->generatingNode();
if (parentNode && (isHTMLOListElement(*parentNode) || isHTMLUListElement(*parentNode)))
return ancestor;
}
return 0;
}
static Node* getGeneratingElementNode(const RenderObject* renderer)
{
Node* node = renderer->generatingNode();
return (node && node->isElementNode()) ? node : 0;
}
static unsigned hashMemory(const void* data, size_t length)
{
return StringHasher::computeHash<UChar>(static_cast<const UChar*>(data), length / sizeof(UChar));
}
static unsigned computeLocalHash(const RenderObject* renderer)
{
Node* generatingElementNode = getGeneratingElementNode(renderer);
ASSERT(generatingElementNode);
RenderObjectPodForHash podForHash;
podForHash.qualifiedNameHash = QualifiedNameHash::hash(toElement(generatingElementNode)->tagQName());
if (RenderStyle* style = renderer->style()) {
podForHash.packedStyleProperties = style->direction();
podForHash.packedStyleProperties |= (style->position() << 1);
podForHash.packedStyleProperties |= (style->floating() << 4);
podForHash.packedStyleProperties |= (style->display() << 6);
podForHash.packedStyleProperties |= (style->width().type() << 11);
podForHash.width = style->width().getFloatValue();
}
return hashMemory(&podForHash, sizeof(podForHash));
}
TextAutosizer::TextAutosizer(Document* document)
: m_document(document)
, m_previouslyAutosized(false)
{
}
unsigned TextAutosizer::getCachedHash(const RenderObject* renderer, bool putInCacheIfAbsent)
{
HashMap<const RenderObject*, unsigned>::const_iterator it = m_hashCache.find(renderer);
if (it != m_hashCache.end())
return it->value;
RenderObject* rendererParent = renderer->parent();
while (rendererParent && !getGeneratingElementNode(rendererParent))
rendererParent = rendererParent->parent();
const unsigned parentHashValue = rendererParent ? getCachedHash(rendererParent, true) : 0;
const unsigned hashes[2] = { parentHashValue, computeLocalHash(renderer) };
const unsigned combinedHashValue = hashMemory(hashes, sizeof(hashes));
if (putInCacheIfAbsent)
m_hashCache.add(renderer, combinedHashValue);
return combinedHashValue;
}
bool TextAutosizer::isApplicable() const
{
return m_document->settings()
&& m_document->settings()->textAutosizingEnabled()
&& m_document->page()
&& m_document->page()->mainFrame()
&& m_document->page()->mainFrame()->loader().stateMachine()->committedFirstRealDocumentLoad();
}
void TextAutosizer::recalculateMultipliers()
{
if (!isApplicable() && !m_previouslyAutosized)
return;
RenderObject* renderer = m_document->renderer();
while (renderer) {
if (renderer->style() && renderer->style()->textAutosizingMultiplier() != 1)
setMultiplier(renderer, 1);
renderer = renderer->nextInPreOrder();
}
m_previouslyAutosized = false;
}
bool TextAutosizer::processSubtree(RenderObject* layoutRoot)
{
TRACE_EVENT0("webkit", "TextAutosizer: check if needed");
if (!isApplicable() || layoutRoot->view()->document().printing())
return false;
LocalFrame* mainFrame = m_document->page()->mainFrame();
TextAutosizingWindowInfo windowInfo;
windowInfo.windowSize = m_document->settings()->textAutosizingWindowSizeOverride();
if (windowInfo.windowSize.isEmpty())
windowInfo.windowSize = mainFrame->view()->unscaledVisibleContentSize(IncludeScrollbars);
windowInfo.minLayoutSize = mainFrame->view()->layoutSize();
for (LocalFrame* frame = m_document->frame(); frame; frame = frame->tree().parent())
windowInfo.minLayoutSize = windowInfo.minLayoutSize.shrunkTo(frame->view()->layoutSize());
RenderBlock* container = layoutRoot->isRenderBlock() ? toRenderBlock(layoutRoot) : layoutRoot->containingBlock();
while (container && !isAutosizingContainer(container))
container = container->containingBlock();
RenderBlock* cluster = container;
while (cluster && (!isAutosizingContainer(cluster) || !isIndependentDescendant(cluster)))
cluster = cluster->containingBlock();
if (!cluster || clusterMultiplier(cluster->style()->writingMode(), windowInfo,
std::numeric_limits<float>::infinity()) == 1.0f)
return false;
TRACE_EVENT0("webkit", "TextAutosizer: process root cluster");
InspectorInstrumentation::willAutosizeText(layoutRoot);
UseCounter::count(*m_document, UseCounter::TextAutosizing);
TextAutosizingClusterInfo clusterInfo(cluster);
processCluster(clusterInfo, container, layoutRoot, windowInfo);
#ifdef AUTOSIZING_CLUSTER_HASH
secondPassProcessStaleNonAutosizedClusters();
m_hashCache.clear();
m_hashToMultiplier.clear();
m_hashesToAutosizeSecondPass.clear();
m_nonAutosizedClusters.clear();
#endif
InspectorInstrumentation::didAutosizeText(layoutRoot);
m_previouslyAutosized = true;
return true;
}
float TextAutosizer::clusterMultiplier(WritingMode writingMode, const TextAutosizingWindowInfo& windowInfo, float textWidth) const
{
int logicalWindowWidth = isHorizontalWritingMode(writingMode) ? windowInfo.windowSize.width() : windowInfo.windowSize.height();
int logicalLayoutWidth = isHorizontalWritingMode(writingMode) ? windowInfo.minLayoutSize.width() : windowInfo.minLayoutSize.height();
float logicalClusterWidth = std::min<float>(textWidth, logicalLayoutWidth);
float multiplier = logicalClusterWidth / logicalWindowWidth;
multiplier *= m_document->settings()->accessibilityFontScaleFactor();
const ViewportDescription& viewportDescription = m_document->page()->mainFrame()->document()->viewportDescription();
if (!viewportDescription.isSpecifiedByAuthor()) {
float deviceScaleAdjustment = m_document->settings()->deviceScaleAdjustment();
multiplier *= deviceScaleAdjustment;
}
return std::max(1.0f, multiplier);
}
void TextAutosizer::processClusterInternal(TextAutosizingClusterInfo& clusterInfo, RenderBlock* container, RenderObject* subtreeRoot, const TextAutosizingWindowInfo& windowInfo, float multiplier)
{
processContainer(multiplier, container, clusterInfo, subtreeRoot, windowInfo);
#ifdef AUTOSIZING_DOM_DEBUG_INFO
writeDebugInfo(clusterInfo.root, AtomicString(String::format("cluster:%f", multiplier)));
#endif
Vector<Vector<TextAutosizingClusterInfo> > narrowDescendantsGroups;
getNarrowDescendantsGroupedByWidth(clusterInfo, narrowDescendantsGroups);
for (size_t i = 0; i < narrowDescendantsGroups.size(); ++i)
processCompositeCluster(narrowDescendantsGroups[i], windowInfo);
}
unsigned TextAutosizer::computeCompositeClusterHash(Vector<TextAutosizingClusterInfo>& clusterInfos)
{
if (clusterInfos.size() == 1 && getGeneratingElementNode(clusterInfos[0].root))
return getCachedHash(clusterInfos[0].root, false);
return 0;
}
void TextAutosizer::addNonAutosizedCluster(unsigned key, TextAutosizingClusterInfo& value)
{
HashMap<unsigned, OwnPtr<Vector<TextAutosizingClusterInfo> > >::const_iterator it = m_nonAutosizedClusters.find(key);
if (it == m_nonAutosizedClusters.end()) {
m_nonAutosizedClusters.add(key, adoptPtr(new Vector<TextAutosizingClusterInfo>(1, value)));
return;
}
it->value->append(value);
}
float TextAutosizer::computeMultiplier(Vector<TextAutosizingClusterInfo>& clusterInfos, const TextAutosizingWindowInfo& windowInfo, float textWidth)
{
#ifdef AUTOSIZING_CLUSTER_HASH
unsigned clusterHash = computeCompositeClusterHash(clusterInfos);
#else
unsigned clusterHash = 0;
#endif
if (clusterHash) {
HashMap<unsigned, float>::iterator it = m_hashToMultiplier.find(clusterHash);
if (it != m_hashToMultiplier.end())
return it->value;
}
if (compositeClusterShouldBeAutosized(clusterInfos, textWidth)) {
float multiplier = clusterMultiplier(clusterInfos[0].root->style()->writingMode(), windowInfo, textWidth);
if (clusterHash) {
if (multiplier > 1 && m_nonAutosizedClusters.contains(clusterHash))
m_hashesToAutosizeSecondPass.append(clusterHash);
m_hashToMultiplier.add(clusterHash, multiplier);
}
return multiplier;
}
if (clusterHash)
addNonAutosizedCluster(clusterHash, clusterInfos[0]);
return 1.0f;
}
void TextAutosizer::processCluster(TextAutosizingClusterInfo& clusterInfo, RenderBlock* container, RenderObject* subtreeRoot, const TextAutosizingWindowInfo& windowInfo)
{
clusterInfo.blockContainingAllText = findDeepestBlockContainingAllText(clusterInfo.root);
float textWidth = clusterInfo.blockContainingAllText->contentLogicalWidth().toFloat();
Vector<TextAutosizingClusterInfo> clusterInfos(1, clusterInfo);
float multiplier = computeMultiplier(clusterInfos, windowInfo, textWidth);
processClusterInternal(clusterInfo, container, subtreeRoot, windowInfo, multiplier);
}
void TextAutosizer::processCompositeCluster(Vector<TextAutosizingClusterInfo>& clusterInfos, const TextAutosizingWindowInfo& windowInfo)
{
if (clusterInfos.isEmpty())
return;
float maxTextWidth = 0;
for (size_t i = 0; i < clusterInfos.size(); ++i) {
TextAutosizingClusterInfo& clusterInfo = clusterInfos[i];
clusterInfo.blockContainingAllText = findDeepestBlockContainingAllText(clusterInfo.root);
maxTextWidth = max<float>(maxTextWidth, clusterInfo.blockContainingAllText->contentLogicalWidth().toFloat());
}
float multiplier = computeMultiplier(clusterInfos, windowInfo, maxTextWidth);
for (size_t i = 0; i < clusterInfos.size(); ++i) {
ASSERT(clusterInfos[i].root->style()->writingMode() == clusterInfos[0].root->style()->writingMode());
processClusterInternal(clusterInfos[i], clusterInfos[i].root, clusterInfos[i].root, windowInfo, multiplier);
}
}
void TextAutosizer::secondPassProcessStaleNonAutosizedClusters()
{
for (size_t i = 0; i < m_hashesToAutosizeSecondPass.size(); ++i) {
unsigned hash = m_hashesToAutosizeSecondPass[i];
float multiplier = m_hashToMultiplier.get(hash);
Vector<TextAutosizingClusterInfo>* val = m_nonAutosizedClusters.get(hash);
for (Vector<TextAutosizingClusterInfo>::iterator it2 = val->begin(); it2 != val->end(); ++it2)
processStaleContainer(multiplier, (*it2).root, *it2);
}
}
void TextAutosizer::processStaleContainer(float multiplier, RenderBlock* cluster, TextAutosizingClusterInfo& clusterInfo)
{
ASSERT(isAutosizingContainer(cluster));
float localMultiplier = containerShouldBeAutosized(cluster) ? multiplier : 1;
RenderObject* descendant = nextInPreOrderSkippingDescendantsOfContainers(cluster, cluster);
while (descendant) {
if (descendant->isText()) {
if (localMultiplier != 1 && descendant->style()->textAutosizingMultiplier() == 1) {
setMultiplier(descendant, localMultiplier);
setMultiplier(descendant->parent(), localMultiplier);
}
} else if (isAutosizingContainer(descendant)) {
RenderBlock* descendantBlock = toRenderBlock(descendant);
if (!isAutosizingCluster(descendantBlock, clusterInfo))
processStaleContainer(multiplier, descendantBlock, clusterInfo);
}
descendant = nextInPreOrderSkippingDescendantsOfContainers(descendant, cluster);
}
}
void TextAutosizer::processContainer(float multiplier, RenderBlock* container, TextAutosizingClusterInfo& clusterInfo, RenderObject* subtreeRoot, const TextAutosizingWindowInfo& windowInfo)
{
ASSERT(isAutosizingContainer(container));
#ifdef AUTOSIZING_DOM_DEBUG_INFO
writeDebugInfo(container, "container");
#endif
float localMultiplier = (multiplier > 1 && containerShouldBeAutosized(container)) ? multiplier: 1;
RenderObject* descendant = nextInPreOrderSkippingDescendantsOfContainers(subtreeRoot, subtreeRoot);
while (descendant) {
if (descendant->isText()) {
if (localMultiplier != 1 && descendant->style()->textAutosizingMultiplier() == 1) {
setMultiplier(descendant, localMultiplier);
setMultiplier(descendant->parent(), localMultiplier);
if (RenderListItem* listItemAncestor = getAncestorListItem(descendant)) {
if (RenderObject* list = getAncestorList(listItemAncestor)) {
if (list->style()->textAutosizingMultiplier() == 1)
setMultiplierForList(list, localMultiplier);
}
}
}
} else if (isAutosizingContainer(descendant)) {
RenderBlock* descendantBlock = toRenderBlock(descendant);
TextAutosizingClusterInfo descendantClusterInfo(descendantBlock);
if (isWiderDescendant(descendantBlock, clusterInfo) || isIndependentDescendant(descendantBlock))
processCluster(descendantClusterInfo, descendantBlock, descendantBlock, windowInfo);
else if (isNarrowDescendant(descendantBlock, clusterInfo)) {
clusterInfo.narrowDescendants.append(descendantClusterInfo);
} else
processContainer(multiplier, descendantBlock, clusterInfo, descendantBlock, windowInfo);
}
descendant = nextInPreOrderSkippingDescendantsOfContainers(descendant, subtreeRoot);
}
}
void TextAutosizer::setMultiplier(RenderObject* renderer, float multiplier)
{
RefPtr<RenderStyle> newStyle = RenderStyle::clone(renderer->style());
newStyle->setTextAutosizingMultiplier(multiplier);
newStyle->setUnique();
renderer->setStyle(newStyle.release());
}
void TextAutosizer::setMultiplierForList(RenderObject* renderer, float multiplier)
{
#ifndef NDEBUG
Node* parentNode = renderer->generatingNode();
ASSERT(parentNode);
ASSERT(isHTMLOListElement(parentNode) || isHTMLUListElement(parentNode));
#endif
setMultiplier(renderer, multiplier);
for (RenderObject* child = renderer->firstChild(); child; child = child->nextSibling()) {
if (child->isListItem() && child->style()->textAutosizingMultiplier() == 1)
setMultiplier(child, multiplier);
}
}
float TextAutosizer::computeAutosizedFontSize(float specifiedSize, float multiplier)
{
const float pleasantSize = 16;
const float gradientAfterPleasantSize = 0.5;
float computedSize;
if (specifiedSize <= pleasantSize)
computedSize = multiplier * specifiedSize;
else {
computedSize = multiplier * pleasantSize + gradientAfterPleasantSize * (specifiedSize - pleasantSize);
if (computedSize < specifiedSize)
computedSize = specifiedSize;
}
return computedSize;
}
bool TextAutosizer::isAutosizingContainer(const RenderObject* renderer)
{
Node* node = renderer->generatingNode();
if ((node && !node->hasChildren())
|| !renderer->isRenderBlock()
|| (renderer->isInline() && !renderer->style()->isDisplayReplacedType()))
return false;
if (renderer->isListItem())
return renderer->isFloating() || renderer->isOutOfFlowPositioned();
Node* parentNode = renderer->parent() ? renderer->parent()->generatingNode() : 0;
if (parentNode && parentNode->isElementNode() && formInputTags().contains(toElement(parentNode)->tagQName()))
return false;
return true;
}
bool TextAutosizer::isNarrowDescendant(const RenderBlock* renderer, TextAutosizingClusterInfo& parentClusterInfo)
{
ASSERT(isAutosizingContainer(renderer));
const float differenceFromMaxWidthDifference = 50;
LayoutUnit contentWidth = renderer->contentLogicalWidth();
LayoutUnit clusterTextWidth = parentClusterInfo.blockContainingAllText->contentLogicalWidth();
LayoutUnit widthDifference = clusterTextWidth - contentWidth;
if (widthDifference - parentClusterInfo.maxAllowedDifferenceFromTextWidth > differenceFromMaxWidthDifference)
return true;
parentClusterInfo.maxAllowedDifferenceFromTextWidth = std::max(widthDifference.toFloat(), parentClusterInfo.maxAllowedDifferenceFromTextWidth);
return false;
}
bool TextAutosizer::isWiderDescendant(const RenderBlock* renderer, const TextAutosizingClusterInfo& parentClusterInfo)
{
ASSERT(isAutosizingContainer(renderer));
LayoutUnit contentWidth = renderer->contentLogicalWidth();
LayoutUnit clusterTextWidth = parentClusterInfo.blockContainingAllText->contentLogicalWidth();
return contentWidth > clusterTextWidth;
}
bool TextAutosizer::isIndependentDescendant(const RenderBlock* renderer)
{
ASSERT(isAutosizingContainer(renderer));
RenderBlock* containingBlock = renderer->containingBlock();
return renderer->isRenderView()
|| renderer->isFloating()
|| renderer->isOutOfFlowPositioned()
|| renderer->isTableCell()
|| renderer->isTableCaption()
|| renderer->isFlexibleBoxIncludingDeprecated()
|| renderer->hasColumns()
|| (containingBlock && containingBlock->isHorizontalWritingMode() != renderer->isHorizontalWritingMode())
|| renderer->style()->isDisplayReplacedType()
|| renderer->isTextArea()
|| renderer->style()->userModify() != READ_ONLY;
}
bool TextAutosizer::isAutosizingCluster(const RenderBlock* renderer, TextAutosizingClusterInfo& parentClusterInfo)
{
ASSERT(isAutosizingContainer(renderer));
return isNarrowDescendant(renderer, parentClusterInfo)
|| isWiderDescendant(renderer, parentClusterInfo)
|| isIndependentDescendant(renderer);
}
bool TextAutosizer::containerShouldBeAutosized(const RenderBlock* container)
{
if (containerContainsOneOfTags(container, formInputTags()))
return false;
if (containerIsRowOfLinks(container))
return false;
if (!container->style()->autoWrap())
return false;
return !contentHeightIsConstrained(container);
}
bool TextAutosizer::containerContainsOneOfTags(const RenderBlock* container, const Vector<QualifiedName>& tags)
{
const RenderObject* renderer = container;
while (renderer) {
const Node* rendererNode = renderer->node();
if (rendererNode && rendererNode->isElementNode()) {
if (tags.contains(toElement(rendererNode)->tagQName()))
return true;
}
renderer = nextInPreOrderSkippingDescendantsOfContainers(renderer, container);
}
return false;
}
bool TextAutosizer::containerIsRowOfLinks(const RenderObject* container)
{
int linkCount = 0;
RenderObject* renderer = container->nextInPreOrder(container);
float matchingFontSize = -1;
while (renderer) {
if (!isAutosizingContainer(renderer)) {
if (renderer->isText() && toRenderText(renderer)->text().impl()->stripWhiteSpace()->length() > 3)
return false;
if (!renderer->isInline())
return false;
if (renderer->isBR())
return false;
}
if (renderer->style()->isLink()) {
if (matchingFontSize < 0)
matchingFontSize = renderer->style()->specifiedFontSize();
else {
if (matchingFontSize != renderer->style()->specifiedFontSize())
return false;
}
linkCount++;
renderer = renderer->nextInPreOrderAfterChildren(container);
} else
renderer = nextInPreOrderSkippingDescendantsOfContainers(renderer, container);
}
return (linkCount >= 3);
}
bool TextAutosizer::contentHeightIsConstrained(const RenderBlock* container)
{
for (; container; container = container->containingBlock()) {
RenderStyle* style = container->style();
if (style->overflowY() >= OSCROLL)
return false;
if (style->height().isSpecified() || style->maxHeight().isSpecified() || container->isOutOfFlowPositioned()) {
return !container->isRoot() && !container->isBody();
}
if (container->isFloating())
return false;
}
return false;
}
bool TextAutosizer::compositeClusterShouldBeAutosized(Vector<TextAutosizingClusterInfo>& clusterInfos, float blockWidth)
{
float totalTextWidth = 0;
const float minLinesOfText = 4;
float minTextWidth = blockWidth * minLinesOfText;
for (size_t i = 0; i < clusterInfos.size(); ++i) {
if (clusterInfos[i].root->isTextArea() || (clusterInfos[i].root->style() && clusterInfos[i].root->style()->userModify() != READ_ONLY))
return true;
measureDescendantTextWidth(clusterInfos[i].blockContainingAllText, clusterInfos[i], minTextWidth, totalTextWidth);
if (totalTextWidth >= minTextWidth)
return true;
}
return false;
}
void TextAutosizer::measureDescendantTextWidth(const RenderBlock* container, TextAutosizingClusterInfo& clusterInfo, float minTextWidth, float& textWidth)
{
bool skipLocalText = !containerShouldBeAutosized(container);
RenderObject* descendant = nextInPreOrderSkippingDescendantsOfContainers(container, container);
while (descendant) {
if (!skipLocalText && descendant->isText()) {
textWidth += toRenderText(descendant)->renderedTextLength() * descendant->style()->specifiedFontSize();
} else if (isAutosizingContainer(descendant)) {
RenderBlock* descendantBlock = toRenderBlock(descendant);
if (!isAutosizingCluster(descendantBlock, clusterInfo))
measureDescendantTextWidth(descendantBlock, clusterInfo, minTextWidth, textWidth);
}
if (textWidth >= minTextWidth)
return;
descendant = nextInPreOrderSkippingDescendantsOfContainers(descendant, container);
}
}
RenderObject* TextAutosizer::nextInPreOrderSkippingDescendantsOfContainers(const RenderObject* current, const RenderObject* stayWithin)
{
if (current == stayWithin || !isAutosizingContainer(current))
return current->nextInPreOrder(stayWithin);
return current->nextInPreOrderAfterChildren(stayWithin);
}
const RenderBlock* TextAutosizer::findDeepestBlockContainingAllText(const RenderBlock* cluster)
{
size_t firstDepth = 0;
const RenderObject* firstTextLeaf = findFirstTextLeafNotInCluster(cluster, firstDepth, FirstToLast);
if (!firstTextLeaf)
return cluster;
size_t lastDepth = 0;
const RenderObject* lastTextLeaf = findFirstTextLeafNotInCluster(cluster, lastDepth, LastToFirst);
ASSERT(lastTextLeaf);
const RenderObject* firstNode = firstTextLeaf;
const RenderObject* lastNode = lastTextLeaf;
while (firstDepth > lastDepth) {
firstNode = firstNode->parent();
--firstDepth;
}
while (lastDepth > firstDepth) {
lastNode = lastNode->parent();
--lastDepth;
}
while (firstNode != lastNode) {
firstNode = firstNode->parent();
lastNode = lastNode->parent();
}
if (firstNode->isRenderBlock())
return toRenderBlock(firstNode);
RenderBlock* containingBlock = firstNode->containingBlock();
ASSERT(containingBlock->isDescendantOf(cluster));
return containingBlock;
}
const RenderObject* TextAutosizer::findFirstTextLeafNotInCluster(const RenderObject* parent, size_t& depth, TraversalDirection direction)
{
if (parent->isEmpty())
return parent->isText() ? parent : 0;
++depth;
const RenderObject* child = (direction == FirstToLast) ? parent->firstChild() : parent->lastChild();
while (child) {
if (!isAutosizingContainer(child) || !isIndependentDescendant(toRenderBlock(child))) {
const RenderObject* leaf = findFirstTextLeafNotInCluster(child, depth, direction);
if (leaf)
return leaf;
}
child = (direction == FirstToLast) ? child->nextSibling() : child->previousSibling();
}
--depth;
return 0;
}
namespace {
bool clusterWiderThanComparisonFn(const TextAutosizingClusterInfo& first, const TextAutosizingClusterInfo& second)
{
return first.root->contentLogicalWidth() > second.root->contentLogicalWidth();
}
}
void TextAutosizer::getNarrowDescendantsGroupedByWidth(const TextAutosizingClusterInfo& parentClusterInfo, Vector<Vector<TextAutosizingClusterInfo> >& groups)
{
ASSERT(parentClusterInfo.blockContainingAllText);
ASSERT(groups.isEmpty());
Vector<TextAutosizingClusterInfo> clusterInfos(parentClusterInfo.narrowDescendants);
if (clusterInfos.isEmpty())
return;
std::sort(clusterInfos.begin(), clusterInfos.end(), &clusterWiderThanComparisonFn);
groups.grow(1);
const float maxWidthDifferenceWithinGroup = 100;
for (size_t i = 0; i < clusterInfos.size(); ++i) {
groups.last().append(clusterInfos[i]);
if (i + 1 < clusterInfos.size()) {
LayoutUnit currentWidth = clusterInfos[i].root->contentLogicalWidth();
LayoutUnit nextWidth = clusterInfos[i + 1].root->contentLogicalWidth();
if (currentWidth - nextWidth > maxWidthDifferenceWithinGroup)
groups.grow(groups.size() + 1);
}
}
}
}