blob: 687e458f1fe04129020de05998ae372b408cfaae [file] [log] [blame]
senorblancod4bb9912015-03-20 08:54:32 -07001/*
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
8#include "sk_tool_utils.h"
senorblancod4bb9912015-03-20 08:54:32 -07009#include "SkBlurImageFilter.h"
10#include "SkColor.h"
11#include "SkDisplacementMapEffect.h"
12#include "SkDropShadowImageFilter.h"
13#include "SkGradientShader.h"
fmalita2f5891e2015-09-25 09:15:55 -070014#include "SkImage.h"
15#include "SkImageSource.h"
senorblancod4bb9912015-03-20 08:54:32 -070016#include "SkMorphologyImageFilter.h"
17#include "SkScalar.h"
fmalita2f5891e2015-09-25 09:15:55 -070018#include "SkSurface.h"
senorblancod4bb9912015-03-20 08:54:32 -070019#include "gm.h"
20
21namespace skiagm {
22
23// This GM draws image filters with a CTM containing shearing / rotation.
24// It checks that the scale portion of the CTM is correctly extracted
25// and applied to the image inputs separately from the non-scale portion.
26
reed9ce9d672016-03-17 10:51:11 -070027static sk_sp<SkImage> make_gradient_circle(int width, int height) {
fmalita2f5891e2015-09-25 09:15:55 -070028 SkScalar x = SkIntToScalar(width / 2);
29 SkScalar y = SkIntToScalar(height / 2);
30 SkScalar radius = SkMinScalar(x, y) * 0.8f;
31
reede8f30622016-03-23 18:59:25 -070032 auto surface(SkSurface::MakeRasterN32Premul(width, height));
fmalita2f5891e2015-09-25 09:15:55 -070033 SkCanvas* canvas = surface->getCanvas();
34
35 canvas->clear(0x00000000);
36 SkColor colors[2];
37 colors[0] = SK_ColorWHITE;
38 colors[1] = SK_ColorBLACK;
fmalita2f5891e2015-09-25 09:15:55 -070039 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -070040 paint.setShader(SkGradientShader::MakeRadial(SkPoint::Make(x, y), radius, colors, nullptr, 2,
41 SkShader::kClamp_TileMode));
fmalita2f5891e2015-09-25 09:15:55 -070042 canvas->drawCircle(x, y, radius, paint);
43
reed9ce9d672016-03-17 10:51:11 -070044 return surface->makeImageSnapshot();
fmalita2f5891e2015-09-25 09:15:55 -070045}
46
senorblancod4bb9912015-03-20 08:54:32 -070047class ImageFiltersTransformedGM : public GM {
48public:
49 ImageFiltersTransformedGM() {
50 this->setBGColor(SK_ColorBLACK);
51 }
52
53protected:
54
mtklein36352bf2015-03-25 18:17:31 -070055 SkString onShortName() override { return SkString("imagefilterstransformed"); }
senorblancod4bb9912015-03-20 08:54:32 -070056
mtklein36352bf2015-03-25 18:17:31 -070057 SkISize onISize() override { return SkISize::Make(420, 240); }
senorblancod4bb9912015-03-20 08:54:32 -070058
mtklein36352bf2015-03-25 18:17:31 -070059 void onOnceBeforeDraw() override {
reed9ce9d672016-03-17 10:51:11 -070060 fCheckerboard = SkImage::MakeFromBitmap(
61 sk_tool_utils::create_checkerboard_bitmap(64, 64, 0xFFA0A0A0, 0xFF404040, 8));
62 fGradientCircle = make_gradient_circle(64, 64);
senorblancod4bb9912015-03-20 08:54:32 -070063 }
64
mtklein36352bf2015-03-25 18:17:31 -070065 void onDraw(SkCanvas* canvas) override {
robertphillips549c8992016-04-01 09:28:51 -070066 sk_sp<SkImageFilter> gradient(SkImageSource::Make(fGradientCircle));
67 sk_sp<SkImageFilter> checkerboard(SkImageSource::Make(fCheckerboard));
senorblancod4bb9912015-03-20 08:54:32 -070068 SkImageFilter* filters[] = {
robertphillips6e7025a2016-04-04 04:31:25 -070069 SkBlurImageFilter::Make(12, 0, nullptr).release(),
senorblancod4bb9912015-03-20 08:54:32 -070070 SkDropShadowImageFilter::Create(0, 15, 8, 0, SK_ColorGREEN,
71 SkDropShadowImageFilter::kDrawShadowAndForeground_ShadowMode),
72 SkDisplacementMapEffect::Create(SkDisplacementMapEffect::kR_ChannelSelectorType,
73 SkDisplacementMapEffect::kR_ChannelSelectorType,
74 12,
75 gradient.get(),
76 checkerboard.get()),
77 SkDilateImageFilter::Create(2, 2, checkerboard.get()),
78 SkErodeImageFilter::Create(2, 2, checkerboard.get()),
79 };
80
81 const SkScalar margin = SkIntToScalar(20);
82 const SkScalar size = SkIntToScalar(60);
83
84 for (size_t j = 0; j < 3; j++) {
85 canvas->save();
86 canvas->translate(margin, 0);
87 for (size_t i = 0; i < SK_ARRAY_COUNT(filters); ++i) {
88 SkPaint paint;
89 paint.setColor(SK_ColorWHITE);
90 paint.setImageFilter(filters[i]);
91 paint.setAntiAlias(true);
92 canvas->save();
93 canvas->translate(size * SK_ScalarHalf, size * SK_ScalarHalf);
94 canvas->scale(SkDoubleToScalar(0.8), SkDoubleToScalar(0.8));
95 if (j == 1) {
96 canvas->rotate(SkIntToScalar(45));
97 } else if (j == 2) {
98 canvas->skew(SkDoubleToScalar(0.5), SkDoubleToScalar(0.2));
99 }
100 canvas->translate(-size * SK_ScalarHalf, -size * SK_ScalarHalf);
101 canvas->drawOval(SkRect::MakeXYWH(0, size * SkDoubleToScalar(0.1),
102 size, size * SkDoubleToScalar(0.6)), paint);
103 canvas->restore();
104 canvas->translate(size + margin, 0);
105 }
106 canvas->restore();
107 canvas->translate(0, size + margin);
108 }
109
110 for (size_t i = 0; i < SK_ARRAY_COUNT(filters); ++i) {
111 SkSafeUnref(filters[i]);
112 }
113 }
114
115private:
reed9ce9d672016-03-17 10:51:11 -0700116 sk_sp<SkImage> fCheckerboard;
117 sk_sp<SkImage> fGradientCircle;
senorblancod4bb9912015-03-20 08:54:32 -0700118 typedef GM INHERITED;
119};
120
121//////////////////////////////////////////////////////////////////////////////
122
123DEF_GM( return new ImageFiltersTransformedGM; )
124
125}