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