1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "cc/animation/animation_host.h"
9 #include "cc/animation/animation_delegate.h"
10 #include "cc/animation/animation_id_provider.h"
11 #include "cc/animation/animation_player.h"
12 #include "cc/animation/animation_registrar.h"
13 #include "cc/animation/animation_timeline.h"
14 #include "cc/animation/element_animations.h"
15 #include "cc/animation/scroll_offset_animation_curve.h"
16 #include "cc/animation/timing_function.h"
17 #include "ui/gfx/geometry/box_f.h"
18 #include "ui/gfx/geometry/scroll_offset.h"
22 class AnimationHost::ScrollOffsetAnimations
: public AnimationDelegate
{
24 explicit ScrollOffsetAnimations(AnimationHost
* animation_host
)
25 : animation_host_(animation_host
),
26 scroll_offset_timeline_(
27 AnimationTimeline::Create(AnimationIdProvider::NextTimelineId())),
28 scroll_offset_animation_player_(
29 AnimationPlayer::Create(AnimationIdProvider::NextPlayerId())) {
30 scroll_offset_timeline_
->set_is_impl_only(true);
31 scroll_offset_animation_player_
->set_layer_animation_delegate(this);
33 animation_host_
->AddAnimationTimeline(scroll_offset_timeline_
.get());
34 scroll_offset_timeline_
->AttachPlayer(
35 scroll_offset_animation_player_
.get());
38 ~ScrollOffsetAnimations() override
{
39 scroll_offset_timeline_
->DetachPlayer(
40 scroll_offset_animation_player_
.get());
41 animation_host_
->RemoveAnimationTimeline(scroll_offset_timeline_
.get());
44 void ScrollAnimationCreate(int layer_id
,
45 const gfx::ScrollOffset
& target_offset
,
46 const gfx::ScrollOffset
& current_offset
) {
47 scoped_ptr
<ScrollOffsetAnimationCurve
> curve
=
48 ScrollOffsetAnimationCurve::Create(target_offset
,
49 EaseInOutTimingFunction::Create());
50 curve
->SetInitialValue(current_offset
);
52 scoped_ptr
<Animation
> animation
= Animation::Create(
53 curve
.Pass(), AnimationIdProvider::NextAnimationId(),
54 AnimationIdProvider::NextGroupId(), Animation::SCROLL_OFFSET
);
55 animation
->set_is_impl_only(true);
57 DCHECK(scroll_offset_animation_player_
);
58 DCHECK(scroll_offset_animation_player_
->animation_timeline());
60 if (scroll_offset_animation_player_
->layer_id() != layer_id
) {
61 if (scroll_offset_animation_player_
->layer_id())
62 scroll_offset_animation_player_
->DetachLayer();
63 scroll_offset_animation_player_
->AttachLayer(layer_id
);
66 scroll_offset_animation_player_
->AddAnimation(animation
.Pass());
69 bool ScrollAnimationUpdateTarget(int layer_id
,
70 const gfx::Vector2dF
& scroll_delta
,
71 const gfx::ScrollOffset
& max_scroll_offset
,
72 base::TimeTicks frame_monotonic_time
) {
73 DCHECK(scroll_offset_animation_player_
);
74 DCHECK_EQ(layer_id
, scroll_offset_animation_player_
->layer_id());
76 Animation
* animation
= scroll_offset_animation_player_
->element_animations()
77 ->layer_animation_controller()
78 ->GetAnimation(Animation::SCROLL_OFFSET
);
80 scroll_offset_animation_player_
->DetachLayer();
84 ScrollOffsetAnimationCurve
* curve
=
85 animation
->curve()->ToScrollOffsetAnimationCurve();
87 gfx::ScrollOffset new_target
=
88 gfx::ScrollOffsetWithDelta(curve
->target_value(), scroll_delta
);
89 new_target
.SetToMax(gfx::ScrollOffset());
90 new_target
.SetToMin(max_scroll_offset
);
92 curve
->UpdateTarget(animation
->TrimTimeToCurrentIteration(
93 frame_monotonic_time
).InSecondsF(),
99 // AnimationDelegate implementation.
100 void NotifyAnimationStarted(base::TimeTicks monotonic_time
,
101 Animation::TargetProperty target_property
,
102 int group
) override
{}
103 void NotifyAnimationFinished(base::TimeTicks monotonic_time
,
104 Animation::TargetProperty target_property
,
105 int group
) override
{
106 DCHECK_EQ(target_property
, Animation::SCROLL_OFFSET
);
107 DCHECK(animation_host_
->mutator_host_client());
108 animation_host_
->mutator_host_client()->ScrollOffsetAnimationFinished();
112 AnimationHost
* animation_host_
;
113 scoped_refptr
<AnimationTimeline
> scroll_offset_timeline_
;
115 // We have just one player for impl-only scroll offset animations.
116 // I.e. only one layer can have an impl-only scroll offset animation at
118 scoped_refptr
<AnimationPlayer
> scroll_offset_animation_player_
;
120 DISALLOW_COPY_AND_ASSIGN(ScrollOffsetAnimations
);
123 scoped_ptr
<AnimationHost
> AnimationHost::Create(
124 ThreadInstance thread_instance
) {
125 return make_scoped_ptr(new AnimationHost(thread_instance
));
128 AnimationHost::AnimationHost(ThreadInstance thread_instance
)
129 : animation_registrar_(AnimationRegistrar::Create()),
130 mutator_host_client_(nullptr),
131 thread_instance_(thread_instance
) {
132 if (thread_instance_
== ThreadInstance::IMPL
)
133 scroll_offset_animations_
=
134 make_scoped_ptr(new ScrollOffsetAnimations(this));
137 AnimationHost::~AnimationHost() {
138 scroll_offset_animations_
= nullptr;
141 DCHECK(!mutator_host_client());
142 DCHECK(layer_to_element_animations_map_
.empty());
145 AnimationTimeline
* AnimationHost::GetTimelineById(int timeline_id
) const {
146 for (auto& timeline
: timelines_
)
147 if (timeline
->id() == timeline_id
)
148 return timeline
.get();
152 void AnimationHost::ClearTimelines() {
153 EraseTimelines(timelines_
.begin(), timelines_
.end());
156 void AnimationHost::EraseTimelines(AnimationTimelineList::iterator begin
,
157 AnimationTimelineList::iterator end
) {
158 for (auto i
= begin
; i
!= end
; ++i
) {
160 timeline
->ClearPlayers();
161 timeline
->SetAnimationHost(nullptr);
164 timelines_
.erase(begin
, end
);
167 void AnimationHost::AddAnimationTimeline(
168 scoped_refptr
<AnimationTimeline
> timeline
) {
169 timeline
->SetAnimationHost(this);
170 timelines_
.push_back(timeline
);
173 void AnimationHost::RemoveAnimationTimeline(
174 scoped_refptr
<AnimationTimeline
> timeline
) {
175 for (auto iter
= timelines_
.begin(); iter
!= timelines_
.end(); ++iter
) {
176 if (iter
->get() != timeline
)
179 EraseTimelines(iter
, iter
+ 1);
184 void AnimationHost::RegisterLayer(int layer_id
, LayerTreeType tree_type
) {
185 ElementAnimations
* element_animations
=
186 GetElementAnimationsForLayerId(layer_id
);
187 if (element_animations
)
188 element_animations
->LayerRegistered(layer_id
, tree_type
);
191 void AnimationHost::UnregisterLayer(int layer_id
, LayerTreeType tree_type
) {
192 ElementAnimations
* element_animations
=
193 GetElementAnimationsForLayerId(layer_id
);
194 if (element_animations
)
195 element_animations
->LayerUnregistered(layer_id
, tree_type
);
198 void AnimationHost::RegisterPlayerForLayer(int layer_id
,
199 AnimationPlayer
* player
) {
203 ElementAnimations
* element_animations
=
204 GetElementAnimationsForLayerId(layer_id
);
205 if (!element_animations
) {
206 auto new_element_animations
= ElementAnimations::Create(this);
207 element_animations
= new_element_animations
.get();
209 layer_to_element_animations_map_
.add(layer_id
,
210 new_element_animations
.Pass());
211 element_animations
->CreateLayerAnimationController(layer_id
);
214 DCHECK(element_animations
);
215 element_animations
->AddPlayer(player
);
218 void AnimationHost::UnregisterPlayerForLayer(int layer_id
,
219 AnimationPlayer
* player
) {
223 ElementAnimations
* element_animations
=
224 GetElementAnimationsForLayerId(layer_id
);
225 DCHECK(element_animations
);
226 element_animations
->RemovePlayer(player
);
228 if (element_animations
->IsEmpty()) {
229 element_animations
->DestroyLayerAnimationController();
230 layer_to_element_animations_map_
.erase(layer_id
);
231 element_animations
= nullptr;
235 void AnimationHost::SetMutatorHostClient(MutatorHostClient
* client
) {
236 if (mutator_host_client_
== client
)
239 mutator_host_client_
= client
;
242 void AnimationHost::SetNeedsCommit() {
243 DCHECK(mutator_host_client_
);
244 mutator_host_client_
->SetMutatorsNeedCommit();
247 void AnimationHost::PushPropertiesTo(AnimationHost
* host_impl
) {
248 PushTimelinesToImplThread(host_impl
);
249 RemoveTimelinesFromImplThread(host_impl
);
250 PushPropertiesToImplThread(host_impl
);
253 void AnimationHost::PushTimelinesToImplThread(AnimationHost
* host_impl
) const {
254 for (auto& timeline
: timelines_
) {
255 AnimationTimeline
* timeline_impl
=
256 host_impl
->GetTimelineById(timeline
->id());
260 scoped_refptr
<AnimationTimeline
> to_add
= timeline
->CreateImplInstance();
261 host_impl
->AddAnimationTimeline(to_add
.get());
265 void AnimationHost::RemoveTimelinesFromImplThread(
266 AnimationHost
* host_impl
) const {
267 AnimationTimelineList
& timelines_impl
= host_impl
->timelines_
;
270 std::partition(timelines_impl
.begin(), timelines_impl
.end(),
271 [this](AnimationTimelineList::value_type timeline_impl
) {
272 return timeline_impl
->is_impl_only() ||
273 GetTimelineById(timeline_impl
->id());
276 host_impl
->EraseTimelines(to_erase
, timelines_impl
.end());
279 void AnimationHost::PushPropertiesToImplThread(AnimationHost
* host_impl
) {
280 // Firstly, sync all players with impl thread to create ElementAnimations and
281 // layer animation controllers.
282 for (auto& timeline
: timelines_
) {
283 AnimationTimeline
* timeline_impl
=
284 host_impl
->GetTimelineById(timeline
->id());
286 timeline
->PushPropertiesTo(timeline_impl
);
289 // Secondly, sync properties for created layer animation controllers.
290 for (auto& kv
: layer_to_element_animations_map_
) {
291 ElementAnimations
* element_animations
= kv
.second
;
292 ElementAnimations
* element_animations_impl
=
293 host_impl
->GetElementAnimationsForLayerId(kv
.first
);
294 if (element_animations_impl
)
295 element_animations
->PushPropertiesTo(element_animations_impl
);
299 LayerAnimationController
* AnimationHost::GetControllerForLayerId(
300 int layer_id
) const {
301 const ElementAnimations
* element_animations
=
302 GetElementAnimationsForLayerId(layer_id
);
303 if (!element_animations
)
306 return element_animations
->layer_animation_controller();
309 ElementAnimations
* AnimationHost::GetElementAnimationsForLayerId(
310 int layer_id
) const {
312 auto iter
= layer_to_element_animations_map_
.find(layer_id
);
313 return iter
== layer_to_element_animations_map_
.end() ? nullptr
317 void AnimationHost::SetSupportsScrollAnimations(
318 bool supports_scroll_animations
) {
319 animation_registrar_
->set_supports_scroll_animations(
320 supports_scroll_animations
);
323 bool AnimationHost::SupportsScrollAnimations() const {
324 return animation_registrar_
->supports_scroll_animations();
327 bool AnimationHost::NeedsAnimateLayers() const {
328 return animation_registrar_
->needs_animate_layers();
331 bool AnimationHost::ActivateAnimations() {
332 return animation_registrar_
->ActivateAnimations();
335 bool AnimationHost::AnimateLayers(base::TimeTicks monotonic_time
) {
336 return animation_registrar_
->AnimateLayers(monotonic_time
);
339 bool AnimationHost::UpdateAnimationState(bool start_ready_animations
,
340 AnimationEventsVector
* events
) {
341 return animation_registrar_
->UpdateAnimationState(start_ready_animations
,
345 scoped_ptr
<AnimationEventsVector
> AnimationHost::CreateEvents() {
346 return animation_registrar_
->CreateEvents();
349 void AnimationHost::SetAnimationEvents(
350 scoped_ptr
<AnimationEventsVector
> events
) {
351 return animation_registrar_
->SetAnimationEvents(events
.Pass());
354 bool AnimationHost::ScrollOffsetAnimationWasInterrupted(int layer_id
) const {
355 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
356 return controller
? controller
->scroll_offset_animation_was_interrupted()
360 bool AnimationHost::IsAnimatingFilterProperty(int layer_id
) const {
361 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
362 return controller
? controller
->IsAnimatingProperty(Animation::FILTER
)
366 bool AnimationHost::IsAnimatingOpacityProperty(int layer_id
) const {
367 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
368 return controller
? controller
->IsAnimatingProperty(Animation::OPACITY
)
372 bool AnimationHost::IsAnimatingTransformProperty(int layer_id
) const {
373 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
374 return controller
? controller
->IsAnimatingProperty(Animation::TRANSFORM
)
378 bool AnimationHost::HasPotentiallyRunningOpacityAnimation(int layer_id
) const {
379 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
383 Animation
* animation
= controller
->GetAnimation(Animation::OPACITY
);
384 return animation
&& !animation
->is_finished();
387 bool AnimationHost::HasPotentiallyRunningTransformAnimation(
388 int layer_id
) const {
389 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
393 Animation
* animation
= controller
->GetAnimation(Animation::TRANSFORM
);
394 return animation
&& !animation
->is_finished();
397 bool AnimationHost::FilterIsAnimatingOnImplOnly(int layer_id
) const {
398 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
402 Animation
* animation
= controller
->GetAnimation(Animation::FILTER
);
403 return animation
&& animation
->is_impl_only();
406 bool AnimationHost::OpacityIsAnimatingOnImplOnly(int layer_id
) const {
407 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
411 Animation
* animation
= controller
->GetAnimation(Animation::OPACITY
);
412 return animation
&& animation
->is_impl_only();
415 bool AnimationHost::TransformIsAnimatingOnImplOnly(int layer_id
) const {
416 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
420 Animation
* animation
= controller
->GetAnimation(Animation::TRANSFORM
);
421 return animation
&& animation
->is_impl_only();
424 bool AnimationHost::HasFilterAnimationThatInflatesBounds(int layer_id
) const {
425 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
426 return controller
? controller
->HasFilterAnimationThatInflatesBounds()
430 bool AnimationHost::HasTransformAnimationThatInflatesBounds(
431 int layer_id
) const {
432 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
433 return controller
? controller
->HasTransformAnimationThatInflatesBounds()
437 bool AnimationHost::HasAnimationThatInflatesBounds(int layer_id
) const {
438 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
439 return controller
? controller
->HasAnimationThatInflatesBounds() : false;
442 bool AnimationHost::FilterAnimationBoundsForBox(int layer_id
,
443 const gfx::BoxF
& box
,
444 gfx::BoxF
* bounds
) const {
445 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
446 return controller
? controller
->FilterAnimationBoundsForBox(box
, bounds
)
450 bool AnimationHost::TransformAnimationBoundsForBox(int layer_id
,
451 const gfx::BoxF
& box
,
452 gfx::BoxF
* bounds
) const {
453 *bounds
= gfx::BoxF();
454 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
455 return controller
? controller
->TransformAnimationBoundsForBox(box
, bounds
)
459 bool AnimationHost::HasOnlyTranslationTransforms(int layer_id
) const {
460 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
461 return controller
? controller
->HasOnlyTranslationTransforms() : true;
464 bool AnimationHost::AnimationsPreserveAxisAlignment(int layer_id
) const {
465 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
466 return controller
? controller
->AnimationsPreserveAxisAlignment() : true;
469 bool AnimationHost::MaximumTargetScale(int layer_id
, float* max_scale
) const {
471 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
472 return controller
? controller
->MaximumTargetScale(max_scale
) : true;
475 bool AnimationHost::AnimationStartScale(int layer_id
,
476 float* start_scale
) const {
478 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
479 return controller
? controller
->AnimationStartScale(start_scale
) : true;
482 bool AnimationHost::HasAnyAnimation(int layer_id
) const {
483 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
484 return controller
? controller
->has_any_animation() : false;
487 bool AnimationHost::HasActiveAnimation(int layer_id
) const {
488 LayerAnimationController
* controller
= GetControllerForLayerId(layer_id
);
489 return controller
? controller
->HasActiveAnimation() : false;
492 void AnimationHost::ImplOnlyScrollAnimationCreate(
494 const gfx::ScrollOffset
& target_offset
,
495 const gfx::ScrollOffset
& current_offset
) {
496 DCHECK(scroll_offset_animations_
);
497 scroll_offset_animations_
->ScrollAnimationCreate(layer_id
, target_offset
,
501 bool AnimationHost::ImplOnlyScrollAnimationUpdateTarget(
503 const gfx::Vector2dF
& scroll_delta
,
504 const gfx::ScrollOffset
& max_scroll_offset
,
505 base::TimeTicks frame_monotonic_time
) {
506 DCHECK(scroll_offset_animations_
);
507 return scroll_offset_animations_
->ScrollAnimationUpdateTarget(
508 layer_id
, scroll_delta
, max_scroll_offset
, frame_monotonic_time
);