Index: src/ports/SkFontHost_win_dw.cpp |
diff --git a/src/ports/SkFontHost_win_dw.cpp b/src/ports/SkFontHost_win_dw.cpp |
deleted file mode 100644 |
index fa21df537c3ee6fbc9619bfd3cfe435c92cb1905..0000000000000000000000000000000000000000 |
--- a/src/ports/SkFontHost_win_dw.cpp |
+++ /dev/null |
@@ -1,623 +0,0 @@ |
-/* |
- * Copyright 2011 Google Inc. |
- * |
- * Use of this source code is governed by a BSD-style license that can be |
- * found in the LICENSE file. |
- */ |
- |
-#include "SkTypes.h" |
-#undef GetGlyphIndices |
- |
-#include "SkDWrite.h" |
-#include "SkDWriteGeometrySink.h" |
-#include "SkEndian.h" |
-#include "SkGlyph.h" |
-#include "SkHRESULT.h" |
-#include "SkMaskGamma.h" |
-#include "SkMatrix22.h" |
-#include "SkOTTable_EBLC.h" |
-#include "SkOTTable_EBSC.h" |
-#include "SkPath.h" |
-#include "SkScalerContext.h" |
-#include "SkScalerContext_win_dw.h" |
-#include "SkTScopedComPtr.h" |
-#include "SkTypeface_win_dw.h" |
- |
-#include <dwrite.h> |
- |
-static bool isLCD(const SkScalerContext::Rec& rec) { |
- return SkMask::kLCD16_Format == rec.fMaskFormat || |
- SkMask::kLCD32_Format == rec.fMaskFormat; |
-} |
- |
-static bool hasBitmapStrike(DWriteFontTypeface* typeface, int size) { |
- { |
- AutoTDWriteTable<SkOTTableEmbeddedBitmapLocation> eblc(typeface->fDWriteFontFace.get()); |
- if (!eblc.fExists) { |
- return false; |
- } |
- if (eblc.fSize < sizeof(SkOTTableEmbeddedBitmapLocation)) { |
- return false; |
- } |
- if (eblc->version != SkOTTableEmbeddedBitmapLocation::version_initial) { |
- return false; |
- } |
- |
- uint32_t numSizes = SkEndianSwap32(eblc->numSizes); |
- if (eblc.fSize < sizeof(SkOTTableEmbeddedBitmapLocation) + |
- sizeof(SkOTTableEmbeddedBitmapLocation::BitmapSizeTable) * numSizes) |
- { |
- return false; |
- } |
- |
- const SkOTTableEmbeddedBitmapLocation::BitmapSizeTable* sizeTable = |
- SkTAfter<const SkOTTableEmbeddedBitmapLocation::BitmapSizeTable>(eblc.get()); |
- for (uint32_t i = 0; i < numSizes; ++i, ++sizeTable) { |
- if (sizeTable->ppemX == size && sizeTable->ppemY == size) { |
- // TODO: determine if we should dig through IndexSubTableArray/IndexSubTable |
- // to determine the actual number of glyphs with bitmaps. |
- |
- // TODO: Ensure that the bitmaps actually cover a significant portion of the strike. |
- |
- //TODO: Endure that the bitmaps are bi-level. |
- if (sizeTable->endGlyphIndex >= sizeTable->startGlyphIndex + 3) { |
- return true; |
- } |
- } |
- } |
- } |
- |
- { |
- AutoTDWriteTable<SkOTTableEmbeddedBitmapScaling> ebsc(typeface->fDWriteFontFace.get()); |
- if (!ebsc.fExists) { |
- return false; |
- } |
- if (ebsc.fSize < sizeof(SkOTTableEmbeddedBitmapScaling)) { |
- return false; |
- } |
- if (ebsc->version != SkOTTableEmbeddedBitmapScaling::version_initial) { |
- return false; |
- } |
- |
- uint32_t numSizes = SkEndianSwap32(ebsc->numSizes); |
- if (ebsc.fSize < sizeof(SkOTTableEmbeddedBitmapScaling) + |
- sizeof(SkOTTableEmbeddedBitmapScaling::BitmapScaleTable) * numSizes) |
- { |
- return false; |
- } |
- |
- const SkOTTableEmbeddedBitmapScaling::BitmapScaleTable* scaleTable = |
- SkTAfter<const SkOTTableEmbeddedBitmapScaling::BitmapScaleTable>(ebsc.get()); |
- for (uint32_t i = 0; i < numSizes; ++i, ++scaleTable) { |
- if (scaleTable->ppemX == size && scaleTable->ppemY == size) { |
- // EBSC tables are normally only found in bitmap only fonts. |
- return true; |
- } |
- } |
- } |
- |
- return false; |
-} |
- |
-static bool bothZero(SkScalar a, SkScalar b) { |
- return 0 == a && 0 == b; |
-} |
- |
-// returns false if there is any non-90-rotation or skew |
-static bool isAxisAligned(const SkScalerContext::Rec& rec) { |
- return 0 == rec.fPreSkewX && |
- (bothZero(rec.fPost2x2[0][1], rec.fPost2x2[1][0]) || |
- bothZero(rec.fPost2x2[0][0], rec.fPost2x2[1][1])); |
-} |
- |
-SkScalerContext_DW::SkScalerContext_DW(DWriteFontTypeface* typeface, |
- const SkDescriptor* desc) |
- : SkScalerContext(typeface, desc) |
- , fTypeface(SkRef(typeface)) |
- , fGlyphCount(-1) { |
- |
- // In general, all glyphs should use CLEARTYPE_NATURAL_SYMMETRIC |
- // except when bi-level rendering is requested or there are embedded |
- // bi-level bitmaps (and the embedded bitmap flag is set and no rotation). |
- // |
- // DirectWrite's IDWriteFontFace::GetRecommendedRenderingMode does not do |
- // this. As a result, determine the actual size of the text and then see if |
- // there are any embedded bi-level bitmaps of that size. If there are, then |
- // force bitmaps by requesting bi-level rendering. |
- // |
- // FreeType allows for separate ppemX and ppemY, but DirectWrite assumes |
- // square pixels and only uses ppemY. Therefore the transform must track any |
- // non-uniform x-scale. |
- // |
- // Also, rotated glyphs should have the same absolute advance widths as |
- // horizontal glyphs and the subpixel flag should not affect glyph shapes. |
- |
- // A is the total matrix. |
- SkMatrix A; |
- fRec.getSingleMatrix(&A); |
- |
- // h is where A maps the horizontal baseline. |
- SkPoint h = SkPoint::Make(SK_Scalar1, 0); |
- A.mapPoints(&h, 1); |
- |
- // G is the Givens Matrix for A (rotational matrix where GA[0][1] == 0). |
- SkMatrix G; |
- SkComputeGivensRotation(h, &G); |
- |
- // GA is the matrix A with rotation removed. |
- SkMatrix GA(G); |
- GA.preConcat(A); |
- |
- // realTextSize is the actual device size we want (as opposed to the size the user requested). |
- // gdiTextSize is the size we request when GDI compatible. |
- // If the scale is negative, this means the matrix will do the flip anyway. |
- SkScalar realTextSize = SkScalarAbs(GA.get(SkMatrix::kMScaleY)); |
- // Due to floating point math, the lower bits are suspect. Round carefully. |
- SkScalar gdiTextSize = SkScalarRoundToScalar(realTextSize * 64.0f) / 64.0f; |
- if (gdiTextSize == 0) { |
- gdiTextSize = SK_Scalar1; |
- } |
- |
- bool bitmapRequested = SkToBool(fRec.fFlags & SkScalerContext::kEmbeddedBitmapText_Flag); |
- bool hasBitmap = false; |
- bool axisAlignedBitmap = false; |
- if (bitmapRequested) { |
- hasBitmap = hasBitmapStrike(typeface, SkScalarTruncToInt(gdiTextSize)); |
- axisAlignedBitmap = isAxisAligned(fRec); |
- } |
- |
- // If the user requested aliased, do so with aliased compatible metrics. |
- if (SkMask::kBW_Format == fRec.fMaskFormat) { |
- fTextSizeRender = gdiTextSize; |
- fRenderingMode = DWRITE_RENDERING_MODE_ALIASED; |
- fTextureType = DWRITE_TEXTURE_ALIASED_1x1; |
- fTextSizeMeasure = gdiTextSize; |
- fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; |
- |
- // If we can use a bitmap, use gdi classic rendering and measurement. |
- // This will not always provide a bitmap, but matches expected behavior. |
- } else if (hasBitmap && axisAlignedBitmap) { |
- fTextSizeRender = gdiTextSize; |
- fRenderingMode = DWRITE_RENDERING_MODE_CLEARTYPE_GDI_CLASSIC; |
- fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; |
- fTextSizeMeasure = gdiTextSize; |
- fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; |
- |
- // If rotated but the horizontal text could have used a bitmap, |
- // render high quality rotated glyphs but measure using bitmap metrics. |
- } else if (hasBitmap) { |
- fTextSizeRender = gdiTextSize; |
- fRenderingMode = DWRITE_RENDERING_MODE_CLEARTYPE_NATURAL_SYMMETRIC; |
- fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; |
- fTextSizeMeasure = gdiTextSize; |
- fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; |
- |
- // The normal case is to use natural symmetric rendering and linear metrics. |
- } else { |
- fTextSizeRender = realTextSize; |
- fRenderingMode = DWRITE_RENDERING_MODE_CLEARTYPE_NATURAL_SYMMETRIC; |
- fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; |
- fTextSizeMeasure = realTextSize; |
- fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; |
- } |
- |
- if (this->isSubpixel()) { |
- fTextSizeMeasure = realTextSize; |
- fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; |
- } |
- |
- // Remove the realTextSize, as that is the text height scale currently in A. |
- SkScalar scale = SkScalarInvert(realTextSize); |
- |
- // fSkXform is the total matrix A without the text height scale. |
- fSkXform = A; |
- fSkXform.preScale(scale, scale); //remove the text height scale. |
- |
- fXform.m11 = SkScalarToFloat(fSkXform.getScaleX()); |
- fXform.m12 = SkScalarToFloat(fSkXform.getSkewY()); |
- fXform.m21 = SkScalarToFloat(fSkXform.getSkewX()); |
- fXform.m22 = SkScalarToFloat(fSkXform.getScaleY()); |
- fXform.dx = 0; |
- fXform.dy = 0; |
- |
- // GsA is the non-rotational part of A without the text height scale. |
- SkMatrix GsA(GA); |
- GsA.preScale(scale, scale); //remove text height scale, G is rotational so reorders with scale. |
- |
- fGsA.m11 = SkScalarToFloat(GsA.get(SkMatrix::kMScaleX)); |
- fGsA.m12 = SkScalarToFloat(GsA.get(SkMatrix::kMSkewY)); // This should be ~0. |
- fGsA.m21 = SkScalarToFloat(GsA.get(SkMatrix::kMSkewX)); |
- fGsA.m22 = SkScalarToFloat(GsA.get(SkMatrix::kMScaleY)); |
- fGsA.dx = 0; |
- fGsA.dy = 0; |
- |
- // fG_inv is G inverse, which is fairly simple since G is 2x2 rotational. |
- fG_inv.setAll(G.get(SkMatrix::kMScaleX), -G.get(SkMatrix::kMSkewX), G.get(SkMatrix::kMTransX), |
- -G.get(SkMatrix::kMSkewY), G.get(SkMatrix::kMScaleY), G.get(SkMatrix::kMTransY), |
- G.get(SkMatrix::kMPersp0), G.get(SkMatrix::kMPersp1), G.get(SkMatrix::kMPersp2)); |
-} |
- |
-SkScalerContext_DW::~SkScalerContext_DW() { |
-} |
- |
-unsigned SkScalerContext_DW::generateGlyphCount() { |
- if (fGlyphCount < 0) { |
- fGlyphCount = fTypeface->fDWriteFontFace->GetGlyphCount(); |
- } |
- return fGlyphCount; |
-} |
- |
-uint16_t SkScalerContext_DW::generateCharToGlyph(SkUnichar uni) { |
- uint16_t index = 0; |
- fTypeface->fDWriteFontFace->GetGlyphIndices(reinterpret_cast<UINT32*>(&uni), 1, &index); |
- return index; |
-} |
- |
-void SkScalerContext_DW::generateAdvance(SkGlyph* glyph) { |
- //Delta is the difference between the right/left side bearing metric |
- //and where the right/left side bearing ends up after hinting. |
- //DirectWrite does not provide this information. |
- glyph->fRsbDelta = 0; |
- glyph->fLsbDelta = 0; |
- |
- glyph->fAdvanceX = 0; |
- glyph->fAdvanceY = 0; |
- |
- uint16_t glyphId = glyph->getGlyphID(); |
- DWRITE_GLYPH_METRICS gm; |
- |
- if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || |
- DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) |
- { |
- HRVM(fTypeface->fDWriteFontFace->GetGdiCompatibleGlyphMetrics( |
- fTextSizeMeasure, |
- 1.0f, // pixelsPerDip |
- &fGsA, |
- DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode, |
- &glyphId, 1, |
- &gm), |
- "Could not get gdi compatible glyph metrics."); |
- } else { |
- HRVM(fTypeface->fDWriteFontFace->GetDesignGlyphMetrics(&glyphId, 1, &gm), |
- "Could not get design metrics."); |
- } |
- |
- DWRITE_FONT_METRICS dwfm; |
- fTypeface->fDWriteFontFace->GetMetrics(&dwfm); |
- SkScalar advanceX = SkScalarMulDiv(fTextSizeMeasure, |
- SkIntToScalar(gm.advanceWidth), |
- SkIntToScalar(dwfm.designUnitsPerEm)); |
- |
- if (!this->isSubpixel()) { |
- advanceX = SkScalarRoundToScalar(advanceX); |
- } |
- |
- SkVector vecs[1] = { { advanceX, 0 } }; |
- if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || |
- DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) |
- { |
- fG_inv.mapVectors(vecs, SK_ARRAY_COUNT(vecs)); |
- } else { |
- fSkXform.mapVectors(vecs, SK_ARRAY_COUNT(vecs)); |
- } |
- |
- glyph->fAdvanceX = SkScalarToFixed(vecs[0].fX); |
- glyph->fAdvanceY = SkScalarToFixed(vecs[0].fY); |
-} |
- |
-void SkScalerContext_DW::generateMetrics(SkGlyph* glyph) { |
- glyph->fWidth = 0; |
- |
- this->generateAdvance(glyph); |
- |
- //Measure raster size. |
- fXform.dx = SkFixedToFloat(glyph->getSubXFixed()); |
- fXform.dy = SkFixedToFloat(glyph->getSubYFixed()); |
- |
- FLOAT advance = 0; |
- |
- UINT16 glyphId = glyph->getGlyphID(); |
- |
- DWRITE_GLYPH_OFFSET offset; |
- offset.advanceOffset = 0.0f; |
- offset.ascenderOffset = 0.0f; |
- |
- DWRITE_GLYPH_RUN run; |
- run.glyphCount = 1; |
- run.glyphAdvances = &advance; |
- run.fontFace = fTypeface->fDWriteFontFace.get(); |
- run.fontEmSize = SkScalarToFloat(fTextSizeRender); |
- run.bidiLevel = 0; |
- run.glyphIndices = &glyphId; |
- run.isSideways = FALSE; |
- run.glyphOffsets = &offset; |
- |
- SkTScopedComPtr<IDWriteGlyphRunAnalysis> glyphRunAnalysis; |
- HRVM(fTypeface->fFactory->CreateGlyphRunAnalysis( |
- &run, |
- 1.0f, // pixelsPerDip, |
- &fXform, |
- fRenderingMode, |
- fMeasuringMode, |
- 0.0f, // baselineOriginX, |
- 0.0f, // baselineOriginY, |
- &glyphRunAnalysis), |
- "Could not create glyph run analysis."); |
- |
- RECT bbox; |
- HRVM(glyphRunAnalysis->GetAlphaTextureBounds(fTextureType, &bbox), |
- "Could not get texture bounds."); |
- |
- glyph->fWidth = SkToU16(bbox.right - bbox.left); |
- glyph->fHeight = SkToU16(bbox.bottom - bbox.top); |
- glyph->fLeft = SkToS16(bbox.left); |
- glyph->fTop = SkToS16(bbox.top); |
-} |
- |
-void SkScalerContext_DW::generateFontMetrics(SkPaint::FontMetrics* mx, |
- SkPaint::FontMetrics* my) { |
- if (!(mx || my)) |
- return; |
- |
- if (mx) { |
- sk_bzero(mx, sizeof(*mx)); |
- } |
- if (my) { |
- sk_bzero(my, sizeof(*my)); |
- } |
- |
- DWRITE_FONT_METRICS dwfm; |
- if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || |
- DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) |
- { |
- fTypeface->fDWriteFontFace->GetGdiCompatibleMetrics( |
- fTextSizeRender, |
- 1.0f, // pixelsPerDip |
- &fXform, |
- &dwfm); |
- } else { |
- fTypeface->fDWriteFontFace->GetMetrics(&dwfm); |
- } |
- |
- SkScalar upem = SkIntToScalar(dwfm.designUnitsPerEm); |
- if (mx) { |
- mx->fTop = -fTextSizeRender * SkIntToScalar(dwfm.ascent) / upem; |
- mx->fAscent = mx->fTop; |
- mx->fDescent = fTextSizeRender * SkIntToScalar(dwfm.descent) / upem; |
- mx->fBottom = mx->fDescent; |
- mx->fLeading = fTextSizeRender * SkIntToScalar(dwfm.lineGap) / upem; |
- mx->fXHeight = fTextSizeRender * SkIntToScalar(dwfm.xHeight) / upem; |
- mx->fUnderlineThickness = fTextSizeRender * SkIntToScalar(dwfm.underlineThickness) / upem; |
- mx->fUnderlinePosition = -(fTextSizeRender * SkIntToScalar(dwfm.underlinePosition) / upem); |
- |
- mx->fFlags |= SkPaint::FontMetrics::kUnderlineThinknessIsValid_Flag; |
- mx->fFlags |= SkPaint::FontMetrics::kUnderlinePositionIsValid_Flag; |
- } |
- |
- if (my) { |
- my->fTop = -fTextSizeRender * SkIntToScalar(dwfm.ascent) / upem; |
- my->fAscent = my->fTop; |
- my->fDescent = fTextSizeRender * SkIntToScalar(dwfm.descent) / upem; |
- my->fBottom = my->fDescent; |
- my->fLeading = fTextSizeRender * SkIntToScalar(dwfm.lineGap) / upem; |
- my->fXHeight = fTextSizeRender * SkIntToScalar(dwfm.xHeight) / upem; |
- my->fUnderlineThickness = fTextSizeRender * SkIntToScalar(dwfm.underlineThickness) / upem; |
- my->fUnderlinePosition = -(fTextSizeRender * SkIntToScalar(dwfm.underlinePosition) / upem); |
- |
- my->fFlags |= SkPaint::FontMetrics::kUnderlineThinknessIsValid_Flag; |
- my->fFlags |= SkPaint::FontMetrics::kUnderlinePositionIsValid_Flag; |
- } |
-} |
- |
-/////////////////////////////////////////////////////////////////////////////// |
- |
-#include "SkColorPriv.h" |
- |
-static void bilevel_to_bw(const uint8_t* SK_RESTRICT src, const SkGlyph& glyph) { |
- const int width = glyph.fWidth; |
- const size_t dstRB = (width + 7) >> 3; |
- uint8_t* SK_RESTRICT dst = static_cast<uint8_t*>(glyph.fImage); |
- |
- int byteCount = width >> 3; |
- int bitCount = width & 7; |
- |
- for (int y = 0; y < glyph.fHeight; ++y) { |
- if (byteCount > 0) { |
- for (int i = 0; i < byteCount; ++i) { |
- unsigned byte = 0; |
- byte |= src[0] & (1 << 7); |
- byte |= src[1] & (1 << 6); |
- byte |= src[2] & (1 << 5); |
- byte |= src[3] & (1 << 4); |
- byte |= src[4] & (1 << 3); |
- byte |= src[5] & (1 << 2); |
- byte |= src[6] & (1 << 1); |
- byte |= src[7] & (1 << 0); |
- dst[i] = byte; |
- src += 8; |
- } |
- } |
- if (bitCount > 0) { |
- unsigned byte = 0; |
- unsigned mask = 0x80; |
- for (int i = 0; i < bitCount; i++) { |
- byte |= (src[i]) & mask; |
- mask >>= 1; |
- } |
- dst[byteCount] = byte; |
- } |
- src += bitCount; |
- dst += dstRB; |
- } |
-} |
- |
-template<bool APPLY_PREBLEND> |
-static void rgb_to_a8(const uint8_t* SK_RESTRICT src, const SkGlyph& glyph, const uint8_t* table8) { |
- const size_t dstRB = glyph.rowBytes(); |
- const U16CPU width = glyph.fWidth; |
- uint8_t* SK_RESTRICT dst = static_cast<uint8_t*>(glyph.fImage); |
- |
- for (U16CPU y = 0; y < glyph.fHeight; y++) { |
- for (U16CPU i = 0; i < width; i++) { |
- U8CPU r = *(src++); |
- U8CPU g = *(src++); |
- U8CPU b = *(src++); |
- dst[i] = sk_apply_lut_if<APPLY_PREBLEND>((r + g + b) / 3, table8); |
- } |
- dst = (uint8_t*)((char*)dst + dstRB); |
- } |
-} |
- |
-template<bool APPLY_PREBLEND> |
-static void rgb_to_lcd16(const uint8_t* SK_RESTRICT src, const SkGlyph& glyph, |
- const uint8_t* tableR, const uint8_t* tableG, const uint8_t* tableB) { |
- const size_t dstRB = glyph.rowBytes(); |
- const U16CPU width = glyph.fWidth; |
- uint16_t* SK_RESTRICT dst = static_cast<uint16_t*>(glyph.fImage); |
- |
- for (U16CPU y = 0; y < glyph.fHeight; y++) { |
- for (U16CPU i = 0; i < width; i++) { |
- U8CPU r = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableR); |
- U8CPU g = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableG); |
- U8CPU b = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableB); |
- dst[i] = SkPack888ToRGB16(r, g, b); |
- } |
- dst = (uint16_t*)((char*)dst + dstRB); |
- } |
-} |
- |
-template<bool APPLY_PREBLEND> |
-static void rgb_to_lcd32(const uint8_t* SK_RESTRICT src, const SkGlyph& glyph, |
- const uint8_t* tableR, const uint8_t* tableG, const uint8_t* tableB) { |
- const size_t dstRB = glyph.rowBytes(); |
- const U16CPU width = glyph.fWidth; |
- SkPMColor* SK_RESTRICT dst = static_cast<SkPMColor*>(glyph.fImage); |
- |
- for (U16CPU y = 0; y < glyph.fHeight; y++) { |
- for (U16CPU i = 0; i < width; i++) { |
- U8CPU r = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableR); |
- U8CPU g = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableG); |
- U8CPU b = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableB); |
- dst[i] = SkPackARGB32(0xFF, r, g, b); |
- } |
- dst = (SkPMColor*)((char*)dst + dstRB); |
- } |
-} |
- |
-const void* SkScalerContext_DW::drawDWMask(const SkGlyph& glyph) { |
- int sizeNeeded = glyph.fWidth * glyph.fHeight; |
- if (DWRITE_RENDERING_MODE_ALIASED != fRenderingMode) { |
- sizeNeeded *= 3; |
- } |
- if (sizeNeeded > fBits.count()) { |
- fBits.setCount(sizeNeeded); |
- } |
- |
- // erase |
- memset(fBits.begin(), 0, sizeNeeded); |
- |
- fXform.dx = SkFixedToFloat(glyph.getSubXFixed()); |
- fXform.dy = SkFixedToFloat(glyph.getSubYFixed()); |
- |
- FLOAT advance = 0.0f; |
- |
- UINT16 index = glyph.getGlyphID(); |
- |
- DWRITE_GLYPH_OFFSET offset; |
- offset.advanceOffset = 0.0f; |
- offset.ascenderOffset = 0.0f; |
- |
- DWRITE_GLYPH_RUN run; |
- run.glyphCount = 1; |
- run.glyphAdvances = &advance; |
- run.fontFace = fTypeface->fDWriteFontFace.get(); |
- run.fontEmSize = SkScalarToFloat(fTextSizeRender); |
- run.bidiLevel = 0; |
- run.glyphIndices = &index; |
- run.isSideways = FALSE; |
- run.glyphOffsets = &offset; |
- |
- SkTScopedComPtr<IDWriteGlyphRunAnalysis> glyphRunAnalysis; |
- HRNM(fTypeface->fFactory->CreateGlyphRunAnalysis(&run, |
- 1.0f, // pixelsPerDip, |
- &fXform, |
- fRenderingMode, |
- fMeasuringMode, |
- 0.0f, // baselineOriginX, |
- 0.0f, // baselineOriginY, |
- &glyphRunAnalysis), |
- "Could not create glyph run analysis."); |
- |
- //NOTE: this assumes that the glyph has already been measured |
- //with an exact same glyph run analysis. |
- RECT bbox; |
- bbox.left = glyph.fLeft; |
- bbox.top = glyph.fTop; |
- bbox.right = glyph.fLeft + glyph.fWidth; |
- bbox.bottom = glyph.fTop + glyph.fHeight; |
- HRNM(glyphRunAnalysis->CreateAlphaTexture(fTextureType, |
- &bbox, |
- fBits.begin(), |
- sizeNeeded), |
- "Could not draw mask."); |
- return fBits.begin(); |
-} |
- |
-void SkScalerContext_DW::generateImage(const SkGlyph& glyph) { |
- //Create the mask. |
- const void* bits = this->drawDWMask(glyph); |
- if (!bits) { |
- sk_bzero(glyph.fImage, glyph.computeImageSize()); |
- return; |
- } |
- |
- //Copy the mask into the glyph. |
- const uint8_t* src = (const uint8_t*)bits; |
- if (DWRITE_RENDERING_MODE_ALIASED == fRenderingMode) { |
- bilevel_to_bw(src, glyph); |
- const_cast<SkGlyph&>(glyph).fMaskFormat = SkMask::kBW_Format; |
- } else if (!isLCD(fRec)) { |
- if (fPreBlend.isApplicable()) { |
- rgb_to_a8<true>(src, glyph, fPreBlend.fG); |
- } else { |
- rgb_to_a8<false>(src, glyph, fPreBlend.fG); |
- } |
- } else if (SkMask::kLCD16_Format == glyph.fMaskFormat) { |
- if (fPreBlend.isApplicable()) { |
- rgb_to_lcd16<true>(src, glyph, fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); |
- } else { |
- rgb_to_lcd16<false>(src, glyph, fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); |
- } |
- } else { |
- SkASSERT(SkMask::kLCD32_Format == glyph.fMaskFormat); |
- if (fPreBlend.isApplicable()) { |
- rgb_to_lcd32<true>(src, glyph, fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); |
- } else { |
- rgb_to_lcd32<false>(src, glyph, fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); |
- } |
- } |
-} |
- |
-void SkScalerContext_DW::generatePath(const SkGlyph& glyph, SkPath* path) { |
- SkASSERT(&glyph && path); |
- |
- path->reset(); |
- |
- SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; |
- HRVM(SkDWriteGeometrySink::Create(path, &geometryToPath), |
- "Could not create geometry to path converter."); |
- uint16_t glyphId = glyph.getGlyphID(); |
- //TODO: convert to<->from DIUs? This would make a difference if hinting. |
- //It may not be needed, it appears that DirectWrite only hints at em size. |
- HRVM(fTypeface->fDWriteFontFace->GetGlyphRunOutline(SkScalarToFloat(fTextSizeRender), |
- &glyphId, |
- NULL, //advances |
- NULL, //offsets |
- 1, //num glyphs |
- FALSE, //sideways |
- FALSE, //rtl |
- geometryToPath.get()), |
- "Could not create glyph outline."); |
- |
- path->transform(fSkXform); |
-} |