/* * Copyright 2014 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/SkFontStyle.h" #include "include/core/SkFontTypes.h" #include "include/core/SkPaint.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/utils/SkRandom.h" #include "tools/ToolUtils.h" /** * Draws text with random parameters. The text draws each get their own clip rect. It is also * used as a bench to measure how well the GPU backend combines draw ops for text draws. */ class VariedTextGM : public skiagm::GM { public: VariedTextGM(bool effectiveClip, bool lcd) : fEffectiveClip(effectiveClip) , fLCD(lcd) { } protected: SkString onShortName() override { SkString name("varied_text"); if (fEffectiveClip) { name.append("_clipped"); } else { name.append("_ignorable_clip"); } if (fLCD) { name.append("_lcd"); } else { name.append("_no_lcd"); } return name; } SkISize onISize() override { return SkISize::Make(640, 480); } void onOnceBeforeDraw() override { fPaint.setAntiAlias(true); fFont.setEdging(fLCD ? SkFont::Edging::kSubpixelAntiAlias : SkFont::Edging::kAntiAlias); SkISize size = this->getISize(); SkScalar w = SkIntToScalar(size.fWidth); SkScalar h = SkIntToScalar(size.fHeight); static_assert(4 == SK_ARRAY_COUNT(fTypefaces), "typeface_cnt"); fTypefaces[0] = ToolUtils::create_portable_typeface("sans-serif", SkFontStyle()); fTypefaces[1] = ToolUtils::create_portable_typeface("sans-serif", SkFontStyle::Bold()); fTypefaces[2] = ToolUtils::create_portable_typeface("serif", SkFontStyle()); fTypefaces[3] = ToolUtils::create_portable_typeface("serif", SkFontStyle::Bold()); SkRandom random; for (int i = 0; i < kCnt; ++i) { int length = random.nextRangeU(kMinLength, kMaxLength); char text[kMaxLength]; for (int j = 0; j < length; ++j) { text[j] = (char)random.nextRangeU('!', 'z'); } fStrings[i].set(text, length); fColors[i] = random.nextU(); fColors[i] |= 0xFF000000; fColors[i] = ToolUtils::color_to_565(fColors[i]); constexpr SkScalar kMinPtSize = 8.f; constexpr SkScalar kMaxPtSize = 32.f; fPtSizes[i] = random.nextRangeScalar(kMinPtSize, kMaxPtSize); fTypefaceIndices[i] = random.nextULessThan(SK_ARRAY_COUNT(fTypefaces)); SkRect r; fPaint.setColor(fColors[i]); fFont.setTypeface(fTypefaces[fTypefaceIndices[i]]); fFont.setSize(fPtSizes[i]); fFont.measureText(fStrings[i].c_str(), fStrings[i].size(), SkTextEncoding::kUTF8, &r); // The set of x,y offsets which place the bounding box inside the GM's border. SkRect safeRect = SkRect::MakeLTRB(-r.fLeft, -r.fTop, w - r.fRight, h - r.fBottom); if (safeRect.isEmpty()) { // If the bounds don't fit then allow any offset in the GM's border. safeRect = SkRect::MakeWH(w, h); } fOffsets[i].fX = random.nextRangeScalar(safeRect.fLeft, safeRect.fRight); fOffsets[i].fY = random.nextRangeScalar(safeRect.fTop, safeRect.fBottom); fClipRects[i] = r; fClipRects[i].offset(fOffsets[i].fX, fOffsets[i].fY); fClipRects[i].outset(2.f, 2.f); if (fEffectiveClip) { fClipRects[i].fRight -= 0.25f * fClipRects[i].width(); } } } void onDraw(SkCanvas* canvas) override { for (int i = 0; i < kCnt; ++i) { fPaint.setColor(fColors[i]); fFont.setSize(fPtSizes[i]); fFont.setTypeface(fTypefaces[fTypefaceIndices[i]]); canvas->save(); canvas->clipRect(fClipRects[i]); canvas->translate(fOffsets[i].fX, fOffsets[i].fY); canvas->drawSimpleText(fStrings[i].c_str(), fStrings[i].size(), SkTextEncoding::kUTF8, 0, 0, fFont, fPaint); canvas->restore(); } // Visualize the clips, but not in bench mode. if (kBench_Mode != this->getMode()) { SkPaint wirePaint; wirePaint.setAntiAlias(true); wirePaint.setStrokeWidth(0); wirePaint.setStyle(SkPaint::kStroke_Style); for (int i = 0; i < kCnt; ++i) { canvas->drawRect(fClipRects[i], wirePaint); } } } bool runAsBench() const override { return true; } private: inline static constexpr int kCnt = 30; inline static constexpr int kMinLength = 15; inline static constexpr int kMaxLength = 40; bool fEffectiveClip; bool fLCD; sk_sp fTypefaces[4]; SkPaint fPaint; SkFont fFont; // precomputed for each text draw SkString fStrings[kCnt]; SkColor fColors[kCnt]; SkScalar fPtSizes[kCnt]; int fTypefaceIndices[kCnt]; SkPoint fOffsets[kCnt]; SkRect fClipRects[kCnt]; using INHERITED = skiagm::GM; }; DEF_GM(return new VariedTextGM(false, false);) DEF_GM(return new VariedTextGM(true, false);) DEF_GM(return new VariedTextGM(false, true);) DEF_GM(return new VariedTextGM(true, true);)