| /* |
| * Copyright (C) Research In Motion Limited 2010-2012. 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 "config.h" |
| |
| #if ENABLE(SVG) |
| #include "SVGTextMetricsBuilder.h" |
| |
| #include "RenderSVGInlineText.h" |
| #include "RenderSVGText.h" |
| #include "SVGTextRunRenderingContext.h" |
| |
| namespace WebCore { |
| |
| SVGTextMetricsBuilder::SVGTextMetricsBuilder() |
| : m_text(0) |
| , m_run(static_cast<const UChar*>(0), 0) |
| , m_textPosition(0) |
| , m_isComplexText(false) |
| , m_totalWidth(0) |
| { |
| } |
| |
| inline bool SVGTextMetricsBuilder::currentCharacterStartsSurrogatePair() const |
| { |
| return U16_IS_LEAD(m_run[m_textPosition]) && int(m_textPosition + 1) < m_run.charactersLength() && U16_IS_TRAIL(m_run[m_textPosition + 1]); |
| } |
| |
| bool SVGTextMetricsBuilder::advance() |
| { |
| m_textPosition += m_currentMetrics.length(); |
| if (int(m_textPosition) >= m_run.charactersLength()) |
| return false; |
| |
| if (m_isComplexText) |
| advanceComplexText(); |
| else |
| advanceSimpleText(); |
| |
| return m_currentMetrics.length() > 0; |
| } |
| |
| void SVGTextMetricsBuilder::advanceSimpleText() |
| { |
| GlyphBuffer glyphBuffer; |
| unsigned metricsLength = m_simpleWidthIterator->advance(m_textPosition + 1, &glyphBuffer); |
| if (!metricsLength) { |
| m_currentMetrics = SVGTextMetrics(); |
| return; |
| } |
| |
| float currentWidth = m_simpleWidthIterator->runWidthSoFar() - m_totalWidth; |
| m_totalWidth = m_simpleWidthIterator->runWidthSoFar(); |
| |
| #if ENABLE(SVG_FONTS) |
| m_currentMetrics = SVGTextMetrics(m_text, m_textPosition, metricsLength, currentWidth, m_simpleWidthIterator->lastGlyphName()); |
| #else |
| m_currentMetrics = SVGTextMetrics(m_text, m_textPosition, metricsLength, currentWidth, emptyString()); |
| #endif |
| } |
| |
| void SVGTextMetricsBuilder::advanceComplexText() |
| { |
| unsigned metricsLength = currentCharacterStartsSurrogatePair() ? 2 : 1; |
| m_currentMetrics = SVGTextMetrics::measureCharacterRange(m_text, m_textPosition, metricsLength); |
| m_complexStartToCurrentMetrics = SVGTextMetrics::measureCharacterRange(m_text, 0, m_textPosition + metricsLength); |
| ASSERT(m_currentMetrics.length() == metricsLength); |
| |
| // Frequent case for Arabic text: when measuring a single character the arabic isolated form is taken |
| // when rendering the glyph "in context" (with it's surrounding characters) it changes due to shaping. |
| // So whenever currentWidth != currentMetrics.width(), we are processing a text run whose length is |
| // not equal to the sum of the individual lengths of the glyphs, when measuring them isolated. |
| float currentWidth = m_complexStartToCurrentMetrics.width() - m_totalWidth; |
| if (currentWidth != m_currentMetrics.width()) |
| m_currentMetrics.setWidth(currentWidth); |
| |
| m_totalWidth = m_complexStartToCurrentMetrics.width(); |
| } |
| |
| void SVGTextMetricsBuilder::initializeMeasurementWithTextRenderer(RenderSVGInlineText* text) |
| { |
| m_text = text; |
| m_textPosition = 0; |
| m_currentMetrics = SVGTextMetrics(); |
| m_complexStartToCurrentMetrics = SVGTextMetrics(); |
| m_totalWidth = 0; |
| |
| const Font& scaledFont = text->scaledFont(); |
| m_run = SVGTextMetrics::constructTextRun(text, text->characters(), 0, text->textLength()); |
| m_isComplexText = scaledFont.codePath(m_run) == Font::Complex; |
| |
| if (m_isComplexText) |
| m_simpleWidthIterator.clear(); |
| else |
| m_simpleWidthIterator = adoptPtr(new WidthIterator(&scaledFont, m_run)); |
| } |
| |
| struct MeasureTextData { |
| MeasureTextData(SVGCharacterDataMap* characterDataMap) |
| : allCharactersMap(characterDataMap) |
| , lastCharacter(0) |
| , processRenderer(false) |
| , valueListPosition(0) |
| , skippedCharacters(0) |
| { |
| } |
| |
| SVGCharacterDataMap* allCharactersMap; |
| const UChar* lastCharacter; |
| bool processRenderer; |
| unsigned valueListPosition; |
| unsigned skippedCharacters; |
| }; |
| |
| void SVGTextMetricsBuilder::measureTextRenderer(RenderSVGInlineText* text, MeasureTextData* data) |
| { |
| ASSERT(text); |
| |
| SVGTextLayoutAttributes* attributes = text->layoutAttributes(); |
| Vector<SVGTextMetrics>* textMetricsValues = &attributes->textMetricsValues(); |
| if (data->processRenderer) { |
| if (data->allCharactersMap) |
| attributes->clear(); |
| else |
| textMetricsValues->clear(); |
| } |
| |
| initializeMeasurementWithTextRenderer(text); |
| bool preserveWhiteSpace = text->style().whiteSpace() == PRE; |
| int surrogatePairCharacters = 0; |
| |
| while (advance()) { |
| const UChar* currentCharacter = m_run.data16(m_textPosition); |
| if (*currentCharacter == ' ' && !preserveWhiteSpace && (!data->lastCharacter || *data->lastCharacter == ' ')) { |
| if (data->processRenderer) |
| textMetricsValues->append(SVGTextMetrics(SVGTextMetrics::SkippedSpaceMetrics)); |
| if (data->allCharactersMap) |
| data->skippedCharacters += m_currentMetrics.length(); |
| continue; |
| } |
| |
| if (data->processRenderer) { |
| if (data->allCharactersMap) { |
| const SVGCharacterDataMap::const_iterator it = data->allCharactersMap->find(data->valueListPosition + m_textPosition - data->skippedCharacters - surrogatePairCharacters + 1); |
| if (it != data->allCharactersMap->end()) |
| attributes->characterDataMap().set(m_textPosition + 1, it->value); |
| } |
| textMetricsValues->append(m_currentMetrics); |
| } |
| |
| if (data->allCharactersMap && currentCharacterStartsSurrogatePair()) |
| surrogatePairCharacters++; |
| |
| data->lastCharacter = currentCharacter; |
| } |
| |
| if (!data->allCharactersMap) |
| return; |
| |
| data->valueListPosition += m_textPosition - data->skippedCharacters; |
| data->skippedCharacters = 0; |
| } |
| |
| void SVGTextMetricsBuilder::walkTree(RenderObject* start, RenderSVGInlineText* stopAtLeaf, MeasureTextData* data) |
| { |
| for (RenderObject* child = start->firstChildSlow(); child; child = child->nextSibling()) { |
| if (child->isSVGInlineText()) { |
| RenderSVGInlineText* text = toRenderSVGInlineText(child); |
| if (stopAtLeaf && stopAtLeaf != text) { |
| data->processRenderer = false; |
| measureTextRenderer(text, data); |
| continue; |
| } |
| |
| data->processRenderer = true; |
| measureTextRenderer(text, data); |
| if (stopAtLeaf) |
| return; |
| |
| continue; |
| } |
| |
| if (!child->isSVGInline()) |
| continue; |
| |
| walkTree(child, stopAtLeaf, data); |
| } |
| } |
| |
| void SVGTextMetricsBuilder::measureTextRenderer(RenderSVGInlineText* text) |
| { |
| ASSERT(text); |
| |
| RenderSVGText* textRoot = RenderSVGText::locateRenderSVGTextAncestor(text); |
| if (!textRoot) |
| return; |
| |
| MeasureTextData data(0); |
| walkTree(textRoot, text, &data); |
| } |
| |
| void SVGTextMetricsBuilder::buildMetricsAndLayoutAttributes(RenderSVGText* textRoot, RenderSVGInlineText* stopAtLeaf, SVGCharacterDataMap& allCharactersMap) |
| { |
| ASSERT(textRoot); |
| MeasureTextData data(&allCharactersMap); |
| walkTree(textRoot, stopAtLeaf, &data); |
| } |
| |
| } |
| |
| #endif // ENABLE(SVG) |