blob: a3c8f05a6451f2053f77763e02f8f49db7811ce1 [file] [log] [blame]
bsalomon@google.com48dd1a22011-10-31 14:18:20 +00001/*
2 * Copyright 2011 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"
Chris Dalton3a778372019-02-07 15:23:36 -07009
10#include "GrContext.h"
bsalomonb62da802015-01-31 07:51:14 -080011#include "SkShader.h"
Mike Kleinb323a5e2017-07-24 15:21:31 -040012#include "SkTraceEvent.h"
Mike Kleinea3f0142019-03-20 11:12:10 -050013#include "ToolUtils.h"
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000014using namespace skiagm;
15
Chris Dalton50e24d72019-02-07 16:20:09 -070016constexpr char GM::kErrorMsg_DrawSkippedGpuOnly[];
17
18static void draw_failure_message(SkCanvas* canvas, const char format[], ...) {
19 SkString failureMsg;
20
21 va_list argp;
22 va_start(argp, format);
23 failureMsg.appendVAList(format, argp);
24 va_end(argp);
25
26 constexpr SkScalar kOffset = 5.0f;
27 canvas->drawColor(SkColorSetRGB(200,0,0));
28 SkFont font;
29 SkRect bounds;
30 font.measureText(failureMsg.c_str(), failureMsg.size(), kUTF8_SkTextEncoding, &bounds);
31 SkPaint textPaint;
32 textPaint.setColor(SK_ColorWHITE);
33 canvas->drawString(failureMsg, kOffset, bounds.height() + kOffset, font, textPaint);
34}
35
36static void draw_gpu_only_message(SkCanvas* canvas) {
37 SkBitmap bmp;
38 bmp.allocN32Pixels(128, 64);
39 SkCanvas bmpCanvas(bmp);
40 bmpCanvas.drawColor(SK_ColorWHITE);
Mike Kleinea3f0142019-03-20 11:12:10 -050041 SkFont font(ToolUtils::create_portable_typeface(), 20);
Chris Dalton50e24d72019-02-07 16:20:09 -070042 SkPaint paint;
43 paint.setColor(SK_ColorRED);
44 bmpCanvas.drawString("GPU Only", 20, 40, font, paint);
45 SkMatrix localM;
46 localM.setRotate(35.f);
47 localM.postTranslate(10.f, 0.f);
48 paint.setShader(SkShader::MakeBitmapShader(
49 bmp, SkShader::kMirror_TileMode, SkShader::kMirror_TileMode, &localM));
50 paint.setFilterQuality(kMedium_SkFilterQuality);
51 canvas->drawPaint(paint);
52}
53
Chris Dalton3a778372019-02-07 15:23:36 -070054GM::GM(SkColor bgColor) {
commit-bot@chromium.orgb21fac12014-02-07 21:13:11 +000055 fMode = kGM_Mode;
Chris Dalton3a778372019-02-07 15:23:36 -070056 fBGColor = bgColor;
reed@google.comaef73612012-11-16 13:41:45 +000057 fCanvasIsDeferred = false;
reed@google.com7775d662012-11-27 15:15:58 +000058 fHaveCalledOnceBeforeDraw = false;
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000059}
tfarina880914c2014-06-09 12:05:34 -070060
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000061GM::~GM() {}
62
Chris Dalton50e24d72019-02-07 16:20:09 -070063DrawResult GM::draw(SkCanvas* canvas, SkString* errorMsg) {
Mike Kleinb323a5e2017-07-24 15:21:31 -040064 TRACE_EVENT1("GM", TRACE_FUNC, "name", TRACE_STR_COPY(this->getName()));
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000065 this->drawBackground(canvas);
Chris Dalton50e24d72019-02-07 16:20:09 -070066 return this->drawContent(canvas, errorMsg);
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000067}
68
Chris Dalton50e24d72019-02-07 16:20:09 -070069DrawResult GM::drawContent(SkCanvas* canvas, SkString* errorMsg) {
Mike Kleinb323a5e2017-07-24 15:21:31 -040070 TRACE_EVENT0("GM", TRACE_FUNC);
reed@google.com7775d662012-11-27 15:15:58 +000071 if (!fHaveCalledOnceBeforeDraw) {
72 fHaveCalledOnceBeforeDraw = true;
73 this->onOnceBeforeDraw();
74 }
Mike Reed36c4fb32018-12-18 11:48:01 -050075 SkAutoCanvasRestore acr(canvas, true);
Chris Dalton50e24d72019-02-07 16:20:09 -070076 DrawResult drawResult = this->onDraw(canvas, errorMsg);
77 if (DrawResult::kOk != drawResult) {
78 if (DrawResult::kFail == drawResult) {
79 draw_failure_message(canvas, "DRAW FAILED: %s", errorMsg->c_str());
80 } else if (SkString(kErrorMsg_DrawSkippedGpuOnly) == *errorMsg) {
81 draw_gpu_only_message(canvas);
82 } else {
83 draw_failure_message(canvas, "DRAW SKIPPED: %s", errorMsg->c_str());
84 }
85 }
86 return drawResult;
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000087}
88
89void GM::drawBackground(SkCanvas* canvas) {
Mike Kleinb323a5e2017-07-24 15:21:31 -040090 TRACE_EVENT0("GM", TRACE_FUNC);
reed@google.com7775d662012-11-27 15:15:58 +000091 if (!fHaveCalledOnceBeforeDraw) {
92 fHaveCalledOnceBeforeDraw = true;
93 this->onOnceBeforeDraw();
94 }
Mike Reed36c4fb32018-12-18 11:48:01 -050095 SkAutoCanvasRestore acr(canvas, true);
Chris Dalton21ca3702019-02-01 12:15:42 -070096 canvas->drawColor(fBGColor, SkBlendMode::kSrc);
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000097}
98
Mike Klein9707e902019-02-07 16:18:22 -050099DrawResult GM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
100 this->onDraw(canvas);
101 return DrawResult::kOk;
102}
103void GM::onDraw(SkCanvas*) { SK_ABORT("Not implemented."); }
104
105
106SkISize SimpleGM::onISize() { return fSize; }
107SkString SimpleGM::onShortName() { return fName; }
108DrawResult SimpleGM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
109 return fDrawProc(canvas, errorMsg);
110}
111
112SkISize SimpleGpuGM::onISize() { return fSize; }
113SkString SimpleGpuGM::onShortName() { return fName; }
114DrawResult SimpleGpuGM::onDraw(GrContext* ctx, GrRenderTargetContext* rtc, SkCanvas* canvas,
115 SkString* errorMsg) {
116 return fDrawProc(ctx, rtc, canvas, errorMsg);
117}
118
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000119const char* GM::getName() {
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000120 if (fShortName.size() == 0) {
121 fShortName = this->onShortName();
122 }
123 return fShortName.c_str();
124}
125
126void GM::setBGColor(SkColor color) {
127 fBGColor = color;
128}
129
Mike Kleincd5104e2019-03-20 11:55:08 -0500130bool GM::animate(const AnimTimer& timer) { return this->onAnimate(timer); }
reedd9adfe62015-02-01 19:01:04 -0800131
Mike Klein9707e902019-02-07 16:18:22 -0500132bool GM::runAsBench() const { return false; }
133void GM::modifyGrContextOptions(GrContextOptions* options) {}
134
135void GM::onOnceBeforeDraw() {}
136
Mike Kleincd5104e2019-03-20 11:55:08 -0500137bool GM::onAnimate(const AnimTimer&) { return false; }
Mike Klein9707e902019-02-07 16:18:22 -0500138bool GM::onHandleKey(SkUnichar uni) { return false; }
139bool GM::onGetControls(SkMetaData*) { return false; }
140void GM::onSetControls(const SkMetaData&) {}
141
reedd9adfe62015-02-01 19:01:04 -0800142/////////////////////////////////////////////////////////////////////////////////////////////
143
reed@google.com2d6ef522012-01-03 17:20:38 +0000144void GM::drawSizeBounds(SkCanvas* canvas, SkColor color) {
145 SkISize size = this->getISize();
146 SkRect r = SkRect::MakeWH(SkIntToScalar(size.width()),
147 SkIntToScalar(size.height()));
148 SkPaint paint;
149 paint.setColor(color);
150 canvas->drawRect(r, paint);
151}
152
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000153// need to explicitly declare this, or we get some weird infinite loop llist
tfarinabcbc1782014-06-18 14:32:48 -0700154template GMRegistry* GMRegistry::gHead;
halcanaryf62c6342015-01-12 15:27:46 -0800155
Mike Klein9707e902019-02-07 16:18:22 -0500156DrawResult GpuGM::onDraw(GrContext* ctx, GrRenderTargetContext* rtc, SkCanvas* canvas,
157 SkString* errorMsg) {
158 this->onDraw(ctx, rtc, canvas);
159 return DrawResult::kOk;
160}
161void GpuGM::onDraw(GrContext*, GrRenderTargetContext*, SkCanvas*) {
162 SK_ABORT("Not implemented.");
163}
164
Chris Dalton50e24d72019-02-07 16:20:09 -0700165DrawResult GpuGM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
Chris Dalton3a778372019-02-07 15:23:36 -0700166 GrContext* ctx = canvas->getGrContext();
167 GrRenderTargetContext* rtc = canvas->internal_private_accessTopLayerRenderTargetContext();
168 if (!ctx || !rtc) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700169 *errorMsg = kErrorMsg_DrawSkippedGpuOnly;
170 return DrawResult::kSkip;
Chris Dalton3a778372019-02-07 15:23:36 -0700171 }
172 if (ctx->abandoned()) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700173 *errorMsg = "GrContext abandoned.";
Mike Klein290690c2019-04-02 18:11:57 -0400174 return DrawResult::kSkip;
Chris Dalton3a778372019-02-07 15:23:36 -0700175 }
Chris Dalton50e24d72019-02-07 16:20:09 -0700176 return this->onDraw(ctx, rtc, canvas, errorMsg);
halcanaryf62c6342015-01-12 15:27:46 -0800177}
Mike Kleinc9eace82018-10-31 10:49:38 -0400178
179template <typename Fn>
180static void mark(SkCanvas* canvas, SkScalar x, SkScalar y, Fn&& fn) {
181 SkPaint alpha;
182 alpha.setAlpha(0x50);
183 canvas->saveLayer(nullptr, &alpha);
184 canvas->translate(x,y);
185 canvas->scale(2,2);
186 fn();
187 canvas->restore();
188}
189
190void MarkGMGood(SkCanvas* canvas, SkScalar x, SkScalar y) {
191 mark(canvas, x,y, [&]{
192 SkPaint paint;
193
194 // A green circle.
195 paint.setColor(SkColorSetRGB(27, 158, 119));
196 canvas->drawCircle(0,0, 12, paint);
197
198 // Cut out a check mark.
199 paint.setBlendMode(SkBlendMode::kSrc);
200 paint.setColor(0x00000000);
201 paint.setStrokeWidth(2);
202 paint.setStyle(SkPaint::kStroke_Style);
203 canvas->drawLine(-6, 0,
204 -1, 5, paint);
205 canvas->drawLine(-1, +5,
206 +7, -5, paint);
207 });
208}
209
210void MarkGMBad(SkCanvas* canvas, SkScalar x, SkScalar y) {
211 mark(canvas, x,y, [&] {
212 SkPaint paint;
213
214 // A red circle.
215 paint.setColor(SkColorSetRGB(231, 41, 138));
216 canvas->drawCircle(0,0, 12, paint);
217
218 // Cut out an 'X'.
219 paint.setBlendMode(SkBlendMode::kSrc);
220 paint.setColor(0x00000000);
221 paint.setStrokeWidth(2);
222 paint.setStyle(SkPaint::kStroke_Style);
223 canvas->drawLine(-5,-5,
224 +5,+5, paint);
225 canvas->drawLine(+5,-5,
226 -5,+5, paint);
227 });
228}