diff options
author | Hal Canary <halcanary@google.com> | 2018-03-28 09:51:00 -0700 |
---|---|---|
committer | Skia Commit-Bot <skia-commit-bot@chromium.org> | 2018-03-29 20:32:15 +0000 |
commit | 1521c8a2b37360c7ac3c091b1992b71026be4ea6 (patch) | |
tree | fade263307063310a0131faf9517ceb31e9630e9 /src/pdf/SkPDFDevice.cpp | |
parent | 04bea3bf76e2cf30c438c876b1c2d6847bcf797f (diff) |
SkPDF/Clusterator: expose to the light of a unit test
Change-Id: I5667da133f608ab42f83daba3424134b8e956b1e
Reviewed-on: https://skia-review.googlesource.com/117006
Reviewed-by: Florin Malita <fmalita@chromium.org>
Commit-Queue: Hal Canary <halcanary@google.com>
Diffstat (limited to 'src/pdf/SkPDFDevice.cpp')
-rw-r--r-- | src/pdf/SkPDFDevice.cpp | 195 |
1 files changed, 8 insertions, 187 deletions
diff --git a/src/pdf/SkPDFDevice.cpp b/src/pdf/SkPDFDevice.cpp index 76d6005210..98e34b004e 100644 --- a/src/pdf/SkPDFDevice.cpp +++ b/src/pdf/SkPDFDevice.cpp @@ -13,6 +13,7 @@ #include "SkBitmapKey.h" #include "SkCanvas.h" #include "SkClipOpPriv.h" +#include "SkClusterator.h" #include "SkColor.h" #include "SkColorFilter.h" #include "SkDraw.h" @@ -1049,164 +1050,8 @@ private: bool fInitialized = false; const bool fDefaultPositioning; }; - -/** Given the m-to-n glyph-to-character mapping data (as returned by - harfbuzz), iterate over the clusters. */ -class Clusterator { -public: - Clusterator() : fClusters(nullptr), fUtf8Text(nullptr), fGlyphCount(0), fTextByteLength(0) {} - explicit Clusterator(uint32_t glyphCount) - : fClusters(nullptr) - , fUtf8Text(nullptr) - , fGlyphCount(glyphCount) - , fTextByteLength(0) {} - // The clusters[] array is an array of offsets into utf8Text[], - // one offset for each glyph. See SkTextBlobBuilder for more info. - Clusterator(const uint32_t* clusters, - const char* utf8Text, - uint32_t glyphCount, - uint32_t textByteLength) - : fClusters(clusters) - , fUtf8Text(utf8Text) - , fGlyphCount(glyphCount) - , fTextByteLength(textByteLength) - { - // "ReversedChars" is how PDF deals with RTL text. - // We set it to true only if all cluster indices are decreasing. - if (fUtf8Text && fClusters && fGlyphCount > 1) { - int clusterCount = 1; - for (uint32_t i = 0; i + 1 < fGlyphCount; ++i) { - if (fClusters[i + 1] != fClusters[i]) { - if (fClusters[i + 1] > fClusters[i]) { - return; - } - ++clusterCount; - } - } - fReversedChars = clusterCount > 1; - } - } - struct Cluster { - const char* fUtf8Text; - uint32_t fTextByteLength; - uint32_t fGlyphIndex; - uint32_t fGlyphCount; - explicit operator bool() const { return fGlyphCount != 0; } - }; - // True if this looks like right-to-left text. - bool reversedChars() const { return fReversedChars; } - Cluster next() { - if (fGlyphIndex >= fGlyphCount) { - return Cluster{nullptr, 0, 0, 0}; - } - if (!fClusters || !fUtf8Text) { - return Cluster{nullptr, 0, fGlyphIndex++, 1}; - } - uint32_t clusterGlyphIndex = fGlyphIndex; - uint32_t cluster = fClusters[clusterGlyphIndex]; - do { - ++fGlyphIndex; - } while (fGlyphIndex < fGlyphCount && cluster == fClusters[fGlyphIndex]); - uint32_t clusterGlyphCount = fGlyphIndex - clusterGlyphIndex; - uint32_t clusterEnd = fTextByteLength; - for (unsigned i = 0; i < fGlyphCount; ++i) { - uint32_t c = fClusters[i]; - if (c > cluster && c < clusterEnd) { - clusterEnd = c; - } - } - uint32_t clusterLen = clusterEnd - cluster; - return Cluster{fUtf8Text + cluster, clusterLen, clusterGlyphIndex, clusterGlyphCount}; - } - -private: - const uint32_t* fClusters; - const char* fUtf8Text; - uint32_t fGlyphCount; - uint32_t fTextByteLength; - uint32_t fGlyphIndex = 0; - bool fReversedChars = false; -}; - -struct TextStorage { - SkAutoTMalloc<char> fUtf8textStorage; - SkAutoTMalloc<uint32_t> fClusterStorage; - SkAutoTMalloc<SkGlyphID> fGlyphStorage; -}; } // namespace -/** Given some unicode text (as passed to drawText(), convert to - glyphs (via primitive shaping), while preserving - glyph-to-character mapping information. */ -static Clusterator make_clusterator( - const void* sourceText, - size_t sourceByteCount, - const SkPaint& paint, - TextStorage* storage, - int glyphCount) { - SkASSERT(SkPaint::kGlyphID_TextEncoding != paint.getTextEncoding()); - SkASSERT(glyphCount == paint.textToGlyphs(sourceText, sourceByteCount, nullptr)); - SkASSERT(glyphCount > 0); - storage->fGlyphStorage.reset(SkToSizeT(glyphCount)); - (void)paint.textToGlyphs(sourceText, sourceByteCount, storage->fGlyphStorage.get()); - storage->fClusterStorage.reset(SkToSizeT(glyphCount)); - uint32_t* clusters = storage->fClusterStorage.get(); - uint32_t utf8ByteCount = 0; - const char* utf8Text = nullptr; - switch (paint.getTextEncoding()) { - case SkPaint::kUTF8_TextEncoding: { - const char* txtPtr = (const char*)sourceText; - for (int i = 0; i < glyphCount; ++i) { - clusters[i] = SkToU32(txtPtr - (const char*)sourceText); - txtPtr += SkUTF8_LeadByteToCount(*(const unsigned char*)txtPtr); - SkASSERT(txtPtr <= (const char*)sourceText + sourceByteCount); - } - SkASSERT(txtPtr == (const char*)sourceText + sourceByteCount); - utf8ByteCount = SkToU32(sourceByteCount); - utf8Text = (const char*)sourceText; - break; - } - case SkPaint::kUTF16_TextEncoding: { - const uint16_t* utf16ptr = (const uint16_t*)sourceText; - int utf16count = SkToInt(sourceByteCount / sizeof(uint16_t)); - utf8ByteCount = SkToU32(SkUTF16_ToUTF8(utf16ptr, utf16count)); - storage->fUtf8textStorage.reset(utf8ByteCount); - char* txtPtr = storage->fUtf8textStorage.get(); - utf8Text = txtPtr; - int clusterIndex = 0; - while (utf16ptr < (const uint16_t*)sourceText + utf16count) { - clusters[clusterIndex++] = SkToU32(txtPtr - utf8Text); - SkUnichar uni = SkUTF16_NextUnichar(&utf16ptr); - txtPtr += SkUTF8_FromUnichar(uni, txtPtr); - } - SkASSERT(clusterIndex == glyphCount); - SkASSERT(txtPtr == storage->fUtf8textStorage.get() + utf8ByteCount); - SkASSERT(utf16ptr == (const uint16_t*)sourceText + utf16count); - break; - } - case SkPaint::kUTF32_TextEncoding: { - const SkUnichar* utf32 = (const SkUnichar*)sourceText; - int utf32count = SkToInt(sourceByteCount / sizeof(SkUnichar)); - SkASSERT(glyphCount == utf32count); - for (int i = 0; i < utf32count; ++i) { - utf8ByteCount += SkToU32(SkUTF8_FromUnichar(utf32[i])); - } - storage->fUtf8textStorage.reset(SkToSizeT(utf8ByteCount)); - char* txtPtr = storage->fUtf8textStorage.get(); - utf8Text = txtPtr; - for (int i = 0; i < utf32count; ++i) { - clusters[i] = SkToU32(txtPtr - utf8Text); - txtPtr += SkUTF8_FromUnichar(utf32[i], txtPtr); - } - break; - } - default: - SkDEBUGFAIL(""); - break; - } - return Clusterator(clusters, utf8Text, SkToU32(glyphCount), utf8ByteCount); -} - static SkUnichar map_glyph(const SkTDArray<SkUnichar>& glyphToUnicode, SkGlyphID glyph) { return SkToInt(glyph) < glyphToUnicode.count() ? glyphToUnicode[SkToInt(glyph)] : -1; } @@ -1355,38 +1200,14 @@ void SkPDFDevice::internalDrawText( if (!metrics) { return; } - int glyphCount = paint.textToGlyphs(sourceText, sourceByteCount, nullptr); - if (glyphCount <= 0) { + SkClusterator clusterator(sourceText, sourceByteCount, paint, + clusters, textByteLength, utf8Text); + const SkGlyphID* glyphs = clusterator.glyphs(); + uint32_t glyphCount = clusterator.glyphCount(); + if (glyphCount == 0) { return; } - // These three heap buffers are only used in the case where no glyphs - // are passed to drawText() (most clients pass glyphs or a textblob). - TextStorage storage; - const SkGlyphID* glyphs = nullptr; - Clusterator clusterator; - if (textByteLength > 0) { - SkASSERT(glyphCount == SkToInt(sourceByteCount / sizeof(SkGlyphID))); - glyphs = (const SkGlyphID*)sourceText; - clusterator = Clusterator(clusters, utf8Text, SkToU32(glyphCount), textByteLength); - SkASSERT(clusters); - SkASSERT(utf8Text); - SkASSERT(srcPaint.getTextEncoding() == SkPaint::kGlyphID_TextEncoding); - SkASSERT(glyphCount == paint.textToGlyphs(sourceText, sourceByteCount, nullptr)); - } else if (SkPaint::kGlyphID_TextEncoding == srcPaint.getTextEncoding()) { - SkASSERT(glyphCount == SkToInt(sourceByteCount / sizeof(SkGlyphID))); - glyphs = (const SkGlyphID*)sourceText; - clusterator = Clusterator(SkToU32(glyphCount)); - SkASSERT(glyphCount == paint.textToGlyphs(sourceText, sourceByteCount, nullptr)); - SkASSERT(nullptr == clusters); - SkASSERT(nullptr == utf8Text); - } else { - SkASSERT(nullptr == clusters); - SkASSERT(nullptr == utf8Text); - clusterator = make_clusterator(sourceText, sourceByteCount, srcPaint, - &storage, glyphCount); - glyphs = storage.fGlyphStorage; - } bool defaultPositioning = (positioning == SkTextBlob::kDefault_Positioning); paint.setHinting(SkPaint::kNo_Hinting); @@ -1406,7 +1227,7 @@ void SkPDFDevice::internalDrawText( /* SkPaint::kRight_Align */ -1.0f; if (defaultPositioning && alignment != SkPaint::kLeft_Align) { SkScalar advance = 0; - for (int i = 0; i < glyphCount; ++i) { + for (uint32_t i = 0; i < glyphCount; ++i) { advance += advanceScale * glyphCache->getGlyphIDAdvance(glyphs[i]).fAdvanceX; } offset.offset(alignmentFactor * advance, 0); @@ -1442,7 +1263,7 @@ void SkPDFDevice::internalDrawText( offset); SkPDFFont* font = nullptr; - while (Clusterator::Cluster c = clusterator.next()) { + while (SkClusterator::Cluster c = clusterator.next()) { int index = c.fGlyphIndex; int glyphLimit = index + c.fGlyphCount; |