1 /*
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/gm.h"
9 #include "include/core/SkBitmap.h"
10 #include "include/core/SkCanvas.h"
11 #include "include/core/SkColor.h"
12 #include "include/core/SkColorPriv.h"
13 #include "include/core/SkColorSpace.h"
14 #include "include/core/SkData.h"
15 #include "include/core/SkEncodedImageFormat.h"
16 #include "include/core/SkFilterQuality.h"
17 #include "include/core/SkFont.h"
18 #include "include/core/SkImage.h"
19 #include "include/core/SkImageEncoder.h"
20 #include "include/core/SkImageInfo.h"
21 #include "include/core/SkPaint.h"
22 #include "include/core/SkPicture.h"
23 #include "include/core/SkPictureRecorder.h"
24 #include "include/core/SkPixmap.h"
25 #include "include/core/SkRect.h"
26 #include "include/core/SkRefCnt.h"
27 #include "include/core/SkScalar.h"
28 #include "include/core/SkSize.h"
29 #include "include/core/SkString.h"
30 #include "include/core/SkSurface.h"
31 #include "include/core/SkTypeface.h"
32 #include "include/core/SkTypes.h"
33 #include "include/private/SkMalloc.h"
34 #include "src/core/SkAutoPixmapStorage.h"
35 #include "src/core/SkReadBuffer.h"
36 #include "src/core/SkWriteBuffer.h"
37 #include "tools/ToolUtils.h"
38
39 #include <functional>
40 #include <utility>
41
42 class GrContext;
43 class GrRenderTargetContext;
44
drawContents(SkSurface * surface,SkColor fillC)45 static void drawContents(SkSurface* surface, SkColor fillC) {
46 SkSize size = SkSize::Make(SkIntToScalar(surface->width()),
47 SkIntToScalar(surface->height()));
48 SkCanvas* canvas = surface->getCanvas();
49
50 SkScalar stroke = size.fWidth / 10;
51 SkScalar radius = (size.fWidth - stroke) / 2;
52
53 SkPaint paint;
54
55 paint.setAntiAlias(true);
56 paint.setColor(fillC);
57 canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint);
58
59 paint.setStyle(SkPaint::kStroke_Style);
60 paint.setStrokeWidth(stroke);
61 paint.setColor(SK_ColorBLACK);
62 canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint);
63 }
64
test_surface(SkCanvas * canvas,SkSurface * surf,bool usePaint)65 static void test_surface(SkCanvas* canvas, SkSurface* surf, bool usePaint) {
66 drawContents(surf, SK_ColorRED);
67 sk_sp<SkImage> imgR = surf->makeImageSnapshot();
68
69 if (true) {
70 sk_sp<SkImage> imgR2 = surf->makeImageSnapshot();
71 SkASSERT(imgR == imgR2);
72 }
73
74 drawContents(surf, SK_ColorGREEN);
75 sk_sp<SkImage> imgG = surf->makeImageSnapshot();
76
77 // since we've drawn after we snapped imgR, imgG will be a different obj
78 SkASSERT(imgR != imgG);
79
80 drawContents(surf, SK_ColorBLUE);
81
82 SkPaint paint;
83 // paint.setFilterBitmap(true);
84 // paint.setAlpha(0x80);
85
86 canvas->drawImage(imgR, 0, 0, usePaint ? &paint : nullptr);
87 canvas->drawImage(imgG, 0, 80, usePaint ? &paint : nullptr);
88 surf->draw(canvas, 0, 160, usePaint ? &paint : nullptr);
89
90 SkRect src1, src2, src3;
91 src1.iset(0, 0, surf->width(), surf->height());
92 src2.iset(-surf->width() / 2, -surf->height() / 2,
93 surf->width(), surf->height());
94 src3.iset(0, 0, surf->width() / 2, surf->height() / 2);
95
96 SkRect dst1, dst2, dst3, dst4;
97 dst1.set(0, 240, 65, 305);
98 dst2.set(0, 320, 65, 385);
99 dst3.set(0, 400, 65, 465);
100 dst4.set(0, 480, 65, 545);
101
102 canvas->drawImageRect(imgR, src1, dst1, usePaint ? &paint : nullptr);
103 canvas->drawImageRect(imgG, src2, dst2, usePaint ? &paint : nullptr);
104 canvas->drawImageRect(imgR, src3, dst3, usePaint ? &paint : nullptr);
105 canvas->drawImageRect(imgG, dst4, usePaint ? &paint : nullptr);
106 }
107
108 class ImageGM : public skiagm::GM {
109 void* fBuffer;
110 size_t fBufferSize;
111 SkSize fSize;
112 enum {
113 W = 64,
114 H = 64,
115 RB = W * 4 + 8,
116 };
117 public:
ImageGM()118 ImageGM() {
119 fBufferSize = RB * H;
120 fBuffer = sk_malloc_throw(fBufferSize);
121 fSize.set(SkIntToScalar(W), SkIntToScalar(H));
122 }
123
~ImageGM()124 ~ImageGM() override {
125 sk_free(fBuffer);
126 }
127
128 protected:
onShortName()129 SkString onShortName() override {
130 return SkString("image-surface");
131 }
132
onISize()133 SkISize onISize() override {
134 return SkISize::Make(960, 1200);
135 }
136
onDraw(SkCanvas * canvas)137 void onDraw(SkCanvas* canvas) override {
138 canvas->scale(2, 2);
139
140 SkFont font(ToolUtils::create_portable_typeface(), 8);
141
142 canvas->drawString("Original Img", 10, 60, font, SkPaint());
143 canvas->drawString("Modified Img", 10, 140, font, SkPaint());
144 canvas->drawString("Cur Surface", 10, 220, font, SkPaint());
145 canvas->drawString("Full Crop", 10, 300, font, SkPaint());
146 canvas->drawString("Over-crop", 10, 380, font, SkPaint());
147 canvas->drawString("Upper-left", 10, 460, font, SkPaint());
148 canvas->drawString("No Crop", 10, 540, font, SkPaint());
149
150 canvas->drawString("Pre-Alloc Img", 80, 10, font, SkPaint());
151 canvas->drawString("New Alloc Img", 160, 10, font, SkPaint());
152 canvas->drawString( "GPU", 265, 10, font, SkPaint());
153
154 canvas->translate(80, 20);
155
156 // since we draw into this directly, we need to start fresh
157 sk_bzero(fBuffer, fBufferSize);
158
159 SkImageInfo info = SkImageInfo::MakeN32Premul(W, H);
160 sk_sp<SkSurface> surf0(SkSurface::MakeRasterDirect(info, fBuffer, RB));
161 sk_sp<SkSurface> surf1(SkSurface::MakeRaster(info));
162 sk_sp<SkSurface> surf2(SkSurface::MakeRenderTarget(canvas->getGrContext(),
163 SkBudgeted::kNo, info));
164
165 test_surface(canvas, surf0.get(), true);
166 canvas->translate(80, 0);
167 test_surface(canvas, surf1.get(), true);
168 if (surf2) {
169 canvas->translate(80, 0);
170 test_surface(canvas, surf2.get(), true);
171 }
172 }
173
174 private:
175 typedef skiagm::GM INHERITED;
176 };
DEF_GM(return new ImageGM;)177 DEF_GM( return new ImageGM; )
178
179 ///////////////////////////////////////////////////////////////////////////////////////////////////
180
181 static void draw_pixmap(SkCanvas* canvas, const SkPixmap& pmap) {
182 SkBitmap bitmap;
183 bitmap.installPixels(pmap);
184 canvas->drawBitmap(bitmap, 0, 0, nullptr);
185 }
186
show_scaled_pixels(SkCanvas * canvas,SkImage * image)187 static void show_scaled_pixels(SkCanvas* canvas, SkImage* image) {
188 SkAutoCanvasRestore acr(canvas, true);
189
190 canvas->drawImage(image, 0, 0, nullptr);
191 canvas->translate(110, 10);
192
193 const SkImageInfo info = SkImageInfo::MakeN32Premul(40, 40);
194 SkAutoPixmapStorage storage;
195 storage.alloc(info);
196
197 const SkImage::CachingHint chints[] = {
198 SkImage::kAllow_CachingHint, SkImage::kDisallow_CachingHint,
199 };
200 const SkFilterQuality qualities[] = {
201 kNone_SkFilterQuality, kLow_SkFilterQuality, kMedium_SkFilterQuality, kHigh_SkFilterQuality,
202 };
203
204 for (auto ch : chints) {
205 canvas->save();
206 for (auto q : qualities) {
207 if (image->scalePixels(storage, q, ch)) {
208 draw_pixmap(canvas, storage);
209 }
210 canvas->translate(70, 0);
211 }
212 canvas->restore();
213 canvas->translate(0, 45);
214 }
215 }
216
draw_contents(SkCanvas * canvas)217 static void draw_contents(SkCanvas* canvas) {
218 SkPaint paint;
219 paint.setStyle(SkPaint::kStroke_Style);
220 paint.setStrokeWidth(20);
221 canvas->drawCircle(50, 50, 35, paint);
222 }
223
make_raster(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))224 static sk_sp<SkImage> make_raster(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
225 auto surface(SkSurface::MakeRaster(info));
226 draw(surface->getCanvas());
227 return surface->makeImageSnapshot();
228 }
229
make_picture(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))230 static sk_sp<SkImage> make_picture(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
231 SkPictureRecorder recorder;
232 draw(recorder.beginRecording(SkRect::MakeIWH(info.width(), info.height())));
233 return SkImage::MakeFromPicture(recorder.finishRecordingAsPicture(),
234 info.dimensions(), nullptr, nullptr, SkImage::BitDepth::kU8,
235 SkColorSpace::MakeSRGB());
236 }
237
make_codec(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))238 static sk_sp<SkImage> make_codec(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
239 sk_sp<SkImage> image(make_raster(info, nullptr, draw));
240 return SkImage::MakeFromEncoded(image->encodeToData());
241 }
242
make_gpu(const SkImageInfo & info,GrContext * ctx,void (* draw)(SkCanvas *))243 static sk_sp<SkImage> make_gpu(const SkImageInfo& info, GrContext* ctx, void (*draw)(SkCanvas*)) {
244 if (!ctx) { return nullptr; }
245 auto surface(SkSurface::MakeRenderTarget(ctx, SkBudgeted::kNo, info));
246 if (!surface) { return nullptr; }
247 draw(surface->getCanvas());
248 return surface->makeImageSnapshot();
249 }
250
251 typedef sk_sp<SkImage> (*ImageMakerProc)(const SkImageInfo&, GrContext*, void (*)(SkCanvas*));
252
253 class ScalePixelsGM : public skiagm::GM {
254 public:
ScalePixelsGM()255 ScalePixelsGM() {}
256
257 protected:
onShortName()258 SkString onShortName() override {
259 return SkString("scale-pixels");
260 }
261
onISize()262 SkISize onISize() override {
263 return SkISize::Make(960, 1200);
264 }
265
onDraw(SkCanvas * canvas)266 void onDraw(SkCanvas* canvas) override {
267 const SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
268
269 const ImageMakerProc procs[] = {
270 make_codec, make_raster, make_picture, make_codec, make_gpu,
271 };
272 for (auto& proc : procs) {
273 sk_sp<SkImage> image(proc(info, canvas->getGrContext(), draw_contents));
274 if (image) {
275 show_scaled_pixels(canvas, image.get());
276 }
277 canvas->translate(0, 120);
278 }
279 }
280
281 private:
282 typedef skiagm::GM INHERITED;
283 };
284 DEF_GM( return new ScalePixelsGM; )
285
286 ///////////////////////////////////////////////////////////////////////////////////////////////////
287
288 DEF_SIMPLE_GPU_GM(new_texture_image, context, rtc, canvas, 280, 60) {
__anon8c6700040202(SkCanvas* canvas) 289 auto render_image = [](SkCanvas* canvas) {
290 canvas->clear(SK_ColorBLUE);
291 SkPaint paint;
292 paint.setColor(SK_ColorRED);
293 canvas->drawRect(SkRect::MakeXYWH(10.f,10.f,10.f,10.f), paint);
294 paint.setColor(SK_ColorGREEN);
295 canvas->drawRect(SkRect::MakeXYWH(30.f,10.f,10.f,10.f), paint);
296 paint.setColor(SK_ColorYELLOW);
297 canvas->drawRect(SkRect::MakeXYWH(10.f,30.f,10.f,10.f), paint);
298 paint.setColor(SK_ColorCYAN);
299 canvas->drawRect(SkRect::MakeXYWH(30.f,30.f,10.f,10.f), paint);
300 };
301
302 static constexpr int kSize = 50;
303 SkBitmap bmp;
304 bmp.allocPixels(SkImageInfo::MakeS32(kSize, kSize, kPremul_SkAlphaType));
305 SkCanvas bmpCanvas(bmp);
306 render_image(&bmpCanvas);
307
308 std::function<sk_sp<SkImage>()> imageFactories[] = {
309 // Create sw raster image.
__anon8c6700040302null310 [bmp] {
311 return SkImage::MakeFromBitmap(bmp);
312 },
313 // Create encoded image.
__anon8c6700040402null314 [bmp] {
315 auto src = SkEncodeBitmap(bmp, SkEncodedImageFormat::kPNG, 100);
316 return SkImage::MakeFromEncoded(std::move(src));
317 },
318 // Create YUV encoded image.
__anon8c6700040502null319 [bmp] {
320 auto src = SkEncodeBitmap(bmp, SkEncodedImageFormat::kJPEG, 100);
321 return SkImage::MakeFromEncoded(std::move(src));
322 },
323 // Create a picture image.
__anon8c6700040602null324 [render_image] {
325 SkPictureRecorder recorder;
326 SkCanvas* canvas = recorder.beginRecording(SkIntToScalar(kSize), SkIntToScalar(kSize));
327 render_image(canvas);
328 sk_sp<SkColorSpace> srgbColorSpace = SkColorSpace::MakeSRGB();
329 return SkImage::MakeFromPicture(recorder.finishRecordingAsPicture(),
330 SkISize::Make(kSize, kSize), nullptr, nullptr,
331 SkImage::BitDepth::kU8, srgbColorSpace);
332 },
333 // Create a texture image
__anon8c6700040702() 334 [context, render_image]() -> sk_sp<SkImage> {
335 auto surface(SkSurface::MakeRenderTarget(context, SkBudgeted::kYes,
336 SkImageInfo::MakeS32(kSize, kSize,
337 kPremul_SkAlphaType)));
338 if (!surface) {
339 return nullptr;
340 }
341 render_image(surface->getCanvas());
342 return surface->makeImageSnapshot();
343 }
344 };
345
346 constexpr SkScalar kPad = 5.f;
347 canvas->translate(kPad, kPad);
348 for (auto factory : imageFactories) {
349 auto image(factory());
350 if (image) {
351 sk_sp<SkImage> texImage(image->makeTextureImage(context));
352 if (texImage) {
353 canvas->drawImage(texImage, 0, 0);
354 }
355 }
356 canvas->translate(kSize + kPad, 0);
357 }
358 }
359
draw_pixmap(SkCanvas * canvas,const SkPixmap & pm,SkScalar x,SkScalar y)360 static void draw_pixmap(SkCanvas* canvas, const SkPixmap& pm, SkScalar x, SkScalar y) {
361 canvas->drawImage(SkImage::MakeRasterCopy(pm), x, y, nullptr);
362 }
363
slam_ff(const SkPixmap & pm)364 static void slam_ff(const SkPixmap& pm) {
365 for (int y = 0; y < pm.height(); ++y) {
366 for (int x = 0; x < pm.width(); ++x) {
367 *pm.writable_addr32(x, y) = *pm.addr32(x, y) | SkPackARGB32(0xFF, 0, 0, 0);
368 }
369 }
370 }
371
372 DEF_SIMPLE_GM(scalepixels_unpremul, canvas, 1080, 280) {
373 SkImageInfo info = SkImageInfo::MakeN32(16, 16, kUnpremul_SkAlphaType);
374 SkAutoPixmapStorage pm;
375 pm.alloc(info);
376 for (int y = 0; y < 16; ++y) {
377 for (int x = 0; x < 16; ++x) {
378 *pm.writable_addr32(x, y) = SkPackARGB32NoCheck(0, (y << 4) | y, (x << 4) | x, 0xFF);
379 }
380 }
381 SkAutoPixmapStorage pm2;
382 pm2.alloc(SkImageInfo::MakeN32(256, 256, kUnpremul_SkAlphaType));
383
384 const SkFilterQuality qualities[] = {
385 kNone_SkFilterQuality, kLow_SkFilterQuality, kMedium_SkFilterQuality, kHigh_SkFilterQuality
386 };
387
388 for (auto fq : qualities) {
389 pm.scalePixels(pm2, fq);
390 slam_ff(pm2);
391 draw_pixmap(canvas, pm2, 10, 10);
392 canvas->translate(pm2.width() + 10.0f, 0);
393 }
394 }
395
396 ///////////////////////////////////////////////////////////////////////////////////////////////////
397
make_lazy_image(SkSurface * surf)398 static sk_sp<SkImage> make_lazy_image(SkSurface* surf) {
399 surf->getCanvas()->drawCircle(100, 100, 100, SkPaint());
400 sk_sp<SkData> data = surf->makeImageSnapshot()->encodeToData();
401 if (!data) {
402 return nullptr;
403 }
404 return SkImage::MakeFromEncoded(std::move(data));
405 }
406
serial_deserial(SkImage * img)407 static sk_sp<SkImage> serial_deserial(SkImage* img) {
408 SkBinaryWriteBuffer writer;
409 writer.writeImage(img);
410 size_t length = writer.bytesWritten();
411 auto data = SkData::MakeUninitialized(length);
412 writer.writeToMemory(data->writable_data());
413
414 SkReadBuffer reader(data->data(), length);
415 return reader.readImage();
416 }
417
418 DEF_SIMPLE_GM_CAN_FAIL(image_subset, canvas, errorMsg, 440, 220) {
419 SkImageInfo info = SkImageInfo::MakeN32Premul(200, 200, nullptr);
420 auto surf = ToolUtils::makeSurface(canvas, info, nullptr);
421 auto img = make_lazy_image(surf.get());
422 if (!img) {
423 *errorMsg = "Failed to make lazy image.";
424 return skiagm::DrawResult::kFail;
425 }
426
427 canvas->drawImage(img, 10, 10, nullptr);
428 auto sub = img->makeSubset({100, 100, 200, 200});
429 canvas->drawImage(sub, 220, 10);
430 sub = serial_deserial(sub.get());
431 canvas->drawImage(sub, 220+110, 10);
432 return skiagm::DrawResult::kOk;
433 }
434