blob: 002c2a83dbd7bf7701a0f3c847a67328877f79fc [file] [log] [blame]
bsalomon7c5c9da2014-06-09 15:11:30 -07001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
9#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040010#include "include/core/SkColor.h"
11#include "include/core/SkFont.h"
12#include "include/core/SkFontStyle.h"
13#include "include/core/SkFontTypes.h"
14#include "include/core/SkPaint.h"
15#include "include/core/SkPoint.h"
16#include "include/core/SkRect.h"
17#include "include/core/SkRefCnt.h"
18#include "include/core/SkScalar.h"
19#include "include/core/SkSize.h"
20#include "include/core/SkString.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050021#include "include/core/SkTypeface.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040022#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "include/utils/SkRandom.h"
24#include "tools/ToolUtils.h"
bsalomon7c5c9da2014-06-09 15:11:30 -070025
26/**
27 * Draws text with random parameters. The text draws each get their own clip rect. It is also
Brian Salomon09d994e2016-12-21 11:14:46 -050028 * used as a bench to measure how well the GPU backend combines draw ops for text draws.
bsalomon7c5c9da2014-06-09 15:11:30 -070029 */
30
31class VariedTextGM : public skiagm::GM {
32public:
33 VariedTextGM(bool effectiveClip, bool lcd)
34 : fEffectiveClip(effectiveClip)
35 , fLCD(lcd) {
bsalomon7c5c9da2014-06-09 15:11:30 -070036 }
37
38protected:
mtklein36352bf2015-03-25 18:17:31 -070039 SkString onShortName() override {
bsalomon7c5c9da2014-06-09 15:11:30 -070040 SkString name("varied_text");
41 if (fEffectiveClip) {
42 name.append("_clipped");
43 } else {
44 name.append("_ignorable_clip");
45 }
46 if (fLCD) {
47 name.append("_lcd");
48 } else {
49 name.append("_no_lcd");
50 }
51 return name;
52 }
53
mtklein36352bf2015-03-25 18:17:31 -070054 SkISize onISize() override {
bsalomon7c5c9da2014-06-09 15:11:30 -070055 return SkISize::Make(640, 480);
56 }
57
mtklein36352bf2015-03-25 18:17:31 -070058 void onOnceBeforeDraw() override {
bsalomon7c5c9da2014-06-09 15:11:30 -070059 fPaint.setAntiAlias(true);
Mike Reed91919132019-01-02 12:21:01 -050060 fFont.setEdging(fLCD ? SkFont::Edging::kSubpixelAntiAlias : SkFont::Edging::kAntiAlias);
bsalomon7c5c9da2014-06-09 15:11:30 -070061
62 SkISize size = this->getISize();
63 SkScalar w = SkIntToScalar(size.fWidth);
64 SkScalar h = SkIntToScalar(size.fHeight);
65
bungeman13b9c952016-05-12 10:09:30 -070066 static_assert(4 == SK_ARRAY_COUNT(fTypefaces), "typeface_cnt");
Mike Kleinea3f0142019-03-20 11:12:10 -050067 fTypefaces[0] = ToolUtils::create_portable_typeface("sans-serif", SkFontStyle());
68 fTypefaces[1] = ToolUtils::create_portable_typeface("sans-serif", SkFontStyle::Bold());
69 fTypefaces[2] = ToolUtils::create_portable_typeface("serif", SkFontStyle());
70 fTypefaces[3] = ToolUtils::create_portable_typeface("serif", SkFontStyle::Bold());
bsalomon7c5c9da2014-06-09 15:11:30 -070071
72 SkRandom random;
73 for (int i = 0; i < kCnt; ++i) {
74 int length = random.nextRangeU(kMinLength, kMaxLength);
75 char text[kMaxLength];
76 for (int j = 0; j < length; ++j) {
77 text[j] = (char)random.nextRangeU('!', 'z');
78 }
79 fStrings[i].set(text, length);
80
81 fColors[i] = random.nextU();
82 fColors[i] |= 0xFF000000;
Mike Kleinea3f0142019-03-20 11:12:10 -050083 fColors[i] = ToolUtils::color_to_565(fColors[i]);
bsalomon7c5c9da2014-06-09 15:11:30 -070084
mtkleindbfd7ab2016-09-01 11:24:54 -070085 constexpr SkScalar kMinPtSize = 8.f;
86 constexpr SkScalar kMaxPtSize = 32.f;
bsalomon7c5c9da2014-06-09 15:11:30 -070087
88 fPtSizes[i] = random.nextRangeScalar(kMinPtSize, kMaxPtSize);
89
bungeman13b9c952016-05-12 10:09:30 -070090 fTypefaceIndices[i] = random.nextULessThan(SK_ARRAY_COUNT(fTypefaces));
bsalomon7c5c9da2014-06-09 15:11:30 -070091
92 SkRect r;
93 fPaint.setColor(fColors[i]);
Mike Reed91919132019-01-02 12:21:01 -050094 fFont.setTypeface(fTypefaces[fTypefaceIndices[i]]);
95 fFont.setSize(fPtSizes[i]);
bsalomon7c5c9da2014-06-09 15:11:30 -070096
Ben Wagner51e15a62019-05-07 15:38:46 -040097 fFont.measureText(fStrings[i].c_str(), fStrings[i].size(), SkTextEncoding::kUTF8, &r);
Ben Wagner12857d42020-09-25 14:17:30 -040098 // The set of x,y offsets which place the bounding box inside the GM's border.
bsalomon7c5c9da2014-06-09 15:11:30 -070099 SkRect safeRect = SkRect::MakeLTRB(-r.fLeft, -r.fTop, w - r.fRight, h - r.fBottom);
100 if (safeRect.isEmpty()) {
Ben Wagner12857d42020-09-25 14:17:30 -0400101 // If the bounds don't fit then allow any offset in the GM's border.
bsalomon7c5c9da2014-06-09 15:11:30 -0700102 safeRect = SkRect::MakeWH(w, h);
103 }
Ben Wagner12857d42020-09-25 14:17:30 -0400104 fOffsets[i].fX = random.nextRangeScalar(safeRect.fLeft, safeRect.fRight);
105 fOffsets[i].fY = random.nextRangeScalar(safeRect.fTop, safeRect.fBottom);
bsalomon7c5c9da2014-06-09 15:11:30 -0700106
107 fClipRects[i] = r;
Ben Wagner12857d42020-09-25 14:17:30 -0400108 fClipRects[i].offset(fOffsets[i].fX, fOffsets[i].fY);
bsalomon7c5c9da2014-06-09 15:11:30 -0700109 fClipRects[i].outset(2.f, 2.f);
110
111 if (fEffectiveClip) {
112 fClipRects[i].fRight -= 0.25f * fClipRects[i].width();
113 }
114 }
115 }
116
mtklein36352bf2015-03-25 18:17:31 -0700117 void onDraw(SkCanvas* canvas) override {
bsalomon7c5c9da2014-06-09 15:11:30 -0700118 for (int i = 0; i < kCnt; ++i) {
119 fPaint.setColor(fColors[i]);
Mike Reed91919132019-01-02 12:21:01 -0500120 fFont.setSize(fPtSizes[i]);
121 fFont.setTypeface(fTypefaces[fTypefaceIndices[i]]);
bsalomon7c5c9da2014-06-09 15:11:30 -0700122
123 canvas->save();
124 canvas->clipRect(fClipRects[i]);
Ben Wagner12857d42020-09-25 14:17:30 -0400125 canvas->translate(fOffsets[i].fX, fOffsets[i].fY);
Ben Wagner51e15a62019-05-07 15:38:46 -0400126 canvas->drawSimpleText(fStrings[i].c_str(), fStrings[i].size(), SkTextEncoding::kUTF8,
Mike Reed91919132019-01-02 12:21:01 -0500127 0, 0, fFont, fPaint);
bsalomon7c5c9da2014-06-09 15:11:30 -0700128 canvas->restore();
129 }
130
131 // Visualize the clips, but not in bench mode.
132 if (kBench_Mode != this->getMode()) {
133 SkPaint wirePaint;
134 wirePaint.setAntiAlias(true);
135 wirePaint.setStrokeWidth(0);
136 wirePaint.setStyle(SkPaint::kStroke_Style);
137 for (int i = 0; i < kCnt; ++i) {
138 canvas->drawRect(fClipRects[i], wirePaint);
139 }
140 }
141 }
142
mtklein36352bf2015-03-25 18:17:31 -0700143 bool runAsBench() const override { return true; }
bsalomon7c5c9da2014-06-09 15:11:30 -0700144
145private:
mtkleindbfd7ab2016-09-01 11:24:54 -0700146 static constexpr int kCnt = 30;
147 static constexpr int kMinLength = 15;
148 static constexpr int kMaxLength = 40;
bsalomon7c5c9da2014-06-09 15:11:30 -0700149
150 bool fEffectiveClip;
151 bool fLCD;
bungeman13b9c952016-05-12 10:09:30 -0700152 sk_sp<SkTypeface> fTypefaces[4];
bsalomon7c5c9da2014-06-09 15:11:30 -0700153 SkPaint fPaint;
Mike Reed91919132019-01-02 12:21:01 -0500154 SkFont fFont;
bsalomon7c5c9da2014-06-09 15:11:30 -0700155
156 // precomputed for each text draw
157 SkString fStrings[kCnt];
158 SkColor fColors[kCnt];
159 SkScalar fPtSizes[kCnt];
bungeman13b9c952016-05-12 10:09:30 -0700160 int fTypefaceIndices[kCnt];
Ben Wagner12857d42020-09-25 14:17:30 -0400161 SkPoint fOffsets[kCnt];
bsalomon7c5c9da2014-06-09 15:11:30 -0700162 SkRect fClipRects[kCnt];
163
John Stiles7571f9e2020-09-02 22:42:33 -0400164 using INHERITED = skiagm::GM;
bsalomon7c5c9da2014-06-09 15:11:30 -0700165};
166
halcanary385fe4d2015-08-26 13:07:48 -0700167DEF_GM(return new VariedTextGM(false, false);)
168DEF_GM(return new VariedTextGM(true, false);)
169DEF_GM(return new VariedTextGM(false, true);)
170DEF_GM(return new VariedTextGM(true, true);)