2013-03-28 13:39:35 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2013 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"
|
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkFontMgr.h"
|
|
|
|
#include "SkGraphics.h"
|
|
|
|
#include "SkTypeface.h"
|
|
|
|
|
2013-08-21 15:20:43 +00:00
|
|
|
#ifdef SK_BUILD_FOR_WIN
|
2013-08-28 20:31:58 +00:00
|
|
|
#include "SkTypeface_win.h"
|
2013-08-21 15:20:43 +00:00
|
|
|
#endif
|
|
|
|
|
2014-10-22 20:20:58 +00:00
|
|
|
static void scale(SkRect* rect, SkScalar scale) {
|
|
|
|
rect->fLeft *= scale;
|
|
|
|
rect->fTop *= scale;
|
|
|
|
rect->fRight *= scale;
|
|
|
|
rect->fBottom *= scale;
|
|
|
|
}
|
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
// limit this just so we don't take too long to draw
|
|
|
|
#define MAX_FAMILIES 30
|
|
|
|
|
|
|
|
static SkScalar drawString(SkCanvas* canvas, const SkString& text, SkScalar x,
|
|
|
|
SkScalar y, const SkPaint& paint) {
|
|
|
|
canvas->drawText(text.c_str(), text.size(), x, y, paint);
|
|
|
|
return x + paint.measureText(text.c_str(), text.size());
|
|
|
|
}
|
|
|
|
|
2014-08-27 14:03:13 +00:00
|
|
|
static SkScalar drawCharacter(SkCanvas* canvas, uint32_t character, SkScalar x,
|
|
|
|
SkScalar y, SkPaint& paint, SkFontMgr* fm,
|
2014-11-10 21:29:33 +00:00
|
|
|
const char* fontName, const char* bcp47[], int bcp47Count,
|
2014-08-27 14:03:13 +00:00
|
|
|
const SkFontStyle& fontStyle) {
|
|
|
|
// find typeface containing the requested character and draw it
|
|
|
|
SkString ch;
|
|
|
|
ch.appendUnichar(character);
|
2014-11-10 21:29:33 +00:00
|
|
|
SkTypeface* typeface = fm->matchFamilyStyleCharacter(fontName, fontStyle,
|
|
|
|
bcp47, bcp47Count, character);
|
2014-08-27 14:03:13 +00:00
|
|
|
SkSafeUnref(paint.setTypeface(typeface));
|
|
|
|
x = drawString(canvas, ch, x, y, paint) + 20;
|
|
|
|
|
|
|
|
if (NULL == typeface) {
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
// repeat the process, but this time use the family name of the typeface
|
|
|
|
// from the first pass. This emulates the behavior in Blink where it
|
|
|
|
// it expects to get the same glyph when following this pattern.
|
|
|
|
SkString familyName;
|
|
|
|
typeface->getFamilyName(&familyName);
|
|
|
|
SkTypeface* typefaceCopy = fm->legacyCreateTypeface(familyName.c_str(), typeface->style());
|
|
|
|
SkSafeUnref(paint.setTypeface(typefaceCopy));
|
|
|
|
return drawString(canvas, ch, x, y, paint) + 20;
|
|
|
|
}
|
|
|
|
|
2014-11-10 21:29:33 +00:00
|
|
|
static const char* zh = "zh";
|
|
|
|
static const char* ja = "ja";
|
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
class FontMgrGM : public skiagm::GM {
|
|
|
|
public:
|
2014-03-31 19:18:07 +00:00
|
|
|
FontMgrGM(SkFontMgr* fontMgr = NULL) {
|
2013-03-28 13:39:35 +00:00
|
|
|
SkGraphics::SetFontCacheLimit(16 * 1024 * 1024);
|
2013-08-21 15:20:43 +00:00
|
|
|
|
|
|
|
fName.set("fontmgr_iter");
|
2014-03-31 19:18:07 +00:00
|
|
|
if (fontMgr) {
|
2013-08-21 15:20:43 +00:00
|
|
|
fName.append("_factory");
|
2014-03-31 19:18:07 +00:00
|
|
|
fFM.reset(fontMgr);
|
2013-08-21 15:20:43 +00:00
|
|
|
} else {
|
|
|
|
fFM.reset(SkFontMgr::RefDefault());
|
|
|
|
}
|
2013-03-28 13:39:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual SkString onShortName() {
|
2013-08-21 15:20:43 +00:00
|
|
|
return fName;
|
2013-03-28 13:39:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual SkISize onISize() {
|
2014-08-27 14:03:13 +00:00
|
|
|
return SkISize::Make(1536, 768);
|
2013-03-28 13:39:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
|
|
|
|
SkScalar y = 20;
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setLCDRenderText(true);
|
|
|
|
paint.setSubpixelText(true);
|
|
|
|
paint.setTextSize(17);
|
2013-03-29 07:01:22 +00:00
|
|
|
|
2013-08-21 15:20:43 +00:00
|
|
|
SkFontMgr* fm = fFM;
|
2013-03-28 13:39:35 +00:00
|
|
|
int count = SkMin32(fm->countFamilies(), MAX_FAMILIES);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
SkString fname;
|
|
|
|
fm->getFamilyName(i, &fname);
|
|
|
|
paint.setTypeface(NULL);
|
|
|
|
(void)drawString(canvas, fname, 20, y, paint);
|
2013-03-29 07:01:22 +00:00
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
SkScalar x = 220;
|
2013-03-29 14:57:22 +00:00
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
SkAutoTUnref<SkFontStyleSet> set(fm->createStyleSet(i));
|
|
|
|
for (int j = 0; j < set->count(); ++j) {
|
|
|
|
SkString sname;
|
|
|
|
SkFontStyle fs;
|
|
|
|
set->getStyle(j, &fs, &sname);
|
2013-04-24 19:14:39 +00:00
|
|
|
sname.appendf(" [%d %d %d]", fs.weight(), fs.width(), fs.isItalic());
|
2013-03-29 07:01:22 +00:00
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
SkSafeUnref(paint.setTypeface(set->createTypeface(j)));
|
|
|
|
x = drawString(canvas, sname, x, y, paint) + 20;
|
2014-08-27 14:03:13 +00:00
|
|
|
|
|
|
|
// check to see that we get different glyphs in japanese and chinese
|
2014-11-10 21:29:33 +00:00
|
|
|
x = drawCharacter(canvas, 0x5203, x, y, paint, fm, fName.c_str(), &zh, 1, fs);
|
|
|
|
x = drawCharacter(canvas, 0x5203, x, y, paint, fm, fName.c_str(), &ja, 1, fs);
|
2014-08-27 14:03:13 +00:00
|
|
|
// check that emoji characters are found
|
2014-11-10 21:29:33 +00:00
|
|
|
x = drawCharacter(canvas, 0x1f601, x, y, paint, fm, fName.c_str(), NULL, 0, fs);
|
2013-03-29 07:01:22 +00:00
|
|
|
}
|
2013-03-28 13:39:35 +00:00
|
|
|
y += 24;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-28 15:03:22 +00:00
|
|
|
virtual uint32_t onGetFlags() const SK_OVERRIDE {
|
|
|
|
// fontdescriptors (and therefore serialization) don't yet understand
|
|
|
|
// these new styles, so skip tests that exercise that for now.
|
2013-05-01 14:21:20 +00:00
|
|
|
|
|
|
|
// If certain fonts are picked up (e.g. Microsoft Jhenghei 20MB for Regular, 12MB for Bold),
|
|
|
|
// the resulting pdf can be ~700MB and crashes Chrome's PDF viewer.
|
|
|
|
|
|
|
|
return kSkipPicture_Flag | kSkipPipe_Flag | kSkipPDF_Flag;
|
2013-03-28 15:03:22 +00:00
|
|
|
}
|
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
private:
|
2013-08-21 15:20:43 +00:00
|
|
|
SkAutoTUnref<SkFontMgr> fFM;
|
|
|
|
SkString fName;
|
2013-03-28 13:39:35 +00:00
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
class FontMgrMatchGM : public skiagm::GM {
|
|
|
|
SkAutoTUnref<SkFontMgr> fFM;
|
|
|
|
|
|
|
|
public:
|
|
|
|
FontMgrMatchGM() : fFM(SkFontMgr::RefDefault()) {
|
|
|
|
SkGraphics::SetFontCacheLimit(16 * 1024 * 1024);
|
|
|
|
}
|
2013-03-30 07:01:27 +00:00
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
protected:
|
|
|
|
virtual SkString onShortName() {
|
|
|
|
return SkString("fontmgr_match");
|
|
|
|
}
|
2013-03-30 07:01:27 +00:00
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
virtual SkISize onISize() {
|
|
|
|
return SkISize::Make(640, 1024);
|
|
|
|
}
|
|
|
|
|
|
|
|
void iterateFamily(SkCanvas* canvas, const SkPaint& paint,
|
|
|
|
SkFontStyleSet* fset) {
|
|
|
|
SkPaint p(paint);
|
|
|
|
SkScalar y = 0;
|
|
|
|
|
|
|
|
for (int j = 0; j < fset->count(); ++j) {
|
|
|
|
SkString sname;
|
|
|
|
SkFontStyle fs;
|
|
|
|
fset->getStyle(j, &fs, &sname);
|
|
|
|
|
|
|
|
sname.appendf(" [%d %d]", fs.weight(), fs.width());
|
|
|
|
|
|
|
|
SkSafeUnref(p.setTypeface(fset->createTypeface(j)));
|
|
|
|
(void)drawString(canvas, sname, 0, y, p);
|
|
|
|
y += 24;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void exploreFamily(SkCanvas* canvas, const SkPaint& paint,
|
|
|
|
SkFontStyleSet* fset) {
|
|
|
|
SkPaint p(paint);
|
|
|
|
SkScalar y = 0;
|
|
|
|
|
|
|
|
for (int weight = 100; weight <= 900; weight += 200) {
|
|
|
|
for (int width = 1; width <= 9; width += 2) {
|
|
|
|
SkFontStyle fs(weight, width, SkFontStyle::kUpright_Slant);
|
|
|
|
SkTypeface* face = fset->matchStyle(fs);
|
|
|
|
if (face) {
|
|
|
|
SkString str;
|
|
|
|
str.printf("request [%d %d]", fs.weight(), fs.width());
|
|
|
|
p.setTypeface(face)->unref();
|
|
|
|
(void)drawString(canvas, str, 0, y, p);
|
|
|
|
y += 24;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-03-30 07:01:27 +00:00
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setLCDRenderText(true);
|
|
|
|
paint.setSubpixelText(true);
|
|
|
|
paint.setTextSize(17);
|
|
|
|
|
|
|
|
static const char* gNames[] = {
|
|
|
|
"Helvetica Neue", "Arial"
|
|
|
|
};
|
|
|
|
|
2013-11-12 15:25:29 +00:00
|
|
|
SkAutoTUnref<SkFontStyleSet> fset;
|
2013-03-30 07:01:27 +00:00
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gNames); ++i) {
|
2013-11-12 15:25:29 +00:00
|
|
|
fset.reset(fFM->matchFamily(gNames[i]));
|
|
|
|
if (fset->count() > 0) {
|
2013-03-29 14:57:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-11-12 15:25:29 +00:00
|
|
|
if (NULL == fset.get()) {
|
2013-03-29 14:57:22 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
canvas->translate(20, 40);
|
|
|
|
this->exploreFamily(canvas, paint, fset);
|
|
|
|
canvas->translate(150, 0);
|
|
|
|
this->iterateFamily(canvas, paint, fset);
|
|
|
|
}
|
2013-03-30 07:01:27 +00:00
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
virtual uint32_t onGetFlags() const SK_OVERRIDE {
|
|
|
|
// fontdescriptors (and therefore serialization) don't yet understand
|
|
|
|
// these new styles, so skip tests that exercise that for now.
|
|
|
|
return kSkipPicture_Flag | kSkipPipe_Flag;
|
|
|
|
}
|
2013-03-30 07:01:27 +00:00
|
|
|
|
2013-03-29 14:57:22 +00:00
|
|
|
private:
|
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
|
2014-10-22 20:20:58 +00:00
|
|
|
class FontMgrBoundsGM : public skiagm::GM {
|
|
|
|
public:
|
|
|
|
FontMgrBoundsGM() {
|
|
|
|
fName.set("fontmgr_bounds");
|
|
|
|
fFM.reset(SkFontMgr::RefDefault());
|
|
|
|
}
|
|
|
|
|
|
|
|
static void show_bounds(SkCanvas* canvas, const SkPaint& paint, SkScalar x, SkScalar y,
|
|
|
|
SkColor boundsColor) {
|
|
|
|
const char str[] = "jyHO[]{}@-_&%$";
|
|
|
|
|
|
|
|
const SkTypeface* tf = paint.getTypeface();
|
|
|
|
for (int i = 0; str[i]; ++i) {
|
|
|
|
canvas->drawText(&str[i], 1, x, y, paint);
|
|
|
|
}
|
|
|
|
|
|
|
|
SkRect r = tf->getBounds();
|
|
|
|
scale(&r, paint.getTextSize());
|
|
|
|
r.offset(x, y);
|
|
|
|
SkPaint p(paint);
|
|
|
|
p.setColor(boundsColor);
|
|
|
|
canvas->drawRect(r, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual SkString onShortName() {
|
|
|
|
return fName;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual SkISize onISize() {
|
|
|
|
return SkISize::Make(1024, 850);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setSubpixelText(true);
|
|
|
|
paint.setTextSize(100);
|
|
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
|
|
|
|
|
|
const SkColor boundsColors[2] = { SK_ColorRED, SK_ColorBLUE };
|
|
|
|
|
|
|
|
SkFontMgr* fm = fFM;
|
|
|
|
int count = SkMin32(fm->countFamilies(), 32);
|
|
|
|
|
|
|
|
int index = 0;
|
|
|
|
SkScalar x = 0, y = 0;
|
|
|
|
|
|
|
|
canvas->translate(80, 120);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
SkAutoTUnref<SkFontStyleSet> set(fm->createStyleSet(i));
|
|
|
|
for (int j = 0; j < set->count(); ++j) {
|
|
|
|
SkSafeUnref(paint.setTypeface(set->createTypeface(j)));
|
|
|
|
if (paint.getTypeface()) {
|
|
|
|
show_bounds(canvas, paint, x, y, boundsColors[index & 1]);
|
|
|
|
index += 1;
|
|
|
|
x += 160;
|
|
|
|
if (0 == (index % 6)) {
|
|
|
|
x = 0;
|
|
|
|
y += 160;
|
|
|
|
}
|
|
|
|
if (index >= 30) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uint32_t onGetFlags() const SK_OVERRIDE {
|
|
|
|
// fontdescriptors (and therefore serialization) don't yet understand
|
|
|
|
// these new styles, so skip tests that exercise that for now.
|
|
|
|
|
|
|
|
// If certain fonts are picked up (e.g. Microsoft Jhenghei 20MB for Regular, 12MB for Bold),
|
|
|
|
// the resulting pdf can be ~700MB and crashes Chrome's PDF viewer.
|
|
|
|
|
|
|
|
return kSkipPicture_Flag | kSkipPipe_Flag | kSkipPDF_Flag;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SkAutoTUnref<SkFontMgr> fFM;
|
|
|
|
SkString fName;
|
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
|
2013-03-28 13:39:35 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
DEF_GM( return SkNEW(FontMgrGM); )
|
2014-10-22 20:20:58 +00:00
|
|
|
DEF_GM( return SkNEW(FontMgrBoundsGM); )
|
2013-03-29 14:57:22 +00:00
|
|
|
DEF_GM( return SkNEW(FontMgrMatchGM); )
|
2013-08-21 15:20:43 +00:00
|
|
|
|
|
|
|
#ifdef SK_BUILD_FOR_WIN
|
2014-03-31 19:18:07 +00:00
|
|
|
DEF_GM( return SkNEW_ARGS(FontMgrGM, (SkFontMgr_New_DirectWrite())); )
|
2013-08-21 15:20:43 +00:00
|
|
|
#endif
|