Chromium Code Reviews| Index: third_party/WebKit/Source/modules/compositorworker/WorkletAnimation.cpp |
| diff --git a/third_party/WebKit/Source/modules/compositorworker/WorkletAnimation.cpp b/third_party/WebKit/Source/modules/compositorworker/WorkletAnimation.cpp |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..d70e5b93cc3c421f26270a00a76676794960e94c |
| --- /dev/null |
| +++ b/third_party/WebKit/Source/modules/compositorworker/WorkletAnimation.cpp |
| @@ -0,0 +1,146 @@ |
| +// Copyright 2017 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +#include "modules/compositorworker/WorkletAnimation.h" |
| + |
| +#include "core/animation/ElementAnimations.h" |
| +#include "core/animation/KeyframeEffectModel.h" |
| +#include "core/animation/ScrollTimeline.h" |
| +#include "core/animation/Timing.h" |
| +#include "core/dom/Node.h" |
| +#include "core/dom/NodeComputedStyle.h" |
| +#include "platform/wtf/text/WTFString.h" |
| +#include "public/platform/Platform.h" |
| +#include "public/platform/WebCompositorSupport.h" |
| + |
| +namespace blink { |
| + |
| +namespace { |
| +bool ValidateEffects(const HeapVector<Member<KeyframeEffectReadOnly>>& effects, |
| + String& error_string) { |
| + if (effects.IsEmpty()) { |
| + error_string = "Effects array must be non-empty"; |
| + return false; |
| + } |
| + |
| + Document& target_document = effects.at(0)->Target()->GetDocument(); |
| + for (const auto& effect : effects) { |
| + if (effect->Target()->GetDocument() != target_document) { |
| + error_string = "All effects must target elements in the same document"; |
| + return false; |
| + } |
| + } |
| + return true; |
| +} |
| + |
| +bool ValidateTimelines(HeapVector<DocumentTimelineOrScrollTimeline>& timelines, |
| + String& error_string) { |
| + if (timelines.IsEmpty()) { |
| + error_string = "Timelines array must be non-empty"; |
| + return false; |
| + } |
| + |
| + for (const auto& timeline : timelines) { |
| + if (timeline.isScrollTimeline()) { |
| + DoubleOrScrollTimelineAutoKeyword time_range; |
| + timeline.getAsScrollTimeline()->timeRange(time_range); |
| + if (time_range.isScrollTimelineAutoKeyword()) { |
| + error_string = "ScrollTimeline timeRange must have non-auto value"; |
| + return false; |
| + } |
| + } |
| + } |
| + return true; |
| +} |
| +} // namespace |
| + |
| +WorkletAnimation* WorkletAnimation::Create( |
| + String animator_name, |
| + const HeapVector<Member<KeyframeEffectReadOnly>>& effects, |
| + HeapVector<DocumentTimelineOrScrollTimeline>& timelines, |
| + RefPtr<SerializedScriptValue> options, |
| + ExceptionState& exception_state) { |
| + String error_string; |
| + if (!ValidateEffects(effects, error_string)) { |
| + exception_state.ThrowDOMException(kNotSupportedError, error_string); |
| + return nullptr; |
| + } |
| + |
| + if (!ValidateTimelines(timelines, error_string)) { |
| + exception_state.ThrowDOMException(kNotSupportedError, error_string); |
| + return nullptr; |
| + } |
| + |
| + Document& document = effects.at(0)->Target()->GetDocument(); |
| + WorkletAnimation* animation = new WorkletAnimation( |
| + animator_name, document, effects, timelines, std::move(options)); |
| + document.Timeline().CompositorTimeline()->PlayerAttached(*animation); |
| + |
| + return animation; |
| +} |
| + |
| +WorkletAnimation::WorkletAnimation( |
| + String animator_name, |
|
nhiroki
2017/09/08 03:58:22
const String& ?
smcgruer
2017/09/11 18:19:00
Done.
|
| + Document& document, |
| + const HeapVector<Member<KeyframeEffectReadOnly>>& effects, |
| + HeapVector<DocumentTimelineOrScrollTimeline>& timelines, |
| + RefPtr<SerializedScriptValue> options) |
| + : animator_name_(animator_name), |
| + play_state_(Animation::kIdle), |
| + document_(document), |
| + effects_(effects), |
| + timelines_(timelines), |
| + options_(std::move(options)) { |
| + DCHECK(Platform::Current()->IsThreadedAnimationEnabled()); |
| + DCHECK(Platform::Current()->CompositorSupport()); |
| + compositor_player_ = CompositorAnimationPlayer::Create(); |
| + compositor_player_->SetAnimationDelegate(this); |
| +} |
| + |
| +String WorkletAnimation::playState() { |
| + return Animation::PlayStateString(play_state_); |
| +} |
| + |
| +void WorkletAnimation::play() { |
| + document_->GetWorkletAnimationController().AttachAnimation(*this); |
| + play_state_ = Animation::kPending; |
|
haraken
2017/09/08 01:11:23
Is there any restriction about the state transitio
smcgruer
2017/09/11 18:19:00
As of yet, no. It is valid to call play() twice in
|
| +} |
| + |
| +void WorkletAnimation::cancel() { |
| + document_->GetWorkletAnimationController().DetachAnimation(*this); |
| + play_state_ = Animation::kIdle; |
| +} |
| + |
| +bool WorkletAnimation::StartOnCompositor() { |
| + // TODO(smcgruer): We need to start all of the effects, not just the first. |
| + double animation_playback_rate = 1; |
| + Element& target = *effects_.at(0)->Target(); |
| + ToKeyframeEffectModelBase(effects_.at(0)->Model()) |
| + ->SnapshotAllCompositorKeyframes(target, target.ComputedStyleRef(), |
| + target.ParentComputedStyle()); |
| + bool success = |
| + effects_.at(0) |
| + ->CheckCanStartAnimationOnCompositor(animation_playback_rate) |
| + .Ok(); |
| + // Currently |StartAnimationOnCompositor| is unable to propagate a |
| + // WorkletAnimation effect to the compositor, so this method is a no-op. |
| + // TODO(smcgruer): Actually create the element animations on the compositor. |
| + play_state_ = success ? Animation::kRunning : Animation::kIdle; |
| + return success; |
| +} |
| + |
| +void WorkletAnimation::Dispose() { |
| + document_->Timeline().CompositorTimeline()->PlayerDestroyed(*this); |
| + compositor_player_->SetAnimationDelegate(nullptr); |
| + compositor_player_ = nullptr; |
| +} |
| + |
| +DEFINE_TRACE(WorkletAnimation) { |
| + visitor->Trace(document_); |
| + visitor->Trace(effects_); |
| + visitor->Trace(timelines_); |
| + WorkletAnimationBase::Trace(visitor); |
| +} |
| + |
| +} // namespace blink |