reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2013 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 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "gm/gm.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 9 | #include "include/core/SkBitmap.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 10 | #include "include/core/SkCanvas.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 11 | #include "include/core/SkColor.h" |
| 12 | #include "include/core/SkImageFilter.h" |
Brian Osman | 01d9566 | 2024-02-28 11:54:05 -0500 | [diff] [blame] | 13 | #include "include/core/SkMatrix.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 14 | #include "include/core/SkPaint.h" |
| 15 | #include "include/core/SkRect.h" |
| 16 | #include "include/core/SkRefCnt.h" |
Brian Osman | 01d9566 | 2024-02-28 11:54:05 -0500 | [diff] [blame] | 17 | #include "include/core/SkSamplingOptions.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 18 | #include "include/core/SkScalar.h" |
| 19 | #include "include/core/SkSize.h" |
| 20 | #include "include/core/SkString.h" |
Michael Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 21 | #include "include/effects/SkImageFilters.h" |
Brian Osman | 01d9566 | 2024-02-28 11:54:05 -0500 | [diff] [blame] | 22 | #include "tools/ToolUtils.h" |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 23 | |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 24 | #include <utility> |
| 25 | |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 26 | static void make_bm(SkBitmap* bm) { |
reed@google.com | eb9a46c | 2014-01-25 16:46:20 +0000 | [diff] [blame] | 27 | bm->allocN32Pixels(100, 100); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 28 | bm->eraseColor(SK_ColorBLUE); |
| 29 | |
| 30 | SkCanvas canvas(*bm); |
| 31 | SkPaint paint; |
| 32 | paint.setAntiAlias(true); |
| 33 | paint.setColor(SK_ColorRED); |
| 34 | canvas.drawCircle(50, 50, 50, paint); |
| 35 | } |
| 36 | |
reed | da420b9 | 2015-12-16 08:38:15 -0800 | [diff] [blame] | 37 | static void draw_1_bitmap(SkCanvas* canvas, const SkBitmap& bm, bool doClip, |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 38 | int dx, int dy, sk_sp<SkImageFilter> filter) { |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 39 | SkAutoCanvasRestore acr(canvas, true); |
| 40 | SkPaint paint; |
| 41 | |
skia.committer@gmail.com | 4d28d98 | 2013-01-17 07:06:06 +0000 | [diff] [blame] | 42 | SkRect clipR = SkRect::MakeXYWH(SkIntToScalar(dx), |
robertphillips@google.com | 914a2f2 | 2013-01-16 19:57:02 +0000 | [diff] [blame] | 43 | SkIntToScalar(dy), |
reed@google.com | 1c711ca | 2013-01-16 19:24:15 +0000 | [diff] [blame] | 44 | SkIntToScalar(bm.width()), |
| 45 | SkIntToScalar(bm.height())); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 46 | |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 47 | paint.setImageFilter(std::move(filter)); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 48 | clipR.inset(5, 5); |
| 49 | |
reed@google.com | 1c711ca | 2013-01-16 19:24:15 +0000 | [diff] [blame] | 50 | canvas->translate(SkIntToScalar(bm.width() + 20), 0); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 51 | |
| 52 | if (doClip) { |
| 53 | canvas->save(); |
| 54 | canvas->clipRect(clipR); |
| 55 | } |
Mike Reed | 568f0ae | 2021-01-24 08:57:23 -0500 | [diff] [blame] | 56 | canvas->drawImage(bm.asImage(), SkIntToScalar(dx), SkIntToScalar(dy), |
| 57 | SkSamplingOptions(), &paint); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 58 | if (doClip) { |
| 59 | canvas->restore(); |
| 60 | } |
| 61 | } |
| 62 | |
| 63 | /** |
| 64 | * Compare output of drawSprite and drawBitmap (esp. clipping and imagefilters) |
| 65 | */ |
| 66 | class SpriteBitmapGM : public skiagm::GM { |
| 67 | public: |
| 68 | SpriteBitmapGM() {} |
| 69 | |
| 70 | protected: |
Leandro Lovisolo | 24fa211 | 2023-08-15 19:05:17 +0000 | [diff] [blame] | 71 | SkString getName() const override { return SkString("spritebitmap"); } |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 72 | |
Leandro Lovisolo | 8f02388 | 2023-08-15 21:13:52 +0000 | [diff] [blame] | 73 | SkISize getISize() override { return SkISize::Make(640, 480); } |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 74 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 75 | void onDraw(SkCanvas* canvas) override { |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 76 | SkBitmap bm; |
| 77 | make_bm(&bm); |
| 78 | |
| 79 | int dx = 10; |
| 80 | int dy = 10; |
| 81 | |
| 82 | SkScalar sigma = 8; |
Michael Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 83 | sk_sp<SkImageFilter> filter(SkImageFilters::Blur(sigma, sigma, nullptr)); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 84 | |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 85 | draw_1_bitmap(canvas, bm, false, dx, dy, nullptr); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 86 | dy += bm.height() + 20; |
reed | da420b9 | 2015-12-16 08:38:15 -0800 | [diff] [blame] | 87 | draw_1_bitmap(canvas, bm, false, dx, dy, filter); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 88 | dy += bm.height() + 20; |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 89 | draw_1_bitmap(canvas, bm, true, dx, dy, nullptr); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 90 | dy += bm.height() + 20; |
reed | da420b9 | 2015-12-16 08:38:15 -0800 | [diff] [blame] | 91 | draw_1_bitmap(canvas, bm, true, dx, dy, filter); |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 92 | } |
| 93 | |
| 94 | private: |
John Stiles | 7571f9e | 2020-09-02 22:42:33 -0400 | [diff] [blame] | 95 | using INHERITED = GM; |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 96 | }; |
reed@google.com | 12fa9ba | 2013-01-16 18:54:15 +0000 | [diff] [blame] | 97 | DEF_GM( return new SpriteBitmapGM; ) |
Brian Osman | 01d9566 | 2024-02-28 11:54:05 -0500 | [diff] [blame] | 98 | |
| 99 | // b/41322892 : The CPU backend tries to detect when an image draw is landing perfectly on pixel |
| 100 | // centers, so it can use a faster sprite-blitting path. That code just assumes that ANY translation |
| 101 | // can be pixel-snapped. Image-shaders used to behave like this, but were fixed long-ago. |
| 102 | // The correct result here is for ALL rectangles to be averaged to grey, rather than a pixel-snapped |
| 103 | // black and white checkerboard. |
| 104 | // |
| 105 | // This GM now tests that linear filtering is preserved for subpixel translation, across several |
| 106 | // different possible methods of drawing. |
| 107 | DEF_SIMPLE_GM_BG(drawimagerect_filter, canvas, 180, 60, SK_ColorWHITE) { |
| 108 | auto image = ToolUtils::create_checkerboard_image(50, 50, SK_ColorWHITE, SK_ColorBLACK, 1); |
| 109 | SkSamplingOptions sampling{SkFilterMode::kLinear}; |
| 110 | |
| 111 | canvas->translate(5, 5); |
| 112 | canvas->drawImage(image, 0.5f, 0.5f, sampling); |
| 113 | |
| 114 | canvas->translate(60, 0); |
| 115 | canvas->drawImageRect(image, {0.5f, 0.5f, 50.5f, 50.5f}, sampling); |
| 116 | |
| 117 | auto shader = image->makeShader(sampling); |
| 118 | canvas->translate(60, 0); |
| 119 | SkPaint paint; |
| 120 | SkMatrix offset = SkMatrix::Translate(0.5f, 0.5f); |
| 121 | paint.setShader(image->makeShader(sampling, &offset)); |
| 122 | canvas->drawRect({0.0f, 0.0f, 50.0f, 50.0f}, paint); |
| 123 | } |