summaryrefslogtreecommitdiffstats
path: root/skia/ext/vector_platform_device_emf_win.cc
diff options
context:
space:
mode:
Diffstat (limited to 'skia/ext/vector_platform_device_emf_win.cc')
-rw-r--r--skia/ext/vector_platform_device_emf_win.cc188
1 files changed, 160 insertions, 28 deletions
diff --git a/skia/ext/vector_platform_device_emf_win.cc b/skia/ext/vector_platform_device_emf_win.cc
index 0789197..eac7977 100644
--- a/skia/ext/vector_platform_device_emf_win.cc
+++ b/skia/ext/vector_platform_device_emf_win.cc
@@ -8,7 +8,9 @@
#include "skia/ext/bitmap_platform_device.h"
#include "skia/ext/skia_utils_win.h"
+#include "third_party/skia/include/core/SkTemplates.h"
#include "third_party/skia/include/core/SkUtils.h"
+#include "third_party/skia/include/ports/SkTypeface_win.h"
namespace skia {
@@ -227,26 +229,27 @@ void VectorPlatformDeviceEmf::drawPath(const SkDraw& draw,
return;
}
HDC dc = BeginPlatformPaint();
- PlatformDevice::LoadPathToDC(dc, path);
- switch (paint.getStyle()) {
- case SkPaint::kFill_Style: {
- BOOL res = StrokeAndFillPath(dc);
- SkASSERT(res != 0);
- break;
- }
- case SkPaint::kStroke_Style: {
- BOOL res = StrokePath(dc);
- SkASSERT(res != 0);
- break;
- }
- case SkPaint::kStrokeAndFill_Style: {
- BOOL res = StrokeAndFillPath(dc);
- SkASSERT(res != 0);
- break;
+ if (PlatformDevice::LoadPathToDC(dc, path)) {
+ switch (paint.getStyle()) {
+ case SkPaint::kFill_Style: {
+ BOOL res = StrokeAndFillPath(dc);
+ SkASSERT(res != 0);
+ break;
+ }
+ case SkPaint::kStroke_Style: {
+ BOOL res = StrokePath(dc);
+ SkASSERT(res != 0);
+ break;
+ }
+ case SkPaint::kStrokeAndFill_Style: {
+ BOOL res = StrokeAndFillPath(dc);
+ SkASSERT(res != 0);
+ break;
+ }
+ default:
+ SkASSERT(false);
+ break;
}
- default:
- SkASSERT(false);
- break;
}
EndPlatformPaint();
Cleanup();
@@ -283,14 +286,108 @@ void VectorPlatformDeviceEmf::drawSprite(const SkDraw& draw,
LoadTransformToDC(hdc_, transform_);
}
+/////////////////////////////////////////////////////////////////////////
+
+static bool gdiCanHandleText(const SkPaint& paint) {
+ return !paint.getShader() &&
+ !paint.getPathEffect() &&
+ (SkPaint::kFill_Style == paint.getStyle()) &&
+ (255 == paint.getAlpha());
+}
+
+class SkGDIFontSetup {
+public:
+ SkGDIFontSetup() : fUseGDI(false) {
+ SkDEBUGCODE(fUseGDIHasBeenCalled = false;)
+ }
+ ~SkGDIFontSetup();
+
+ // can only be called once
+ bool useGDI(HDC hdc, const SkPaint&);
+
+private:
+ HDC fHDC;
+ HFONT fNewFont;
+ HFONT fSavedFont;
+ COLORREF fSavedTextColor;
+ bool fUseGDI;
+ SkDEBUGCODE(bool fUseGDIHasBeenCalled;)
+};
+
+bool SkGDIFontSetup::useGDI(HDC hdc, const SkPaint& paint) {
+ SkASSERT(!fUseGDIHasBeenCalled);
+ SkDEBUGCODE(fUseGDIHasBeenCalled = true;)
+
+ fUseGDI = gdiCanHandleText(paint);
+ if (fUseGDI) {
+ fSavedTextColor = GetTextColor(hdc);
+ SetTextColor(hdc, skia::SkColorToCOLORREF(paint.getColor()));
+
+ LOGFONT lf;
+ SkLOGFONTFromTypeface(paint.getTypeface(), &lf);
+ lf.lfHeight = -SkScalarRound(paint.getTextSize());
+ fNewFont = CreateFontIndirect(&lf);
+ fSavedFont = (HFONT)::SelectObject(hdc, fNewFont);
+ }
+ return fUseGDI;
+}
+
+SkGDIFontSetup::~SkGDIFontSetup() {
+ if (fUseGDI) {
+ ::SelectObject(fHDC, fSavedFont);
+ ::DeleteObject(fNewFont);
+ SetTextColor(fHDC, fSavedTextColor);
+ }
+}
+
+static SkScalar getAscent(const SkPaint& paint) {
+ SkPaint::FontMetrics fm;
+ paint.getFontMetrics(&fm);
+ return fm.fAscent;
+}
+
+// return the options int for ExtTextOut. Only valid if the paint's text encoding
+// is not UTF8 (in which case ExtTextOut can't be used).
+static UINT getTextOutOptions(const SkPaint& paint) {
+ if (SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding()) {
+ return ETO_GLYPH_INDEX;
+ } else {
+ SkASSERT(SkPaint::kUTF16_TextEncoding == paint.getTextEncoding());
+ return 0;
+ }
+}
+
void VectorPlatformDeviceEmf::drawText(const SkDraw& draw,
const void* text,
size_t byteLength,
SkScalar x,
SkScalar y,
const SkPaint& paint) {
- // This function isn't used in the code. Verify this assumption.
- SkASSERT(false);
+ SkGDIFontSetup setup;
+ if (SkPaint::kUTF8_TextEncoding != paint.getTextEncoding()
+ && setup.useGDI(hdc_, paint)) {
+ UINT options = getTextOutOptions(paint);
+ UINT count = byteLength >> 1;
+ ExtTextOut(hdc_, SkScalarRound(x), SkScalarRound(y + getAscent(paint)),
+ options, 0, reinterpret_cast<const wchar_t*>(text), count, NULL);
+ } else {
+ SkPath path;
+ paint.getTextPath(text, byteLength, x, y, &path);
+ drawPath(draw, path, paint);
+ }
+}
+
+static size_t size_utf8(const char* text) {
+ return SkUTF8_CountUTF8Bytes(text);
+}
+
+static size_t size_utf16(const char* text) {
+ uint16_t c = *reinterpret_cast<const uint16_t*>(text);
+ return SkUTF16_IsHighSurrogate(c) ? 4 : 2;
+}
+
+static size_t size_glyphid(const char* text) {
+ return 2;
}
void VectorPlatformDeviceEmf::drawPosText(const SkDraw& draw,
@@ -300,8 +397,46 @@ void VectorPlatformDeviceEmf::drawPosText(const SkDraw& draw,
SkScalar constY,
int scalarsPerPos,
const SkPaint& paint) {
- // This function isn't used in the code. Verify this assumption.
- SkASSERT(false);
+ SkGDIFontSetup setup;
+ if (2 == scalarsPerPos
+ && SkPaint::kUTF8_TextEncoding != paint.getTextEncoding()
+ && setup.useGDI(hdc_, paint)) {
+ int startX = SkScalarRound(pos[0]);
+ int startY = SkScalarRound(pos[1] + getAscent(paint));
+ const int count = len >> 1;
+ SkAutoSTMalloc<64, INT> storage(count);
+ INT* advances = storage.get();
+ for (int i = 0; i < count - 1; ++i) {
+ advances[i] = SkScalarRound(pos[2] - pos[0]);
+ pos += 2;
+ }
+ ExtTextOut(hdc_, startX, startY, getTextOutOptions(paint), 0,
+ reinterpret_cast<const wchar_t*>(text), count, advances);
+ } else {
+ size_t (*bytesPerCodePoint)(const char*);
+ switch (paint.getTextEncoding()) {
+ case SkPaint::kUTF8_TextEncoding:
+ bytesPerCodePoint = size_utf8;
+ break;
+ case SkPaint::kUTF16_TextEncoding:
+ bytesPerCodePoint = size_utf16;
+ break;
+ default:
+ SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
+ bytesPerCodePoint = size_glyphid;
+ break;
+ }
+
+ const char* curr = reinterpret_cast<const char*>(text);
+ const char* stop = curr + len;
+ while (curr < stop) {
+ SkScalar y = (1 == scalarsPerPos) ? constY : pos[1];
+ size_t bytes = bytesPerCodePoint(curr);
+ drawText(draw, curr, bytes, pos[0], y, paint);
+ curr += bytes;
+ pos += scalarsPerPos;
+ }
+ }
}
void VectorPlatformDeviceEmf::drawTextOnPath(const SkDraw& draw,
@@ -409,11 +544,9 @@ bool VectorPlatformDeviceEmf::ApplyPaint(const SkPaint& paint) {
// The path effect should be processed before arriving here.
SkASSERT(!paint.getPathEffect());
- // These aren't used in the code. Verify this assumption.
- SkASSERT(!paint.getColorFilter());
+ // This isn't used in the code. Verify this assumption.
SkASSERT(!paint.getRasterizer());
// Reuse code to load Win32 Fonts.
- SkASSERT(!paint.getTypeface());
return true;
}
@@ -717,7 +850,7 @@ void VectorPlatformDeviceEmf::InternalDrawBitmap(const SkBitmap& bitmap,
DeleteObject(hbitmap);
DeleteDC(bitmap_dc);
} else {
- BOOL result = StretchDIBits(dc,
+ int nCopied = StretchDIBits(dc,
x, y, // Destination origin.
src_size_x, src_size_y,
0, 0, // Source origin.
@@ -726,7 +859,6 @@ void VectorPlatformDeviceEmf::InternalDrawBitmap(const SkBitmap& bitmap,
reinterpret_cast<const BITMAPINFO*>(&hdr),
DIB_RGB_COLORS,
SRCCOPY);
- SkASSERT(result);
}
EndPlatformPaint();
Cleanup();