reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2015 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" |
| 9 | #include "SkCanvas.h" |
| 10 | #include "SkImage.h" |
Brian Osman | df7e075 | 2017-04-26 16:20:28 -0400 | [diff] [blame] | 11 | #include "SkImageGenerator.h" |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 12 | #include "SkImage_Base.h" |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 13 | #include "SkMakeUnique.h" |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 14 | #include "SkPictureRecorder.h" |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 15 | #include "SkSurface.h" |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 16 | |
| 17 | #if SK_SUPPORT_GPU |
| 18 | #include "GrContext.h" |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 19 | #include "GrContextPriv.h" |
| 20 | #include "GrSurfaceContext.h" |
| 21 | #include "GrSurfaceProxy.h" |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 22 | #include "GrTexture.h" |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 23 | #include "GrTextureProxy.h" |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 24 | #include "../src/image/SkImage_Gpu.h" |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 25 | #endif |
| 26 | |
| 27 | static void draw_something(SkCanvas* canvas, const SkRect& bounds) { |
| 28 | SkPaint paint; |
| 29 | paint.setAntiAlias(true); |
| 30 | paint.setColor(SK_ColorRED); |
| 31 | paint.setStyle(SkPaint::kStroke_Style); |
| 32 | paint.setStrokeWidth(10); |
| 33 | canvas->drawRect(bounds, paint); |
| 34 | paint.setStyle(SkPaint::kFill_Style); |
| 35 | paint.setColor(SK_ColorBLUE); |
| 36 | canvas->drawOval(bounds, paint); |
| 37 | } |
| 38 | |
| 39 | /* |
| 40 | * Exercise drawing pictures inside an image, showing that the image version is pixelated |
| 41 | * (correctly) when it is inside an image. |
| 42 | */ |
| 43 | class ImagePictGM : public skiagm::GM { |
reed | ca2622b | 2016-03-18 07:25:55 -0700 | [diff] [blame] | 44 | sk_sp<SkPicture> fPicture; |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 45 | sk_sp<SkImage> fImage0; |
| 46 | sk_sp<SkImage> fImage1; |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 47 | public: |
| 48 | ImagePictGM() {} |
| 49 | |
| 50 | protected: |
| 51 | SkString onShortName() override { |
| 52 | return SkString("image-picture"); |
| 53 | } |
| 54 | |
| 55 | SkISize onISize() override { |
| 56 | return SkISize::Make(850, 450); |
| 57 | } |
| 58 | |
| 59 | void onOnceBeforeDraw() override { |
| 60 | const SkRect bounds = SkRect::MakeXYWH(100, 100, 100, 100); |
| 61 | SkPictureRecorder recorder; |
| 62 | draw_something(recorder.beginRecording(bounds), bounds); |
reed | ca2622b | 2016-03-18 07:25:55 -0700 | [diff] [blame] | 63 | fPicture = recorder.finishRecordingAsPicture(); |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 64 | |
| 65 | // extract enough just for the oval. |
| 66 | const SkISize size = SkISize::Make(100, 100); |
Matt Sarett | 77a7a1b | 2017-02-07 13:56:11 -0500 | [diff] [blame] | 67 | auto srgbColorSpace = SkColorSpace::MakeSRGB(); |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 68 | |
| 69 | SkMatrix matrix; |
| 70 | matrix.setTranslate(-100, -100); |
Matt Sarett | e94255d | 2017-01-09 12:38:59 -0500 | [diff] [blame] | 71 | fImage0 = SkImage::MakeFromPicture(fPicture, size, &matrix, nullptr, |
| 72 | SkImage::BitDepth::kU8, srgbColorSpace); |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 73 | matrix.postTranslate(-50, -50); |
| 74 | matrix.postRotate(45); |
| 75 | matrix.postTranslate(50, 50); |
Matt Sarett | e94255d | 2017-01-09 12:38:59 -0500 | [diff] [blame] | 76 | fImage1 = SkImage::MakeFromPicture(fPicture, size, &matrix, nullptr, |
| 77 | SkImage::BitDepth::kU8, srgbColorSpace); |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 78 | } |
| 79 | |
| 80 | void drawSet(SkCanvas* canvas) const { |
| 81 | SkMatrix matrix = SkMatrix::MakeTrans(-100, -100); |
| 82 | canvas->drawPicture(fPicture, &matrix, nullptr); |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 83 | canvas->drawImage(fImage0.get(), 150, 0); |
| 84 | canvas->drawImage(fImage1.get(), 300, 0); |
reed | d5b7563 | 2015-08-13 09:37:45 -0700 | [diff] [blame] | 85 | } |
| 86 | |
| 87 | void onDraw(SkCanvas* canvas) override { |
| 88 | canvas->translate(20, 20); |
| 89 | |
| 90 | this->drawSet(canvas); |
| 91 | |
| 92 | canvas->save(); |
| 93 | canvas->translate(0, 130); |
| 94 | canvas->scale(0.25f, 0.25f); |
| 95 | this->drawSet(canvas); |
| 96 | canvas->restore(); |
| 97 | |
| 98 | canvas->save(); |
| 99 | canvas->translate(0, 200); |
| 100 | canvas->scale(2, 2); |
| 101 | this->drawSet(canvas); |
| 102 | canvas->restore(); |
| 103 | } |
| 104 | |
| 105 | private: |
| 106 | typedef skiagm::GM INHERITED; |
| 107 | }; |
| 108 | DEF_GM( return new ImagePictGM; ) |
| 109 | |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 110 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 111 | |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 112 | static std::unique_ptr<SkImageGenerator> make_pic_generator(GrContext*, sk_sp<SkPicture> pic) { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 113 | SkMatrix matrix; |
| 114 | matrix.setTranslate(-100, -100); |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 115 | return SkImageGenerator::MakeFromPicture({ 100, 100 }, std::move(pic), &matrix, nullptr, |
Matt Sarett | e94255d | 2017-01-09 12:38:59 -0500 | [diff] [blame] | 116 | SkImage::BitDepth::kU8, |
Matt Sarett | 77a7a1b | 2017-02-07 13:56:11 -0500 | [diff] [blame] | 117 | SkColorSpace::MakeSRGB()); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 118 | } |
| 119 | |
| 120 | class RasterGenerator : public SkImageGenerator { |
| 121 | public: |
Mike Reed | 4edb5d2 | 2017-04-17 11:02:51 -0400 | [diff] [blame] | 122 | RasterGenerator(const SkBitmap& bm) : SkImageGenerator(bm.info()), fBM(bm) |
| 123 | {} |
| 124 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 125 | protected: |
| 126 | bool onGetPixels(const SkImageInfo& info, void* pixels, size_t rowBytes, |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 127 | SkPMColor* ctable, int* ctableCount) override { |
| 128 | SkASSERT(fBM.width() == info.width()); |
| 129 | SkASSERT(fBM.height() == info.height()); |
| 130 | |
| 131 | if (info.colorType() == kIndex_8_SkColorType) { |
| 132 | if (SkColorTable* ct = fBM.getColorTable()) { |
| 133 | if (ctable) { |
| 134 | memcpy(ctable, ct->readColors(), ct->count() * sizeof(SkPMColor)); |
| 135 | } |
| 136 | if (ctableCount) { |
| 137 | *ctableCount = ct->count(); |
| 138 | } |
| 139 | |
| 140 | for (int y = 0; y < info.height(); ++y) { |
| 141 | memcpy(pixels, fBM.getAddr8(0, y), fBM.width()); |
| 142 | pixels = (char*)pixels + rowBytes; |
| 143 | } |
| 144 | return true; |
| 145 | } else { |
| 146 | return false; |
| 147 | } |
| 148 | } else { |
| 149 | return fBM.readPixels(info, pixels, rowBytes, 0, 0); |
| 150 | } |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 151 | } |
| 152 | private: |
| 153 | SkBitmap fBM; |
| 154 | }; |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 155 | static std::unique_ptr<SkImageGenerator> make_ras_generator(GrContext*, sk_sp<SkPicture> pic) { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 156 | SkBitmap bm; |
| 157 | bm.allocN32Pixels(100, 100); |
| 158 | SkCanvas canvas(bm); |
| 159 | canvas.clear(0); |
| 160 | canvas.translate(-100, -100); |
| 161 | canvas.drawPicture(pic); |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 162 | return skstd::make_unique<RasterGenerator>(bm); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 163 | } |
| 164 | |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 165 | // so we can create a color-table |
| 166 | static int find_closest(SkPMColor c, const SkPMColor table[], int count) { |
| 167 | const int cr = SkGetPackedR32(c); |
| 168 | const int cg = SkGetPackedG32(c); |
| 169 | const int cb = SkGetPackedB32(c); |
| 170 | |
| 171 | int minDist = 999999999; |
| 172 | int index = 0; |
| 173 | for (int i = 0; i < count; ++i) { |
| 174 | int dr = SkAbs32((int)SkGetPackedR32(table[i]) - cr); |
| 175 | int dg = SkAbs32((int)SkGetPackedG32(table[i]) - cg); |
| 176 | int db = SkAbs32((int)SkGetPackedB32(table[i]) - cb); |
| 177 | int dist = dr + dg + db; |
| 178 | if (dist < minDist) { |
| 179 | minDist = dist; |
| 180 | index = i; |
| 181 | } |
| 182 | } |
| 183 | return index; |
| 184 | } |
| 185 | |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 186 | static std::unique_ptr<SkImageGenerator> make_ctable_generator(GrContext*, sk_sp<SkPicture> pic) { |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 187 | SkBitmap bm; |
| 188 | bm.allocN32Pixels(100, 100); |
| 189 | SkCanvas canvas(bm); |
| 190 | canvas.clear(0); |
| 191 | canvas.translate(-100, -100); |
| 192 | canvas.drawPicture(pic); |
| 193 | |
| 194 | const SkPMColor colors[] = { |
| 195 | SkPreMultiplyColor(SK_ColorRED), |
| 196 | SkPreMultiplyColor(0), |
| 197 | SkPreMultiplyColor(SK_ColorBLUE), |
| 198 | }; |
| 199 | const int count = SK_ARRAY_COUNT(colors); |
| 200 | SkImageInfo info = SkImageInfo::Make(100, 100, kIndex_8_SkColorType, kPremul_SkAlphaType); |
| 201 | |
| 202 | SkBitmap bm2; |
Mike Reed | 6b3155c | 2017-04-03 14:41:44 -0400 | [diff] [blame] | 203 | bm2.allocPixels(info, SkColorTable::Make(colors, count)); |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 204 | for (int y = 0; y < info.height(); ++y) { |
| 205 | for (int x = 0; x < info.width(); ++x) { |
| 206 | *bm2.getAddr8(x, y) = find_closest(*bm.getAddr32(x, y), colors, count); |
| 207 | } |
| 208 | } |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 209 | return skstd::make_unique<RasterGenerator>(bm2); |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 210 | } |
| 211 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 212 | class EmptyGenerator : public SkImageGenerator { |
| 213 | public: |
| 214 | EmptyGenerator(const SkImageInfo& info) : SkImageGenerator(info) {} |
| 215 | }; |
| 216 | |
| 217 | #if SK_SUPPORT_GPU |
| 218 | class TextureGenerator : public SkImageGenerator { |
| 219 | public: |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 220 | TextureGenerator(GrContext* ctx, const SkImageInfo& info, sk_sp<SkPicture> pic) |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 221 | : SkImageGenerator(info) |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 222 | , fCtx(SkRef(ctx)) { |
| 223 | |
| 224 | sk_sp<SkSurface> surface(SkSurface::MakeRenderTarget(ctx, SkBudgeted::kNo, info)); |
scroggo | e6f0d6e | 2016-05-13 07:25:44 -0700 | [diff] [blame] | 225 | if (surface) { |
| 226 | surface->getCanvas()->clear(0); |
| 227 | surface->getCanvas()->translate(-100, -100); |
| 228 | surface->getCanvas()->drawPicture(pic); |
| 229 | sk_sp<SkImage> image(surface->makeImageSnapshot()); |
Robert Phillips | 6de9904 | 2017-01-31 11:31:39 -0500 | [diff] [blame] | 230 | fProxy = as_IB(image)->asTextureProxyRef(); |
scroggo | e6f0d6e | 2016-05-13 07:25:44 -0700 | [diff] [blame] | 231 | } |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 232 | } |
| 233 | protected: |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 234 | sk_sp<GrTextureProxy> onGenerateTexture(GrContext* ctx, const SkImageInfo& info, |
| 235 | const SkIPoint& origin) override { |
| 236 | SkASSERT(ctx); |
| 237 | SkASSERT(ctx == fCtx.get()); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 238 | |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 239 | if (!fProxy) { |
scroggo | e6f0d6e | 2016-05-13 07:25:44 -0700 | [diff] [blame] | 240 | return nullptr; |
| 241 | } |
| 242 | |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 243 | if (origin.fX == 0 && origin.fY == 0 && |
| 244 | info.width() == fProxy->width() && info.height() == fProxy->height()) { |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 245 | return fProxy; |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 246 | } |
| 247 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 248 | // need to copy the subset into a new texture |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 249 | GrSurfaceDesc desc = fProxy->desc(); |
Brian Osman | 222e9ad | 2016-12-14 15:42:36 -0500 | [diff] [blame] | 250 | desc.fWidth = info.width(); |
| 251 | desc.fHeight = info.height(); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 252 | |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 253 | sk_sp<GrSurfaceContext> dstContext(fCtx->contextPriv().makeDeferredSurfaceContext( |
| 254 | desc, |
| 255 | SkBackingFit::kExact, |
| 256 | SkBudgeted::kNo)); |
| 257 | if (!dstContext) { |
| 258 | return nullptr; |
| 259 | } |
| 260 | |
| 261 | if (!dstContext->copy( |
| 262 | fProxy.get(), |
| 263 | SkIRect::MakeXYWH(origin.x(), origin.y(), info.width(), info.height()), |
| 264 | SkIPoint::Make(0, 0))) { |
| 265 | return nullptr; |
| 266 | } |
| 267 | |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 268 | return dstContext->asTextureProxyRef(); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 269 | } |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 270 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 271 | private: |
Robert Phillips | e2f7d18 | 2016-12-15 09:23:05 -0500 | [diff] [blame] | 272 | sk_sp<GrContext> fCtx; |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 273 | sk_sp<GrTextureProxy> fProxy; |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 274 | }; |
Robert Phillips | 4447b64 | 2017-03-03 11:10:18 -0500 | [diff] [blame] | 275 | |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 276 | static std::unique_ptr<SkImageGenerator> make_tex_generator(GrContext* ctx, sk_sp<SkPicture> pic) { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 277 | const SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100); |
| 278 | |
| 279 | if (!ctx) { |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 280 | return skstd::make_unique<EmptyGenerator>(info); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 281 | } |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 282 | return skstd::make_unique<TextureGenerator>(ctx, info, pic); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 283 | } |
| 284 | #endif |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 285 | |
| 286 | class ImageCacheratorGM : public skiagm::GM { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 287 | SkString fName; |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 288 | std::unique_ptr<SkImageGenerator> (*fFactory)(GrContext*, sk_sp<SkPicture>); |
reed | ca2622b | 2016-03-18 07:25:55 -0700 | [diff] [blame] | 289 | sk_sp<SkPicture> fPicture; |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 290 | sk_sp<SkImage> fImage; |
| 291 | sk_sp<SkImage> fImageSubset; |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 292 | |
| 293 | public: |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 294 | ImageCacheratorGM(const char suffix[], |
| 295 | std::unique_ptr<SkImageGenerator> (*factory)(GrContext*, sk_sp<SkPicture>)) |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 296 | : fFactory(factory) |
| 297 | { |
| 298 | fName.printf("image-cacherator-from-%s", suffix); |
| 299 | } |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 300 | |
| 301 | protected: |
| 302 | SkString onShortName() override { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 303 | return fName; |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 304 | } |
| 305 | |
| 306 | SkISize onISize() override { |
reed | a32cc95 | 2015-08-19 06:07:29 -0700 | [diff] [blame] | 307 | return SkISize::Make(960, 450); |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 308 | } |
| 309 | |
| 310 | void onOnceBeforeDraw() override { |
| 311 | const SkRect bounds = SkRect::MakeXYWH(100, 100, 100, 100); |
| 312 | SkPictureRecorder recorder; |
| 313 | draw_something(recorder.beginRecording(bounds), bounds); |
reed | ca2622b | 2016-03-18 07:25:55 -0700 | [diff] [blame] | 314 | fPicture = recorder.finishRecordingAsPicture(); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 315 | } |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 316 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 317 | void makeCaches(GrContext* ctx) { |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 318 | auto gen = fFactory(ctx, fPicture); |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 319 | fImage = SkImage::MakeFromGenerator(std::move(gen)); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 320 | |
| 321 | const SkIRect subset = SkIRect::MakeLTRB(50, 50, 100, 100); |
| 322 | |
Mike Reed | 185130c | 2017-02-15 15:14:16 -0500 | [diff] [blame] | 323 | gen = fFactory(ctx, fPicture); |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 324 | fImageSubset = SkImage::MakeFromGenerator(std::move(gen), &subset); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 325 | |
Brian Osman | df7e075 | 2017-04-26 16:20:28 -0400 | [diff] [blame] | 326 | SkASSERT(fImage->dimensions() == SkISize::Make(100, 100)); |
| 327 | SkASSERT(fImageSubset->dimensions() == SkISize::Make(50, 50)); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 328 | } |
| 329 | |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 330 | static void draw_as_bitmap(SkCanvas* canvas, SkImage* image, SkScalar x, SkScalar y) { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 331 | SkBitmap bitmap; |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 332 | as_IB(image)->getROPixels(&bitmap, canvas->imageInfo().colorSpace()); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 333 | canvas->drawBitmap(bitmap, x, y); |
| 334 | } |
| 335 | |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 336 | static void draw_as_tex(SkCanvas* canvas, SkImage* image, SkScalar x, SkScalar y) { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 337 | #if SK_SUPPORT_GPU |
Brian Osman | 7992da3 | 2016-11-18 11:28:24 -0500 | [diff] [blame] | 338 | sk_sp<SkColorSpace> texColorSpace; |
Robert Phillips | 4f358be | 2017-03-23 08:21:00 -0400 | [diff] [blame] | 339 | sk_sp<GrTextureProxy> proxy( |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 340 | as_IB(image)->asTextureProxyRef(canvas->getGrContext(), GrSamplerParams::ClampBilerp(), |
| 341 | canvas->imageInfo().colorSpace(), &texColorSpace, |
| 342 | nullptr)); |
Robert Phillips | 4f358be | 2017-03-23 08:21:00 -0400 | [diff] [blame] | 343 | if (!proxy) { |
reed | a32cc95 | 2015-08-19 06:07:29 -0700 | [diff] [blame] | 344 | // show placeholder if we have no texture |
| 345 | SkPaint paint; |
| 346 | paint.setStyle(SkPaint::kStroke_Style); |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 347 | SkRect r = SkRect::MakeXYWH(x, y, SkIntToScalar(image->width()), |
| 348 | SkIntToScalar(image->width())); |
reed | a32cc95 | 2015-08-19 06:07:29 -0700 | [diff] [blame] | 349 | canvas->drawRect(r, paint); |
| 350 | canvas->drawLine(r.left(), r.top(), r.right(), r.bottom(), paint); |
| 351 | canvas->drawLine(r.left(), r.bottom(), r.right(), r.top(), paint); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 352 | return; |
| 353 | } |
Robert Phillips | 0ae6faa | 2017-03-21 16:22:00 -0400 | [diff] [blame] | 354 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 355 | // No API to draw a GrTexture directly, so we cheat and create a private image subclass |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 356 | sk_sp<SkImage> texImage(new SkImage_Gpu(canvas->getGrContext(), image->uniqueID(), |
| 357 | kPremul_SkAlphaType, std::move(proxy), |
| 358 | std::move(texColorSpace), SkBudgeted::kNo)); |
| 359 | canvas->drawImage(texImage.get(), x, y); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 360 | #endif |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 361 | } |
| 362 | |
| 363 | void drawSet(SkCanvas* canvas) const { |
| 364 | SkMatrix matrix = SkMatrix::MakeTrans(-100, -100); |
| 365 | canvas->drawPicture(fPicture, &matrix, nullptr); |
| 366 | |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 367 | // Draw the tex first, so it doesn't hit a lucky cache from the raster version. This |
| 368 | // way we also can force the generateTexture call. |
| 369 | |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 370 | draw_as_tex(canvas, fImage.get(), 310, 0); |
| 371 | draw_as_tex(canvas, fImageSubset.get(), 310+101, 0); |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 372 | |
Brian Osman | a28e2b0 | 2017-04-24 16:44:03 -0400 | [diff] [blame] | 373 | draw_as_bitmap(canvas, fImage.get(), 150, 0); |
| 374 | draw_as_bitmap(canvas, fImageSubset.get(), 150+101, 0); |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 375 | } |
| 376 | |
| 377 | void onDraw(SkCanvas* canvas) override { |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 378 | this->makeCaches(canvas->getGrContext()); |
| 379 | |
reed | 8f34372 | 2015-08-13 13:32:39 -0700 | [diff] [blame] | 380 | canvas->translate(20, 20); |
| 381 | |
| 382 | this->drawSet(canvas); |
| 383 | |
| 384 | canvas->save(); |
| 385 | canvas->translate(0, 130); |
| 386 | canvas->scale(0.25f, 0.25f); |
| 387 | this->drawSet(canvas); |
| 388 | canvas->restore(); |
| 389 | |
| 390 | canvas->save(); |
| 391 | canvas->translate(0, 200); |
| 392 | canvas->scale(2, 2); |
| 393 | this->drawSet(canvas); |
| 394 | canvas->restore(); |
| 395 | } |
| 396 | |
| 397 | private: |
| 398 | typedef skiagm::GM INHERITED; |
| 399 | }; |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 400 | DEF_GM( return new ImageCacheratorGM("picture", make_pic_generator); ) |
| 401 | DEF_GM( return new ImageCacheratorGM("raster", make_ras_generator); ) |
reed | c4a83e2 | 2015-09-11 11:47:27 -0700 | [diff] [blame] | 402 | DEF_GM( return new ImageCacheratorGM("ctable", make_ctable_generator); ) |
reed | 935d6cf | 2015-08-18 11:16:09 -0700 | [diff] [blame] | 403 | #if SK_SUPPORT_GPU |
| 404 | DEF_GM( return new ImageCacheratorGM("texture", make_tex_generator); ) |
| 405 | #endif |