reed | 1a9caff | 2015-09-02 19:05:10 -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 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "include/core/SkCanvas.h" |
| 9 | #include "include/core/SkDrawable.h" |
| 10 | #include "include/core/SkPath.h" |
| 11 | #include "include/core/SkRSXform.h" |
| 12 | #include "include/core/SkString.h" |
| 13 | #include "include/core/SkSurface.h" |
| 14 | #include "include/effects/SkGradientShader.h" |
| 15 | #include "include/utils/SkRandom.h" |
| 16 | #include "include/utils/SkTextUtils.h" |
| 17 | #include "samplecode/Sample.h" |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 18 | |
Brian Osman | d1e67e7 | 2017-03-15 12:19:37 -0400 | [diff] [blame] | 19 | const SkBlendMode gModes[] = { |
| 20 | SkBlendMode::kSrcOver, |
| 21 | SkBlendMode::kSrc, |
| 22 | SkBlendMode::kSrcIn, |
| 23 | SkBlendMode::kSrcOut, |
| 24 | SkBlendMode::kSrcATop, |
| 25 | SkBlendMode::kDstOver, |
| 26 | SkBlendMode::kDstIn, |
| 27 | SkBlendMode::kDstOut, |
| 28 | SkBlendMode::kDstATop, |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 29 | }; |
| 30 | const int N_Modes = SK_ARRAY_COUNT(gModes); |
| 31 | |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 32 | static SkRandom gRand; |
| 33 | |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 34 | struct ModeButton { |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 35 | SkString fLabel; |
| 36 | SkColor fColor; |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 37 | SkRect fRect; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 38 | |
| 39 | public: |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 40 | void init(const char label[], const SkRect& rect) { |
| 41 | fLabel = label; |
| 42 | fRect = rect; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 43 | fColor = (gRand.nextU() & 0x7F7F7F7F) | SkColorSetARGB(0xFF, 0, 0, 0x80); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 44 | } |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 45 | |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 46 | void draw(SkCanvas* canvas) { |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 47 | SkPaint paint; |
| 48 | paint.setAntiAlias(true); |
| 49 | paint.setColor(fColor); |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 50 | canvas->drawRoundRect(fRect, 8, 8, paint); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 51 | |
| 52 | paint.setColor(0xFFFFFFFF); |
Mike Reed | b579f07 | 2019-01-03 15:45:53 -0500 | [diff] [blame] | 53 | SkFont font; |
| 54 | font.setSize(16); |
| 55 | font.setEdging(SkFont::Edging::kSubpixelAntiAlias); |
| 56 | SkTextUtils::DrawString(canvas, fLabel.c_str(), fRect.centerX(), fRect.fTop + 0.68f * fRect.height(), |
| 57 | font, paint, SkTextUtils::kCenter_Align); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 58 | } |
| 59 | |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 60 | bool hitTest(SkScalar x, SkScalar y) { |
Mike Reed | 9ea6315 | 2019-08-22 16:19:50 -0400 | [diff] [blame] | 61 | return fRect.intersects({x - 1, y - 1, x + 1, y + 1}); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 62 | } |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 63 | }; |
| 64 | |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 65 | class ModeDrawable : public SkDrawable { |
| 66 | public: |
reed | 374772b | 2016-10-05 17:33:02 -0700 | [diff] [blame] | 67 | ModeDrawable() : fMode(SkBlendMode::kSrcOver), fLoc(SkPoint::Make(0, 0)) {} |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 68 | |
reed | 374772b | 2016-10-05 17:33:02 -0700 | [diff] [blame] | 69 | SkBlendMode fMode; |
| 70 | SkPoint fLoc; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 71 | |
| 72 | bool hitTest(SkScalar x, SkScalar y) { |
| 73 | SkRect target = SkRect::MakeXYWH(x - fLoc.x() - 1, y - fLoc.y() - 1, 3, 3); |
| 74 | return this->getBounds().intersects(target); |
| 75 | } |
| 76 | }; |
| 77 | |
| 78 | class CircDrawable : public ModeDrawable { |
| 79 | SkPaint fPaint; |
| 80 | SkRect fBounds; |
| 81 | |
| 82 | public: |
| 83 | CircDrawable(SkScalar size, SkColor c) { |
| 84 | const SkColor colors[] = { 0, c }; |
reed | 8a21c9f | 2016-03-08 18:50:00 -0800 | [diff] [blame] | 85 | fPaint.setShader(SkGradientShader::MakeRadial(SkPoint::Make(size/2, size/2), size/2, |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 86 | colors, nullptr, 2, |
Mike Reed | fae8fce | 2019-04-03 10:27:45 -0400 | [diff] [blame] | 87 | SkTileMode::kClamp)); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 88 | fBounds = SkRect::MakeWH(size, size); |
| 89 | } |
| 90 | |
| 91 | protected: |
| 92 | SkRect onGetBounds() override { |
| 93 | return fBounds; |
| 94 | } |
| 95 | |
| 96 | void onDraw(SkCanvas* canvas) override { |
reed | 374772b | 2016-10-05 17:33:02 -0700 | [diff] [blame] | 97 | fPaint.setBlendMode(fMode); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 98 | canvas->save(); |
| 99 | canvas->translate(fLoc.x(), fLoc.y()); |
| 100 | canvas->drawOval(fBounds, fPaint); |
| 101 | canvas->restore(); |
| 102 | } |
| 103 | }; |
| 104 | |
Ben Wagner | b2c4ea6 | 2018-08-08 11:36:17 -0400 | [diff] [blame] | 105 | class XferDemo : public Sample { |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 106 | enum { |
| 107 | N = 4 |
| 108 | }; |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 109 | |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 110 | SkRect fModeRect[N_Modes]; |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 111 | ModeButton fModeButtons[N_Modes]; |
Hal Canary | 2db8361 | 2016-11-04 13:02:54 -0400 | [diff] [blame] | 112 | sk_sp<CircDrawable> fDrs[N]; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 113 | CircDrawable* fSelected; |
| 114 | |
| 115 | void addButtons() { |
| 116 | SkScalar x = 10; |
| 117 | SkScalar y = 10; |
| 118 | for (int i = 0; i < N_Modes; ++i) { |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 119 | fModeButtons[i].init(SkBlendMode_Name(gModes[i]), SkRect::MakeXYWH(x, y, 70, 25)); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 120 | fModeRect[i] = SkRect::MakeXYWH(x, y + 28, 70, 2); |
| 121 | x += 80; |
| 122 | } |
| 123 | } |
| 124 | |
| 125 | public: |
| 126 | XferDemo() { |
| 127 | const SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorBLACK }; |
| 128 | for (int i = 0; i < N; ++i) { |
| 129 | fDrs[i].reset(new CircDrawable(200, colors[i])); |
| 130 | fDrs[i]->fLoc.set(100.f + i * 100, 100.f + i * 100); |
reed | 374772b | 2016-10-05 17:33:02 -0700 | [diff] [blame] | 131 | fDrs[i]->fMode = SkBlendMode::kSrcOver; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 132 | } |
| 133 | fSelected = nullptr; |
| 134 | |
| 135 | this->addButtons(); |
| 136 | } |
| 137 | |
| 138 | protected: |
Hal Canary | 8a02731 | 2019-07-03 10:55:44 -0400 | [diff] [blame] | 139 | SkString name() override { return SkString("XferDemo"); } |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 140 | |
| 141 | void onDrawContent(SkCanvas* canvas) override { |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 142 | for (int i = 0; i < N_Modes; ++i) { |
| 143 | fModeButtons[i].draw(canvas); |
| 144 | } |
| 145 | |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 146 | SkPaint paint; |
| 147 | if (fSelected) { |
| 148 | for (int i = 0; i < N_Modes; ++i) { |
Brian Osman | d1e67e7 | 2017-03-15 12:19:37 -0400 | [diff] [blame] | 149 | if (fSelected->fMode == gModes[i]) { |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 150 | canvas->drawRect(fModeRect[i], paint); |
| 151 | break; |
| 152 | } |
| 153 | } |
| 154 | } |
| 155 | |
| 156 | canvas->saveLayer(nullptr, nullptr); |
| 157 | for (int i = 0; i < N; ++i) { |
| 158 | fDrs[i]->draw(canvas); |
| 159 | } |
| 160 | canvas->restore(); |
| 161 | } |
| 162 | |
Hal Canary | b1f411a | 2019-08-29 10:39:22 -0400 | [diff] [blame] | 163 | Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey) override { |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 164 | // Check mode buttons first |
| 165 | for (int i = 0; i < N_Modes; ++i) { |
| 166 | if (fModeButtons[i].hitTest(x, y)) { |
Hal Canary | fcf6359 | 2019-07-12 11:32:43 -0400 | [diff] [blame] | 167 | Click* click = new Click(); |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 168 | click->fMeta.setS32("mode", i); |
| 169 | return click; |
| 170 | } |
| 171 | } |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 172 | fSelected = nullptr; |
| 173 | for (int i = N - 1; i >= 0; --i) { |
| 174 | if (fDrs[i]->hitTest(x, y)) { |
Hal Canary | 2db8361 | 2016-11-04 13:02:54 -0400 | [diff] [blame] | 175 | fSelected = fDrs[i].get(); |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 176 | break; |
| 177 | } |
| 178 | } |
Hal Canary | fcf6359 | 2019-07-12 11:32:43 -0400 | [diff] [blame] | 179 | return fSelected ? new Click() : nullptr; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 180 | } |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 181 | |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 182 | bool onClick(Click* click) override { |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 183 | int32_t mode; |
| 184 | if (click->fMeta.findS32("mode", &mode)) { |
Hal Canary | b1f411a | 2019-08-29 10:39:22 -0400 | [diff] [blame] | 185 | if (fSelected && skui::InputState::kUp == click->fState) { |
Brian Osman | ede860e | 2017-11-22 16:36:07 -0500 | [diff] [blame] | 186 | fSelected->fMode = gModes[mode]; |
| 187 | } |
| 188 | } else { |
| 189 | fSelected->fLoc.fX += click->fCurr.fX - click->fPrev.fX; |
| 190 | fSelected->fLoc.fY += click->fCurr.fY - click->fPrev.fY; |
| 191 | } |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 192 | return true; |
| 193 | } |
| 194 | |
| 195 | private: |
John Stiles | 7571f9e | 2020-09-02 22:42:33 -0400 | [diff] [blame] | 196 | using INHERITED = Sample; |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 197 | }; |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 198 | DEF_SAMPLE( return new XferDemo; ) |
reed | 1a9caff | 2015-09-02 19:05:10 -0700 | [diff] [blame] | 199 | |
| 200 | ////////////////////////////////////////////////////////////////////////////// |
| 201 | |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 202 | #include "tools/Resources.h" |
| 203 | |
| 204 | class CubicResamplerDemo : public Sample { |
| 205 | struct Rec { |
| 206 | sk_sp<SkImage> fImage; |
| 207 | SkRect fBounds; |
| 208 | |
Mike Reed | 15b95d6 | 2020-11-06 09:50:47 -0500 | [diff] [blame] | 209 | void draw(SkCanvas* canvas, SkCubicResampler cubic) const { |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 210 | SkRect r = fBounds; |
| 211 | SkPaint paint; |
| 212 | |
| 213 | SkMatrix lm = SkMatrix::Translate(r.x(), r.y()) |
| 214 | * SkMatrix::Scale(10, 10); |
Mike Reed | 99c9446 | 2020-12-08 13:16:56 -0500 | [diff] [blame] | 215 | paint.setShader(fImage->makeShader(SkSamplingOptions(), lm)); |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 216 | canvas->drawRect(r, paint); |
| 217 | |
| 218 | r.offset(r.width() + 10, 0); |
| 219 | lm.postTranslate(r.width() + 10, 0); |
| 220 | |
Mike Reed | 99c9446 | 2020-12-08 13:16:56 -0500 | [diff] [blame] | 221 | paint.setShader(fImage->makeShader(SkSamplingOptions{ SkFilterMode::kLinear, |
Mike Reed | a03f8bf | 2020-11-20 18:45:36 -0500 | [diff] [blame] | 222 | SkMipmapMode::kNone }, |
Mike Reed | 99c9446 | 2020-12-08 13:16:56 -0500 | [diff] [blame] | 223 | lm)); |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 224 | canvas->drawRect(r, paint); |
| 225 | |
| 226 | r.offset(r.width() + 10, 0); |
| 227 | lm.postTranslate(r.width() + 10, 0); |
| 228 | |
Mike Reed | a03f8bf | 2020-11-20 18:45:36 -0500 | [diff] [blame] | 229 | paint.setShader(fImage->makeShader(SkTileMode::kClamp, SkTileMode::kClamp, |
| 230 | SkSamplingOptions(cubic), &lm)); |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 231 | canvas->drawRect(r, paint); |
| 232 | } |
| 233 | }; |
| 234 | std::vector<Rec> fRecs; |
| 235 | |
| 236 | public: |
| 237 | CubicResamplerDemo() { |
| 238 | const char* names[] = { |
| 239 | "images/mandrill_128.png", |
| 240 | "images/rle.bmp", |
| 241 | "images/example_4.png", |
| 242 | }; |
| 243 | SkRect r = {10, 10, 200, 200}; |
| 244 | for (auto name : names) { |
| 245 | fRecs.push_back({GetResourceAsImage(name), r}); |
| 246 | r.offset(0, r.height() + 10); |
| 247 | } |
| 248 | |
| 249 | fDomain.setXYWH(r.fLeft + 3*r.width() + 40, 50, 200, 200); |
| 250 | fCubic = {.3f, .5f}; |
| 251 | } |
| 252 | |
| 253 | protected: |
| 254 | SkString name() override { return SkString("CubicResampler"); } |
| 255 | |
| 256 | void onDrawContent(SkCanvas* canvas) override { |
| 257 | for (const auto& rec : fRecs) { |
| 258 | rec.draw(canvas, fCubic); |
| 259 | } |
| 260 | |
| 261 | SkPaint paint; |
| 262 | paint.setAntiAlias(true); |
| 263 | paint.setStroke(true); |
| 264 | canvas->drawRect(fDomain, paint); |
| 265 | |
| 266 | paint.setColor(SK_ColorRED); |
| 267 | paint.setStroke(false); |
| 268 | SkPoint loc = SkMatrix::MakeRectToRect({0,0,1,1}, fDomain, SkMatrix::kFill_ScaleToFit) |
| 269 | .mapXY(fCubic.B, fCubic.C); |
| 270 | canvas->drawCircle(loc.fX, loc.fY, 8, paint); |
| 271 | |
| 272 | SkString str; |
| 273 | str.printf("B=%4.2f C=%4.2f", fCubic.B, fCubic.C); |
| 274 | SkFont font; |
| 275 | font.setSize(25); |
| 276 | font.setEdging(SkFont::Edging::kAntiAlias); |
| 277 | paint.setColor(SK_ColorBLACK); |
| 278 | canvas->drawSimpleText(str.c_str(), str.size(), SkTextEncoding::kUTF8, |
| 279 | fDomain.fLeft + 10, fDomain.fBottom + 40, font, paint); |
| 280 | } |
| 281 | |
| 282 | static float pin_unitize(float min, float max, float value) { |
| 283 | return (std::min(std::max(value, min), max) - min) / (max - min); |
| 284 | } |
| 285 | static SkPoint pin_unitize(const SkRect& r, SkPoint p) { |
| 286 | return { |
| 287 | pin_unitize(r.fLeft, r.fRight, p.fX), |
| 288 | pin_unitize(r.fTop, r.fBottom, p.fY), |
| 289 | }; |
| 290 | } |
| 291 | |
| 292 | Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey) override { |
| 293 | if (fDomain.contains(x, y)) { |
| 294 | return new Click([this](Click* click) { |
| 295 | auto [B, C] = pin_unitize(fDomain, click->fCurr); |
| 296 | fCubic = {B, C}; |
| 297 | return true; |
| 298 | }); |
| 299 | } |
| 300 | return nullptr; |
| 301 | } |
| 302 | |
| 303 | private: |
| 304 | SkRect fDomain; |
| 305 | SkImage::CubicResampler fCubic; |
| 306 | |
John Stiles | 7571f9e | 2020-09-02 22:42:33 -0400 | [diff] [blame] | 307 | using INHERITED = Sample; |
Mike Reed | bd2b2c1 | 2020-07-24 15:30:49 -0400 | [diff] [blame] | 308 | }; |
| 309 | DEF_SAMPLE( return new CubicResamplerDemo; ) |