| Index: Source/core/svg/SVGGlyphMap.h
|
| diff --git a/Source/core/svg/SVGGlyphMap.h b/Source/core/svg/SVGGlyphMap.h
|
| index 8e20f0644ca99495c52a663c0d664ab0c063bafb..d46489cfd722eeb4238e3d77ac5e67a8893ea63e 100644
|
| --- a/Source/core/svg/SVGGlyphMap.h
|
| +++ b/Source/core/svg/SVGGlyphMap.h
|
| @@ -21,6 +21,7 @@
|
| #define SVGGlyphMap_h
|
|
|
| #if ENABLE(SVG_FONTS)
|
| +#include "core/svg/SVGParserUtilities.h"
|
| #include "platform/fonts/Latin1TextIterator.h"
|
| #include "platform/fonts/SVGGlyph.h"
|
| #include "platform/text/SurrogatePairAwareTextIterator.h"
|
| @@ -49,16 +50,16 @@ class SVGGlyphMap {
|
| public:
|
| SVGGlyphMap() : m_currentPriority(0) { }
|
|
|
| - void addGlyph(const String& glyphName, const String& unicodeString, SVGGlyph glyph)
|
| + void addGlyph(const String& glyphIdentifier, const String& unicodeString, SVGGlyph glyph)
|
| {
|
| - ASSERT(!glyphName.isEmpty() || !unicodeString.isEmpty());
|
| + ASSERT(!glyphIdentifier.isEmpty() || !unicodeString.isEmpty());
|
|
|
| - bool hasGlyphName = !glyphName.isEmpty();
|
| + bool hasGlyphIdentifier = !glyphIdentifier.isEmpty();
|
| if (unicodeString.isEmpty()) {
|
| - // Register named glyph in the named glyph map and in the glyph table.
|
| - ASSERT(hasGlyphName);
|
| + // Register glyphs with 'id's in the id glyph map and in the glyph table.
|
| + ASSERT(hasGlyphIdentifier);
|
| appendToGlyphTable(glyph);
|
| - m_namedGlyphs.add(glyphName, glyph.tableEntry);
|
| + m_idGlyphs.add(glyphIdentifier, glyph.tableEntry);
|
| return;
|
| }
|
|
|
| @@ -83,8 +84,10 @@ public:
|
|
|
| // If the glyph is named, also add it to the named glyph name, and to the glyph table in both cases.
|
| appendToGlyphTable(lastGlyph);
|
| - if (hasGlyphName)
|
| - m_namedGlyphs.add(glyphName, lastGlyph.tableEntry);
|
| + if (!lastGlyph.glyphName.isEmpty())
|
| + m_namedGlyphs.add(lastGlyph.glyphName, lastGlyph.tableEntry);
|
| + if (hasGlyphIdentifier)
|
| + m_idGlyphs.add(glyphIdentifier, lastGlyph.tableEntry);
|
| }
|
|
|
| void appendToGlyphTable(SVGGlyph& glyph)
|
| @@ -120,13 +123,48 @@ public:
|
| std::sort(glyphs.begin(), glyphs.end(), compareGlyphPriority);
|
| }
|
|
|
| + void collectGlyphsForStringExact(const String& string, Vector<SVGGlyph>& glyphs) const
|
| + {
|
| + unsigned length = string.length();
|
| +
|
| + if (!length)
|
| + return;
|
| +
|
| + RefPtr<GlyphMapNode> node;
|
| + if (string.is8Bit()) {
|
| + Latin1TextIterator textIterator(string.characters8(), 0, length, length);
|
| + node = findNode(textIterator);
|
| + } else {
|
| + SurrogatePairAwareTextIterator textIterator(string.characters16(), 0, length, length);
|
| + node = findNode(textIterator);
|
| + }
|
| +
|
| + if (node)
|
| + glyphs.appendVector(node->glyphs);
|
| + }
|
| +
|
| + void collectGlyphsForUnicodeRange(const UnicodeRange& unicodeRange, Vector<SVGGlyph>& glyphs) const
|
| + {
|
| + for (unsigned character = unicodeRange.first; character <= unicodeRange.second; ++character) {
|
| + if (RefPtr<GlyphMapNode> node = m_rootLayer.get(character))
|
| + glyphs.appendVector(node->glyphs);
|
| + }
|
| + }
|
| +
|
| void clear()
|
| {
|
| m_rootLayer.clear();
|
| m_glyphTable.clear();
|
| + m_idGlyphs.clear();
|
| + m_namedGlyphs.clear();
|
| m_currentPriority = 0;
|
| }
|
|
|
| + void dropNamedGlyphMap()
|
| + {
|
| + m_namedGlyphs.clear();
|
| + }
|
| +
|
| const SVGGlyph& svgGlyphForGlyph(Glyph glyph) const
|
| {
|
| if (!glyph || glyph > m_glyphTable.size()) {
|
| @@ -136,6 +174,11 @@ public:
|
| return m_glyphTable[glyph - 1];
|
| }
|
|
|
| + const SVGGlyph& glyphIdentifierForAltGlyphReference(const String& glyphIdentifier) const
|
| + {
|
| + return svgGlyphForGlyph(m_idGlyphs.get(glyphIdentifier));
|
| + }
|
| +
|
| const SVGGlyph& glyphIdentifierForGlyphName(const String& glyphName) const
|
| {
|
| return svgGlyphForGlyph(m_namedGlyphs.get(glyphName));
|
| @@ -164,6 +207,25 @@ private:
|
| }
|
|
|
| template<typename Iterator>
|
| + PassRefPtr<GlyphMapNode> findNode(Iterator& textIterator) const
|
| + {
|
| + const GlyphMapLayer* currentLayer = &m_rootLayer;
|
| +
|
| + RefPtr<GlyphMapNode> node;
|
| + UChar32 character = 0;
|
| + unsigned clusterLength = 0;
|
| + while (textIterator.consume(character, clusterLength)) {
|
| + node = currentLayer->get(character);
|
| + if (!node)
|
| + break;
|
| + currentLayer = &node->children;
|
| + textIterator.advance(clusterLength);
|
| + }
|
| +
|
| + return node.release();
|
| + }
|
| +
|
| + template<typename Iterator>
|
| void collectGlyphsForIterator(Iterator& textIterator, Vector<SVGGlyph>& glyphs)
|
| {
|
| GlyphMapLayer* currentLayer = &m_rootLayer;
|
| @@ -183,6 +245,7 @@ private:
|
| GlyphMapLayer m_rootLayer;
|
| Vector<SVGGlyph> m_glyphTable;
|
| HashMap<String, Glyph> m_namedGlyphs;
|
| + HashMap<String, Glyph> m_idGlyphs;
|
| int m_currentPriority;
|
| };
|
|
|
|
|