bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +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 | */ |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 7 | |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 8 | #include "gm.h" |
Mike Klein | 33d2055 | 2017-03-22 13:47:51 -0400 | [diff] [blame] | 9 | #include "sk_tool_utils.h" |
robertphillips@google.com | b706117 | 2013-09-06 14:16:12 +0000 | [diff] [blame] | 10 | #include "SkBlurMask.h" |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 11 | #include "SkBlurMaskFilter.h" |
robertphillips@google.com | b706117 | 2013-09-06 14:16:12 +0000 | [diff] [blame] | 12 | #include "SkColorPriv.h" |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 13 | #include "SkGradientShader.h" |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 14 | #include "SkImage.h" |
bsalomon | 84a4e5a | 2016-02-29 11:41:52 -0800 | [diff] [blame] | 15 | #include "SkImage_Base.h" |
halcanary | 4dbbd04 | 2016-06-07 17:21:10 -0700 | [diff] [blame] | 16 | #include "SkMathPriv.h" |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 17 | #include "SkShader.h" |
| 18 | #include "SkSurface.h" |
| 19 | |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 20 | |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 21 | static SkBitmap make_chessbm(int w, int h) { |
| 22 | SkBitmap bm; |
reed@google.com | eb9a46c | 2014-01-25 16:46:20 +0000 | [diff] [blame] | 23 | bm.allocN32Pixels(w, h); |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 24 | |
| 25 | for (int y = 0; y < bm.height(); y++) { |
| 26 | uint32_t* p = bm.getAddr32(0, y); |
| 27 | for (int x = 0; x < bm.width(); x++) { |
| 28 | p[x] = ((x + y) & 1) ? SK_ColorWHITE : SK_ColorBLACK; |
| 29 | } |
| 30 | } |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 31 | return bm; |
| 32 | } |
| 33 | |
bsalomon | 7cf36cc | 2016-07-14 09:33:42 -0700 | [diff] [blame] | 34 | // Creates a bitmap and a matching image. |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 35 | static sk_sp<SkImage> makebm(SkCanvas* origCanvas, SkBitmap* resultBM, int w, int h) { |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 36 | SkImageInfo info = SkImageInfo::MakeN32Premul(w, h); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 37 | |
Mike Reed | 46596ae | 2018-01-02 15:40:29 -0500 | [diff] [blame] | 38 | auto surface(sk_tool_utils::makeSurface(origCanvas, info)); |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 39 | SkCanvas* canvas = surface->getCanvas(); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 40 | |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 41 | canvas->clear(SK_ColorTRANSPARENT); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 42 | |
| 43 | SkScalar wScalar = SkIntToScalar(w); |
| 44 | SkScalar hScalar = SkIntToScalar(h); |
| 45 | |
| 46 | SkPoint pt = { wScalar / 2, hScalar / 2 }; |
| 47 | |
| 48 | SkScalar radius = 4 * SkMaxScalar(wScalar, hScalar); |
| 49 | |
| 50 | SkColor colors[] = { SK_ColorRED, SK_ColorYELLOW, |
| 51 | SK_ColorGREEN, SK_ColorMAGENTA, |
| 52 | SK_ColorBLUE, SK_ColorCYAN, |
| 53 | SK_ColorRED}; |
| 54 | |
| 55 | SkScalar pos[] = {0, |
| 56 | SK_Scalar1 / 6, |
| 57 | 2 * SK_Scalar1 / 6, |
| 58 | 3 * SK_Scalar1 / 6, |
| 59 | 4 * SK_Scalar1 / 6, |
| 60 | 5 * SK_Scalar1 / 6, |
| 61 | SK_Scalar1}; |
| 62 | |
| 63 | SkPaint paint; |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 64 | SkRect rect = SkRect::MakeWH(wScalar, hScalar); |
| 65 | SkMatrix mat = SkMatrix::I(); |
| 66 | for (int i = 0; i < 4; ++i) { |
reed | 2ad1aa6 | 2016-03-09 09:50:50 -0800 | [diff] [blame] | 67 | paint.setShader(SkGradientShader::MakeRadial( |
commit-bot@chromium.org | 9c9005a | 2014-04-28 14:55:39 +0000 | [diff] [blame] | 68 | pt, radius, |
| 69 | colors, pos, |
| 70 | SK_ARRAY_COUNT(colors), |
| 71 | SkShader::kRepeat_TileMode, |
reed | 2ad1aa6 | 2016-03-09 09:50:50 -0800 | [diff] [blame] | 72 | 0, &mat)); |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 73 | canvas->drawRect(rect, paint); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 74 | rect.inset(wScalar / 8, hScalar / 8); |
| 75 | mat.postScale(SK_Scalar1 / 4, SK_Scalar1 / 4); |
| 76 | } |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 77 | |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 78 | auto image = surface->makeImageSnapshot(); |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 79 | |
| 80 | SkBitmap tempBM; |
| 81 | |
Cary Clark | 4f5a79c | 2018-02-07 15:51:00 -0500 | [diff] [blame] | 82 | image->asLegacyBitmap(&tempBM); |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 83 | |
| 84 | // Let backends know we won't change this, so they don't have to deep copy it defensively. |
| 85 | tempBM.setImmutable(); |
| 86 | *resultBM = tempBM; |
| 87 | |
| 88 | return image; |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 89 | } |
| 90 | |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 91 | static void bitmapproc(SkCanvas* canvas, SkImage*, const SkBitmap& bm, const SkIRect& srcR, |
| 92 | const SkRect& dstR, const SkPaint* paint) { |
| 93 | canvas->drawBitmapRect(bm, srcR, dstR, paint); |
| 94 | } |
| 95 | |
| 96 | static void bitmapsubsetproc(SkCanvas* canvas, SkImage*, const SkBitmap& bm, const SkIRect& srcR, |
| 97 | const SkRect& dstR, const SkPaint* paint) { |
| 98 | if (!bm.bounds().contains(srcR)) { |
| 99 | bitmapproc(canvas, nullptr, bm, srcR, dstR, paint); |
| 100 | return; |
| 101 | } |
| 102 | |
| 103 | SkBitmap subset; |
| 104 | if (bm.extractSubset(&subset, srcR)) { |
| 105 | canvas->drawBitmapRect(subset, dstR, paint); |
| 106 | } |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 107 | } |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 108 | |
reed | 84984ef | 2015-07-17 07:09:43 -0700 | [diff] [blame] | 109 | static void imageproc(SkCanvas* canvas, SkImage* image, const SkBitmap&, const SkIRect& srcR, |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 110 | const SkRect& dstR, const SkPaint* paint) { |
| 111 | canvas->drawImageRect(image, srcR, dstR, paint); |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 112 | } |
| 113 | |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 114 | static void imagesubsetproc(SkCanvas* canvas, SkImage* image, const SkBitmap& bm, |
| 115 | const SkIRect& srcR, const SkRect& dstR, const SkPaint* paint) { |
| 116 | if (!image->bounds().contains(srcR)) { |
| 117 | imageproc(canvas, image, bm, srcR, dstR, paint); |
| 118 | return; |
| 119 | } |
| 120 | |
| 121 | if (sk_sp<SkImage> subset = image->makeSubset(srcR)) { |
| 122 | canvas->drawImageRect(subset, dstR, paint); |
| 123 | } |
| 124 | } |
| 125 | |
| 126 | typedef void DrawRectRectProc(SkCanvas*, SkImage*, const SkBitmap&, const SkIRect&, const SkRect&, |
| 127 | const SkPaint*); |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 128 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 129 | constexpr int gSize = 1024; |
| 130 | constexpr int gBmpSize = 2048; |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 131 | |
| 132 | class DrawBitmapRectGM : public skiagm::GM { |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 133 | public: |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 134 | DrawBitmapRectGM(DrawRectRectProc proc, const char suffix[]) : fProc(proc) { |
| 135 | fName.set("drawbitmaprect"); |
| 136 | if (suffix) { |
| 137 | fName.append(suffix); |
| 138 | } |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 139 | } |
| 140 | |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 141 | DrawRectRectProc* fProc; |
| 142 | SkBitmap fLargeBitmap; |
| 143 | sk_sp<SkImage> fImage; |
| 144 | SkString fName; |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 145 | |
| 146 | protected: |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 147 | SkString onShortName() override { return fName; } |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 148 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 149 | SkISize onISize() override { return SkISize::Make(gSize, gSize); } |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 150 | |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 151 | void setupImage(SkCanvas* canvas) { |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 152 | fImage = makebm(canvas, &fLargeBitmap, gBmpSize, gBmpSize); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 153 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 154 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 155 | void onDraw(SkCanvas* canvas) override { |
Brian Salomon | 5426850 | 2018-05-24 13:22:01 -0400 | [diff] [blame] | 156 | if (!fImage || !fImage->isValid(canvas->getGrContext())) { |
robertphillips | 24eb7a8 | 2015-09-24 08:47:49 -0700 | [diff] [blame] | 157 | this->setupImage(canvas); |
| 158 | } |
| 159 | |
vandebo@chromium.org | 663515b | 2012-01-05 18:45:27 +0000 | [diff] [blame] | 160 | SkRect dstRect = { 0, 0, SkIntToScalar(64), SkIntToScalar(64)}; |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 161 | const int kMaxSrcRectSize = 1 << (SkNextLog2(gBmpSize) + 2); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 162 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 163 | const int kPadX = 30; |
| 164 | const int kPadY = 40; |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 165 | SkPaint paint; |
| 166 | paint.setAlpha(0x20); |
bsalomon | 7cf36cc | 2016-07-14 09:33:42 -0700 | [diff] [blame] | 167 | canvas->drawImageRect(fImage, SkRect::MakeIWH(gSize, gSize), &paint); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 168 | canvas->translate(SK_Scalar1 * kPadX / 2, |
| 169 | SK_Scalar1 * kPadY / 2); |
| 170 | SkPaint blackPaint; |
| 171 | SkScalar titleHeight = SK_Scalar1 * 24; |
| 172 | blackPaint.setColor(SK_ColorBLACK); |
| 173 | blackPaint.setTextSize(titleHeight); |
| 174 | blackPaint.setAntiAlias(true); |
caryclark | 1818acb | 2015-07-24 12:09:25 -0700 | [diff] [blame] | 175 | sk_tool_utils::set_portable_typeface(&blackPaint); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 176 | SkString title; |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 177 | title.printf("Bitmap size: %d x %d", gBmpSize, gBmpSize); |
Cary Clark | 2a475ea | 2017-04-28 15:35:12 -0400 | [diff] [blame] | 178 | canvas->drawString(title, 0, |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 179 | titleHeight, blackPaint); |
| 180 | |
| 181 | canvas->translate(0, SK_Scalar1 * kPadY / 2 + titleHeight); |
| 182 | int rowCount = 0; |
| 183 | canvas->save(); |
| 184 | for (int w = 1; w <= kMaxSrcRectSize; w *= 4) { |
| 185 | for (int h = 1; h <= kMaxSrcRectSize; h *= 4) { |
| 186 | |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 187 | SkIRect srcRect = SkIRect::MakeXYWH((gBmpSize - w) / 2, (gBmpSize - h) / 2, w, h); |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 188 | fProc(canvas, fImage.get(), fLargeBitmap, srcRect, dstRect, nullptr); |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 189 | |
| 190 | SkString label; |
| 191 | label.appendf("%d x %d", w, h); |
| 192 | blackPaint.setAntiAlias(true); |
| 193 | blackPaint.setStyle(SkPaint::kFill_Style); |
| 194 | blackPaint.setTextSize(SK_Scalar1 * 10); |
| 195 | SkScalar baseline = dstRect.height() + |
| 196 | blackPaint.getTextSize() + SK_Scalar1 * 3; |
Cary Clark | 2a475ea | 2017-04-28 15:35:12 -0400 | [diff] [blame] | 197 | canvas->drawString(label, |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 198 | 0, baseline, |
| 199 | blackPaint); |
| 200 | blackPaint.setStyle(SkPaint::kStroke_Style); |
| 201 | blackPaint.setStrokeWidth(SK_Scalar1); |
| 202 | blackPaint.setAntiAlias(false); |
| 203 | canvas->drawRect(dstRect, blackPaint); |
| 204 | |
| 205 | canvas->translate(dstRect.width() + SK_Scalar1 * kPadX, 0); |
| 206 | ++rowCount; |
| 207 | if ((dstRect.width() + kPadX) * rowCount > gSize) { |
| 208 | canvas->restore(); |
| 209 | canvas->translate(0, dstRect.height() + SK_Scalar1 * kPadY); |
| 210 | canvas->save(); |
| 211 | rowCount = 0; |
| 212 | } |
| 213 | } |
| 214 | } |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 215 | |
| 216 | { |
| 217 | // test the following code path: |
| 218 | // SkGpuDevice::drawPath() -> SkGpuDevice::drawWithMaskFilter() |
| 219 | SkIRect srcRect; |
| 220 | SkPaint paint; |
| 221 | SkBitmap bm; |
| 222 | |
| 223 | bm = make_chessbm(5, 5); |
reed | 93a1215 | 2015-03-16 10:08:34 -0700 | [diff] [blame] | 224 | paint.setFilterQuality(kLow_SkFilterQuality); |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 225 | |
| 226 | srcRect.setXYWH(1, 1, 3, 3); |
Mike Reed | 1be1f8d | 2018-03-14 13:01:17 -0400 | [diff] [blame] | 227 | paint.setMaskFilter(SkMaskFilter::MakeBlur( |
commit-bot@chromium.org | e396455 | 2014-04-28 16:25:35 +0000 | [diff] [blame] | 228 | kNormal_SkBlurStyle, |
Mike Reed | 1be1f8d | 2018-03-14 13:01:17 -0400 | [diff] [blame] | 229 | SkBlurMask::ConvertRadiusToSigma(SkIntToScalar(5)))); |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 230 | |
| 231 | sk_sp<SkImage> image(SkImage::MakeFromBitmap(bm)); |
| 232 | fProc(canvas, image.get(), bm, srcRect, dstRect, &paint); |
bsalomon@google.com | 7d30a21 | 2012-04-25 15:52:27 +0000 | [diff] [blame] | 233 | } |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 234 | } |
| 235 | |
| 236 | private: |
reed | f803da1 | 2015-01-23 05:58:07 -0800 | [diff] [blame] | 237 | typedef skiagm::GM INHERITED; |
bsalomon@google.com | fb30951 | 2011-11-30 14:13:48 +0000 | [diff] [blame] | 238 | }; |
| 239 | |
fmalita | ab83da7 | 2016-08-26 13:04:14 -0700 | [diff] [blame] | 240 | DEF_GM( return new DrawBitmapRectGM(bitmapproc , nullptr); ) |
| 241 | DEF_GM( return new DrawBitmapRectGM(bitmapsubsetproc, "-subset"); ) |
| 242 | DEF_GM( return new DrawBitmapRectGM(imageproc , "-imagerect"); ) |
| 243 | DEF_GM( return new DrawBitmapRectGM(imagesubsetproc , "-imagerect-subset"); ) |