blob: 2819a813ae792f80d016c64d5f85f501f9eaf620 [file] [log] [blame]
reed@google.com71121732012-09-18 15:14:33 +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 */
reed2ad1aa62016-03-09 09:50:50 -08007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner6a34f3a2019-05-01 10:59:30 -04009#include "include/core/SkBitmap.h"
10#include "include/core/SkBlendMode.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkCanvas.h"
Ben Wagner6a34f3a2019-05-01 10:59:30 -040012#include "include/core/SkColor.h"
13#include "include/core/SkPaint.h"
14#include "include/core/SkPoint.h"
15#include "include/core/SkRect.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "include/core/SkShader.h"
Ben Wagner6a34f3a2019-05-01 10:59:30 -040017#include "include/core/SkSize.h"
18#include "include/core/SkString.h"
Mike Reed6dbeac52021-01-13 13:14:23 -050019#include "include/core/SkSurface.h"
Ben Wagner6a34f3a2019-05-01 10:59:30 -040020#include "include/core/SkTileMode.h"
21#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "include/effects/SkGradientShader.h"
reed@google.com71121732012-09-18 15:14:33 +000023
Mike Reed6dbeac52021-01-13 13:14:23 -050024static sk_sp<SkImage> make_image() {
25 auto surf = SkSurface::MakeRasterN32Premul(64, 64);
26 auto canvas = surf->getCanvas();
reed@google.com71121732012-09-18 15:14:33 +000027
Mike Reed6dbeac52021-01-13 13:14:23 -050028 canvas->drawColor(SK_ColorRED);
reed@google.com71121732012-09-18 15:14:33 +000029 SkPaint paint;
30 paint.setAntiAlias(true);
31 const SkPoint pts[] = { { 0, 0 }, { 64, 64 } };
32 const SkColor colors[] = { SK_ColorWHITE, SK_ColorBLUE };
Mike Reedfae8fce2019-04-03 10:27:45 -040033 paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp));
Mike Reed6dbeac52021-01-13 13:14:23 -050034 canvas->drawCircle(32, 32, 32, paint);
35
36 return surf->makeImageSnapshot();
reed@google.com71121732012-09-18 15:14:33 +000037}
38
39class DrawBitmapRect2 : public skiagm::GM {
40 bool fUseIRect;
41public:
42 DrawBitmapRect2(bool useIRect) : fUseIRect(useIRect) {
43 }
44
45protected:
mtklein36352bf2015-03-25 18:17:31 -070046 SkString onShortName() override {
reed@google.com71121732012-09-18 15:14:33 +000047 SkString str;
48 str.printf("bitmaprect_%s", fUseIRect ? "i" : "s");
49 return str;
50 }
51
mtklein36352bf2015-03-25 18:17:31 -070052 SkISize onISize() override {
reed@google.com71121732012-09-18 15:14:33 +000053 return SkISize::Make(640, 480);
54 }
55
mtklein36352bf2015-03-25 18:17:31 -070056 void onDraw(SkCanvas* canvas) override {
Mike Kleind46dce32018-08-16 10:17:03 -040057 canvas->drawColor(0xFFCCCCCC);
reed@google.com71121732012-09-18 15:14:33 +000058
59 const SkIRect src[] = {
60 { 0, 0, 32, 32 },
61 { 0, 0, 80, 80 },
62 { 32, 32, 96, 96 },
63 { -32, -32, 32, 32, }
64 };
65
66 SkPaint paint;
67 paint.setStyle(SkPaint::kStroke_Style);
Mike Reede02d7f82021-01-21 22:25:21 -050068 auto sampling = SkSamplingOptions();
reed@google.com71121732012-09-18 15:14:33 +000069
Mike Reed6dbeac52021-01-13 13:14:23 -050070 auto image = make_image();
reed@google.com71121732012-09-18 15:14:33 +000071
72 SkRect dstR = { 0, 200, 128, 380 };
73
74 canvas->translate(16, 40);
75 for (size_t i = 0; i < SK_ARRAY_COUNT(src); i++) {
76 SkRect srcR;
77 srcR.set(src[i]);
78
Mike Reede02d7f82021-01-21 22:25:21 -050079 canvas->drawImage(image, 0, 0, sampling, &paint);
robertphillips@google.com21a95f12012-09-26 13:10:19 +000080 if (!fUseIRect) {
Mike Reede02d7f82021-01-21 22:25:21 -050081 canvas->drawImageRect(image.get(), srcR, dstR, sampling, &paint,
Mike Reed6dbeac52021-01-13 13:14:23 -050082 SkCanvas::kStrict_SrcRectConstraint);
reed@google.com71121732012-09-18 15:14:33 +000083 } else {
Mike Reede02d7f82021-01-21 22:25:21 -050084 canvas->drawImageRect(image.get(), SkRect::Make(src[i]), dstR, sampling, &paint,
85 SkCanvas::kStrict_SrcRectConstraint);
reed@google.com71121732012-09-18 15:14:33 +000086 }
87
88 canvas->drawRect(dstR, paint);
89 canvas->drawRect(srcR, paint);
90
91 canvas->translate(160, 0);
92 }
93 }
94
95private:
John Stiles7571f9e2020-09-02 22:42:33 -040096 using INHERITED = skiagm::GM;
reed@google.com71121732012-09-18 15:14:33 +000097};
98
99//////////////////////////////////////////////////////////////////////////////
reed776c0cd2015-01-27 07:26:51 -0800100
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000101static void make_3x3_bitmap(SkBitmap* bitmap) {
reed776c0cd2015-01-27 07:26:51 -0800102 const int xSize = 3;
103 const int ySize = 3;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000104
reed776c0cd2015-01-27 07:26:51 -0800105 const SkColor textureData[xSize][ySize] = {
robertphillips@google.com93f03322012-12-03 17:35:19 +0000106 { SK_ColorRED, SK_ColorWHITE, SK_ColorBLUE },
107 { SK_ColorGREEN, SK_ColorBLACK, SK_ColorCYAN },
108 { SK_ColorYELLOW, SK_ColorGRAY, SK_ColorMAGENTA }
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000109 };
110
reed776c0cd2015-01-27 07:26:51 -0800111 bitmap->allocN32Pixels(xSize, ySize, true);
112 SkCanvas canvas(*bitmap);
113 SkPaint paint;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000114
reed776c0cd2015-01-27 07:26:51 -0800115 for (int y = 0; y < ySize; y++) {
116 for (int x = 0; x < xSize; x++) {
117 paint.setColor(textureData[x][y]);
118 canvas.drawIRect(SkIRect::MakeXYWH(x, y, 1, 1), paint);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000119 }
120 }
121}
122
reeda5517e22015-07-14 10:54:12 -0700123// This GM attempts to make visible any issues drawBitmapRect may have
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000124// with partial source rects. In this case the eight pixels on the border
125// should be half the width/height of the central pixel, i.e.:
126// __|____|__
127// | |
128// __|____|__
129// | |
130class DrawBitmapRect3 : public skiagm::GM {
131public:
132 DrawBitmapRect3() {
133 this->setBGColor(SK_ColorBLACK);
134 }
135
136protected:
mtklein36352bf2015-03-25 18:17:31 -0700137 SkString onShortName() override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000138 SkString str;
139 str.printf("3x3bitmaprect");
140 return str;
141 }
142
mtklein36352bf2015-03-25 18:17:31 -0700143 SkISize onISize() override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000144 return SkISize::Make(640, 480);
145 }
146
mtklein36352bf2015-03-25 18:17:31 -0700147 void onDraw(SkCanvas* canvas) override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000148
149 SkBitmap bitmap;
150 make_3x3_bitmap(&bitmap);
151
152 SkRect srcR = { 0.5f, 0.5f, 2.5f, 2.5f };
153 SkRect dstR = { 100, 100, 300, 200 };
154
Mike Reedfa582c82021-01-23 22:07:05 -0500155 canvas->drawImageRect(bitmap.asImage(), srcR, dstR, SkSamplingOptions(),
156 nullptr, SkCanvas::kStrict_SrcRectConstraint);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000157 }
158
159private:
John Stiles7571f9e2020-09-02 22:42:33 -0400160 using INHERITED = skiagm::GM;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000161};
162
163//////////////////////////////////////////////////////////////////////////////
Mike Reedfa582c82021-01-23 22:07:05 -0500164static sk_sp<SkImage> make_big_bitmap() {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000165
mtkleindbfd7ab2016-09-01 11:24:54 -0700166 constexpr int gXSize = 4096;
167 constexpr int gYSize = 4096;
168 constexpr int gBorderWidth = 10;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000169
Mike Reedfa582c82021-01-23 22:07:05 -0500170 SkBitmap bitmap;
171 bitmap.allocN32Pixels(gXSize, gYSize);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000172 for (int y = 0; y < gYSize; ++y) {
173 for (int x = 0; x < gXSize; ++x) {
skia.committer@gmail.com44d49882012-09-27 02:01:04 +0000174 if (x <= gBorderWidth || x >= gXSize-gBorderWidth ||
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000175 y <= gBorderWidth || y >= gYSize-gBorderWidth) {
Mike Reedfa582c82021-01-23 22:07:05 -0500176 *bitmap.getAddr32(x, y) = SkPreMultiplyColor(0x88FFFFFF);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000177 } else {
Mike Reedfa582c82021-01-23 22:07:05 -0500178 *bitmap.getAddr32(x, y) = SkPreMultiplyColor(0x88FF0000);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000179 }
180 }
181 }
Mike Reedfa582c82021-01-23 22:07:05 -0500182 bitmap.setImmutable();
183 return bitmap.asImage();
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000184}
185
186// This GM attempts to reveal any issues we may have when the GPU has to
187// break up a large texture in order to draw it. The XOR transfer mode will
188// create stripes in the image if there is imprecision in the destination
189// tile placement.
190class DrawBitmapRect4 : public skiagm::GM {
191 bool fUseIRect;
Mike Reedfa582c82021-01-23 22:07:05 -0500192 sk_sp<SkImage> fBigImage;
commit-bot@chromium.org9a558d42014-05-30 15:06:24 +0000193
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000194public:
195 DrawBitmapRect4(bool useIRect) : fUseIRect(useIRect) {
196 this->setBGColor(0x88444444);
197 }
198
199protected:
mtklein36352bf2015-03-25 18:17:31 -0700200 SkString onShortName() override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000201 SkString str;
202 str.printf("bigbitmaprect_%s", fUseIRect ? "i" : "s");
203 return str;
204 }
205
mtklein36352bf2015-03-25 18:17:31 -0700206 SkISize onISize() override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000207 return SkISize::Make(640, 480);
208 }
209
mtklein36352bf2015-03-25 18:17:31 -0700210 void onOnceBeforeDraw() override {
Mike Reedfa582c82021-01-23 22:07:05 -0500211 fBigImage = make_big_bitmap();
commit-bot@chromium.org9a558d42014-05-30 15:06:24 +0000212 }
213
mtklein36352bf2015-03-25 18:17:31 -0700214 void onDraw(SkCanvas* canvas) override {
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000215 SkPaint paint;
216 paint.setAlpha(128);
reed374772b2016-10-05 17:33:02 -0700217 paint.setBlendMode(SkBlendMode::kXor);
Mike Reedfa582c82021-01-23 22:07:05 -0500218 SkSamplingOptions sampling;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000219
robertphillips@google.comffad46b2012-10-01 14:32:51 +0000220 SkRect srcR1 = { 0.0f, 0.0f, 4096.0f, 2040.0f };
221 SkRect dstR1 = { 10.1f, 10.1f, 629.9f, 400.9f };
222
223 SkRect srcR2 = { 4085.0f, 10.0f, 4087.0f, 12.0f };
224 SkRect dstR2 = { 10, 410, 30, 430 };
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000225
226 if (!fUseIRect) {
Mike Reedfa582c82021-01-23 22:07:05 -0500227 canvas->drawImageRect(fBigImage, srcR1, dstR1, sampling, &paint,
reed84984ef2015-07-17 07:09:43 -0700228 SkCanvas::kStrict_SrcRectConstraint);
Mike Reedfa582c82021-01-23 22:07:05 -0500229 canvas->drawImageRect(fBigImage, srcR2, dstR2, sampling, &paint,
reed84984ef2015-07-17 07:09:43 -0700230 SkCanvas::kStrict_SrcRectConstraint);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000231 } else {
Mike Reedfa582c82021-01-23 22:07:05 -0500232 canvas->drawImageRect(fBigImage, SkRect::Make(srcR1.roundOut()), dstR1, sampling,
233 &paint, SkCanvas::kStrict_SrcRectConstraint);
234 canvas->drawImageRect(fBigImage, SkRect::Make(srcR2.roundOut()), dstR2, sampling,
235 &paint, SkCanvas::kStrict_SrcRectConstraint);
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000236 }
237 }
238
239private:
John Stiles7571f9e2020-09-02 22:42:33 -0400240 using INHERITED = skiagm::GM;
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000241};
242
reedf7869012014-12-01 13:54:01 -0800243class BitmapRectRounding : public skiagm::GM {
244 SkBitmap fBM;
245
246public:
247 BitmapRectRounding() {}
248
249protected:
mtklein36352bf2015-03-25 18:17:31 -0700250 SkString onShortName() override {
reedf7869012014-12-01 13:54:01 -0800251 SkString str;
252 str.printf("bitmaprect_rounding");
253 return str;
254 }
255
mtklein36352bf2015-03-25 18:17:31 -0700256 SkISize onISize() override {
reedf7869012014-12-01 13:54:01 -0800257 return SkISize::Make(640, 480);
258 }
259
mtklein36352bf2015-03-25 18:17:31 -0700260 void onOnceBeforeDraw() override {
reedf7869012014-12-01 13:54:01 -0800261 fBM.allocN32Pixels(10, 10);
262 fBM.eraseColor(SK_ColorBLUE);
263 }
264
265 // This choice of coordinates and matrix land the bottom edge of the clip (and bitmap dst)
266 // at exactly 1/2 pixel boundary. However, drawBitmapRect may lose precision along the way.
267 // If it does, we may see a red-line at the bottom, instead of the bitmap exactly matching
268 // the clip (in which case we should see all blue).
269 // The correct image should be all blue.
mtklein36352bf2015-03-25 18:17:31 -0700270 void onDraw(SkCanvas* canvas) override {
reedf7869012014-12-01 13:54:01 -0800271 SkPaint paint;
272 paint.setColor(SK_ColorRED);
273
274 const SkRect r = SkRect::MakeXYWH(1, 1, 110, 114);
275 canvas->scale(0.9f, 0.9f);
276
277 // the drawRect shows the same problem as clipRect(r) followed by drawcolor(red)
278 canvas->drawRect(r, paint);
Mike Reedfa582c82021-01-23 22:07:05 -0500279 canvas->drawImageRect(fBM.asImage(), r, SkSamplingOptions());
reedf7869012014-12-01 13:54:01 -0800280 }
mtklein1c402922015-01-23 11:07:07 -0800281
reedf7869012014-12-01 13:54:01 -0800282private:
John Stiles7571f9e2020-09-02 22:42:33 -0400283 using INHERITED = skiagm::GM;
reedf7869012014-12-01 13:54:01 -0800284};
reed03939122014-12-15 13:42:51 -0800285DEF_GM( return new BitmapRectRounding; )
reedf7869012014-12-01 13:54:01 -0800286
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000287//////////////////////////////////////////////////////////////////////////////
reed@google.com71121732012-09-18 15:14:33 +0000288
scroggo96f16e82015-12-10 13:31:59 -0800289DEF_GM( return new DrawBitmapRect2(false); )
290DEF_GM( return new DrawBitmapRect2(true); )
291DEF_GM( return new DrawBitmapRect3(); )
robertphillips@google.com21a95f12012-09-26 13:10:19 +0000292
robertphillips@google.com653b0d62012-09-26 15:28:04 +0000293#ifndef SK_BUILD_FOR_ANDROID
scroggo96f16e82015-12-10 13:31:59 -0800294DEF_GM( return new DrawBitmapRect4(false); )
295DEF_GM( return new DrawBitmapRect4(true); )
robertphillips@google.com653b0d62012-09-26 15:28:04 +0000296#endif