blob: ad3be77b448c56c2919d48629186b5981e1af1be [file] [log] [blame]
Florin Malita76a076b2018-02-15 18:40:48 -05001/*
2 * Copyright 2018 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 "SlideDir.h"
9
Mike Kleincd5104e2019-03-20 11:55:08 -050010#include "AnimTimer.h"
Florin Malita76a076b2018-02-15 18:40:48 -050011#include "SkCanvas.h"
Florin Malita836f8222018-02-20 10:31:01 -050012#include "SkCubicMap.h"
Florin Malita76a076b2018-02-15 18:40:48 -050013#include "SkMakeUnique.h"
Florin Malita76a076b2018-02-15 18:40:48 -050014#include "SkSGDraw.h"
15#include "SkSGGroup.h"
Florin Malitafb4bce82019-04-01 13:40:58 -040016#include "SkSGPaint.h"
Florin Malita97b3d2b2018-02-20 11:26:15 -050017#include "SkSGPlane.h"
Florin Malita65fce9e2018-02-19 13:25:18 -050018#include "SkSGRect.h"
Florin Malita76a076b2018-02-15 18:40:48 -050019#include "SkSGRenderNode.h"
20#include "SkSGScene.h"
21#include "SkSGText.h"
22#include "SkSGTransform.h"
23#include "SkTypeface.h"
24
Florin Malita65fce9e2018-02-19 13:25:18 -050025#include <cmath>
Ben Wagnerf08d1d02018-06-18 15:11:00 -040026#include <utility>
Florin Malita65fce9e2018-02-19 13:25:18 -050027
Florin Malita76a076b2018-02-15 18:40:48 -050028namespace {
29
Florin Malita65fce9e2018-02-19 13:25:18 -050030static constexpr float kAspectRatio = 1.5f;
31static constexpr float kLabelSize = 12.0f;
32static constexpr SkSize kPadding = { 12.0f , 24.0f };
Florin Malita836f8222018-02-20 10:31:01 -050033
34static constexpr float kFocusDuration = 500;
35static constexpr SkSize kFocusInset = { 100.0f, 100.0f };
36static constexpr SkPoint kFocusCtrl0 = { 0.3f, 1.0f };
37static constexpr SkPoint kFocusCtrl1 = { 0.0f, 1.0f };
38static constexpr SkColor kFocusShade = 0xa0000000;
Florin Malita65fce9e2018-02-19 13:25:18 -050039
40// TODO: better unfocus binding?
41static constexpr SkUnichar kUnfocusKey = ' ';
Florin Malita76a076b2018-02-15 18:40:48 -050042
43class SlideAdapter final : public sksg::RenderNode {
44public:
45 explicit SlideAdapter(sk_sp<Slide> slide)
46 : fSlide(std::move(slide)) {
47 SkASSERT(fSlide);
48 }
49
50 std::unique_ptr<sksg::Animator> makeForwardingAnimator() {
51 // Trivial sksg::Animator -> skottie::Animation tick adapter
52 class ForwardingAnimator final : public sksg::Animator {
53 public:
54 explicit ForwardingAnimator(sk_sp<SlideAdapter> adapter)
55 : fAdapter(std::move(adapter)) {}
56
57 protected:
58 void onTick(float t) override {
59 fAdapter->tick(SkScalarRoundToInt(t));
60 }
61
62 private:
63 sk_sp<SlideAdapter> fAdapter;
64 };
65
66 return skstd::make_unique<ForwardingAnimator>(sk_ref_sp(this));
67 }
68
69protected:
70 SkRect onRevalidate(sksg::InvalidationController* ic, const SkMatrix& ctm) override {
71 const auto isize = fSlide->getDimensions();
72 return SkRect::MakeIWH(isize.width(), isize.height());
73 }
74
Florin Malitac0132ff2018-08-09 07:40:01 -040075 void onRender(SkCanvas* canvas, const RenderContext* ctx) const override {
Florin Malita6127c4c2018-04-05 15:15:59 -040076 SkAutoCanvasRestore acr(canvas, true);
77 canvas->clipRect(SkRect::Make(fSlide->getDimensions()), true);
Florin Malitac0132ff2018-08-09 07:40:01 -040078
79 // TODO: commit the context?
Florin Malita76a076b2018-02-15 18:40:48 -050080 fSlide->draw(canvas);
81 }
82
Florin Malitaeb46bd82019-02-12 09:33:21 -050083 const RenderNode* onNodeAt(const SkPoint&) const override { return nullptr; }
84
Florin Malita76a076b2018-02-15 18:40:48 -050085private:
86 void tick(SkMSec t) {
Mike Kleincd5104e2019-03-20 11:55:08 -050087 fSlide->animate(AnimTimer(t * 1e6));
Florin Malita76a076b2018-02-15 18:40:48 -050088 this->invalidate();
89 }
90
91 const sk_sp<Slide> fSlide;
92
93 using INHERITED = sksg::RenderNode;
94};
95
Florin Malita65fce9e2018-02-19 13:25:18 -050096SkMatrix SlideMatrix(const sk_sp<Slide>& slide, const SkRect& dst) {
97 const auto slideSize = slide->getDimensions();
98 return SkMatrix::MakeRectToRect(SkRect::MakeIWH(slideSize.width(), slideSize.height()),
99 dst,
100 SkMatrix::kCenter_ScaleToFit);
101}
102
Florin Malita76a076b2018-02-15 18:40:48 -0500103} // namespace
104
105struct SlideDir::Rec {
Florin Malita760a0522019-01-10 15:24:15 -0500106 sk_sp<Slide> fSlide;
107 sk_sp<sksg::RenderNode> fSlideRoot;
108 sk_sp<sksg::Matrix<SkMatrix>> fMatrix;
109 SkRect fRect;
Florin Malita65fce9e2018-02-19 13:25:18 -0500110};
111
112class SlideDir::FocusController final : public sksg::Animator {
113public:
114 FocusController(const SlideDir* dir, const SkRect& focusRect)
115 : fDir(dir)
116 , fRect(focusRect)
117 , fTarget(nullptr)
Florin Malita34336e32019-03-06 16:34:21 -0500118 , fMap(kFocusCtrl1, kFocusCtrl0)
Florin Malita836f8222018-02-20 10:31:01 -0500119 , fState(State::kIdle) {
Florin Malita836f8222018-02-20 10:31:01 -0500120 fShadePaint = sksg::Color::Make(kFocusShade);
Florin Malita97b3d2b2018-02-20 11:26:15 -0500121 fShade = sksg::Draw::Make(sksg::Plane::Make(), fShadePaint);
Florin Malita836f8222018-02-20 10:31:01 -0500122 }
Florin Malita65fce9e2018-02-19 13:25:18 -0500123
124 bool hasFocus() const { return fState == State::kFocused; }
125
126 void startFocus(const Rec* target) {
Florin Malita836f8222018-02-20 10:31:01 -0500127 if (fState != State::kIdle)
128 return;
129
Florin Malita65fce9e2018-02-19 13:25:18 -0500130 fTarget = target;
Florin Malita836f8222018-02-20 10:31:01 -0500131
132 // Move the shade & slide to front.
Florin Malita919e2092019-01-09 15:37:57 -0500133 fDir->fRoot->removeChild(fTarget->fSlideRoot);
Florin Malita836f8222018-02-20 10:31:01 -0500134 fDir->fRoot->addChild(fShade);
Florin Malita919e2092019-01-09 15:37:57 -0500135 fDir->fRoot->addChild(fTarget->fSlideRoot);
Florin Malita836f8222018-02-20 10:31:01 -0500136
Florin Malita65fce9e2018-02-19 13:25:18 -0500137 fM0 = SlideMatrix(fTarget->fSlide, fTarget->fRect);
138 fM1 = SlideMatrix(fTarget->fSlide, fRect);
139
Florin Malita836f8222018-02-20 10:31:01 -0500140 fOpacity0 = 0;
141 fOpacity1 = 1;
142
Florin Malita65fce9e2018-02-19 13:25:18 -0500143 fTimeBase = 0;
144 fState = State::kFocusing;
Florin Malita836f8222018-02-20 10:31:01 -0500145
146 // Push initial state to the scene graph.
147 this->onTick(fTimeBase);
Florin Malita65fce9e2018-02-19 13:25:18 -0500148 }
149
150 void startUnfocus() {
151 SkASSERT(fTarget);
152
Ben Wagnerf08d1d02018-06-18 15:11:00 -0400153 using std::swap;
154 swap(fM0, fM1);
155 swap(fOpacity0, fOpacity1);
Florin Malita65fce9e2018-02-19 13:25:18 -0500156
157 fTimeBase = 0;
158 fState = State::kUnfocusing;
159 }
160
161 bool onMouse(SkScalar x, SkScalar y, sk_app::Window::InputState state, uint32_t modifiers) {
162 SkASSERT(fTarget);
163
Florin Malitaeb420452018-02-20 11:44:43 -0500164 if (!fRect.contains(x, y)) {
Florin Malita836f8222018-02-20 10:31:01 -0500165 this->startUnfocus();
166 return true;
167 }
168
Florin Malita65fce9e2018-02-19 13:25:18 -0500169 // Map coords to slide space.
170 const auto xform = SkMatrix::MakeRectToRect(fRect,
171 SkRect::MakeSize(fDir->fWinSize),
172 SkMatrix::kCenter_ScaleToFit);
173 const auto pt = xform.mapXY(x, y);
174
175 return fTarget->fSlide->onMouse(pt.x(), pt.y(), state, modifiers);
176 }
177
178 bool onChar(SkUnichar c) {
179 SkASSERT(fTarget);
180
181 return fTarget->fSlide->onChar(c);
182 }
183
184protected:
185 void onTick(float t) {
186 if (!this->isAnimating())
187 return;
188
189 if (!fTimeBase) {
190 fTimeBase = t;
191 }
192
Florin Malita836f8222018-02-20 10:31:01 -0500193 const auto rel_t = (t - fTimeBase) / kFocusDuration,
194 map_t = SkTPin(fMap.computeYFromX(rel_t), 0.0f, 1.0f);
Florin Malita65fce9e2018-02-19 13:25:18 -0500195
196 SkMatrix m;
197 for (int i = 0; i < 9; ++i) {
Florin Malita836f8222018-02-20 10:31:01 -0500198 m[i] = fM0[i] + map_t * (fM1[i] - fM0[i]);
Florin Malita65fce9e2018-02-19 13:25:18 -0500199 }
200
201 SkASSERT(fTarget);
Florin Malita919e2092019-01-09 15:37:57 -0500202 fTarget->fMatrix->setMatrix(m);
Florin Malita65fce9e2018-02-19 13:25:18 -0500203
Florin Malita836f8222018-02-20 10:31:01 -0500204 const auto shadeOpacity = fOpacity0 + map_t * (fOpacity1 - fOpacity0);
205 fShadePaint->setOpacity(shadeOpacity);
206
Florin Malita65fce9e2018-02-19 13:25:18 -0500207 if (rel_t < 1)
208 return;
209
210 switch (fState) {
211 case State::kFocusing:
212 fState = State::kFocused;
213 break;
214 case State::kUnfocusing:
215 fState = State::kIdle;
Florin Malita836f8222018-02-20 10:31:01 -0500216 fDir->fRoot->removeChild(fShade);
Florin Malita65fce9e2018-02-19 13:25:18 -0500217 break;
218
219 case State::kIdle:
220 case State::kFocused:
221 SkASSERT(false);
222 break;
223 }
224 }
225
226private:
227 enum class State {
228 kIdle,
229 kFocusing,
230 kUnfocusing,
231 kFocused,
232 };
233
234 bool isAnimating() const { return fState == State::kFocusing || fState == State::kUnfocusing; }
235
Florin Malita836f8222018-02-20 10:31:01 -0500236 const SlideDir* fDir;
237 const SkRect fRect;
238 const Rec* fTarget;
239
240 SkCubicMap fMap;
241 sk_sp<sksg::RenderNode> fShade;
242 sk_sp<sksg::PaintNode> fShadePaint;
Florin Malita65fce9e2018-02-19 13:25:18 -0500243
244 SkMatrix fM0 = SkMatrix::I(),
245 fM1 = SkMatrix::I();
Florin Malita836f8222018-02-20 10:31:01 -0500246 float fOpacity0 = 0,
247 fOpacity1 = 1,
248 fTimeBase = 0;
Florin Malita65fce9e2018-02-19 13:25:18 -0500249 State fState = State::kIdle;
250
251 using INHERITED = sksg::Animator;
Florin Malita76a076b2018-02-15 18:40:48 -0500252};
253
Brian Salomon343553a2018-09-05 15:41:23 -0400254SlideDir::SlideDir(const SkString& name, SkTArray<sk_sp<Slide>>&& slides, int columns)
Florin Malita76a076b2018-02-15 18:40:48 -0500255 : fSlides(std::move(slides))
256 , fColumns(columns) {
257 fName = name;
258}
259
Florin Malita65fce9e2018-02-19 13:25:18 -0500260static sk_sp<sksg::RenderNode> MakeLabel(const SkString& txt,
261 const SkPoint& pos,
262 const SkMatrix& dstXform) {
263 const auto size = kLabelSize / std::sqrt(dstXform.getScaleX() * dstXform.getScaleY());
Florin Malita76a076b2018-02-15 18:40:48 -0500264 auto text = sksg::Text::Make(nullptr, txt);
Florin Malitaf7d6ac12018-11-21 16:03:58 -0500265 text->setEdging(SkFont::Edging::kAntiAlias);
Florin Malita65fce9e2018-02-19 13:25:18 -0500266 text->setSize(size);
Mike Reed3a42ec02018-10-30 12:53:21 -0400267 text->setAlign(SkTextUtils::kCenter_Align);
Florin Malita65fce9e2018-02-19 13:25:18 -0500268 text->setPosition(pos + SkPoint::Make(0, size));
Florin Malita76a076b2018-02-15 18:40:48 -0500269
270 return sksg::Draw::Make(std::move(text), sksg::Color::Make(SK_ColorBLACK));
271}
272
273void SlideDir::load(SkScalar winWidth, SkScalar winHeight) {
274 // Build a global scene using transformed animation fragments:
275 //
276 // [Group(root)]
277 // [Transform]
278 // [Group]
279 // [AnimationWrapper]
280 // [Draw]
281 // [Text]
282 // [Color]
283 // [Transform]
284 // [Group]
285 // [AnimationWrapper]
286 // [Draw]
287 // [Text]
288 // [Color]
289 // ...
290 //
291
Florin Malita65fce9e2018-02-19 13:25:18 -0500292 fWinSize = SkSize::Make(winWidth, winHeight);
293 const auto cellWidth = winWidth / fColumns;
294 fCellSize = SkSize::Make(cellWidth, cellWidth / kAspectRatio);
Florin Malita76a076b2018-02-15 18:40:48 -0500295
296 sksg::AnimatorList sceneAnimators;
Florin Malita65fce9e2018-02-19 13:25:18 -0500297 fRoot = sksg::Group::Make();
Florin Malita76a076b2018-02-15 18:40:48 -0500298
299 for (int i = 0; i < fSlides.count(); ++i) {
300 const auto& slide = fSlides[i];
301 slide->load(winWidth, winHeight);
302
303 const auto slideSize = slide->getDimensions();
Florin Malita65fce9e2018-02-19 13:25:18 -0500304 const auto cell = SkRect::MakeXYWH(fCellSize.width() * (i % fColumns),
305 fCellSize.height() * (i / fColumns),
306 fCellSize.width(),
307 fCellSize.height()),
Florin Malita76a076b2018-02-15 18:40:48 -0500308 slideRect = cell.makeInset(kPadding.width(), kPadding.height());
309
Florin Malita760a0522019-01-10 15:24:15 -0500310 auto slideMatrix = sksg::Matrix<SkMatrix>::Make(SlideMatrix(slide, slideRect));
Florin Malita65fce9e2018-02-19 13:25:18 -0500311 auto adapter = sk_make_sp<SlideAdapter>(slide);
312 auto slideGrp = sksg::Group::Make();
313 slideGrp->addChild(sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeIWH(slideSize.width(),
314 slideSize.height())),
315 sksg::Color::Make(0xfff0f0f0)));
316 slideGrp->addChild(adapter);
317 slideGrp->addChild(MakeLabel(slide->getName(),
318 SkPoint::Make(slideSize.width() / 2, slideSize.height()),
Florin Malita919e2092019-01-09 15:37:57 -0500319 slideMatrix->getMatrix()));
320 auto slideRoot = sksg::TransformEffect::Make(std::move(slideGrp), slideMatrix);
Florin Malita76a076b2018-02-15 18:40:48 -0500321
322 sceneAnimators.push_back(adapter->makeForwardingAnimator());
Florin Malita76a076b2018-02-15 18:40:48 -0500323
Florin Malita919e2092019-01-09 15:37:57 -0500324 fRoot->addChild(slideRoot);
325 fRecs.push_back({ slide, slideRoot, slideMatrix, slideRect });
Florin Malita76a076b2018-02-15 18:40:48 -0500326 }
327
Florin Malita65fce9e2018-02-19 13:25:18 -0500328 fScene = sksg::Scene::Make(fRoot, std::move(sceneAnimators));
329
330 const auto focusRect = SkRect::MakeSize(fWinSize).makeInset(kFocusInset.width(),
331 kFocusInset.height());
332 fFocusController = skstd::make_unique<FocusController>(this, focusRect);
Florin Malita76a076b2018-02-15 18:40:48 -0500333}
334
335void SlideDir::unload() {
336 for (const auto& slide : fSlides) {
337 slide->unload();
338 }
339
340 fRecs.reset();
341 fScene.reset();
Florin Malita65fce9e2018-02-19 13:25:18 -0500342 fFocusController.reset();
343 fRoot.reset();
Florin Malita76a076b2018-02-15 18:40:48 -0500344 fTimeBase = 0;
345}
346
347SkISize SlideDir::getDimensions() const {
Florin Malita60d3bfc2018-02-20 16:49:20 -0500348 return SkSize::Make(fWinSize.width(),
349 fCellSize.height() * (1 + (fSlides.count() - 1) / fColumns)).toCeil();
Florin Malita76a076b2018-02-15 18:40:48 -0500350}
351
352void SlideDir::draw(SkCanvas* canvas) {
353 fScene->render(canvas);
354}
355
Mike Kleincd5104e2019-03-20 11:55:08 -0500356bool SlideDir::animate(const AnimTimer& timer) {
Florin Malita76a076b2018-02-15 18:40:48 -0500357 if (fTimeBase == 0) {
358 // Reset the animation time.
359 fTimeBase = timer.msec();
360 }
Florin Malita65fce9e2018-02-19 13:25:18 -0500361
362 const auto t = timer.msec() - fTimeBase;
363 fScene->animate(t);
364 fFocusController->tick(t);
Florin Malita76a076b2018-02-15 18:40:48 -0500365
366 return true;
367}
368
369bool SlideDir::onChar(SkUnichar c) {
Florin Malita65fce9e2018-02-19 13:25:18 -0500370 if (fFocusController->hasFocus()) {
371 if (c == kUnfocusKey) {
372 fFocusController->startUnfocus();
373 return true;
374 }
375 return fFocusController->onChar(c);
376 }
377
Florin Malita76a076b2018-02-15 18:40:48 -0500378 return false;
379}
380
Florin Malita65fce9e2018-02-19 13:25:18 -0500381bool SlideDir::onMouse(SkScalar x, SkScalar y, sk_app::Window::InputState state,
382 uint32_t modifiers) {
383 if (state == sk_app::Window::kMove_InputState || modifiers)
384 return false;
385
386 if (fFocusController->hasFocus()) {
387 return fFocusController->onMouse(x, y, state, modifiers);
388 }
389
390 const auto* cell = this->findCell(x, y);
391 if (!cell)
392 return false;
393
394 static constexpr SkScalar kClickMoveTolerance = 4;
395
396 switch (state) {
397 case sk_app::Window::kDown_InputState:
398 fTrackingCell = cell;
399 fTrackingPos = SkPoint::Make(x, y);
400 break;
401 case sk_app::Window::kUp_InputState:
402 if (cell == fTrackingCell &&
403 SkPoint::Distance(fTrackingPos, SkPoint::Make(x, y)) < kClickMoveTolerance) {
Florin Malita65fce9e2018-02-19 13:25:18 -0500404 fFocusController->startFocus(cell);
405 }
406 break;
407 default:
408 break;
409 }
410
Florin Malita76a076b2018-02-15 18:40:48 -0500411 return false;
412}
Florin Malita65fce9e2018-02-19 13:25:18 -0500413
414const SlideDir::Rec* SlideDir::findCell(float x, float y) const {
415 // TODO: use SG hit testing instead of layout info?
416 const auto size = this->getDimensions();
417 if (x < 0 || y < 0 || x >= size.width() || y >= size.height()) {
418 return nullptr;
419 }
420
421 const int col = static_cast<int>(x / fCellSize.width()),
422 row = static_cast<int>(y / fCellSize.height()),
423 idx = row * fColumns + col;
424
Florin Malita60d3bfc2018-02-20 16:49:20 -0500425 return idx < fRecs.count() ? &fRecs[idx] : nullptr;
Florin Malita65fce9e2018-02-19 13:25:18 -0500426}