blob: 35ef2c1d8b30191fc0b988013c8ba6dfedcfc092 [file] [log] [blame]
vjiaoblack955e8792016-08-05 07:55:01 -07001/*
2 * Copyright 2016 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 "SampleCode.h"
9#include "SkPictureRecorder.h"
10#include "SkShadowPaintFilterCanvas.h"
11#include "SkShadowShader.h"
12#include "SkSurface.h"
13
14#ifdef SK_EXPERIMENTAL_SHADOWING
15
vjiaoblack955e8792016-08-05 07:55:01 -070016class ShadowingView : public SampleView {
17public:
vjiaoblack56f33ea2016-08-26 08:49:46 -070018 ShadowingView()
19 : fSceneChanged(true)
20 , fLightsChanged(true)
21 , fMoveLight(false)
22 , fClearShadowMaps(false)
23 , fSelectedRectID(-1)
24 , fSelectedSliderID(-1)
vjiaoblack1e5abf12016-09-07 07:48:12 -070025 , fLightDepth(400.0f) {
vjiaoblack955e8792016-08-05 07:55:01 -070026 this->setBGColor(0xFFCCCCCC);
vjiaoblack56f33ea2016-08-26 08:49:46 -070027
vjiaoblack1e5abf12016-09-07 07:48:12 -070028 this->updateLights(100, 100);
vjiaoblack955e8792016-08-05 07:55:01 -070029
30 fTestRects[0].fColor = 0xFFEE8888;
31 fTestRects[0].fDepth = 80;
32 fTestRects[0].fGeometry = SkRect::MakeLTRB(200,150,350,300);
33
34 fTestRects[1].fColor = 0xFF88EE88;
35 fTestRects[1].fDepth = 160;
36 fTestRects[1].fGeometry = SkRect::MakeLTRB(150,200,300,350);
37
38 fTestRects[2].fColor = 0xFF8888EE;
39 fTestRects[2].fDepth = 240;
40 fTestRects[2].fGeometry = SkRect::MakeLTRB(100,100,250,250);
41
vjiaoblacke6f5d562016-08-25 06:30:23 -070042 fSliders[0].fGeometry = SkRect::MakeLTRB(20, 400, 30, 420);
43 fSliders[0].fOffset = 0.0f;
44 fSliders[0].fScale = 0.1f;
45
46 fSliders[1].fGeometry = SkRect::MakeLTRB(100, 420, 110, 440);
47 fSliders[1].fOffset = 0.0f;
48 fSliders[1].fScale = 10.0f;
49
50 fSliders[2].fGeometry = SkRect::MakeLTRB(0, 440, 10, 460);
51 fSliders[2].fOffset = 0.0f;
52 fSliders[2].fScale = 0.0025f;
53
vjiaoblack56f33ea2016-08-26 08:49:46 -070054 fShadowParams.fShadowRadius = 4.0f;
vjiaoblacke6f5d562016-08-25 06:30:23 -070055 fShadowParams.fBiasingConstant = 0.3f;
vjiaoblack1e5abf12016-09-07 07:48:12 -070056 fShadowParams.fMinVariance = 2048; // we need a higher min variance for point lights
vjiaoblacke6f5d562016-08-25 06:30:23 -070057 fShadowParams.fType = SkShadowParams::kVariance_ShadowType;
vjiaoblack955e8792016-08-05 07:55:01 -070058 }
59
60protected:
61 bool onQuery(SkEvent *evt) override {
62 if (SampleCode::TitleQ(*evt)) {
63 SampleCode::TitleR(evt, "shadowing");
64 return true;
65 }
66
67 SkUnichar uni;
68 if (SampleCode::CharQ(*evt, &uni)) {
69 switch (uni) {
70 case 'L':
71 fMoveLight = !fMoveLight;
72 break;
vjiaoblack904527d2016-08-09 09:32:09 -070073 case 'd':
74 // Raster generated shadow maps have their origin in the UL corner
75 // GPU shadow maps can have an arbitrary origin.
76 // We override the 'd' keypress so that when the device is cycled,
77 // the shadow maps will be re-generated according to the new backend.
78 fClearShadowMaps = true;
79 break;
vjiaoblack56f33ea2016-08-26 08:49:46 -070080 case 'q':
81 fLightDepth += 5.0f;
82 fMoveLight = true;
vjiaoblackbb106062016-08-31 11:15:21 -070083 break;
vjiaoblacke6f5d562016-08-25 06:30:23 -070084 case 'B':
85 if (SkShadowParams::kVariance_ShadowType == fShadowParams.fType) {
86 fShadowParams.fType = SkShadowParams::kNoBlur_ShadowType;
87 } else if (SkShadowParams::kNoBlur_ShadowType ==
88 fShadowParams.fType) {
89 fShadowParams.fType = SkShadowParams::kVariance_ShadowType;
90 }
91 fLightsChanged = true;
92 break;
vjiaoblack56f33ea2016-08-26 08:49:46 -070093 case 'w':
94 fLightDepth -= 5.0f;
95 fMoveLight = true;
96 break;
vjiaoblack955e8792016-08-05 07:55:01 -070097 default:
98 break;
99 }
100 }
101 return this->INHERITED::onQuery(evt);
102 }
103
104 sk_sp<SkPicture> makeTestPicture(int width, int height) {
105 SkPictureRecorder recorder;
106
107 // LONG RANGE TODO: eventually add SkBBHFactory (bounding box factory)
108 SkCanvas* canvas = recorder.beginRecording(SkRect::MakeIWH(width, height));
109
110 SkASSERT(canvas->getTotalMatrix().isIdentity());
111 SkPaint paint;
112 paint.setColor(SK_ColorGRAY);
113
114 // LONG RANGE TODO: tag occluders
115 // LONG RANGE TODO: track number of IDs we need (hopefully less than 256)
116 // and determinate the mapping from z to id
117
118 // universal receiver, "ground"
119 canvas->drawRect(SkRect::MakeIWH(width, height), paint);
120
121 for (int i = 0; i < kNumTestRects; i++) {
122 paint.setColor(fTestRects[i].fColor);
123 if (i == 0) {
124 canvas->translateZ(fTestRects[0].fDepth);
125 } else {
126 canvas->translateZ(fTestRects[i].fDepth - fTestRects[i-1].fDepth);
127 }
128 canvas->drawRect(fTestRects[i].fGeometry, paint);
129 }
130
131 return recorder.finishRecordingAsPicture();
132 }
133
vjiaoblack955e8792016-08-05 07:55:01 -0700134 void onDrawContent(SkCanvas *canvas) override {
vjiaoblack904527d2016-08-09 09:32:09 -0700135 if (fSceneChanged) {
vjiaoblack955e8792016-08-05 07:55:01 -0700136 fPicture = this->makeTestPicture(kWidth, kHeight);
vjiaoblack955e8792016-08-05 07:55:01 -0700137 }
138
vjiaoblack904527d2016-08-09 09:32:09 -0700139 if (fSceneChanged || fLightsChanged || fClearShadowMaps) {
140 for (int i = 0; i < fLights->numLights(); i++) {
141 fLights->light(i).setShadowMap(nullptr);
142 }
vjiaoblacke6f5d562016-08-25 06:30:23 -0700143
vjiaoblack904527d2016-08-09 09:32:09 -0700144 fSceneChanged = false;
145 fLightsChanged = false;
146 fClearShadowMaps = false;
vjiaoblack955e8792016-08-05 07:55:01 -0700147 }
vjiaoblackd707f2d2016-08-16 05:38:45 -0700148
149 canvas->setLights(fLights);
vjiaoblacke6f5d562016-08-25 06:30:23 -0700150 canvas->drawShadowedPicture(fPicture, nullptr, nullptr, fShadowParams);
151
152 for (int i = 0; i < kNumSliders; i++) {
153 SkPaint paint;
154 paint.setColor(SK_ColorBLACK);
155 canvas->drawRect(fSliders[i].fGeometry, paint);
156 }
vjiaoblack955e8792016-08-05 07:55:01 -0700157 }
158
159 SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override {
160 return new SkView::Click(this);
161 }
162
vjiaoblack56f33ea2016-08-26 08:49:46 -0700163 void updateLights(int x, int y) {
164 SkLights::Builder builder;
vjiaoblack1e5abf12016-09-07 07:48:12 -0700165 builder.setAmbientLightColor(SkColor3f::Make(0.2f, 0.2f, 0.2f));
vjiaoblack56f33ea2016-08-26 08:49:46 -0700166 builder.add(SkLights::Light::MakePoint(SkColor3f::Make(0.2f, 0.4f, 0.6f),
167 SkVector3::Make(x - 50,
168 350 - y,
169 fLightDepth),
vjiaoblackbb106062016-08-31 11:15:21 -0700170 100000));
vjiaoblack56f33ea2016-08-26 08:49:46 -0700171 builder.add(SkLights::Light::MakePoint(SkColor3f::Make(0.6f, 0.4f, 0.2f),
172 SkVector3::Make(x + 50,
173 450 - y,
174 fLightDepth),
vjiaoblackbb106062016-08-31 11:15:21 -0700175 100000));
vjiaoblack56f33ea2016-08-26 08:49:46 -0700176 fLights = builder.finish();
177 }
vjiaoblacke6f5d562016-08-25 06:30:23 -0700178
vjiaoblack56f33ea2016-08-26 08:49:46 -0700179 void updateFromSelectedSlider() {
180 SkScalar newValue = fSliders[fSelectedSliderID].fGeometry.fLeft *
181 fSliders[fSelectedSliderID].fScale +
182 fSliders[fSelectedSliderID].fOffset;
183
184 switch (fSelectedSliderID) {
vjiaoblacke6f5d562016-08-25 06:30:23 -0700185 case 0:
186 fShadowParams.fShadowRadius = newValue;
187 break;
188 case 1:
189 fShadowParams.fMinVariance = newValue;
190 break;
191 case 2:
192 fShadowParams.fBiasingConstant = newValue;
193 break;
194 default:
195 break;
196 }
197 }
198
vjiaoblack955e8792016-08-05 07:55:01 -0700199 bool onClick(Click *click) override {
200 SkScalar x = click->fCurr.fX;
201 SkScalar y = click->fCurr.fY;
202
203 SkScalar dx = x - click->fPrev.fX;
204 SkScalar dy = y - click->fPrev.fY;
205
206 if (fMoveLight) {
207 if (dx != 0 || dy != 0) {
vjiaoblack56f33ea2016-08-26 08:49:46 -0700208 this->updateLights(x, y);
vjiaoblack955e8792016-08-05 07:55:01 -0700209 fLightsChanged = true;
210 this->inval(nullptr);
211 }
212 return true;
213 }
214
215 if (click->fState == Click::State::kUp_State) {
vjiaoblack56f33ea2016-08-26 08:49:46 -0700216 fSelectedRectID = -1;
217 fSelectedSliderID = -1;
vjiaoblack955e8792016-08-05 07:55:01 -0700218 return true;
219 }
220
vjiaoblack56f33ea2016-08-26 08:49:46 -0700221 if (fSelectedRectID > -1) {
222 fTestRects[fSelectedRectID].fGeometry.offset(dx, dy);
vjiaoblack955e8792016-08-05 07:55:01 -0700223
224 fSceneChanged = true;
225 this->inval(nullptr);
226 return true;
227 }
228
vjiaoblack56f33ea2016-08-26 08:49:46 -0700229 if (fSelectedSliderID > -1) {
230 fSliders[fSelectedSliderID].fGeometry.offset(dx, 0);
vjiaoblacke6f5d562016-08-25 06:30:23 -0700231
232 this->updateFromSelectedSlider();
233
234 fLightsChanged = true;
235 this->inval(nullptr);
236 return true;
237 }
238
vjiaoblack955e8792016-08-05 07:55:01 -0700239 // assume last elements are highest
240 for (int i = kNumTestRects - 1; i >= 0; i--) {
241 if (fTestRects[i].fGeometry.contains(SkRect::MakeXYWH(x, y, 1, 1))) {
vjiaoblack56f33ea2016-08-26 08:49:46 -0700242 fSelectedRectID = i;
vjiaoblack955e8792016-08-05 07:55:01 -0700243 fTestRects[i].fGeometry.offset(dx, dy);
244
245 fSceneChanged = true;
246 this->inval(nullptr);
247 break;
248 }
249 }
250
vjiaoblacke6f5d562016-08-25 06:30:23 -0700251 for (int i = 0; i <= kNumSliders; i++) {
252 if (fSliders[i].fGeometry.contains(SkRect::MakeXYWH(x, y, 1, 1))) {
vjiaoblack56f33ea2016-08-26 08:49:46 -0700253 fSelectedSliderID = i;
vjiaoblacke6f5d562016-08-25 06:30:23 -0700254 fSliders[i].fGeometry.offset(dx, 0);
255
256 this->updateFromSelectedSlider();
257
258 fLightsChanged = true;
259
260 this->inval(nullptr);
261 break;
262 }
263 }
264
vjiaoblack955e8792016-08-05 07:55:01 -0700265 return true;
266 }
267
268private:
269 static constexpr int kNumTestRects = 3;
vjiaoblacke6f5d562016-08-25 06:30:23 -0700270 static constexpr int kNumSliders = 3;
vjiaoblack955e8792016-08-05 07:55:01 -0700271
272 static const int kWidth = 400;
273 static const int kHeight = 400;
274
vjiaoblack56f33ea2016-08-26 08:49:46 -0700275 bool fSceneChanged;
276 bool fLightsChanged;
277 bool fMoveLight;
278 bool fClearShadowMaps;
279
vjiaoblack955e8792016-08-05 07:55:01 -0700280 struct {
281 SkRect fGeometry;
282 int fDepth;
283 SkColor fColor;
284 } fTestRects[kNumTestRects];
vjiaoblack56f33ea2016-08-26 08:49:46 -0700285 int fSelectedRectID;
vjiaoblacke6f5d562016-08-25 06:30:23 -0700286
287 struct {
288 SkRect fGeometry;
289 SkScalar fOffset;
290 SkScalar fScale;
291 } fSliders[kNumSliders];
vjiaoblack56f33ea2016-08-26 08:49:46 -0700292 int fSelectedSliderID;
vjiaoblacke6f5d562016-08-25 06:30:23 -0700293
vjiaoblack56f33ea2016-08-26 08:49:46 -0700294 SkScalar fLightDepth;
vjiaoblack955e8792016-08-05 07:55:01 -0700295
vjiaoblacke6f5d562016-08-25 06:30:23 -0700296 sk_sp<SkPicture> fPicture;
297 SkShadowParams fShadowParams;
vjiaoblack955e8792016-08-05 07:55:01 -0700298 sk_sp<SkLights> fLights;
299
300 typedef SampleView INHERITED;
301};
302
303//////////////////////////////////////////////////////////////////////////////
304static SkView* MyFactory() { return new ShadowingView; }
305static SkViewRegister reg(MyFactory);
306
307#endif