1
2 /*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8 #include "gm.h"
9 #include "SkCanvas.h"
10 #include "SkGradientShader.h"
11 #include "SkGraphics.h"
12 #include "SkPath.h"
13 #include "SkRegion.h"
14 #include "SkShader.h"
15
make_bitmap(SkBitmap * bitmap)16 static void make_bitmap(SkBitmap* bitmap) {
17 bitmap->allocN32Pixels(64, 64);
18
19 SkCanvas canvas(*bitmap);
20
21 canvas.drawColor(SK_ColorRED);
22 SkPaint paint;
23 paint.setAntiAlias(true);
24 const SkPoint pts[] = { { 0, 0 }, { 64, 64 } };
25 const SkColor colors[] = { SK_ColorWHITE, SK_ColorBLUE };
26 paint.setShader(SkGradientShader::CreateLinear(pts, colors, nullptr, 2,
27 SkShader::kClamp_TileMode))->unref();
28 canvas.drawCircle(32, 32, 32, paint);
29 }
30
31 class DrawBitmapRect2 : public skiagm::GM {
32 bool fUseIRect;
33 public:
DrawBitmapRect2(bool useIRect)34 DrawBitmapRect2(bool useIRect) : fUseIRect(useIRect) {
35 }
36
37 protected:
onShortName()38 SkString onShortName() override {
39 SkString str;
40 str.printf("bitmaprect_%s", fUseIRect ? "i" : "s");
41 return str;
42 }
43
onISize()44 SkISize onISize() override {
45 return SkISize::Make(640, 480);
46 }
47
onDraw(SkCanvas * canvas)48 void onDraw(SkCanvas* canvas) override {
49 canvas->drawColor(sk_tool_utils::color_to_565(0xFFCCCCCC));
50
51 const SkIRect src[] = {
52 { 0, 0, 32, 32 },
53 { 0, 0, 80, 80 },
54 { 32, 32, 96, 96 },
55 { -32, -32, 32, 32, }
56 };
57
58 SkPaint paint;
59 paint.setStyle(SkPaint::kStroke_Style);
60
61 SkBitmap bitmap;
62 make_bitmap(&bitmap);
63
64 SkRect dstR = { 0, 200, 128, 380 };
65
66 canvas->translate(16, 40);
67 for (size_t i = 0; i < SK_ARRAY_COUNT(src); i++) {
68 SkRect srcR;
69 srcR.set(src[i]);
70
71 canvas->drawBitmap(bitmap, 0, 0, &paint);
72 if (!fUseIRect) {
73 canvas->drawBitmapRect(bitmap, srcR, dstR, &paint,
74 SkCanvas::kStrict_SrcRectConstraint);
75 } else {
76 canvas->drawBitmapRect(bitmap, src[i], dstR, &paint);
77 }
78
79 canvas->drawRect(dstR, paint);
80 canvas->drawRect(srcR, paint);
81
82 canvas->translate(160, 0);
83 }
84 }
85
86 private:
87 typedef skiagm::GM INHERITED;
88 };
89
90 //////////////////////////////////////////////////////////////////////////////
91
make_3x3_bitmap(SkBitmap * bitmap)92 static void make_3x3_bitmap(SkBitmap* bitmap) {
93 const int xSize = 3;
94 const int ySize = 3;
95
96 const SkColor textureData[xSize][ySize] = {
97 { SK_ColorRED, SK_ColorWHITE, SK_ColorBLUE },
98 { SK_ColorGREEN, SK_ColorBLACK, SK_ColorCYAN },
99 { SK_ColorYELLOW, SK_ColorGRAY, SK_ColorMAGENTA }
100 };
101
102 bitmap->allocN32Pixels(xSize, ySize, true);
103 SkCanvas canvas(*bitmap);
104 SkPaint paint;
105
106 for (int y = 0; y < ySize; y++) {
107 for (int x = 0; x < xSize; x++) {
108 paint.setColor(textureData[x][y]);
109 canvas.drawIRect(SkIRect::MakeXYWH(x, y, 1, 1), paint);
110 }
111 }
112 }
113
114 // This GM attempts to make visible any issues drawBitmapRect may have
115 // with partial source rects. In this case the eight pixels on the border
116 // should be half the width/height of the central pixel, i.e.:
117 // __|____|__
118 // | |
119 // __|____|__
120 // | |
121 class DrawBitmapRect3 : public skiagm::GM {
122 public:
DrawBitmapRect3()123 DrawBitmapRect3() {
124 this->setBGColor(SK_ColorBLACK);
125 }
126
127 protected:
onShortName()128 SkString onShortName() override {
129 SkString str;
130 str.printf("3x3bitmaprect");
131 return str;
132 }
133
onISize()134 SkISize onISize() override {
135 return SkISize::Make(640, 480);
136 }
137
onDraw(SkCanvas * canvas)138 void onDraw(SkCanvas* canvas) override {
139
140 SkBitmap bitmap;
141 make_3x3_bitmap(&bitmap);
142
143 SkRect srcR = { 0.5f, 0.5f, 2.5f, 2.5f };
144 SkRect dstR = { 100, 100, 300, 200 };
145
146 canvas->drawBitmapRect(bitmap, srcR, dstR, nullptr, SkCanvas::kStrict_SrcRectConstraint);
147 }
148
149 private:
150 typedef skiagm::GM INHERITED;
151 };
152
153 //////////////////////////////////////////////////////////////////////////////
make_big_bitmap(SkBitmap * bitmap)154 static void make_big_bitmap(SkBitmap* bitmap) {
155
156 static const int gXSize = 4096;
157 static const int gYSize = 4096;
158 static const int gBorderWidth = 10;
159
160 bitmap->allocN32Pixels(gXSize, gYSize);
161 for (int y = 0; y < gYSize; ++y) {
162 for (int x = 0; x < gXSize; ++x) {
163 if (x <= gBorderWidth || x >= gXSize-gBorderWidth ||
164 y <= gBorderWidth || y >= gYSize-gBorderWidth) {
165 *bitmap->getAddr32(x, y) = SkPreMultiplyColor(0x88FFFFFF);
166 } else {
167 *bitmap->getAddr32(x, y) = SkPreMultiplyColor(0x88FF0000);
168 }
169 }
170 }
171 }
172
173 // This GM attempts to reveal any issues we may have when the GPU has to
174 // break up a large texture in order to draw it. The XOR transfer mode will
175 // create stripes in the image if there is imprecision in the destination
176 // tile placement.
177 class DrawBitmapRect4 : public skiagm::GM {
178 bool fUseIRect;
179 SkBitmap fBigBitmap;
180
181 public:
DrawBitmapRect4(bool useIRect)182 DrawBitmapRect4(bool useIRect) : fUseIRect(useIRect) {
183 this->setBGColor(0x88444444);
184 }
185
186 protected:
onShortName()187 SkString onShortName() override {
188 SkString str;
189 str.printf("bigbitmaprect_%s", fUseIRect ? "i" : "s");
190 return str;
191 }
192
onISize()193 SkISize onISize() override {
194 return SkISize::Make(640, 480);
195 }
196
onOnceBeforeDraw()197 void onOnceBeforeDraw() override {
198 make_big_bitmap(&fBigBitmap);
199 }
200
onDraw(SkCanvas * canvas)201 void onDraw(SkCanvas* canvas) override {
202
203 SkXfermode* mode = SkXfermode::Create(SkXfermode::kXor_Mode);
204
205 SkPaint paint;
206 paint.setAlpha(128);
207 paint.setXfermode(mode)->unref();
208
209 SkRect srcR1 = { 0.0f, 0.0f, 4096.0f, 2040.0f };
210 SkRect dstR1 = { 10.1f, 10.1f, 629.9f, 400.9f };
211
212 SkRect srcR2 = { 4085.0f, 10.0f, 4087.0f, 12.0f };
213 SkRect dstR2 = { 10, 410, 30, 430 };
214
215 if (!fUseIRect) {
216 canvas->drawBitmapRect(fBigBitmap, srcR1, dstR1, &paint,
217 SkCanvas::kStrict_SrcRectConstraint);
218 canvas->drawBitmapRect(fBigBitmap, srcR2, dstR2, &paint,
219 SkCanvas::kStrict_SrcRectConstraint);
220 } else {
221 canvas->drawBitmapRect(fBigBitmap, srcR1.roundOut(), dstR1, &paint);
222 canvas->drawBitmapRect(fBigBitmap, srcR2.roundOut(), dstR2, &paint);
223 }
224 }
225
226 private:
227 typedef skiagm::GM INHERITED;
228 };
229
230 class BitmapRectRounding : public skiagm::GM {
231 SkBitmap fBM;
232
233 public:
BitmapRectRounding()234 BitmapRectRounding() {}
235
236 protected:
onShortName()237 SkString onShortName() override {
238 SkString str;
239 str.printf("bitmaprect_rounding");
240 return str;
241 }
242
onISize()243 SkISize onISize() override {
244 return SkISize::Make(640, 480);
245 }
246
onOnceBeforeDraw()247 void onOnceBeforeDraw() override {
248 fBM.allocN32Pixels(10, 10);
249 fBM.eraseColor(SK_ColorBLUE);
250 }
251
252 // This choice of coordinates and matrix land the bottom edge of the clip (and bitmap dst)
253 // at exactly 1/2 pixel boundary. However, drawBitmapRect may lose precision along the way.
254 // If it does, we may see a red-line at the bottom, instead of the bitmap exactly matching
255 // the clip (in which case we should see all blue).
256 // The correct image should be all blue.
onDraw(SkCanvas * canvas)257 void onDraw(SkCanvas* canvas) override {
258 SkPaint paint;
259 paint.setColor(SK_ColorRED);
260
261 const SkRect r = SkRect::MakeXYWH(1, 1, 110, 114);
262 canvas->scale(0.9f, 0.9f);
263
264 // the drawRect shows the same problem as clipRect(r) followed by drawcolor(red)
265 canvas->drawRect(r, paint);
266 canvas->drawBitmapRect(fBM, r, nullptr);
267 }
268
269 private:
270 typedef skiagm::GM INHERITED;
271 };
272 DEF_GM( return new BitmapRectRounding; )
273
274 //////////////////////////////////////////////////////////////////////////////
275
276 DEF_GM( return new DrawBitmapRect2(false); )
277 DEF_GM( return new DrawBitmapRect2(true); )
278 DEF_GM( return new DrawBitmapRect3(); )
279
280 #ifndef SK_BUILD_FOR_ANDROID
281 DEF_GM( return new DrawBitmapRect4(false); )
282 DEF_GM( return new DrawBitmapRect4(true); )
283 #endif
284
285