2012-07-27 20:39:19 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "gm.h"
|
2016-07-06 18:55:05 +00:00
|
|
|
#include "Resources.h"
|
2012-07-27 20:39:19 +00:00
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkString.h"
|
2016-07-06 18:55:05 +00:00
|
|
|
#include "SkSurfaceProps.h"
|
2012-07-27 20:39:19 +00:00
|
|
|
#include "SkTypeface.h"
|
|
|
|
#include "SkTypes.h"
|
|
|
|
|
2013-10-30 15:07:03 +00:00
|
|
|
static void getGlyphPositions(const SkPaint& paint, const uint16_t glyphs[],
|
|
|
|
int count, SkScalar x, SkScalar y, SkPoint pos[]) {
|
|
|
|
SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
|
|
|
|
|
|
|
|
SkAutoSTMalloc<128, SkScalar> widthStorage(count);
|
|
|
|
SkScalar* widths = widthStorage.get();
|
|
|
|
paint.getTextWidths(glyphs, count * sizeof(uint16_t), widths);
|
2013-10-31 07:01:53 +00:00
|
|
|
|
2013-10-30 15:07:03 +00:00
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
pos[i].set(x, y);
|
|
|
|
x += widths[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void applyKerning(SkPoint pos[], const int32_t adjustments[], int count,
|
|
|
|
const SkPaint& paint) {
|
|
|
|
SkScalar scale = paint.getTextSize() / paint.getTypeface()->getUnitsPerEm();
|
|
|
|
|
|
|
|
SkScalar globalAdj = 0;
|
|
|
|
for (int i = 0; i < count - 1; ++i) {
|
|
|
|
globalAdj += adjustments[i] * scale;
|
|
|
|
pos[i + 1].fX += globalAdj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void drawKernText(SkCanvas* canvas, const void* text, size_t len,
|
|
|
|
SkScalar x, SkScalar y, const SkPaint& paint) {
|
|
|
|
SkTypeface* face = paint.getTypeface();
|
|
|
|
if (!face) {
|
|
|
|
canvas->drawText(text, len, x, y, paint);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkAutoSTMalloc<128, uint16_t> glyphStorage(len);
|
|
|
|
uint16_t* glyphs = glyphStorage.get();
|
|
|
|
int glyphCount = paint.textToGlyphs(text, len, glyphs);
|
|
|
|
if (glyphCount < 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkAutoSTMalloc<128, int32_t> adjustmentStorage(glyphCount - 1);
|
|
|
|
int32_t* adjustments = adjustmentStorage.get();
|
|
|
|
if (!face->getKerningPairAdjustments(glyphs, glyphCount, adjustments)) {
|
|
|
|
canvas->drawText(text, len, x, y, paint);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkPaint glyphPaint(paint);
|
|
|
|
glyphPaint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
|
|
|
|
|
|
|
|
SkAutoSTMalloc<128, SkPoint> posStorage(glyphCount);
|
|
|
|
SkPoint* pos = posStorage.get();
|
|
|
|
getGlyphPositions(glyphPaint, glyphs, glyphCount, x, y, pos);
|
|
|
|
|
|
|
|
applyKerning(pos, adjustments, glyphCount, glyphPaint);
|
|
|
|
canvas->drawPosText(glyphs, glyphCount * sizeof(uint16_t), pos, glyphPaint);
|
|
|
|
}
|
|
|
|
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr struct {
|
2013-02-26 16:57:16 +00:00
|
|
|
const char* fName;
|
|
|
|
SkTypeface::Style fStyle;
|
|
|
|
} gFaceStyles[] = {
|
|
|
|
{ "sans-serif", SkTypeface::kNormal },
|
|
|
|
{ "sans-serif", SkTypeface::kBold },
|
|
|
|
{ "sans-serif", SkTypeface::kItalic },
|
|
|
|
{ "sans-serif", SkTypeface::kBoldItalic },
|
|
|
|
{ "serif", SkTypeface::kNormal },
|
|
|
|
{ "serif", SkTypeface::kBold },
|
|
|
|
{ "serif", SkTypeface::kItalic },
|
|
|
|
{ "serif", SkTypeface::kBoldItalic },
|
|
|
|
{ "monospace", SkTypeface::kNormal },
|
|
|
|
{ "monospace", SkTypeface::kBold },
|
|
|
|
{ "monospace", SkTypeface::kItalic },
|
|
|
|
{ "monospace", SkTypeface::kBoldItalic },
|
2012-07-27 20:39:19 +00:00
|
|
|
};
|
|
|
|
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr int gFaceStylesCount = SK_ARRAY_COUNT(gFaceStyles);
|
2013-02-26 16:57:16 +00:00
|
|
|
|
|
|
|
class TypefaceStylesGM : public skiagm::GM {
|
2016-05-12 17:09:30 +00:00
|
|
|
sk_sp<SkTypeface> fFaces[gFaceStylesCount];
|
2013-10-30 15:07:03 +00:00
|
|
|
bool fApplyKerning;
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-02-26 16:57:16 +00:00
|
|
|
public:
|
2015-05-21 13:15:28 +00:00
|
|
|
TypefaceStylesGM(bool applyKerning)
|
|
|
|
: fApplyKerning(applyKerning) {
|
|
|
|
memset(fFaces, 0, sizeof(fFaces));
|
2013-02-26 16:57:16 +00:00
|
|
|
}
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-02-26 16:57:16 +00:00
|
|
|
protected:
|
2015-05-21 13:15:28 +00:00
|
|
|
void onOnceBeforeDraw() override {
|
|
|
|
for (int i = 0; i < gFaceStylesCount; i++) {
|
2016-05-12 17:09:30 +00:00
|
|
|
fFaces[i] = SkTypeface::MakeFromName(
|
2016-05-31 18:42:36 +00:00
|
|
|
sk_tool_utils::platform_font_name(
|
|
|
|
gFaceStyles[i].fName), SkFontStyle::FromOldStyle(gFaceStyles[i].fStyle));
|
2015-05-21 13:15:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkString onShortName() override {
|
2013-10-30 15:07:03 +00:00
|
|
|
SkString name("typefacestyles");
|
|
|
|
if (fApplyKerning) {
|
|
|
|
name.append("_kerning");
|
|
|
|
}
|
2015-07-24 18:08:01 +00:00
|
|
|
name.append(sk_tool_utils::major_platform_os_name());
|
2013-10-30 15:07:03 +00:00
|
|
|
return name;
|
2013-02-26 16:57:16 +00:00
|
|
|
}
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkISize onISize() override {
|
2013-02-26 16:57:16 +00:00
|
|
|
return SkISize::Make(640, 480);
|
|
|
|
}
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onDraw(SkCanvas* canvas) override {
|
2013-02-26 16:57:16 +00:00
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setTextSize(SkIntToScalar(30));
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-10-30 15:07:03 +00:00
|
|
|
const char* text = fApplyKerning ? "Type AWAY" : "Hamburgefons";
|
2013-02-26 16:57:16 +00:00
|
|
|
const size_t textLen = strlen(text);
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-02-26 16:57:16 +00:00
|
|
|
SkScalar x = SkIntToScalar(10);
|
2015-08-27 14:41:13 +00:00
|
|
|
SkScalar dy = paint.getFontMetrics(nullptr);
|
2013-02-26 16:57:16 +00:00
|
|
|
SkScalar y = dy;
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-10-30 15:07:03 +00:00
|
|
|
if (fApplyKerning) {
|
|
|
|
paint.setSubpixelText(true);
|
|
|
|
} else {
|
|
|
|
paint.setLinearText(true);
|
|
|
|
}
|
2013-02-26 16:57:16 +00:00
|
|
|
for (int i = 0; i < gFaceStylesCount; i++) {
|
|
|
|
paint.setTypeface(fFaces[i]);
|
|
|
|
canvas->drawText(text, textLen, x, y, paint);
|
2013-10-30 15:07:03 +00:00
|
|
|
if (fApplyKerning) {
|
|
|
|
drawKernText(canvas, text, textLen, x + 240, y, paint);
|
|
|
|
}
|
2013-02-26 16:57:16 +00:00
|
|
|
y += dy;
|
|
|
|
}
|
|
|
|
}
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2013-02-26 16:57:16 +00:00
|
|
|
private:
|
|
|
|
typedef skiagm::GM INHERITED;
|
|
|
|
};
|
2013-02-27 07:10:10 +00:00
|
|
|
|
2016-07-06 18:55:05 +00:00
|
|
|
class TypefaceRenderingGM : public skiagm::GM {
|
|
|
|
sk_sp<SkTypeface> fFace;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void onOnceBeforeDraw() override {
|
|
|
|
fFace = MakeResourceAsTypeface("/fonts/hintgasp.ttf");
|
|
|
|
}
|
|
|
|
|
|
|
|
SkString onShortName() override {
|
|
|
|
SkString name("typefacerendering");
|
|
|
|
name.append(sk_tool_utils::major_platform_os_name());
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkISize onISize() override {
|
2016-07-12 13:55:25 +00:00
|
|
|
return SkISize::Make(640, 680);
|
2016-07-06 18:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void onDraw(SkCanvas* canvas) override {
|
|
|
|
struct AliasType {
|
|
|
|
bool antiAlias;
|
|
|
|
bool subpixelAntitalias;
|
|
|
|
bool inLayer;
|
|
|
|
} constexpr aliasTypes[] {
|
2016-07-13 21:00:39 +00:00
|
|
|
#ifndef SK_BUILD_FOR_IOS
|
|
|
|
// This gm crashes on iOS when drawing an embedded bitmap when requesting aliased rendering.
|
|
|
|
// The crash looks like
|
|
|
|
// libTrueTypeScaler.dylib`<redacted> + 80
|
|
|
|
// stop reason = EXC_BAD_ACCESS (code=EXC_ARM_DA_ALIGN, address=...)
|
|
|
|
// -> 0x330b19d0 <+80>: strd r2, r3, [r5, #36]
|
|
|
|
// 0x330b19d4 <+84>: movs r3, #0x0
|
|
|
|
// 0x330b19d6 <+86>: add r2, sp, #0x28
|
|
|
|
// 0x330b19d8 <+88>: ldr r0, [r4, #0x4]
|
|
|
|
// Disable testing embedded bitmaps on iOS for now.
|
|
|
|
// See https://bug.skia.org/5530 .
|
2016-07-06 18:55:05 +00:00
|
|
|
{ false, false, false }, // aliased
|
2016-07-13 21:00:39 +00:00
|
|
|
#endif
|
2016-07-06 18:55:05 +00:00
|
|
|
{ true, false, false }, // anti-aliased
|
|
|
|
{ true, true , false }, // subpixel anti-aliased
|
|
|
|
{ true, false, true }, // anti-aliased in layer (flat pixel geometry)
|
|
|
|
{ true, true , true }, // subpixel anti-aliased in layer (flat pixel geometry)
|
|
|
|
};
|
|
|
|
|
|
|
|
// The hintgasp.ttf is designed for the following sizes to be different.
|
|
|
|
// GASP_DOGRAY 0x0002 0<=ppem<=10
|
|
|
|
// GASP_SYMMETRIC_SMOOTHING 0x0008 0<=ppem<=10
|
|
|
|
// GASP_GRIDFIT 0x0001 11<=ppem<=12
|
|
|
|
// GASP_SYMMETRIC_GRIDFIT 0x0004 11<=ppem<=12
|
|
|
|
// GASP_DOGRAY|GASP_GRIDFIT 0x0003 13<=ppem<=14
|
|
|
|
// GASP_SYMMETRIC_SMOOTHING|GASP_SYMMETRIC_GRIDFIT 0x000C 13<=ppem<=14
|
|
|
|
// (neither) 0x0000 15<=ppem
|
2016-07-12 13:55:25 +00:00
|
|
|
// Odd sizes have embedded bitmaps.
|
|
|
|
constexpr SkScalar textSizes[] = { 9, 10, 11, 12, 13, 14, 15, 16 };
|
2016-07-06 18:55:05 +00:00
|
|
|
|
|
|
|
constexpr SkPaint::Hinting hintingTypes[] = { SkPaint::kNo_Hinting,
|
|
|
|
SkPaint::kSlight_Hinting,
|
|
|
|
SkPaint::kNormal_Hinting,
|
|
|
|
SkPaint::kFull_Hinting };
|
|
|
|
|
|
|
|
struct SubpixelType {
|
|
|
|
bool requested;
|
|
|
|
SkVector offset;
|
|
|
|
} constexpr subpixelTypes[] = {
|
|
|
|
{ false, { 0.00, 0.00 } },
|
|
|
|
{ true , { 0.00, 0.00 } },
|
|
|
|
{ true , { 0.25, 0.00 } },
|
|
|
|
{ true , { 0.25, 0.25 } },
|
|
|
|
};
|
|
|
|
|
|
|
|
constexpr bool rotateABitTypes[] = { false, true };
|
|
|
|
|
|
|
|
SkPaint paint;
|
2016-07-12 13:55:25 +00:00
|
|
|
paint.setTypeface(fFace);
|
|
|
|
paint.setEmbeddedBitmapText(true);
|
|
|
|
|
2016-07-06 18:55:05 +00:00
|
|
|
SkScalar x = 0;
|
|
|
|
SkScalar xMax = x;
|
|
|
|
SkScalar xBase = 0;
|
|
|
|
SkScalar y = 0; // The baseline of the previous output
|
|
|
|
for (const SubpixelType subpixel : subpixelTypes) {
|
|
|
|
y = 0;
|
|
|
|
paint.setSubpixelText(subpixel.requested);
|
|
|
|
|
|
|
|
for (const AliasType& alias : aliasTypes) {
|
|
|
|
paint.setAntiAlias(alias.antiAlias);
|
|
|
|
paint.setLCDRenderText(alias.subpixelAntitalias);
|
|
|
|
SkAutoCanvasRestore acr(canvas, false);
|
|
|
|
if (alias.inLayer) {
|
|
|
|
canvas->saveLayer(nullptr, &paint);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const SkScalar& textSize : textSizes) {
|
|
|
|
x = xBase + 5;
|
|
|
|
paint.setTextSize(textSize);
|
|
|
|
|
|
|
|
SkScalar dy = SkScalarCeilToScalar(paint.getFontMetrics(nullptr));
|
|
|
|
y += dy;
|
|
|
|
for (const SkPaint::Hinting& hinting : hintingTypes) {
|
|
|
|
paint.setHinting(hinting);
|
|
|
|
|
|
|
|
for (const bool& rotateABit : rotateABitTypes) {
|
|
|
|
SkAutoCanvasRestore acr(canvas, true);
|
|
|
|
if (rotateABit) {
|
2016-07-12 22:01:19 +00:00
|
|
|
canvas->rotate(2, x + subpixel.offset.x(),
|
|
|
|
y + subpixel.offset.y());
|
2016-07-06 18:55:05 +00:00
|
|
|
}
|
|
|
|
canvas->drawText("A", 1, x + subpixel.offset.x(),
|
|
|
|
y + subpixel.offset.y(), paint);
|
|
|
|
|
|
|
|
SkScalar dx = SkScalarCeilToScalar(paint.measureText("A", 1)) + 5;
|
|
|
|
x += dx;
|
|
|
|
xMax = SkTMax(x, xMax);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
y += 10;
|
|
|
|
}
|
|
|
|
xBase = xMax;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef skiagm::GM INHERITED;
|
|
|
|
};
|
|
|
|
|
2013-02-26 16:57:16 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2012-07-27 20:39:19 +00:00
|
|
|
|
2013-10-30 15:07:03 +00:00
|
|
|
DEF_GM( return new TypefaceStylesGM(false); )
|
|
|
|
DEF_GM( return new TypefaceStylesGM(true); )
|
2016-07-06 18:55:05 +00:00
|
|
|
DEF_GM( return new TypefaceRenderingGM(); )
|