383 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			383 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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/gm.h"
 | |
| #include "include/core/SkCanvas.h"
 | |
| #include "include/core/SkColor.h"
 | |
| #include "include/core/SkFont.h"
 | |
| #include "include/core/SkFontMetrics.h"
 | |
| #include "include/core/SkFontMgr.h"
 | |
| #include "include/core/SkFontStyle.h"
 | |
| #include "include/core/SkFontTypes.h"
 | |
| #include "include/core/SkGraphics.h"
 | |
| #include "include/core/SkPaint.h"
 | |
| #include "include/core/SkPath.h"
 | |
| #include "include/core/SkPoint.h"
 | |
| #include "include/core/SkRect.h"
 | |
| #include "include/core/SkRefCnt.h"
 | |
| #include "include/core/SkScalar.h"
 | |
| #include "include/core/SkSize.h"
 | |
| #include "include/core/SkString.h"
 | |
| #include "include/core/SkTypeface.h"
 | |
| #include "include/core/SkTypes.h"
 | |
| #include "include/effects/SkDashPathEffect.h"
 | |
| #include "src/core/SkFontPriv.h"
 | |
| #include "tools/SkMetaData.h"
 | |
| #include "tools/ToolUtils.h"
 | |
| 
 | |
| #include <utility>
 | |
| 
 | |
| // 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 SkFont& font) {
 | |
|     canvas->drawString(text, x, y, font, SkPaint());
 | |
|     return x + font.measureText(text.c_str(), text.size(), SkTextEncoding::kUTF8);
 | |
| }
 | |
| 
 | |
| static SkScalar drawCharacter(SkCanvas* canvas, uint32_t character, SkScalar x,
 | |
|                               SkScalar y, const SkFont& origFont, SkFontMgr* fm,
 | |
|                               const char* fontName, const char* bcp47[], int bcp47Count,
 | |
|                               const SkFontStyle& fontStyle) {
 | |
|     SkFont font = origFont;
 | |
|     // find typeface containing the requested character and draw it
 | |
|     SkString ch;
 | |
|     ch.appendUnichar(character);
 | |
|     sk_sp<SkTypeface> typeface(fm->matchFamilyStyleCharacter(fontName, fontStyle,
 | |
|                                                              bcp47, bcp47Count, character));
 | |
|     font.setTypeface(typeface);
 | |
|     x = drawString(canvas, ch, x, y, font) + 20;
 | |
| 
 | |
|     if (nullptr == 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);
 | |
|     font.setTypeface(fm->legacyMakeTypeface(familyName.c_str(), typeface->fontStyle()));
 | |
|     return drawString(canvas, ch, x, y, font) + 20;
 | |
| }
 | |
| 
 | |
| static const char* zh = "zh";
 | |
| static const char* ja = "ja";
 | |
| 
 | |
| class FontMgrGM : public skiagm::GM {
 | |
|     sk_sp<SkFontMgr> fFM;
 | |
| 
 | |
|     void onOnceBeforeDraw() override {
 | |
|         SkGraphics::SetFontCacheLimit(16 * 1024 * 1024);
 | |
|         fFM = SkFontMgr::RefDefault();
 | |
|     }
 | |
| 
 | |
|     SkString onShortName() override { return SkString("fontmgr_iter"); }
 | |
| 
 | |
|     SkISize onISize() override { return {1536, 768}; }
 | |
| 
 | |
|     void onDraw(SkCanvas* canvas) override {
 | |
|         SkScalar y = 20;
 | |
|         SkFont font;
 | |
|         font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
 | |
|         font.setSubpixel(true);
 | |
|         font.setSize(17);
 | |
| 
 | |
|         SkFontMgr* fm = fFM.get();
 | |
|         int count = std::min(fm->countFamilies(), MAX_FAMILIES);
 | |
| 
 | |
|         for (int i = 0; i < count; ++i) {
 | |
|             SkString familyName;
 | |
|             fm->getFamilyName(i, &familyName);
 | |
|             font.setTypeface(nullptr);
 | |
|             (void)drawString(canvas, familyName, 20, y, font);
 | |
| 
 | |
|             SkScalar x = 220;
 | |
| 
 | |
|             sk_sp<SkFontStyleSet> set(fm->createStyleSet(i));
 | |
|             for (int j = 0; j < set->count(); ++j) {
 | |
|                 SkString sname;
 | |
|                 SkFontStyle fs;
 | |
|                 set->getStyle(j, &fs, &sname);
 | |
|                 sname.appendf(" [%d %d %d]", fs.weight(), fs.width(), fs.slant());
 | |
| 
 | |
|                 font.setTypeface(sk_sp<SkTypeface>(set->createTypeface(j)));
 | |
|                 x = drawString(canvas, sname, x, y, font) + 20;
 | |
| 
 | |
|                 // check to see that we get different glyphs in japanese and chinese
 | |
|                 x = drawCharacter(canvas, 0x5203, x, y, font, fm, familyName.c_str(), &zh, 1, fs);
 | |
|                 x = drawCharacter(canvas, 0x5203, x, y, font, fm, familyName.c_str(), &ja, 1, fs);
 | |
|                 // check that emoji characters are found
 | |
|                 x = drawCharacter(canvas, 0x1f601, x, y, font, fm, familyName.c_str(), nullptr,0, fs);
 | |
|             }
 | |
|             y += 24;
 | |
|         }
 | |
|     }
 | |
| };
 | |
| 
 | |
| class FontMgrMatchGM : public skiagm::GM {
 | |
|     sk_sp<SkFontMgr> fFM;
 | |
| 
 | |
|     void onOnceBeforeDraw() override {
 | |
|         fFM = SkFontMgr::RefDefault();
 | |
|         SkGraphics::SetFontCacheLimit(16 * 1024 * 1024);
 | |
|     }
 | |
| 
 | |
|     SkString onShortName() override { return SkString("fontmgr_match"); }
 | |
| 
 | |
|     SkISize onISize() override { return {640, 1024}; }
 | |
| 
 | |
|     void iterateFamily(SkCanvas* canvas, const SkFont& font, SkFontStyleSet* fset) {
 | |
|         SkFont f(font);
 | |
|         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());
 | |
| 
 | |
|             f.setTypeface(sk_sp<SkTypeface>(fset->createTypeface(j)));
 | |
|             (void)drawString(canvas, sname, 0, y, f);
 | |
|             y += 24;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void exploreFamily(SkCanvas* canvas, const SkFont& font, SkFontStyleSet* fset) {
 | |
|         SkFont f(font);
 | |
|         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);
 | |
|                 sk_sp<SkTypeface> face(fset->matchStyle(fs));
 | |
|                 if (face) {
 | |
|                     SkString str;
 | |
|                     str.printf("request [%d %d]", fs.weight(), fs.width());
 | |
|                     f.setTypeface(std::move(face));
 | |
|                     (void)drawString(canvas, str, 0, y, f);
 | |
|                     y += 24;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
 | |
|         SkFont font;
 | |
|         font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
 | |
|         font.setSubpixel(true);
 | |
|         font.setSize(17);
 | |
| 
 | |
|         const char* gNames[] = {
 | |
|             "Helvetica Neue", "Arial", "sans"
 | |
|         };
 | |
| 
 | |
|         sk_sp<SkFontStyleSet> fset;
 | |
|         for (size_t i = 0; i < SK_ARRAY_COUNT(gNames); ++i) {
 | |
|             fset.reset(fFM->matchFamily(gNames[i]));
 | |
|             if (fset->count() > 0) {
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         if (nullptr == fset.get()) {
 | |
|             *errorMsg = "No SkFontStyleSet";
 | |
|             return DrawResult::kFail;
 | |
|         }
 | |
| 
 | |
|         canvas->translate(20, 40);
 | |
|         this->exploreFamily(canvas, font, fset.get());
 | |
|         canvas->translate(150, 0);
 | |
|         this->iterateFamily(canvas, font, fset.get());
 | |
|         return DrawResult::kOk;
 | |
|     }
 | |
| };
 | |
| 
 | |
| class FontMgrBoundsGM : public skiagm::GM {
 | |
| public:
 | |
|     FontMgrBoundsGM(float scale, float skew) : fScaleX(scale) , fSkewX(skew) {}
 | |
| 
 | |
| private:
 | |
|     SkString onShortName() override {
 | |
|         if (fScaleX != 1 || fSkewX != 0) {
 | |
|             return SkStringPrintf("fontmgr_bounds_%g_%g", fScaleX, fSkewX);
 | |
|         }
 | |
|         return SkString("fontmgr_bounds");
 | |
|     }
 | |
| 
 | |
|     void onOnceBeforeDraw() override {
 | |
|         fFM = SkFontMgr::RefDefault();
 | |
|     }
 | |
| 
 | |
|     bool onGetControls(SkMetaData* controls) override {
 | |
|         controls->setBool("Label Bounds", fLabelBounds);
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     void onSetControls(const SkMetaData& controls) override {
 | |
|         controls.findBool("Label Bounds", &fLabelBounds);
 | |
|     }
 | |
| 
 | |
|     static SkRect show_bounds(SkCanvas* canvas, const SkFont& font, SkScalar x, SkScalar y,
 | |
|                               SkColor boundsColor, bool labelBounds)
 | |
|     {
 | |
|         SkGlyphID left = 0, right = 0, top = 0, bottom = 0;
 | |
|         SkRect min = SkRect::MakeLTRB(SK_ScalarInfinity, SK_ScalarInfinity,
 | |
|                                       SK_ScalarNegativeInfinity, SK_ScalarNegativeInfinity);
 | |
|         {
 | |
|             int numGlyphs = font.getTypefaceOrDefault()->countGlyphs();
 | |
|             for (int i = 0; i < numGlyphs; ++i) {
 | |
|                 SkGlyphID glyphId = i;
 | |
|                 SkRect cur;
 | |
|                 font.getBounds(&glyphId, 1, &cur, nullptr);
 | |
|                 if (cur.fLeft   < min.fLeft  ) { min.fLeft   = cur.fLeft;   left   = i; }
 | |
|                 if (cur.fTop    < min.fTop   ) { min.fTop    = cur.fTop ;   top    = i; }
 | |
|                 if (min.fRight  < cur.fRight ) { min.fRight  = cur.fRight;  right  = i; }
 | |
|                 if (min.fBottom < cur.fBottom) { min.fBottom = cur.fBottom; bottom = i; }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         SkRect fontBounds = SkFontPriv::GetFontBounds(font);
 | |
| 
 | |
|         SkRect drawBounds = min;
 | |
|         drawBounds.join(fontBounds);
 | |
| 
 | |
|         SkAutoCanvasRestore acr(canvas, true);
 | |
|         canvas->translate(x - drawBounds.left(), y);
 | |
| 
 | |
|         SkPaint boundsPaint;
 | |
|         boundsPaint.setAntiAlias(true);
 | |
|         boundsPaint.setColor(boundsColor);
 | |
|         boundsPaint.setStyle(SkPaint::kStroke_Style);
 | |
|         canvas->drawRect(fontBounds, boundsPaint);
 | |
| 
 | |
|         const SkScalar intervals[] = { 10.f, 10.f };
 | |
|         boundsPaint.setPathEffect(SkDashPathEffect::Make(intervals, 2, 0.f));
 | |
|         canvas->drawRect(min, boundsPaint);
 | |
| 
 | |
|         SkFontMetrics fm;
 | |
|         font.getMetrics(&fm);
 | |
|         SkPaint metricsPaint(boundsPaint);
 | |
|         metricsPaint.setStyle(SkPaint::kFill_Style);
 | |
|         metricsPaint.setAlphaf(0.25f);
 | |
|         if ((fm.fFlags & SkFontMetrics::kUnderlinePositionIsValid_Flag) &&
 | |
|             (fm.fFlags & SkFontMetrics::kUnderlineThicknessIsValid_Flag))
 | |
|         {
 | |
|             SkRect underline{ min.fLeft,  fm.fUnderlinePosition,
 | |
|                               min.fRight, fm.fUnderlinePosition + fm.fUnderlineThickness };
 | |
|             canvas->drawRect(underline, metricsPaint);
 | |
|         }
 | |
| 
 | |
|         if ((fm.fFlags & SkFontMetrics::kStrikeoutPositionIsValid_Flag) &&
 | |
|             (fm.fFlags & SkFontMetrics::kStrikeoutThicknessIsValid_Flag))
 | |
|         {
 | |
|             SkRect strikeout{ min.fLeft,  fm.fStrikeoutPosition - fm.fStrikeoutThickness,
 | |
|                               min.fRight, fm.fStrikeoutPosition };
 | |
|             canvas->drawRect(strikeout, metricsPaint);
 | |
|         }
 | |
| 
 | |
|         struct GlyphToDraw {
 | |
|             SkGlyphID id;
 | |
|             SkPoint location;
 | |
|             SkScalar rotation;
 | |
|         } glyphsToDraw [] = {
 | |
|             {left,   {min.left(),    min.centerY()}, 270},
 | |
|             {right,  {min.right(),   min.centerY()},  90},
 | |
|             {top,    {min.centerX(), min.top()    },   0},
 | |
|             {bottom, {min.centerX(), min.bottom() }, 180},
 | |
|         };
 | |
| 
 | |
|         SkFont labelFont;
 | |
|         labelFont.setEdging(SkFont::Edging::kAntiAlias);
 | |
|         labelFont.setTypeface(ToolUtils::create_portable_typeface());
 | |
| 
 | |
|         if (labelBounds) {
 | |
|             SkString name;
 | |
|             font.getTypefaceOrDefault()->getFamilyName(&name);
 | |
|             canvas->drawString(name, min.fLeft, min.fBottom, labelFont, SkPaint());
 | |
|         }
 | |
|         for (const GlyphToDraw& glyphToDraw : glyphsToDraw) {
 | |
|             SkPath path;
 | |
|             font.getPath(glyphToDraw.id, &path);
 | |
|             SkPaint::Style style = path.isEmpty() ? SkPaint::kFill_Style : SkPaint::kStroke_Style;
 | |
|             SkPaint glyphPaint;
 | |
|             glyphPaint.setStyle(style);
 | |
|             canvas->drawSimpleText(&glyphToDraw.id, sizeof(glyphToDraw.id),
 | |
|                                    SkTextEncoding::kGlyphID, 0, 0, font, glyphPaint);
 | |
| 
 | |
|             if (labelBounds) {
 | |
|                 SkAutoCanvasRestore acr2(canvas, true);
 | |
|                 canvas->translate(glyphToDraw.location.fX, glyphToDraw.location.fY);
 | |
|                 canvas->rotate(glyphToDraw.rotation);
 | |
|                 SkString glyphStr;
 | |
|                 glyphStr.appendS32(glyphToDraw.id);
 | |
|                 canvas->drawString(glyphStr, 0, 0, labelFont, SkPaint());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return drawBounds;
 | |
|     }
 | |
| 
 | |
|     SkISize onISize() override { return {1024, 850}; }
 | |
| 
 | |
|     void onDraw(SkCanvas* canvas) override {
 | |
|         SkFont font;
 | |
|         font.setEdging(SkFont::Edging::kAntiAlias);
 | |
|         font.setSubpixel(true);
 | |
|         font.setSize(100);
 | |
|         font.setScaleX(fScaleX);
 | |
|         font.setSkewX(fSkewX);
 | |
| 
 | |
|         const SkColor boundsColors[2] = { SK_ColorRED, SK_ColorBLUE };
 | |
| 
 | |
|         SkFontMgr* fm = fFM.get();
 | |
|         int count = std::min(fm->countFamilies(), 32);
 | |
| 
 | |
|         int index = 0;
 | |
|         SkScalar x = 0, y = 0;
 | |
| 
 | |
|         canvas->translate(10, 120);
 | |
| 
 | |
|         for (int i = 0; i < count; ++i) {
 | |
|             sk_sp<SkFontStyleSet> set(fm->createStyleSet(i));
 | |
|             for (int j = 0; j < set->count() && j < 3; ++j) {
 | |
|                 font.setTypeface(sk_sp<SkTypeface>(set->createTypeface(j)));
 | |
|                 // Fonts with lots of glyphs are interesting, but can take a long time to find
 | |
|                 // the glyphs which make up the maximum extent.
 | |
|                 SkTypeface* typeface = font.getTypefaceOrDefault();
 | |
|                 if (typeface && 0 < typeface->countGlyphs() && typeface->countGlyphs() < 1000) {
 | |
|                     SkColor color = boundsColors[index & 1];
 | |
|                     SkRect drawBounds = show_bounds(canvas, font, x, y, color, fLabelBounds);
 | |
|                     x += drawBounds.width() + 20;
 | |
|                     index += 1;
 | |
|                     if (x > 900) {
 | |
|                         x = 0;
 | |
|                         y += 160;
 | |
|                     }
 | |
|                     if (y >= 700) {
 | |
|                         return;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     sk_sp<SkFontMgr> fFM;
 | |
|     const SkScalar fScaleX;
 | |
|     const SkScalar fSkewX;
 | |
|     bool fLabelBounds = false;
 | |
| };
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| DEF_GM(return new FontMgrGM;)
 | |
| DEF_GM(return new FontMgrMatchGM;)
 | |
| DEF_GM(return new FontMgrBoundsGM(1, 0);)
 | |
| DEF_GM(return new FontMgrBoundsGM(0.75f, 0);)
 | |
| DEF_GM(return new FontMgrBoundsGM(1, -0.25f);)
 |