| /* |
| * Copyright 2016 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "gm/gm.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkFont.h" |
| #include "include/core/SkFontTypes.h" |
| #include "include/core/SkImageFilter.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkPoint.h" |
| #include "include/core/SkRRect.h" |
| #include "include/core/SkRect.h" |
| #include "include/core/SkScalar.h" |
| #include "include/core/SkSize.h" |
| #include "include/core/SkString.h" |
| #include "include/core/SkTileMode.h" |
| #include "include/core/SkTypes.h" |
| #include "include/effects/SkImageFilters.h" |
| #include "include/private/base/SkTPin.h" |
| #include "src/base/SkRandom.h" |
| #include "tools/DecodeUtils.h" |
| #include "tools/fonts/FontToolUtils.h" |
| #include "tools/timer/TimeUtils.h" |
| |
| static const SkScalar kBlurMax = 7.0f; |
| static const int kNumNodes = 30; |
| static const int kWidth = 512; |
| static const int kHeight = 512; |
| static const SkScalar kBlurAnimationDuration = 4.0f; // in secs |
| |
| // This GM draws a lot of layers with animating BlurImageFilters |
| class AnimatedImageBlurs : public skiagm::GM { |
| public: |
| AnimatedImageBlurs() : fLastTime(0.0f) { |
| this->setBGColor(0xFFCCCCCC); |
| } |
| |
| protected: |
| bool runAsBench() const override { return true; } |
| |
| SkString getName() const override { return SkString("animated-image-blurs"); } |
| |
| SkISize getISize() override { return SkISize::Make(kWidth, kHeight); } |
| |
| void onOnceBeforeDraw() override { |
| for (int i = 0; i < kNumNodes; ++i) { |
| fNodes[i].init(&fRand); |
| } |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| SkPaint paint; |
| paint.setAntiAlias(true); |
| |
| for (int i = 0; i < kNumNodes; ++i) { |
| SkPaint layerPaint; |
| layerPaint.setImageFilter(SkImageFilters::Blur(fNodes[i].sigma(), fNodes[i].sigma(), |
| nullptr)); |
| |
| canvas->saveLayer(nullptr, &layerPaint); |
| // The rect is outset to block the circle case |
| SkRect rect = SkRect::MakeLTRB(fNodes[i].pos().fX - fNodes[i].size()-0.5f, |
| fNodes[i].pos().fY - fNodes[i].size()-0.5f, |
| fNodes[i].pos().fX + fNodes[i].size()+0.5f, |
| fNodes[i].pos().fY + fNodes[i].size()+0.5f); |
| SkRRect rrect = SkRRect::MakeRectXY(rect, fNodes[i].size(), fNodes[i].size()); |
| canvas->drawRRect(rrect, paint); |
| canvas->restore(); |
| } |
| } |
| |
| bool onAnimate(double nanos) override { |
| if (0.0f != fLastTime) { |
| for (int i = 0; i < kNumNodes; ++i) { |
| fNodes[i].update(nanos, fLastTime); |
| } |
| } |
| |
| fLastTime = 1e-9 * nanos; |
| return true; |
| } |
| |
| private: |
| class Node { |
| public: |
| Node() |
| : fSize(0.0f) |
| , fPos { 0.0f, 0.0f } |
| , fDir { 1.0f, 0.0f } |
| , fBlurOffset(0.0f) |
| , fBlur(fBlurOffset) |
| , fSpeed(0.0f) { |
| } |
| |
| void init(SkRandom* rand) { |
| fSize = rand->nextRangeF(10.0f, 60.f); |
| fPos.fX = rand->nextRangeF(fSize, kWidth - fSize); |
| fPos.fY = rand->nextRangeF(fSize, kHeight - fSize); |
| fDir.fX = rand->nextRangeF(-1.0f, 1.0f); |
| fDir.fY = SkScalarSqrt(1.0f - fDir.fX * fDir.fX); |
| if (rand->nextBool()) { |
| fDir.fY = -fDir.fY; |
| } |
| fBlurOffset = rand->nextRangeF(0.0f, kBlurMax); |
| fBlur = fBlurOffset; |
| fSpeed = rand->nextRangeF(20.0f, 60.0f); |
| } |
| |
| void update(double nanos, SkScalar lastTime) { |
| SkScalar deltaTime = 1e-9 * nanos - lastTime; |
| |
| fPos.fX += deltaTime * fSpeed * fDir.fX; |
| fPos.fY += deltaTime * fSpeed * fDir.fY; |
| if (fPos.fX >= kWidth || fPos.fX < 0.0f) { |
| fPos.fX = SkTPin<SkScalar>(fPos.fX, 0.0f, kWidth); |
| fDir.fX = -fDir.fX; |
| } |
| if (fPos.fY >= kHeight || fPos.fY < 0.0f) { |
| fPos.fY = SkTPin<SkScalar>(fPos.fY, 0.0f, kHeight); |
| fDir.fY = -fDir.fY; |
| } |
| |
| fBlur = TimeUtils::PingPong(1e-9 * nanos, kBlurAnimationDuration, fBlurOffset, 0.0f, kBlurMax); |
| } |
| |
| SkScalar sigma() const { return fBlur; } |
| const SkPoint& pos() const { return fPos; } |
| SkScalar size() const { return fSize; } |
| |
| private: |
| SkScalar fSize; |
| SkPoint fPos; |
| SkVector fDir; |
| SkScalar fBlurOffset; |
| SkScalar fBlur; |
| SkScalar fSpeed; |
| }; |
| |
| Node fNodes[kNumNodes]; |
| SkRandom fRand; |
| SkScalar fLastTime; |
| |
| using INHERITED = GM; |
| }; |
| |
| // This GM draws an image with a tiled blur that animates from large to small sigmas |
| class AnimatedTiledImageBlur : public skiagm::GM { |
| static constexpr float kMaxBlurSigma = 250.f; |
| static constexpr float kAnimationDuration = 12.f; // seconds |
| public: |
| AnimatedTiledImageBlur() : fBlurSigma(0.3f * kMaxBlurSigma) { |
| this->setBGColor(0xFFCCCCCC); |
| } |
| |
| protected: |
| bool runAsBench() const override { return true; } |
| |
| SkString getName() const override { return SkString("animated-tiled-image-blur"); } |
| |
| SkISize getISize() override { return SkISize::Make(530, 530); } |
| |
| void onOnceBeforeDraw() override { |
| fImage = ToolUtils::GetResourceAsImage("images/mandrill_512.png"); |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| auto drawBlurredImage = [&](float tx, float ty, SkTileMode tileMode) { |
| SkPaint paint; |
| SkRect rect = SkRect::MakeIWH(250, 250); |
| canvas->save(); |
| canvas->translate(tx, ty); |
| paint.setImageFilter(SkImageFilters::Blur(fBlurSigma, fBlurSigma, tileMode, |
| nullptr, rect)); |
| canvas->drawImageRect(fImage, rect, SkFilterMode::kLinear, &paint); |
| canvas->restore(); |
| }; |
| |
| drawBlurredImage(10.f, 10.f, SkTileMode::kDecal); |
| drawBlurredImage(270.f, 10.f, SkTileMode::kClamp); |
| drawBlurredImage(10.f, 270.f, SkTileMode::kRepeat); |
| drawBlurredImage(270.f, 270.f, SkTileMode::kMirror); |
| } |
| |
| bool onAnimate(double nanos) override { |
| fBlurSigma = TimeUtils::PingPong(1e-9 * nanos, kAnimationDuration, |
| 0.f, 0.0f, kMaxBlurSigma); |
| return true; |
| } |
| |
| private: |
| sk_sp<SkImage> fImage; |
| SkScalar fBlurSigma; |
| }; |
| |
| class AnimatedBackdropBlur final : public skiagm::GM { |
| public: |
| SkString getName() const override { return SkString("animated-backdrop-blur"); } |
| |
| SkISize getISize() override { return {512, 1024}; } |
| |
| void onOnceBeforeDraw() override { |
| fFont = SkFont(ToolUtils::DefaultPortableTypeface(), 20); |
| fImage = ToolUtils::GetResourceAsImage("images/color_wheel.png"); |
| const SkRect crop{0, 100, 512, 400}; |
| fFilter = SkImageFilters::Crop(crop, SkTileMode::kDecal, |
| SkImageFilters::Blur(30, 30, |
| SkImageFilters::Crop(crop, SkTileMode::kMirror, nullptr))); |
| |
| fLayerRec = SkCanvas::SaveLayerRec(nullptr, nullptr, fFilter.get(), 0); |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| static constexpr const char* txts[] = { |
| "Lorem ipsum dolor sit amet,", |
| "consectetur adipiscing elit,", |
| "sed do eiusmod tempor incididunt", |
| "ut labore et dolore magna aliqua.", |
| "", |
| "", |
| "Ut enim ad minim veniam,", |
| "quis nostrud exercitation ullamco laboris", |
| "nisi ut aliquip ex ea commodo consequat.", |
| "", |
| "", |
| "Duis aute irure dolor in reprehenderit", |
| "in voluptate velit esse cillum dolore", |
| "eu fugiat nulla pariatur." |
| }; |
| |
| SkPaint paint; |
| float voffset = fVOffset; |
| for (const auto& txt : txts) { |
| canvas->drawSimpleText( |
| txt, strlen(txt), SkTextEncoding::kUTF8, 0, voffset, fFont, paint); |
| voffset += fFont.getSize(); |
| } |
| |
| float dstHeight = fImage->height() * 128.f / fImage->width(); |
| canvas->drawImageRect(fImage.get(), SkRect::MakeXYWH(16.f, fVOffset, 128.f, dstHeight), |
| SkFilterMode::kLinear); |
| |
| canvas->saveLayer(fLayerRec); |
| canvas->restore(); |
| } |
| |
| bool onAnimate(double nanos) override { |
| fVOffset = TimeUtils::PingPong(nanos * 1e-9, 6, 0, 350, 0); |
| |
| return true; |
| } |
| |
| private: |
| sk_sp<SkImageFilter> fFilter; |
| sk_sp<SkImage> fImage; |
| SkCanvas::SaveLayerRec fLayerRec; |
| SkFont fFont; |
| float fVOffset = 0; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| DEF_GM(return new AnimatedImageBlurs;) |
| DEF_GM(return new AnimatedTiledImageBlur;) |
| DEF_GM(return new AnimatedBackdropBlur;) |