blob: 4553fbe880fe9964e0c4486815ef9af64fdd51b4 [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "core/animation/AnimationTimeline.h"
#include "core/animation/ActiveAnimations.h"
#include "core/animation/AnimationClock.h"
#include "core/dom/Document.h"
#include "core/frame/FrameView.h"
#include "core/page/Page.h"
#include "platform/TraceEvent.h"
namespace WebCore {
// This value represents 1 frame at 30Hz plus a little bit of wiggle room.
// TODO: Plumb a nominal framerate through and derive this value from that.
const double AnimationTimeline::s_minimumDelay = 0.04;
PassRefPtrWillBeRawPtr<AnimationTimeline> AnimationTimeline::create(Document* document, PassOwnPtrWillBeRawPtr<PlatformTiming> timing)
{
return adoptRefWillBeNoop(new AnimationTimeline(document, timing));
}
AnimationTimeline::AnimationTimeline(Document* document, PassOwnPtrWillBeRawPtr<PlatformTiming> timing)
: m_document(document)
{
if (!timing)
m_timing = adoptPtrWillBeNoop(new AnimationTimelineTiming(this));
else
m_timing = timing;
ASSERT(document);
}
AnimationTimeline::~AnimationTimeline()
{
#if !ENABLE(OILPAN)
for (WillBeHeapHashSet<RawPtrWillBeWeakMember<AnimationPlayer> >::iterator it = m_players.begin(); it != m_players.end(); ++it)
(*it)->timelineDestroyed();
#endif
}
AnimationPlayer* AnimationTimeline::createAnimationPlayer(AnimationNode* child)
{
RefPtrWillBeRawPtr<AnimationPlayer> player = AnimationPlayer::create(m_document->contextDocument().get(), *this, child);
AnimationPlayer* result = player.get();
m_players.add(result);
setOutdatedAnimationPlayer(result);
return result;
}
AnimationPlayer* AnimationTimeline::play(AnimationNode* child)
{
if (!m_document)
return 0;
AnimationPlayer* player = createAnimationPlayer(child);
m_document->compositorPendingAnimations().add(player);
return player;
}
void AnimationTimeline::wake()
{
m_timing->serviceOnNextFrame();
}
void AnimationTimeline::serviceAnimations(TimingUpdateReason reason)
{
TRACE_EVENT0("webkit", "AnimationTimeline::serviceAnimations");
m_timing->cancelWake();
double timeToNextEffect = std::numeric_limits<double>::infinity();
WillBeHeapVector<RawPtrWillBeMember<AnimationPlayer> > players;
for (WillBeHeapHashSet<RefPtrWillBeMember<AnimationPlayer> >::iterator it = m_playersNeedingUpdate.begin(); it != m_playersNeedingUpdate.end(); ++it)
players.append(it->get());
std::sort(players.begin(), players.end(), AnimationPlayer::hasLowerPriority);
for (size_t i = 0; i < players.size(); ++i) {
AnimationPlayer* player = players[i];
if (player->update(reason))
timeToNextEffect = std::min(timeToNextEffect, player->timeToEffectChange());
else
m_playersNeedingUpdate.remove(player);
}
if (timeToNextEffect < s_minimumDelay)
m_timing->serviceOnNextFrame();
else if (timeToNextEffect != std::numeric_limits<double>::infinity())
m_timing->wakeAfter(timeToNextEffect - s_minimumDelay);
ASSERT(!hasOutdatedAnimationPlayer());
}
void AnimationTimeline::AnimationTimelineTiming::wakeAfter(double duration)
{
m_timer.startOneShot(duration, FROM_HERE);
}
void AnimationTimeline::AnimationTimelineTiming::cancelWake()
{
m_timer.stop();
}
void AnimationTimeline::AnimationTimelineTiming::serviceOnNextFrame()
{
if (m_timeline->m_document && m_timeline->m_document->view())
m_timeline->m_document->view()->scheduleAnimation();
}
void AnimationTimeline::AnimationTimelineTiming::trace(Visitor* visitor)
{
visitor->trace(m_timeline);
AnimationTimeline::PlatformTiming::trace(visitor);
}
double AnimationTimeline::currentTime(bool& isNull)
{
return currentTimeInternal(isNull) * 1000;
}
double AnimationTimeline::currentTimeInternal(bool& isNull)
{
if (!m_document) {
isNull = true;
return std::numeric_limits<double>::quiet_NaN();
}
double result = m_document->animationClock().currentTime() - zeroTime();
isNull = std::isnan(result);
return result;
}
double AnimationTimeline::currentTime()
{
return currentTimeInternal() * 1000;
}
double AnimationTimeline::currentTimeInternal()
{
bool isNull;
return currentTimeInternal(isNull);
}
double AnimationTimeline::effectiveTime()
{
double time = currentTimeInternal();
return std::isnan(time) ? 0 : time;
}
void AnimationTimeline::pauseAnimationsForTesting(double pauseTime)
{
for (WillBeHeapHashSet<RefPtrWillBeMember<AnimationPlayer> >::iterator it = m_playersNeedingUpdate.begin(); it != m_playersNeedingUpdate.end(); ++it)
(*it)->pauseForTesting(pauseTime);
serviceAnimations(TimingUpdateOnDemand);
}
bool AnimationTimeline::hasOutdatedAnimationPlayer() const
{
for (WillBeHeapHashSet<RefPtrWillBeMember<AnimationPlayer> >::iterator it = m_playersNeedingUpdate.begin(); it != m_playersNeedingUpdate.end(); ++it) {
if ((*it)->outdated())
return true;
}
return false;
}
void AnimationTimeline::setOutdatedAnimationPlayer(AnimationPlayer* player)
{
ASSERT(player->outdated());
m_playersNeedingUpdate.add(player);
if (m_document && m_document->page() && !m_document->page()->animator().isServicingAnimations())
m_timing->serviceOnNextFrame();
}
size_t AnimationTimeline::numberOfActiveAnimationsForTesting() const
{
// Includes all players whose directly associated timed items
// are current or in effect.
size_t count = 0;
for (WillBeHeapHashSet<RefPtrWillBeMember<AnimationPlayer> >::iterator it = m_playersNeedingUpdate.begin(); it != m_playersNeedingUpdate.end(); ++it) {
const AnimationNode* animationNode = (*it)->source();
if ((*it)->hasStartTime())
count += (animationNode && (animationNode->isCurrent() || animationNode->isInEffect()));
}
return count;
}
#if !ENABLE(OILPAN)
void AnimationTimeline::detachFromDocument()
{
// FIXME: AnimationTimeline should keep Document alive.
m_document = nullptr;
}
#endif
void AnimationTimeline::trace(Visitor* visitor)
{
visitor->trace(m_document);
visitor->trace(m_timing);
visitor->trace(m_playersNeedingUpdate);
visitor->trace(m_players);
}
} // namespace