tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +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" |
| 9 | #include "SkCanvas.h" |
| 10 | #include "SkPath.h" |
| 11 | |
reed | 02f9ed7 | 2016-09-06 09:06:18 -0700 | [diff] [blame] | 12 | static void do_draw(SkCanvas* canvas, const SkRect& r) { |
| 13 | SkPaint paint; |
| 14 | paint.setXfermodeMode(SkXfermode::kSrc_Mode); |
| 15 | paint.setColor(0x800000FF); |
| 16 | canvas->drawRect(r, paint); |
| 17 | } |
| 18 | |
| 19 | /** |
| 20 | * Exercise kDontClipToLayer_Legacy_SaveLayerFlag flag, which does not limit the clip to the |
| 21 | * layer's bounds. Thus when a draw occurs, it can (depending on "where" it is) draw into the layer |
| 22 | * and/or draw onto the surrounding portions of the canvas, or both. |
| 23 | * |
| 24 | * This GM has a 100x100 rectangle (r), which its going to draw. However first it creates a layer |
| 25 | * with this flag covering 1/2 of the rectangle (upper half). Then it draws the rect in SRC mode. |
| 26 | * |
| 27 | * The portion of the draw that intersects the layer should see the SRC draw, apply it to the layer |
| 28 | * and then during restore, it will SRC_OVER that layer onto the canvas (SRC_OVER since the layer |
| 29 | * has no paint, so it gets the default xfermode during restore). |
| 30 | * |
| 31 | * The portion of the draw below the layer draws directly into the canvas. Since it is in SRC mode, |
| 32 | * it will wrote 0x80 to the canvas' alpha, making it appear darker when shown in the window. |
| 33 | * The portion in the layer, will end up SRC_OVERing the 0x80 layer pixels onto the canvas, so |
| 34 | * they will appear lighter (since the canvas was erased to white initially). |
| 35 | * |
| 36 | * Thus the expected result is the upper half to be light-blue w/ 0xFF for its alpha, and |
| 37 | * the lower half to be darker blue with 0x80 for its alpha. |
| 38 | */ |
| 39 | DEF_SIMPLE_GM(dont_clip_to_layer, canvas, 120, 120) { |
| 40 | SkRect r { 10, 10, 110, 110 }; |
| 41 | SkRect r0 = SkRect::MakeXYWH(r.left(), r.top(), r.width(), r.height()/2); |
| 42 | |
| 43 | SkCanvas::SaveLayerRec rec; |
| 44 | rec.fPaint = nullptr; |
| 45 | rec.fBounds = &r0; |
| 46 | rec.fBackdrop = nullptr; |
| 47 | rec.fSaveLayerFlags = 1 << 31;//SkCanvas::kDontClipToLayer_Legacy_SaveLayerFlag; |
| 48 | canvas->saveLayer(rec); |
| 49 | do_draw(canvas, r); |
| 50 | canvas->restore(); |
| 51 | } |
| 52 | |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 53 | /** Draw a 2px border around the target, then red behind the target; |
| 54 | set the clip to match the target, then draw >> the target in blue. |
| 55 | */ |
| 56 | |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 57 | static void draw(SkCanvas* canvas, SkRect& target, int x, int y) { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 58 | SkPaint borderPaint; |
| 59 | borderPaint.setColor(SkColorSetRGB(0x0, 0xDD, 0x0)); |
| 60 | borderPaint.setAntiAlias(true); |
| 61 | SkPaint backgroundPaint; |
| 62 | backgroundPaint.setColor(SkColorSetRGB(0xDD, 0x0, 0x0)); |
| 63 | backgroundPaint.setAntiAlias(true); |
| 64 | SkPaint foregroundPaint; |
| 65 | foregroundPaint.setColor(SkColorSetRGB(0x0, 0x0, 0xDD)); |
| 66 | foregroundPaint.setAntiAlias(true); |
| 67 | |
| 68 | canvas->save(); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 69 | canvas->translate(SkIntToScalar(x), SkIntToScalar(y)); |
| 70 | target.inset(SkIntToScalar(-2), SkIntToScalar(-2)); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 71 | canvas->drawRect(target, borderPaint); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 72 | target.inset(SkIntToScalar(2), SkIntToScalar(2)); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 73 | canvas->drawRect(target, backgroundPaint); |
| 74 | canvas->clipRect(target, SkRegion::kIntersect_Op, true); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 75 | target.inset(SkIntToScalar(-4), SkIntToScalar(-4)); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 76 | canvas->drawRect(target, foregroundPaint); |
| 77 | canvas->restore(); |
| 78 | } |
| 79 | |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 80 | static void draw_square(SkCanvas* canvas, int x, int y) { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 81 | SkRect target (SkRect::MakeWH(10 * SK_Scalar1, 10 * SK_Scalar1)); |
| 82 | draw(canvas, target, x, y); |
| 83 | } |
| 84 | |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 85 | static void draw_column(SkCanvas* canvas, int x, int y) { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 86 | SkRect target (SkRect::MakeWH(1 * SK_Scalar1, 10 * SK_Scalar1)); |
| 87 | draw(canvas, target, x, y); |
| 88 | } |
| 89 | |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 90 | static void draw_bar(SkCanvas* canvas, int x, int y) { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 91 | SkRect target (SkRect::MakeWH(10 * SK_Scalar1, 1 * SK_Scalar1)); |
| 92 | draw(canvas, target, x, y); |
| 93 | } |
| 94 | |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 95 | static void draw_rect_tests(SkCanvas* canvas) { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 96 | draw_square(canvas, 10, 10); |
| 97 | draw_column(canvas, 30, 10); |
| 98 | draw_bar(canvas, 10, 30); |
| 99 | } |
| 100 | |
| 101 | /** |
| 102 | Test a set of clipping problems discovered while writing blitAntiRect, |
| 103 | and test all the code paths through the clipping blitters. |
| 104 | Each region should show as a blue center surrounded by a 2px green |
| 105 | border, with no red. |
| 106 | */ |
| 107 | |
reed@google.com | ff26b7e | 2013-05-23 17:04:19 +0000 | [diff] [blame] | 108 | class AAClipGM : public skiagm::GM { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 109 | public: |
| 110 | AAClipGM() { |
reed@google.com | 7112173 | 2012-09-18 15:14:33 +0000 | [diff] [blame] | 111 | |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 112 | } |
| 113 | |
| 114 | protected: |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 115 | SkString onShortName() override { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 116 | return SkString("aaclip"); |
| 117 | } |
| 118 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 119 | SkISize onISize() override { |
commit-bot@chromium.org | 5d0b150 | 2014-04-14 15:02:19 +0000 | [diff] [blame] | 120 | return SkISize::Make(240, 120); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 121 | } |
| 122 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 123 | void onDraw(SkCanvas* canvas) override { |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 124 | // Initial pixel-boundary-aligned draw |
| 125 | draw_rect_tests(canvas); |
| 126 | |
| 127 | // Repeat 4x with .2, .4, .6, .8 px offsets |
| 128 | canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 129 | canvas->translate(SkIntToScalar(50), 0); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 130 | draw_rect_tests(canvas); |
| 131 | |
| 132 | canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 133 | canvas->translate(SkIntToScalar(50), 0); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 134 | draw_rect_tests(canvas); |
| 135 | |
| 136 | canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 137 | canvas->translate(SkIntToScalar(50), 0); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 138 | draw_rect_tests(canvas); |
| 139 | |
| 140 | canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); |
tomhudson@google.com | abfa8d1 | 2011-12-28 19:40:48 +0000 | [diff] [blame] | 141 | canvas->translate(SkIntToScalar(50), 0); |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 142 | draw_rect_tests(canvas); |
| 143 | } |
| 144 | |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 145 | private: |
reed@google.com | ff26b7e | 2013-05-23 17:04:19 +0000 | [diff] [blame] | 146 | typedef skiagm::GM INHERITED; |
tomhudson@google.com | ef279d3 | 2011-12-21 14:27:14 +0000 | [diff] [blame] | 147 | }; |
| 148 | |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 149 | DEF_GM(return new AAClipGM;) |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 150 | |
| 151 | ///////////////////////////////////////////////////////////////////////// |
| 152 | |
| 153 | #ifdef SK_BUILD_FOR_MAC |
| 154 | |
| 155 | static SkCanvas* make_canvas(const SkBitmap& bm) { |
| 156 | const SkImageInfo& info = bm.info(); |
| 157 | if (info.bytesPerPixel() == 4) { |
| 158 | return SkCanvas::NewRasterDirectN32(info.width(), info.height(), |
| 159 | (SkPMColor*)bm.getPixels(), |
| 160 | bm.rowBytes()); |
| 161 | } else { |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 162 | return new SkCanvas(bm); |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 163 | } |
| 164 | } |
| 165 | |
| 166 | #include "SkCGUtils.h" |
| 167 | static void test_image(SkCanvas* canvas, const SkImageInfo& info) { |
| 168 | SkBitmap bm; |
| 169 | bm.allocPixels(info); |
| 170 | |
| 171 | SkAutoTUnref<SkCanvas> newc(make_canvas(bm)); |
| 172 | if (info.isOpaque()) { |
| 173 | bm.eraseColor(SK_ColorGREEN); |
| 174 | } else { |
| 175 | bm.eraseColor(0); |
| 176 | } |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 177 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 178 | SkPaint paint; |
| 179 | paint.setAntiAlias(true); |
| 180 | paint.setColor(SK_ColorBLUE); |
| 181 | newc->drawCircle(50, 50, 49, paint); |
| 182 | canvas->drawBitmap(bm, 10, 10); |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 183 | |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 184 | CGImageRef image = SkCreateCGImageRefWithColorspace(bm, nullptr); |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 185 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 186 | SkBitmap bm2; |
| 187 | SkCreateBitmapFromCGImage(&bm2, image); |
| 188 | CGImageRelease(image); |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 189 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 190 | canvas->drawBitmap(bm2, 10, 120); |
| 191 | } |
| 192 | |
| 193 | class CGImageGM : public skiagm::GM { |
| 194 | public: |
| 195 | CGImageGM() {} |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 196 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 197 | protected: |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 198 | SkString onShortName() override { |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 199 | return SkString("cgimage"); |
| 200 | } |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 201 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 202 | SkISize onISize() override { |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 203 | return SkISize::Make(800, 250); |
| 204 | } |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 205 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 206 | void onDraw(SkCanvas* canvas) override { |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 207 | const struct { |
| 208 | SkColorType fCT; |
| 209 | SkAlphaType fAT; |
| 210 | } rec[] = { |
| 211 | { kRGB_565_SkColorType, kOpaque_SkAlphaType }, |
| 212 | |
| 213 | { kRGBA_8888_SkColorType, kPremul_SkAlphaType }, |
| 214 | { kRGBA_8888_SkColorType, kUnpremul_SkAlphaType }, |
| 215 | { kRGBA_8888_SkColorType, kOpaque_SkAlphaType }, |
| 216 | |
| 217 | { kBGRA_8888_SkColorType, kPremul_SkAlphaType }, |
| 218 | { kBGRA_8888_SkColorType, kUnpremul_SkAlphaType }, |
| 219 | { kBGRA_8888_SkColorType, kOpaque_SkAlphaType }, |
| 220 | }; |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 221 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 222 | for (size_t i = 0; i < SK_ARRAY_COUNT(rec); ++i) { |
| 223 | SkImageInfo info = SkImageInfo::Make(100, 100, rec[i].fCT, rec[i].fAT); |
| 224 | test_image(canvas, info); |
| 225 | canvas->translate(info.width() + 10, 0); |
| 226 | } |
| 227 | } |
skia.committer@gmail.com | ede0c5c | 2014-04-23 03:04:11 +0000 | [diff] [blame] | 228 | |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 229 | private: |
| 230 | typedef skiagm::GM INHERITED; |
| 231 | }; |
| 232 | |
bsalomon@google.com | 2a9e3ad | 2014-04-23 18:05:36 +0000 | [diff] [blame] | 233 | #if 0 // Disabled pending fix from reed@ |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 234 | DEF_GM( return new CGImageGM; ) |
bsalomon@google.com | 2a9e3ad | 2014-04-23 18:05:36 +0000 | [diff] [blame] | 235 | #endif |
commit-bot@chromium.org | 60b5dce | 2014-04-22 20:24:33 +0000 | [diff] [blame] | 236 | #endif |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 237 | |
| 238 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 239 | |
halcanary | 6950de6 | 2015-11-07 05:29:00 -0800 | [diff] [blame] | 240 | // https://bug.skia.org/3716 |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 241 | class ClipCubicGM : public skiagm::GM { |
| 242 | const SkScalar W = 100; |
| 243 | const SkScalar H = 240; |
| 244 | |
| 245 | SkPath fVPath, fHPath; |
| 246 | public: |
| 247 | ClipCubicGM() { |
| 248 | fVPath.moveTo(W, 0); |
| 249 | fVPath.cubicTo(W, H-10, 0, 10, 0, H); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 250 | |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 251 | SkMatrix pivot; |
| 252 | pivot.setRotate(90, W/2, H/2); |
| 253 | fVPath.transform(pivot, &fHPath); |
| 254 | } |
| 255 | |
| 256 | protected: |
| 257 | SkString onShortName() override { |
| 258 | return SkString("clipcubic"); |
| 259 | } |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 260 | |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 261 | SkISize onISize() override { |
| 262 | return SkISize::Make(400, 410); |
| 263 | } |
| 264 | |
| 265 | void doDraw(SkCanvas* canvas, const SkPath& path) { |
| 266 | SkPaint paint; |
| 267 | paint.setAntiAlias(true); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 268 | |
caryclark | 1259601 | 2015-07-29 05:27:47 -0700 | [diff] [blame] | 269 | paint.setColor(sk_tool_utils::color_to_565(0xFFCCCCCC)); |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 270 | canvas->drawPath(path, paint); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 271 | |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 272 | paint.setColor(SK_ColorRED); |
| 273 | paint.setStyle(SkPaint::kStroke_Style); |
| 274 | canvas->drawPath(path, paint); |
| 275 | } |
| 276 | |
| 277 | void drawAndClip(SkCanvas* canvas, const SkPath& path, SkScalar dx, SkScalar dy) { |
| 278 | SkAutoCanvasRestore acr(canvas, true); |
| 279 | |
| 280 | SkRect r = SkRect::MakeXYWH(0, H/4, W, H/2); |
| 281 | SkPaint paint; |
caryclark | 1259601 | 2015-07-29 05:27:47 -0700 | [diff] [blame] | 282 | paint.setColor(sk_tool_utils::color_to_565(0xFF8888FF)); |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 283 | |
| 284 | canvas->drawRect(r, paint); |
| 285 | this->doDraw(canvas, path); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 286 | |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 287 | canvas->translate(dx, dy); |
| 288 | |
| 289 | canvas->drawRect(r, paint); |
| 290 | canvas->clipRect(r); |
| 291 | this->doDraw(canvas, path); |
| 292 | } |
| 293 | |
| 294 | void onDraw(SkCanvas* canvas) override { |
| 295 | canvas->translate(80, 10); |
| 296 | this->drawAndClip(canvas, fVPath, 200, 0); |
| 297 | canvas->translate(0, 200); |
| 298 | this->drawAndClip(canvas, fHPath, 200, 0); |
| 299 | } |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 300 | |
reed | 8f76cb9 | 2015-04-22 17:38:23 -0700 | [diff] [blame] | 301 | private: |
| 302 | typedef skiagm::GM INHERITED; |
| 303 | }; |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 304 | DEF_GM(return new ClipCubicGM;) |