mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 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.h" |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 9 | #include "SkData.h" |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 10 | #include "SkCanvas.h" |
| 11 | #include "SkRandom.h" |
| 12 | #include "SkStream.h" |
| 13 | #include "SkSurface.h" |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 14 | |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 15 | #if SK_SUPPORT_GPU |
| 16 | #include "GrContext.h" |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 17 | #endif |
reed@google.com | 97af1a6 | 2012-08-28 12:19:02 +0000 | [diff] [blame] | 18 | |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 19 | static void drawJpeg(SkCanvas* canvas, const SkISize& size) { |
scroggo@google.com | ccd7afb | 2013-05-28 16:45:07 +0000 | [diff] [blame] | 20 | // TODO: Make this draw a file that is checked in, so it can |
| 21 | // be exercised on machines other than mike's. Will require a |
| 22 | // rebaseline. |
reed | 48925e3 | 2014-09-18 13:57:05 -0700 | [diff] [blame] | 23 | SkAutoDataUnref data(SkData::NewFromFileName("/Users/mike/Downloads/skia.google.jpeg")); |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 24 | if (nullptr == data.get()) { |
reed | 48925e3 | 2014-09-18 13:57:05 -0700 | [diff] [blame] | 25 | return; |
| 26 | } |
reed | 871872f | 2015-06-22 12:48:26 -0700 | [diff] [blame] | 27 | SkImage* image = SkImage::NewFromEncoded(data); |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 28 | if (image) { |
| 29 | SkAutoCanvasRestore acr(canvas, true); |
| 30 | canvas->scale(size.width() * 1.0f / image->width(), |
| 31 | size.height() * 1.0f / image->height()); |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 32 | canvas->drawImage(image, 0, 0, nullptr); |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 33 | image->unref(); |
| 34 | } |
| 35 | } |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 36 | |
| 37 | static void drawContents(SkSurface* surface, SkColor fillC) { |
skia.committer@gmail.com | 04ba448 | 2012-09-07 02:01:30 +0000 | [diff] [blame] | 38 | SkSize size = SkSize::Make(SkIntToScalar(surface->width()), |
robertphillips@google.com | 94acc70 | 2012-09-06 18:43:21 +0000 | [diff] [blame] | 39 | SkIntToScalar(surface->height())); |
mike@reedtribe.org | d2782ed | 2012-07-31 02:45:15 +0000 | [diff] [blame] | 40 | SkCanvas* canvas = surface->getCanvas(); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 41 | |
| 42 | SkScalar stroke = size.fWidth / 10; |
| 43 | SkScalar radius = (size.fWidth - stroke) / 2; |
| 44 | |
| 45 | SkPaint paint; |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 46 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 47 | paint.setAntiAlias(true); |
| 48 | paint.setColor(fillC); |
| 49 | canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 50 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 51 | paint.setStyle(SkPaint::kStroke_Style); |
| 52 | paint.setStrokeWidth(stroke); |
| 53 | paint.setColor(SK_ColorBLACK); |
| 54 | canvas->drawCircle(size.fWidth/2, size.fHeight/2, radius, paint); |
| 55 | } |
| 56 | |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 57 | static void test_surface(SkCanvas* canvas, SkSurface* surf, bool usePaint) { |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 58 | drawContents(surf, SK_ColorRED); |
junov@chromium.org | 5ee449a | 2013-04-12 20:20:50 +0000 | [diff] [blame] | 59 | SkImage* imgR = surf->newImageSnapshot(); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 60 | |
reed@google.com | 97af1a6 | 2012-08-28 12:19:02 +0000 | [diff] [blame] | 61 | if (true) { |
junov@chromium.org | 5ee449a | 2013-04-12 20:20:50 +0000 | [diff] [blame] | 62 | SkImage* imgR2 = surf->newImageSnapshot(); |
reed@google.com | 97af1a6 | 2012-08-28 12:19:02 +0000 | [diff] [blame] | 63 | SkASSERT(imgR == imgR2); |
| 64 | imgR2->unref(); |
| 65 | } |
| 66 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 67 | drawContents(surf, SK_ColorGREEN); |
junov@chromium.org | 5ee449a | 2013-04-12 20:20:50 +0000 | [diff] [blame] | 68 | SkImage* imgG = surf->newImageSnapshot(); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 69 | |
reed@google.com | 97af1a6 | 2012-08-28 12:19:02 +0000 | [diff] [blame] | 70 | // since we've drawn after we snapped imgR, imgG will be a different obj |
| 71 | SkASSERT(imgR != imgG); |
| 72 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 73 | drawContents(surf, SK_ColorBLUE); |
| 74 | |
mike@reedtribe.org | d2782ed | 2012-07-31 02:45:15 +0000 | [diff] [blame] | 75 | SkPaint paint; |
| 76 | // paint.setFilterBitmap(true); |
| 77 | // paint.setAlpha(0x80); |
| 78 | |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 79 | canvas->drawImage(imgR, 0, 0, usePaint ? &paint : nullptr); |
| 80 | canvas->drawImage(imgG, 0, 80, usePaint ? &paint : nullptr); |
| 81 | surf->draw(canvas, 0, 160, usePaint ? &paint : nullptr); |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 82 | |
| 83 | SkRect src1, src2, src3; |
| 84 | src1.iset(0, 0, surf->width(), surf->height()); |
skia.committer@gmail.com | 7f1af50 | 2013-07-24 07:01:12 +0000 | [diff] [blame] | 85 | src2.iset(-surf->width() / 2, -surf->height() / 2, |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 86 | surf->width(), surf->height()); |
| 87 | src3.iset(0, 0, surf->width() / 2, surf->height() / 2); |
| 88 | |
| 89 | SkRect dst1, dst2, dst3, dst4; |
| 90 | dst1.set(0, 240, 65, 305); |
| 91 | dst2.set(0, 320, 65, 385); |
| 92 | dst3.set(0, 400, 65, 465); |
| 93 | dst4.set(0, 480, 65, 545); |
| 94 | |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 95 | canvas->drawImageRect(imgR, src1, dst1, usePaint ? &paint : nullptr); |
| 96 | canvas->drawImageRect(imgG, src2, dst2, usePaint ? &paint : nullptr); |
| 97 | canvas->drawImageRect(imgR, src3, dst3, usePaint ? &paint : nullptr); |
| 98 | canvas->drawImageRect(imgG, dst4, usePaint ? &paint : nullptr); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 99 | |
| 100 | imgG->unref(); |
| 101 | imgR->unref(); |
| 102 | } |
| 103 | |
| 104 | class ImageGM : public skiagm::GM { |
| 105 | void* fBuffer; |
reed@google.com | 58b21ec | 2012-07-30 18:20:12 +0000 | [diff] [blame] | 106 | size_t fBufferSize; |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 107 | SkSize fSize; |
| 108 | enum { |
| 109 | W = 64, |
| 110 | H = 64, |
| 111 | RB = W * 4 + 8, |
| 112 | }; |
| 113 | public: |
| 114 | ImageGM() { |
reed@google.com | 58b21ec | 2012-07-30 18:20:12 +0000 | [diff] [blame] | 115 | fBufferSize = RB * H; |
| 116 | fBuffer = sk_malloc_throw(fBufferSize); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 117 | fSize.set(SkIntToScalar(W), SkIntToScalar(H)); |
| 118 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 119 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 120 | virtual ~ImageGM() { |
| 121 | sk_free(fBuffer); |
| 122 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 123 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 124 | protected: |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 125 | SkString onShortName() override { |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 126 | return SkString("image-surface"); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 127 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 128 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 129 | SkISize onISize() override { |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 130 | return SkISize::Make(960, 1200); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 131 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 132 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 133 | void onDraw(SkCanvas* canvas) override { |
mike@reedtribe.org | d829b5c | 2012-07-31 03:57:11 +0000 | [diff] [blame] | 134 | drawJpeg(canvas, this->getISize()); |
| 135 | |
mike@reedtribe.org | d2782ed | 2012-07-31 02:45:15 +0000 | [diff] [blame] | 136 | canvas->scale(2, 2); |
| 137 | |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 138 | static const char* kLabel1 = "Original Img"; |
| 139 | static const char* kLabel2 = "Modified Img"; |
| 140 | static const char* kLabel3 = "Cur Surface"; |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 141 | static const char* kLabel4 = "Full Crop"; |
| 142 | static const char* kLabel5 = "Over-crop"; |
| 143 | static const char* kLabel6 = "Upper-left"; |
| 144 | static const char* kLabel7 = "No Crop"; |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 145 | |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 146 | static const char* kLabel8 = "Pre-Alloc Img"; |
| 147 | static const char* kLabel9 = "New Alloc Img"; |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 148 | static const char* kLabel10 = "GPU"; |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 149 | |
| 150 | SkPaint textPaint; |
commit-bot@chromium.org | cae54f1 | 2014-04-11 18:34:35 +0000 | [diff] [blame] | 151 | textPaint.setAntiAlias(true); |
caryclark | 1818acb | 2015-07-24 12:09:25 -0700 | [diff] [blame] | 152 | sk_tool_utils::set_portable_typeface(&textPaint); |
commit-bot@chromium.org | cae54f1 | 2014-04-11 18:34:35 +0000 | [diff] [blame] | 153 | textPaint.setTextSize(8); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 154 | |
| 155 | canvas->drawText(kLabel1, strlen(kLabel1), 10, 60, textPaint); |
| 156 | canvas->drawText(kLabel2, strlen(kLabel2), 10, 140, textPaint); |
| 157 | canvas->drawText(kLabel3, strlen(kLabel3), 10, 220, textPaint); |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 158 | canvas->drawText(kLabel4, strlen(kLabel4), 10, 300, textPaint); |
| 159 | canvas->drawText(kLabel5, strlen(kLabel5), 10, 380, textPaint); |
| 160 | canvas->drawText(kLabel6, strlen(kLabel6), 10, 460, textPaint); |
| 161 | canvas->drawText(kLabel7, strlen(kLabel7), 10, 540, textPaint); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 162 | |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 163 | canvas->drawText(kLabel8, strlen(kLabel8), 80, 10, textPaint); |
| 164 | canvas->drawText(kLabel9, strlen(kLabel9), 160, 10, textPaint); |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 165 | canvas->drawText(kLabel10, strlen(kLabel10), 265, 10, textPaint); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 166 | |
| 167 | canvas->translate(80, 20); |
| 168 | |
reed@google.com | 58b21ec | 2012-07-30 18:20:12 +0000 | [diff] [blame] | 169 | // since we draw into this directly, we need to start fresh |
| 170 | sk_bzero(fBuffer, fBufferSize); |
| 171 | |
commit-bot@chromium.org | 32678d9 | 2014-01-15 02:38:22 +0000 | [diff] [blame] | 172 | SkImageInfo info = SkImageInfo::MakeN32Premul(W, H); |
mike@reedtribe.org | b947625 | 2012-11-15 02:37:45 +0000 | [diff] [blame] | 173 | SkAutoTUnref<SkSurface> surf0(SkSurface::NewRasterDirect(info, fBuffer, RB)); |
| 174 | SkAutoTUnref<SkSurface> surf1(SkSurface::NewRaster(info)); |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 175 | SkAutoTUnref<SkSurface> surf2; // gpu |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 176 | |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 177 | #if SK_SUPPORT_GPU |
| 178 | surf2.reset(SkSurface::NewRenderTarget(canvas->getGrContext(), |
| 179 | SkSurface::kNo_Budgeted, info)); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 180 | #endif |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 181 | |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 182 | test_surface(canvas, surf0, true); |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 183 | canvas->translate(80, 0); |
commit-bot@chromium.org | dfec28d | 2013-07-23 15:52:16 +0000 | [diff] [blame] | 184 | test_surface(canvas, surf1, true); |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 185 | if (surf2) { |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 186 | canvas->translate(80, 0); |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 187 | test_surface(canvas, surf2, true); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 188 | } |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 189 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 190 | |
mike@reedtribe.org | 70e3590 | 2012-07-29 20:38:16 +0000 | [diff] [blame] | 191 | private: |
| 192 | typedef skiagm::GM INHERITED; |
| 193 | }; |
reed | a9cb871 | 2015-01-16 14:21:40 -0800 | [diff] [blame] | 194 | DEF_GM( return new ImageGM; ) |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 195 | |
| 196 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 197 | |
| 198 | #include "SkPictureRecorder.h" |
| 199 | |
| 200 | static void draw_pixmap(SkCanvas* canvas, const SkPixmap& pmap) { |
| 201 | SkBitmap bitmap; |
| 202 | bitmap.installPixels(pmap.info(), (void*)pmap.addr(), pmap.rowBytes()); |
| 203 | canvas->drawBitmap(bitmap, 0, 0, nullptr); |
| 204 | } |
| 205 | |
| 206 | static void show_scaled_pixels(SkCanvas* canvas, SkImage* image) { |
| 207 | SkAutoCanvasRestore acr(canvas, true); |
| 208 | |
| 209 | canvas->drawImage(image, 0, 0, nullptr); |
| 210 | canvas->translate(110, 10); |
| 211 | |
| 212 | const SkImageInfo info = SkImageInfo::MakeN32Premul(40, 40); |
| 213 | SkAutoPixmapStorage storage; |
| 214 | storage.alloc(info); |
| 215 | |
| 216 | const SkImage::CachingHint chints[] = { |
reed | 6868c3f | 2015-11-24 11:44:47 -0800 | [diff] [blame] | 217 | SkImage::kAllow_CachingHint, SkImage::kDisallow_CachingHint, |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 218 | }; |
| 219 | const SkFilterQuality qualities[] = { |
| 220 | kNone_SkFilterQuality, kLow_SkFilterQuality, kMedium_SkFilterQuality, kHigh_SkFilterQuality, |
| 221 | }; |
| 222 | |
| 223 | for (auto ch : chints) { |
| 224 | canvas->save(); |
| 225 | for (auto q : qualities) { |
| 226 | if (image->scalePixels(storage, q, ch)) { |
| 227 | draw_pixmap(canvas, storage); |
| 228 | } |
| 229 | canvas->translate(70, 0); |
| 230 | } |
| 231 | canvas->restore(); |
| 232 | canvas->translate(0, 45); |
| 233 | } |
| 234 | } |
| 235 | |
| 236 | static void draw_contents(SkCanvas* canvas) { |
| 237 | SkPaint paint; |
| 238 | paint.setStyle(SkPaint::kStroke_Style); |
| 239 | paint.setStrokeWidth(20); |
| 240 | canvas->drawCircle(50, 50, 35, paint); |
| 241 | } |
| 242 | |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 243 | static SkImage* make_raster(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) { |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 244 | SkAutoTUnref<SkSurface> surface(SkSurface::NewRaster(info)); |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 245 | draw(surface->getCanvas()); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 246 | return surface->newImageSnapshot(); |
| 247 | } |
| 248 | |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 249 | static SkImage* make_picture(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) { |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 250 | SkPictureRecorder recorder; |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 251 | draw(recorder.beginRecording(SkRect::MakeIWH(info.width(), info.height()))); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 252 | SkAutoTUnref<SkPicture> pict(recorder.endRecording()); |
| 253 | return SkImage::NewFromPicture(pict, info.dimensions(), nullptr, nullptr); |
| 254 | } |
| 255 | |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 256 | static SkImage* make_codec(const SkImageInfo& info, GrContext*, void (*draw)(SkCanvas*)) { |
| 257 | SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw)); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 258 | SkAutoTUnref<SkData> data(image->encode()); |
| 259 | return SkImage::NewFromEncoded(data); |
| 260 | } |
| 261 | |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 262 | static SkImage* make_gpu(const SkImageInfo& info, GrContext* ctx, void (*draw)(SkCanvas*)) { |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 263 | if (!ctx) { return nullptr; } |
| 264 | SkAutoTUnref<SkSurface> surface(SkSurface::NewRenderTarget(ctx, SkSurface::kNo_Budgeted, info)); |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 265 | draw(surface->getCanvas()); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 266 | return surface->newImageSnapshot(); |
| 267 | } |
| 268 | |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 269 | typedef SkImage* (*ImageMakerProc)(const SkImageInfo&, GrContext*, void (*)(SkCanvas*)); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 270 | |
| 271 | class ScalePixelsGM : public skiagm::GM { |
| 272 | public: |
| 273 | ScalePixelsGM() {} |
| 274 | |
| 275 | protected: |
| 276 | SkString onShortName() override { |
| 277 | return SkString("scale-pixels"); |
| 278 | } |
| 279 | |
| 280 | SkISize onISize() override { |
| 281 | return SkISize::Make(960, 1200); |
| 282 | } |
| 283 | |
| 284 | void onDraw(SkCanvas* canvas) override { |
| 285 | const SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100); |
| 286 | |
| 287 | const ImageMakerProc procs[] = { |
reed | 6868c3f | 2015-11-24 11:44:47 -0800 | [diff] [blame] | 288 | make_codec, make_raster, make_picture, make_codec, make_gpu, |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 289 | }; |
| 290 | for (auto& proc : procs) { |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 291 | SkAutoTUnref<SkImage> image(proc(info, canvas->getGrContext(), draw_contents)); |
reed | 0955303 | 2015-11-23 12:32:16 -0800 | [diff] [blame] | 292 | if (image) { |
| 293 | show_scaled_pixels(canvas, image); |
| 294 | } |
| 295 | canvas->translate(0, 120); |
| 296 | } |
| 297 | } |
| 298 | |
| 299 | private: |
| 300 | typedef skiagm::GM INHERITED; |
| 301 | }; |
| 302 | DEF_GM( return new ScalePixelsGM; ) |
reed | 7850eb2 | 2015-12-02 14:19:47 -0800 | [diff] [blame] | 303 | |
| 304 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 305 | |
| 306 | #include "SkImageGenerator.h" |
| 307 | |
| 308 | static SkImageInfo make_info(SkImage* img) { |
| 309 | return SkImageInfo::MakeN32(img->width(), img->height(), |
| 310 | img->isOpaque() ? kOpaque_SkAlphaType : kPremul_SkAlphaType); |
| 311 | } |
| 312 | |
| 313 | // Its simple, but I wonder if we should expose this formally? |
| 314 | // |
| 315 | class ImageGeneratorFromImage : public SkImageGenerator { |
| 316 | public: |
| 317 | ImageGeneratorFromImage(SkImage* img) : INHERITED(make_info(img)), fImg(SkRef(img)) {} |
| 318 | |
| 319 | protected: |
| 320 | bool onGetPixels(const SkImageInfo& info, void* pixels, size_t rowBytes, SkPMColor ctable[], |
| 321 | int* ctableCount) override { |
| 322 | return fImg->readPixels(info, pixels, rowBytes, 0, 0); |
| 323 | } |
| 324 | |
| 325 | private: |
| 326 | SkAutoTUnref<SkImage> fImg; |
| 327 | |
| 328 | typedef SkImageGenerator INHERITED; |
| 329 | }; |
| 330 | |
| 331 | static void draw_opaque_contents(SkCanvas* canvas) { |
| 332 | canvas->drawColor(0xFFFF8844); |
| 333 | |
| 334 | SkPaint paint; |
| 335 | paint.setStyle(SkPaint::kStroke_Style); |
| 336 | paint.setStrokeWidth(20); |
| 337 | canvas->drawCircle(50, 50, 35, paint); |
| 338 | } |
| 339 | |
| 340 | static SkImageGenerator* gen_raster(const SkImageInfo& info) { |
| 341 | SkAutoTUnref<SkSurface> surface(SkSurface::NewRaster(info)); |
| 342 | draw_opaque_contents(surface->getCanvas()); |
| 343 | SkAutoTUnref<SkImage> img(surface->newImageSnapshot()); |
| 344 | return new ImageGeneratorFromImage(img); |
| 345 | } |
| 346 | |
| 347 | static SkImageGenerator* gen_picture(const SkImageInfo& info) { |
| 348 | SkPictureRecorder recorder; |
| 349 | draw_opaque_contents(recorder.beginRecording(SkRect::MakeIWH(info.width(), info.height()))); |
| 350 | SkAutoTUnref<SkPicture> pict(recorder.endRecording()); |
| 351 | return SkImageGenerator::NewFromPicture(info.dimensions(), pict, nullptr, nullptr); |
| 352 | } |
| 353 | |
| 354 | static SkImageGenerator* gen_png(const SkImageInfo& info) { |
| 355 | SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw_opaque_contents)); |
| 356 | SkAutoTUnref<SkData> data(image->encode(SkImageEncoder::kPNG_Type, 100)); |
| 357 | return SkImageGenerator::NewFromEncoded(data); |
| 358 | } |
| 359 | |
| 360 | static SkImageGenerator* gen_jpg(const SkImageInfo& info) { |
| 361 | SkAutoTUnref<SkImage> image(make_raster(info, nullptr, draw_opaque_contents)); |
| 362 | SkAutoTUnref<SkData> data(image->encode(SkImageEncoder::kJPEG_Type, 100)); |
| 363 | return SkImageGenerator::NewFromEncoded(data); |
| 364 | } |
| 365 | |
| 366 | typedef SkImageGenerator* (*GeneratorMakerProc)(const SkImageInfo&); |
| 367 | |
| 368 | static void show_scaled_generator(SkCanvas* canvas, SkImageGenerator* gen) { |
| 369 | const SkImageInfo genInfo = gen->getInfo(); |
| 370 | |
| 371 | SkAutoCanvasRestore acr(canvas, true); |
| 372 | |
| 373 | SkBitmap bm; |
| 374 | bm.allocPixels(genInfo); |
| 375 | if (gen->getPixels(bm.info(), bm.getPixels(), bm.rowBytes())) { |
| 376 | canvas->drawBitmap(bm, 0, 0, nullptr); |
| 377 | } |
| 378 | canvas->translate(110, 0); |
| 379 | |
| 380 | const float scales[] = { 0.75f, 0.5f, 0.25f }; |
| 381 | for (auto scale : scales) { |
| 382 | SkImageGenerator::SupportedSizes sizes; |
| 383 | if (gen->computeScaledDimensions(scale, &sizes)) { |
| 384 | const SkImageInfo info = SkImageInfo::MakeN32Premul(sizes.fSizes[0].width(), |
| 385 | sizes.fSizes[0].height()); |
| 386 | bm.allocPixels(info); |
| 387 | SkPixmap pmap; |
| 388 | bm.peekPixels(&pmap); |
| 389 | if (gen->generateScaledPixels(pmap)) { |
| 390 | canvas->drawBitmap(bm, 0, SkIntToScalar(genInfo.height() - info.height())/2); |
| 391 | } |
| 392 | } |
| 393 | canvas->translate(100, 0); |
| 394 | } |
| 395 | } |
| 396 | |
| 397 | class ScaleGeneratorGM : public skiagm::GM { |
| 398 | public: |
| 399 | ScaleGeneratorGM() {} |
| 400 | |
| 401 | protected: |
| 402 | SkString onShortName() override { |
| 403 | return SkString("scale-generator"); |
| 404 | } |
| 405 | |
| 406 | SkISize onISize() override { |
| 407 | return SkISize::Make(500, 500); |
| 408 | } |
| 409 | |
| 410 | void onDraw(SkCanvas* canvas) override { |
| 411 | canvas->translate(10, 10); |
| 412 | |
| 413 | // explicitly make it opaque, so we can test JPEG (which is only ever opaque) |
| 414 | const SkImageInfo info = SkImageInfo::MakeN32(100, 100, kOpaque_SkAlphaType); |
| 415 | |
| 416 | const GeneratorMakerProc procs[] = { |
| 417 | gen_raster, gen_picture, gen_png, gen_jpg, |
| 418 | }; |
| 419 | for (auto& proc : procs) { |
| 420 | SkAutoTDelete<SkImageGenerator> gen(proc(info)); |
| 421 | if (gen) { |
| 422 | show_scaled_generator(canvas, gen); |
| 423 | } |
| 424 | canvas->translate(0, 120); |
| 425 | } |
| 426 | } |
| 427 | |
| 428 | private: |
| 429 | typedef skiagm::GM INHERITED; |
| 430 | }; |
| 431 | DEF_GM( return new ScaleGeneratorGM; ) |