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; |
}; |