• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 <functional>
9 #include "gm.h"
10 #include "SkData.h"
11 #include "SkCanvas.h"
12 #include "SkRandom.h"
13 #include "SkStream.h"
14 #include "SkSurface.h"
15 
16 #if SK_SUPPORT_GPU
17 #include "GrContext.h"
18 #endif
19 
drawJpeg(SkCanvas * canvas,const SkISize & size)20 static void drawJpeg(SkCanvas* canvas, const SkISize& size) {
21     // TODO: Make this draw a file that is checked in, so it can
22     // be exercised on machines other than mike's. Will require a
23     // rebaseline.
24     SkAutoDataUnref data(SkData::NewFromFileName("/Users/mike/Downloads/skia.google.jpeg"));
25     if (nullptr == data.get()) {
26         return;
27     }
28     SkImage* image = SkImage::NewFromEncoded(data);
29     if (image) {
30         SkAutoCanvasRestore acr(canvas, true);
31         canvas->scale(size.width() * 1.0f / image->width(),
32                       size.height() * 1.0f / image->height());
33         canvas->drawImage(image, 0, 0, nullptr);
34         image->unref();
35     }
36 }
37 
drawContents(SkSurface * surface,SkColor fillC)38 static void drawContents(SkSurface* surface, SkColor fillC) {
39     SkSize size = SkSize::Make(SkIntToScalar(surface->width()),
40                                SkIntToScalar(surface->height()));
41     SkCanvas* canvas = surface->getCanvas();
42 
43     SkScalar stroke = size.fWidth / 10;
44     SkScalar radius = (size.fWidth - stroke) / 2;
45 
46     SkPaint paint;
47 
48     paint.setAntiAlias(true);
49     paint.setColor(fillC);
50     canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint);
51 
52     paint.setStyle(SkPaint::kStroke_Style);
53     paint.setStrokeWidth(stroke);
54     paint.setColor(SK_ColorBLACK);
55     canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint);
56 }
57 
test_surface(SkCanvas * canvas,SkSurface * surf,bool usePaint)58 static void test_surface(SkCanvas* canvas, SkSurface* surf, bool usePaint) {
59     drawContents(surf, SK_ColorRED);
60     SkImage* imgR = surf->newImageSnapshot();
61 
62     if (true) {
63         SkImage* imgR2 = surf->newImageSnapshot();
64         SkASSERT(imgR == imgR2);
65         imgR2->unref();
66     }
67 
68     drawContents(surf, SK_ColorGREEN);
69     SkImage* imgG = surf->newImageSnapshot();
70 
71     // since we've drawn after we snapped imgR, imgG will be a different obj
72     SkASSERT(imgR != imgG);
73 
74     drawContents(surf, SK_ColorBLUE);
75 
76     SkPaint paint;
77 //    paint.setFilterBitmap(true);
78 //    paint.setAlpha(0x80);
79 
80     canvas->drawImage(imgR, 0, 0, usePaint ? &paint : nullptr);
81     canvas->drawImage(imgG, 0, 80, usePaint ? &paint : nullptr);
82     surf->draw(canvas, 0, 160, usePaint ? &paint : nullptr);
83 
84     SkRect src1, src2, src3;
85     src1.iset(0, 0, surf->width(), surf->height());
86     src2.iset(-surf->width() / 2, -surf->height() / 2,
87              surf->width(), surf->height());
88     src3.iset(0, 0, surf->width() / 2, surf->height() / 2);
89 
90     SkRect dst1, dst2, dst3, dst4;
91     dst1.set(0, 240, 65, 305);
92     dst2.set(0, 320, 65, 385);
93     dst3.set(0, 400, 65, 465);
94     dst4.set(0, 480, 65, 545);
95 
96     canvas->drawImageRect(imgR, src1, dst1, usePaint ? &paint : nullptr);
97     canvas->drawImageRect(imgG, src2, dst2, usePaint ? &paint : nullptr);
98     canvas->drawImageRect(imgR, src3, dst3, usePaint ? &paint : nullptr);
99     canvas->drawImageRect(imgG, dst4, usePaint ? &paint : nullptr);
100 
101     imgG->unref();
102     imgR->unref();
103 }
104 
105 class ImageGM : public skiagm::GM {
106     void*   fBuffer;
107     size_t  fBufferSize;
108     SkSize  fSize;
109     enum {
110         W = 64,
111         H = 64,
112         RB = W * 4 + 8,
113     };
114 public:
ImageGM()115     ImageGM() {
116         fBufferSize = RB * H;
117         fBuffer = sk_malloc_throw(fBufferSize);
118         fSize.set(SkIntToScalar(W), SkIntToScalar(H));
119     }
120 
~ImageGM()121     virtual ~ImageGM() {
122         sk_free(fBuffer);
123     }
124 
125 protected:
onShortName()126     SkString onShortName() override {
127         return SkString("image-surface");
128     }
129 
onISize()130     SkISize onISize() override {
131         return SkISize::Make(960, 1200);
132     }
133 
onDraw(SkCanvas * canvas)134     void onDraw(SkCanvas* canvas) override {
135         drawJpeg(canvas, this->getISize());
136 
137         canvas->scale(2, 2);
138 
139         static const char* kLabel1 = "Original Img";
140         static const char* kLabel2 = "Modified Img";
141         static const char* kLabel3 = "Cur Surface";
142         static const char* kLabel4 = "Full Crop";
143         static const char* kLabel5 = "Over-crop";
144         static const char* kLabel6 = "Upper-left";
145         static const char* kLabel7 = "No Crop";
146 
147         static const char* kLabel8 = "Pre-Alloc Img";
148         static const char* kLabel9 = "New Alloc Img";
149         static const char* kLabel10 = "GPU";
150 
151         SkPaint textPaint;
152         textPaint.setAntiAlias(true);
153         sk_tool_utils::set_portable_typeface(&textPaint);
154         textPaint.setTextSize(8);
155 
156         canvas->drawText(kLabel1, strlen(kLabel1), 10,  60, textPaint);
157         canvas->drawText(kLabel2, strlen(kLabel2), 10, 140, textPaint);
158         canvas->drawText(kLabel3, strlen(kLabel3), 10, 220, textPaint);
159         canvas->drawText(kLabel4, strlen(kLabel4), 10, 300, textPaint);
160         canvas->drawText(kLabel5, strlen(kLabel5), 10, 380, textPaint);
161         canvas->drawText(kLabel6, strlen(kLabel6), 10, 460, textPaint);
162         canvas->drawText(kLabel7, strlen(kLabel7), 10, 540, textPaint);
163 
164         canvas->drawText(kLabel8, strlen(kLabel8),  80, 10, textPaint);
165         canvas->drawText(kLabel9, strlen(kLabel9), 160, 10, textPaint);
166         canvas->drawText(kLabel10, strlen(kLabel10), 265, 10, textPaint);
167 
168         canvas->translate(80, 20);
169 
170         // since we draw into this directly, we need to start fresh
171         sk_bzero(fBuffer, fBufferSize);
172 
173         SkImageInfo info = SkImageInfo::MakeN32Premul(W, H);
174         SkAutoTUnref<SkSurface> surf0(SkSurface::NewRasterDirect(info, fBuffer, RB));
175         SkAutoTUnref<SkSurface> surf1(SkSurface::NewRaster(info));
176         SkAutoTUnref<SkSurface> surf2;  // gpu
177 
178 #if SK_SUPPORT_GPU
179         surf2.reset(SkSurface::NewRenderTarget(canvas->getGrContext(),
180                                                SkBudgeted::kNo, info));
181 #endif
182 
183         test_surface(canvas, surf0, true);
184         canvas->translate(80, 0);
185         test_surface(canvas, surf1, true);
186         if (surf2) {
187             canvas->translate(80, 0);
188             test_surface(canvas, surf2, true);
189         }
190     }
191 
192 private:
193     typedef skiagm::GM INHERITED;
194 };
DEF_GM(return new ImageGM;)195 DEF_GM( return new ImageGM; )
196 
197 ///////////////////////////////////////////////////////////////////////////////////////////////////
198 
199 #include "SkPictureRecorder.h"
200 
201 static void draw_pixmap(SkCanvas* canvas, const SkPixmap& pmap) {
202     SkBitmap bitmap;
203     bitmap.installPixels(pmap);
204     canvas->drawBitmap(bitmap, 0, 0, nullptr);
205 }
206 
show_scaled_pixels(SkCanvas * canvas,SkImage * image)207 static void show_scaled_pixels(SkCanvas* canvas, SkImage* image) {
208     SkAutoCanvasRestore acr(canvas, true);
209 
210     canvas->drawImage(image, 0, 0, nullptr);
211     canvas->translate(110, 10);
212 
213     const SkImageInfo info = SkImageInfo::MakeN32Premul(40, 40);
214     SkAutoPixmapStorage storage;
215     storage.alloc(info);
216 
217     const SkImage::CachingHint chints[] = {
218         SkImage::kAllow_CachingHint, SkImage::kDisallow_CachingHint,
219     };
220     const SkFilterQuality qualities[] = {
221         kNone_SkFilterQuality, kLow_SkFilterQuality, kMedium_SkFilterQuality, kHigh_SkFilterQuality,
222     };
223 
224     for (auto ch : chints) {
225         canvas->save();
226         for (auto q : qualities) {
227             if (image->scalePixels(storage, q, ch)) {
228                 draw_pixmap(canvas, storage);
229             }
230             canvas->translate(70, 0);
231         }
232         canvas->restore();
233         canvas->translate(0, 45);
234     }
235 }
236 
draw_contents(SkCanvas * canvas)237 static void draw_contents(SkCanvas* canvas) {
238     SkPaint paint;
239     paint.setStyle(SkPaint::kStroke_Style);
240     paint.setStrokeWidth(20);
241     canvas->drawCircle(50, 50, 35, paint);
242 }
243 
make_raster(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))244 static SkImage* make_raster(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
245     SkAutoTUnref<SkSurface> surface(SkSurface::NewRaster(info));
246     draw(surface->getCanvas());
247     return surface->newImageSnapshot();
248 }
249 
make_picture(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))250 static SkImage* make_picture(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
251     SkPictureRecorder recorder;
252     draw(recorder.beginRecording(SkRect::MakeIWH(info.width(), info.height())));
253     SkAutoTUnref<SkPicture> pict(recorder.endRecording());
254     return SkImage::NewFromPicture(pict, info.dimensions(), nullptr, nullptr);
255 }
256 
make_codec(const SkImageInfo & info,GrContext *,void (* draw)(SkCanvas *))257 static SkImage* make_codec(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) {
258     SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw));
259     SkAutoTUnref<SkData> data(image->encode());
260     return SkImage::NewFromEncoded(data);
261 }
262 
make_gpu(const SkImageInfo & info,GrContext * ctx,void (* draw)(SkCanvas *))263 static SkImage* make_gpu(const SkImageInfo& info, GrContext* ctx, void (*draw)(SkCanvas*)) {
264     if (!ctx) { return nullptr; }
265     SkAutoTUnref<SkSurface> surface(SkSurface::NewRenderTarget(ctx, SkBudgeted::kNo, info));
266     draw(surface->getCanvas());
267     return surface->newImageSnapshot();
268 }
269 
270 typedef SkImage* (*ImageMakerProc)(const SkImageInfo&, GrContext*, void (*)(SkCanvas*));
271 
272 class ScalePixelsGM : public skiagm::GM {
273 public:
ScalePixelsGM()274     ScalePixelsGM() {}
275 
276 protected:
onShortName()277     SkString onShortName() override {
278         return SkString("scale-pixels");
279     }
280 
onISize()281     SkISize onISize() override {
282         return SkISize::Make(960, 1200);
283     }
284 
onDraw(SkCanvas * canvas)285     void onDraw(SkCanvas* canvas) override {
286         const SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
287 
288         const ImageMakerProc procs[] = {
289             make_codec, make_raster, make_picture, make_codec, make_gpu,
290         };
291         for (auto& proc : procs) {
292             SkAutoTUnref<SkImage> image(proc(info, canvas->getGrContext(), draw_contents));
293             if (image) {
294                 show_scaled_pixels(canvas, image);
295             }
296             canvas->translate(0, 120);
297         }
298     }
299 
300 private:
301     typedef skiagm::GM INHERITED;
302 };
DEF_GM(return new ScalePixelsGM;)303 DEF_GM( return new ScalePixelsGM; )
304 
305 ///////////////////////////////////////////////////////////////////////////////////////////////////
306 
307 #include "SkImageGenerator.h"
308 
309 static SkImageInfo make_info(SkImage* img) {
310     return SkImageInfo::MakeN32(img->width(), img->height(),
311                                 img->isOpaque() ? kOpaque_SkAlphaType : kPremul_SkAlphaType);
312 }
313 
314 // Its simple, but I wonder if we should expose this formally?
315 //
316 class ImageGeneratorFromImage : public SkImageGenerator {
317 public:
ImageGeneratorFromImage(SkImage * img)318     ImageGeneratorFromImage(SkImage* img) : INHERITED(make_info(img)), fImg(SkRef(img)) {}
319 
320 protected:
onGetPixels(const SkImageInfo & info,void * pixels,size_t rowBytes,SkPMColor ctable[],int * ctableCount)321     bool onGetPixels(const SkImageInfo& info, void* pixels, size_t rowBytes, SkPMColor ctable[],
322                      int* ctableCount) override {
323         return fImg->readPixels(info, pixels, rowBytes, 0, 0);
324     }
325 
326 private:
327     SkAutoTUnref<SkImage> fImg;
328 
329     typedef SkImageGenerator INHERITED;
330 };
331 
draw_opaque_contents(SkCanvas * canvas)332 static void draw_opaque_contents(SkCanvas* canvas) {
333     canvas->drawColor(0xFFFF8844);
334 
335     SkPaint paint;
336     paint.setStyle(SkPaint::kStroke_Style);
337     paint.setStrokeWidth(20);
338     canvas->drawCircle(50, 50, 35, paint);
339 }
340 
gen_raster(const SkImageInfo & info)341 static SkImageGenerator* gen_raster(const SkImageInfo& info) {
342     SkAutoTUnref<SkSurface> surface(SkSurface::NewRaster(info));
343     draw_opaque_contents(surface->getCanvas());
344     SkAutoTUnref<SkImage> img(surface->newImageSnapshot());
345     return new ImageGeneratorFromImage(img);
346 }
347 
gen_picture(const SkImageInfo & info)348 static SkImageGenerator* gen_picture(const SkImageInfo& info) {
349     SkPictureRecorder recorder;
350     draw_opaque_contents(recorder.beginRecording(SkRect::MakeIWH(info.width(), info.height())));
351     SkAutoTUnref<SkPicture> pict(recorder.endRecording());
352     return SkImageGenerator::NewFromPicture(info.dimensions(), pict, nullptr, nullptr);
353 }
354 
gen_png(const SkImageInfo & info)355 static SkImageGenerator* gen_png(const SkImageInfo& info) {
356     SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw_opaque_contents));
357     SkAutoTUnref<SkData> data(image->encode(SkImageEncoder::kPNG_Type, 100));
358     return SkImageGenerator::NewFromEncoded(data);
359 }
360 
gen_jpg(const SkImageInfo & info)361 static SkImageGenerator* gen_jpg(const SkImageInfo& info) {
362     SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw_opaque_contents));
363     SkAutoTUnref<SkData> data(image->encode(SkImageEncoder::kJPEG_Type, 100));
364     return SkImageGenerator::NewFromEncoded(data);
365 }
366 
367 typedef SkImageGenerator* (*GeneratorMakerProc)(const SkImageInfo&);
368 
show_scaled_generator(SkCanvas * canvas,SkImageGenerator * gen)369 static void show_scaled_generator(SkCanvas* canvas, SkImageGenerator* gen) {
370     const SkImageInfo genInfo = gen->getInfo();
371 
372     SkAutoCanvasRestore acr(canvas, true);
373 
374     SkBitmap bm;
375     bm.allocPixels(genInfo);
376     if (gen->getPixels(bm.info(), bm.getPixels(), bm.rowBytes())) {
377         canvas->drawBitmap(bm, 0, 0, nullptr);
378     }
379     canvas->translate(110, 0);
380 
381     const float scales[] = { 0.75f, 0.5f, 0.25f };
382     for (auto scale : scales) {
383         SkImageGenerator::SupportedSizes sizes;
384         if (gen->computeScaledDimensions(scale, &sizes)) {
385             const SkImageInfo info = SkImageInfo::MakeN32Premul(sizes.fSizes[0].width(),
386                                                                 sizes.fSizes[0].height());
387             bm.allocPixels(info);
388             SkPixmap pmap;
389             bm.peekPixels(&pmap);
390             if (gen->generateScaledPixels(pmap)) {
391                 canvas->drawBitmap(bm, 0, SkIntToScalar(genInfo.height() - info.height())/2);
392             }
393         }
394         canvas->translate(100, 0);
395     }
396 }
397 
398 class ScaleGeneratorGM : public skiagm::GM {
399 public:
ScaleGeneratorGM()400     ScaleGeneratorGM() {}
401 
402 protected:
onShortName()403     SkString onShortName() override {
404         return SkString("scale-generator");
405     }
406 
onISize()407     SkISize onISize() override {
408         return SkISize::Make(500, 500);
409     }
410 
onDraw(SkCanvas * canvas)411     void onDraw(SkCanvas* canvas) override {
412         canvas->translate(10, 10);
413 
414         // explicitly make it opaque, so we can test JPEG (which is only ever opaque)
415         const SkImageInfo info = SkImageInfo::MakeN32(100, 100, kOpaque_SkAlphaType);
416 
417         const GeneratorMakerProc procs[] = {
418             gen_raster, gen_picture, gen_png, gen_jpg,
419         };
420         for (auto& proc : procs) {
421             SkAutoTDelete<SkImageGenerator> gen(proc(info));
422             if (gen) {
423                 show_scaled_generator(canvas, gen);
424             }
425             canvas->translate(0, 120);
426         }
427     }
428 
429 private:
430     typedef skiagm::GM INHERITED;
431 };
432 DEF_GM( return new ScaleGeneratorGM; )
433 
434 #if SK_SUPPORT_GPU
435 #include "GrContextFactory.h"
436 #endif
437 
438 DEF_SIMPLE_GM(new_texture_image, canvas, 225, 60) {
439     GrContext* context = nullptr;
440 #if SK_SUPPORT_GPU
441     context = canvas->getGrContext();
442     GrContextFactory factory;
443 #endif
444     if (!context) {
445         skiagm::GM::DrawGpuOnlyMessage(canvas);
446         return;
447     }
448 
__anon56a0b04b0202(SkCanvas* canvas) 449     auto render_image = [](SkCanvas* canvas) {
450         canvas->clear(SK_ColorBLUE);
451         SkPaint paint;
452         paint.setColor(SK_ColorRED);
453         canvas->drawRect(SkRect::MakeXYWH(10.f,10.f,10.f,10.f), paint);
454         paint.setColor(SK_ColorGREEN);
455         canvas->drawRect(SkRect::MakeXYWH(30.f,10.f,10.f,10.f), paint);
456         paint.setColor(SK_ColorYELLOW);
457         canvas->drawRect(SkRect::MakeXYWH(10.f,30.f,10.f,10.f), paint);
458         paint.setColor(SK_ColorCYAN);
459         canvas->drawRect(SkRect::MakeXYWH(30.f,30.f,10.f,10.f), paint);
460     };
461 
462     static const int kSize = 50;
463     SkBitmap bmp;
464     bmp.allocN32Pixels(kSize, kSize);
465     SkCanvas bmpCanvas(bmp);
466     render_image(&bmpCanvas);
467 
468     std::function<SkImage*()> imageFactories[] = {
469         // Create sw raster image.
__anon56a0b04b0302null470         [bmp] {
471             return SkImage::NewFromBitmap(bmp);
472         },
473         // Create encoded image.
__anon56a0b04b0402null474         [bmp] {
475             SkAutoTUnref<SkData> src(
476                 SkImageEncoder::EncodeData(bmp, SkImageEncoder::kPNG_Type, 100));
477             return SkImage::NewFromEncoded(src);
478         },
479         // Create a picture image.
__anon56a0b04b0502null480         [render_image] {
481             SkPictureRecorder recorder;
482             SkCanvas* canvas = recorder.beginRecording(SkIntToScalar(kSize), SkIntToScalar(kSize));
483             render_image(canvas);
484             SkAutoTUnref<SkPicture> picture(recorder.endRecording());
485             return SkImage::NewFromPicture(picture, SkISize::Make(kSize, kSize), nullptr, nullptr);
486         },
487         // Create a texture image
__anon56a0b04b0602() 488         [context, render_image]() -> SkImage* {
489             SkAutoTUnref<SkSurface> surface(
490                 SkSurface::NewRenderTarget(context, SkBudgeted::kYes,
491                                            SkImageInfo::MakeN32Premul(kSize, kSize)));
492             if (!surface) {
493                 return nullptr;
494             }
495             render_image(surface->getCanvas());
496             return surface->newImageSnapshot();
497         }
498     };
499 
500     static const SkScalar kPad = 5.f;
501     canvas->translate(kPad, kPad);
502     for (auto factory : imageFactories) {
503         SkAutoTUnref<SkImage> image(factory());
504         if (!image) {
505             continue;
506         }
507         if (context) {
508             SkAutoTUnref<SkImage> texImage(image->newTextureImage(context));
509             if (texImage) {
510                 canvas->drawImage(texImage, 0, 0);
511             }
512         }
513         canvas->translate(image->width() + kPad, 0);
514     }
515 }
516