1 /* 2 * Copyright 2014 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #include "gm.h" 9 #include "sk_tool_utils.h" 10 #include "SkCanvas.h" 11 #include "SkPath.h" 12 #include "SkTypeface.h" 13 #include "SkRandom.h" 14 15 /** 16 * Draws text with random parameters. The text draws each get their own clip rect. It is also 17 * used as a bench to measure how well the GPU backend combines draw ops for text draws. 18 */ 19 20 class VariedTextGM : public skiagm::GM { 21 public: VariedTextGM(bool effectiveClip,bool lcd)22 VariedTextGM(bool effectiveClip, bool lcd) 23 : fEffectiveClip(effectiveClip) 24 , fLCD(lcd) { 25 } 26 27 protected: onShortName()28 SkString onShortName() override { 29 SkString name("varied_text"); 30 if (fEffectiveClip) { 31 name.append("_clipped"); 32 } else { 33 name.append("_ignorable_clip"); 34 } 35 if (fLCD) { 36 name.append("_lcd"); 37 } else { 38 name.append("_no_lcd"); 39 } 40 return name; 41 } 42 onISize()43 SkISize onISize() override { 44 return SkISize::Make(640, 480); 45 } 46 onOnceBeforeDraw()47 void onOnceBeforeDraw() override { 48 fPaint.setAntiAlias(true); 49 fPaint.setLCDRenderText(fLCD); 50 51 SkISize size = this->getISize(); 52 SkScalar w = SkIntToScalar(size.fWidth); 53 SkScalar h = SkIntToScalar(size.fHeight); 54 55 static_assert(4 == SK_ARRAY_COUNT(fTypefaces), "typeface_cnt"); 56 fTypefaces[0] = sk_tool_utils::create_portable_typeface("sans-serif", SkFontStyle()); 57 fTypefaces[1] = sk_tool_utils::create_portable_typeface("sans-serif", 58 SkFontStyle::FromOldStyle(SkTypeface::kBold)); 59 fTypefaces[2] = sk_tool_utils::create_portable_typeface("serif", SkFontStyle()); 60 fTypefaces[3] = sk_tool_utils::create_portable_typeface("serif", 61 SkFontStyle::FromOldStyle(SkTypeface::kBold)); 62 63 SkRandom random; 64 for (int i = 0; i < kCnt; ++i) { 65 int length = random.nextRangeU(kMinLength, kMaxLength); 66 char text[kMaxLength]; 67 for (int j = 0; j < length; ++j) { 68 text[j] = (char)random.nextRangeU('!', 'z'); 69 } 70 fStrings[i].set(text, length); 71 72 fColors[i] = random.nextU(); 73 fColors[i] |= 0xFF000000; 74 fColors[i] = sk_tool_utils::color_to_565(fColors[i]); 75 76 constexpr SkScalar kMinPtSize = 8.f; 77 constexpr SkScalar kMaxPtSize = 32.f; 78 79 fPtSizes[i] = random.nextRangeScalar(kMinPtSize, kMaxPtSize); 80 81 fTypefaceIndices[i] = random.nextULessThan(SK_ARRAY_COUNT(fTypefaces)); 82 83 SkRect r; 84 fPaint.setColor(fColors[i]); 85 fPaint.setTypeface(fTypefaces[fTypefaceIndices[i]]); 86 fPaint.setTextSize(fPtSizes[i]); 87 88 fPaint.measureText(fStrings[i].c_str(), fStrings[i].size(), &r); 89 // safeRect is set of x,y positions where we can draw the string without hitting 90 // the GM's border. 91 SkRect safeRect = SkRect::MakeLTRB(-r.fLeft, -r.fTop, w - r.fRight, h - r.fBottom); 92 if (safeRect.isEmpty()) { 93 // If we don't fit then just don't worry about how we get cliped to the device 94 // border. 95 safeRect = SkRect::MakeWH(w, h); 96 } 97 fPositions[i].fX = random.nextRangeScalar(safeRect.fLeft, safeRect.fRight); 98 fPositions[i].fY = random.nextRangeScalar(safeRect.fTop, safeRect.fBottom); 99 100 fClipRects[i] = r; 101 fClipRects[i].offset(fPositions[i].fX, fPositions[i].fY); 102 fClipRects[i].outset(2.f, 2.f); 103 104 if (fEffectiveClip) { 105 fClipRects[i].fRight -= 0.25f * fClipRects[i].width(); 106 } 107 } 108 } 109 onDraw(SkCanvas * canvas)110 void onDraw(SkCanvas* canvas) override { 111 for (int i = 0; i < kCnt; ++i) { 112 fPaint.setColor(fColors[i]); 113 fPaint.setTextSize(fPtSizes[i]); 114 fPaint.setTypeface(fTypefaces[fTypefaceIndices[i]]); 115 116 canvas->save(); 117 canvas->clipRect(fClipRects[i]); 118 canvas->translate(fPositions[i].fX, fPositions[i].fY); 119 canvas->drawText(fStrings[i].c_str(), fStrings[i].size(), 0, 0, fPaint); 120 canvas->restore(); 121 } 122 123 // Visualize the clips, but not in bench mode. 124 if (kBench_Mode != this->getMode()) { 125 SkPaint wirePaint; 126 wirePaint.setAntiAlias(true); 127 wirePaint.setStrokeWidth(0); 128 wirePaint.setStyle(SkPaint::kStroke_Style); 129 for (int i = 0; i < kCnt; ++i) { 130 canvas->drawRect(fClipRects[i], wirePaint); 131 } 132 } 133 } 134 runAsBench() const135 bool runAsBench() const override { return true; } 136 137 private: 138 static constexpr int kCnt = 30; 139 static constexpr int kMinLength = 15; 140 static constexpr int kMaxLength = 40; 141 142 bool fEffectiveClip; 143 bool fLCD; 144 sk_sp<SkTypeface> fTypefaces[4]; 145 SkPaint fPaint; 146 147 // precomputed for each text draw 148 SkString fStrings[kCnt]; 149 SkColor fColors[kCnt]; 150 SkScalar fPtSizes[kCnt]; 151 int fTypefaceIndices[kCnt]; 152 SkPoint fPositions[kCnt]; 153 SkRect fClipRects[kCnt]; 154 155 typedef skiagm::GM INHERITED; 156 }; 157 158 DEF_GM(return new VariedTextGM(false, false);) 159 DEF_GM(return new VariedTextGM(true, false);) 160 DEF_GM(return new VariedTextGM(false, true);) 161 DEF_GM(return new VariedTextGM(true, true);) 162