• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25 
26 #include "config.h"
27 #include "core/svg/animation/SMILTimeContainer.h"
28 
29 #include "core/animation/AnimationClock.h"
30 #include "core/animation/AnimationTimeline.h"
31 #include "core/dom/ElementTraversal.h"
32 #include "core/frame/FrameView.h"
33 #include "core/svg/SVGSVGElement.h"
34 #include "core/svg/animation/SVGSMILElement.h"
35 
36 namespace blink {
37 
38 static const double initialFrameDelay = 0.025;
39 
40 #if !ENABLE(OILPAN)
41 // Every entry-point that calls updateAnimations() should instantiate a
42 // DiscardScope to prevent deletion of the ownerElement (and hence itself.)
43 class DiscardScope {
44 public:
DiscardScope(SVGSVGElement & timeContainerOwner)45     explicit DiscardScope(SVGSVGElement& timeContainerOwner) : m_discardScopeElement(&timeContainerOwner) { }
46 
47 private:
48     RefPtr<SVGSVGElement> m_discardScopeElement;
49 };
50 #endif
51 
SMILTimeContainer(SVGSVGElement & owner)52 SMILTimeContainer::SMILTimeContainer(SVGSVGElement& owner)
53     : m_beginTime(0)
54     , m_pauseTime(0)
55     , m_resumeTime(0)
56     , m_accumulatedActiveTime(0)
57     , m_presetStartTime(0)
58     , m_frameSchedulingState(Idle)
59     , m_documentOrderIndexesDirty(false)
60     , m_wakeupTimer(this, &SMILTimeContainer::wakeupTimerFired)
61     , m_ownerSVGElement(owner)
62 #if ENABLE(ASSERT)
63     , m_preventScheduledAnimationsChanges(false)
64 #endif
65 {
66 }
67 
~SMILTimeContainer()68 SMILTimeContainer::~SMILTimeContainer()
69 {
70     cancelAnimationFrame();
71     ASSERT(!m_wakeupTimer.isActive());
72 #if ENABLE(ASSERT)
73     ASSERT(!m_preventScheduledAnimationsChanges);
74 #endif
75 }
76 
schedule(SVGSMILElement * animation,SVGElement * target,const QualifiedName & attributeName)77 void SMILTimeContainer::schedule(SVGSMILElement* animation, SVGElement* target, const QualifiedName& attributeName)
78 {
79     ASSERT(animation->timeContainer() == this);
80     ASSERT(target);
81     ASSERT(animation->hasValidAttributeName());
82 
83 #if ENABLE(ASSERT)
84     ASSERT(!m_preventScheduledAnimationsChanges);
85 #endif
86 
87     ElementAttributePair key(target, attributeName);
88     OwnPtrWillBeMember<AnimationsLinkedHashSet>& scheduled = m_scheduledAnimations.add(key, nullptr).storedValue->value;
89     if (!scheduled)
90         scheduled = adoptPtrWillBeNoop(new AnimationsLinkedHashSet);
91     ASSERT(!scheduled->contains(animation));
92     scheduled->add(animation);
93 
94     SMILTime nextFireTime = animation->nextProgressTime();
95     if (nextFireTime.isFinite())
96         notifyIntervalsChanged();
97 }
98 
unschedule(SVGSMILElement * animation,SVGElement * target,const QualifiedName & attributeName)99 void SMILTimeContainer::unschedule(SVGSMILElement* animation, SVGElement* target, const QualifiedName& attributeName)
100 {
101     ASSERT(animation->timeContainer() == this);
102 
103 #if ENABLE(ASSERT)
104     ASSERT(!m_preventScheduledAnimationsChanges);
105 #endif
106 
107     ElementAttributePair key(target, attributeName);
108     GroupedAnimationsMap::iterator it = m_scheduledAnimations.find(key);
109     ASSERT(it != m_scheduledAnimations.end());
110     AnimationsLinkedHashSet* scheduled = it->value.get();
111     ASSERT(scheduled);
112     AnimationsLinkedHashSet::iterator itAnimation = scheduled->find(animation);
113     ASSERT(itAnimation != scheduled->end());
114     scheduled->remove(itAnimation);
115 
116     if (scheduled->isEmpty())
117         m_scheduledAnimations.remove(it);
118 }
119 
hasAnimations() const120 bool SMILTimeContainer::hasAnimations() const
121 {
122     return !m_scheduledAnimations.isEmpty();
123 }
124 
hasPendingSynchronization() const125 bool SMILTimeContainer::hasPendingSynchronization() const
126 {
127     return m_frameSchedulingState == SynchronizeAnimations && m_wakeupTimer.isActive() && !m_wakeupTimer.nextFireInterval();
128 }
129 
notifyIntervalsChanged()130 void SMILTimeContainer::notifyIntervalsChanged()
131 {
132     if (!isStarted())
133         return;
134     // Schedule updateAnimations() to be called asynchronously so multiple intervals
135     // can change with updateAnimations() only called once at the end.
136     if (hasPendingSynchronization())
137         return;
138     cancelAnimationFrame();
139     scheduleWakeUp(0, SynchronizeAnimations);
140 }
141 
elapsed() const142 SMILTime SMILTimeContainer::elapsed() const
143 {
144     if (!m_beginTime)
145         return 0;
146 
147     if (isPaused())
148         return m_accumulatedActiveTime;
149 
150     return currentTime() + m_accumulatedActiveTime - lastResumeTime();
151 }
152 
isPaused() const153 bool SMILTimeContainer::isPaused() const
154 {
155     return m_pauseTime;
156 }
157 
isStarted() const158 bool SMILTimeContainer::isStarted() const
159 {
160     return m_beginTime;
161 }
162 
begin()163 void SMILTimeContainer::begin()
164 {
165     RELEASE_ASSERT(!m_beginTime);
166     double now = currentTime();
167 
168     // If 'm_presetStartTime' is set, the timeline was modified via setElapsed() before the document began.
169     // In this case pass on 'seekToTime=true' to updateAnimations().
170     m_beginTime = now - m_presetStartTime;
171 #if !ENABLE(OILPAN)
172     DiscardScope discardScope(m_ownerSVGElement);
173 #endif
174     SMILTime earliestFireTime = updateAnimations(SMILTime(m_presetStartTime), m_presetStartTime ? true : false);
175     m_presetStartTime = 0;
176 
177     if (m_pauseTime) {
178         m_pauseTime = now;
179         // If updateAnimations() caused new syncbase instance to be generated,
180         // we don't want to cancel those. Excepting that, no frame should've
181         // been scheduled at this point.
182         ASSERT(m_frameSchedulingState == Idle || m_frameSchedulingState == SynchronizeAnimations);
183     } else if (!hasPendingSynchronization()) {
184         ASSERT(isTimelineRunning());
185         // If the timeline is running, and there's pending animation updates,
186         // always perform the first update after the timeline was started using
187         // the wake-up mechanism.
188         if (earliestFireTime.isFinite()) {
189             SMILTime delay = earliestFireTime - elapsed();
190             scheduleWakeUp(std::max(initialFrameDelay, delay.value()), SynchronizeAnimations);
191         }
192     }
193 }
194 
pause()195 void SMILTimeContainer::pause()
196 {
197     ASSERT(!isPaused());
198     m_pauseTime = currentTime();
199 
200     if (m_beginTime) {
201         m_accumulatedActiveTime += m_pauseTime - lastResumeTime();
202         cancelAnimationFrame();
203     }
204     m_resumeTime = 0;
205 }
206 
resume()207 void SMILTimeContainer::resume()
208 {
209     ASSERT(isPaused());
210     m_resumeTime = currentTime();
211 
212     m_pauseTime = 0;
213     scheduleWakeUp(0, SynchronizeAnimations);
214 }
215 
setElapsed(SMILTime time)216 void SMILTimeContainer::setElapsed(SMILTime time)
217 {
218     // If the documment didn't begin yet, record a new start time, we'll seek to once its possible.
219     if (!m_beginTime) {
220         m_presetStartTime = time.value();
221         return;
222     }
223 
224     cancelAnimationFrame();
225 
226     double now = currentTime();
227     m_beginTime = now - time.value();
228     m_resumeTime = 0;
229     if (m_pauseTime) {
230         m_pauseTime = now;
231         m_accumulatedActiveTime = time.value();
232     } else {
233         m_accumulatedActiveTime = 0;
234     }
235 
236 #if ENABLE(ASSERT)
237     m_preventScheduledAnimationsChanges = true;
238 #endif
239     GroupedAnimationsMap::iterator end = m_scheduledAnimations.end();
240     for (GroupedAnimationsMap::iterator it = m_scheduledAnimations.begin(); it != end; ++it) {
241         if (!it->key.first)
242             continue;
243 
244         AnimationsLinkedHashSet* scheduled = it->value.get();
245         for (AnimationsLinkedHashSet::const_iterator itAnimation = scheduled->begin(), itAnimationEnd = scheduled->end(); itAnimation != itAnimationEnd; ++itAnimation)
246             (*itAnimation)->reset();
247     }
248 #if ENABLE(ASSERT)
249     m_preventScheduledAnimationsChanges = false;
250 #endif
251 
252     updateAnimationsAndScheduleFrameIfNeeded(time, true);
253 }
254 
isTimelineRunning() const255 bool SMILTimeContainer::isTimelineRunning() const
256 {
257     return m_beginTime && !isPaused();
258 }
259 
scheduleAnimationFrame(SMILTime fireTime)260 void SMILTimeContainer::scheduleAnimationFrame(SMILTime fireTime)
261 {
262     ASSERT(isTimelineRunning() && fireTime.isFinite());
263     ASSERT(!m_wakeupTimer.isActive());
264 
265     SMILTime delay = fireTime - elapsed();
266     if (delay.value() < AnimationTimeline::s_minimumDelay) {
267         serviceOnNextFrame();
268     } else {
269         scheduleWakeUp(delay.value() - AnimationTimeline::s_minimumDelay, FutureAnimationFrame);
270     }
271 }
272 
cancelAnimationFrame()273 void SMILTimeContainer::cancelAnimationFrame()
274 {
275     m_frameSchedulingState = Idle;
276     m_wakeupTimer.stop();
277 }
278 
scheduleWakeUp(double delayTime,FrameSchedulingState frameSchedulingState)279 void SMILTimeContainer::scheduleWakeUp(double delayTime, FrameSchedulingState frameSchedulingState)
280 {
281     ASSERT(frameSchedulingState == SynchronizeAnimations || frameSchedulingState == FutureAnimationFrame);
282     m_wakeupTimer.startOneShot(delayTime, FROM_HERE);
283     m_frameSchedulingState = frameSchedulingState;
284 }
285 
wakeupTimerFired(Timer<SMILTimeContainer> *)286 void SMILTimeContainer::wakeupTimerFired(Timer<SMILTimeContainer>*)
287 {
288     ASSERT(m_frameSchedulingState == SynchronizeAnimations || m_frameSchedulingState == FutureAnimationFrame);
289     if (m_frameSchedulingState == FutureAnimationFrame) {
290         ASSERT(isTimelineRunning());
291         m_frameSchedulingState = Idle;
292         serviceOnNextFrame();
293     } else {
294         m_frameSchedulingState = Idle;
295         updateAnimationsAndScheduleFrameIfNeeded(elapsed());
296     }
297 }
298 
updateDocumentOrderIndexes()299 void SMILTimeContainer::updateDocumentOrderIndexes()
300 {
301     unsigned timingElementCount = 0;
302     for (SVGSMILElement* element = Traversal<SVGSMILElement>::firstWithin(m_ownerSVGElement); element; element = Traversal<SVGSMILElement>::next(*element, &m_ownerSVGElement))
303         element->setDocumentOrderIndex(timingElementCount++);
304     m_documentOrderIndexesDirty = false;
305 }
306 
307 struct PriorityCompare {
PriorityCompareblink::PriorityCompare308     PriorityCompare(SMILTime elapsed) : m_elapsed(elapsed) {}
operator ()blink::PriorityCompare309     bool operator()(const RefPtrWillBeMember<SVGSMILElement>& a, const RefPtrWillBeMember<SVGSMILElement>& b)
310     {
311         // FIXME: This should also consider possible timing relations between the elements.
312         SMILTime aBegin = a->intervalBegin();
313         SMILTime bBegin = b->intervalBegin();
314         // Frozen elements need to be prioritized based on their previous interval.
315         aBegin = a->isFrozen() && m_elapsed < aBegin ? a->previousIntervalBegin() : aBegin;
316         bBegin = b->isFrozen() && m_elapsed < bBegin ? b->previousIntervalBegin() : bBegin;
317         if (aBegin == bBegin)
318             return a->documentOrderIndex() < b->documentOrderIndex();
319         return aBegin < bBegin;
320     }
321     SMILTime m_elapsed;
322 };
323 
document() const324 Document& SMILTimeContainer::document() const
325 {
326     return m_ownerSVGElement.document();
327 }
328 
currentTime() const329 double SMILTimeContainer::currentTime() const
330 {
331     return document().animationClock().currentTime();
332 }
333 
serviceOnNextFrame()334 void SMILTimeContainer::serviceOnNextFrame()
335 {
336     if (document().view()) {
337         document().view()->scheduleAnimation();
338         m_frameSchedulingState = AnimationFrame;
339     }
340 }
341 
serviceAnimations(double monotonicAnimationStartTime)342 void SMILTimeContainer::serviceAnimations(double monotonicAnimationStartTime)
343 {
344     if (m_frameSchedulingState != AnimationFrame)
345         return;
346 
347     m_frameSchedulingState = Idle;
348     updateAnimationsAndScheduleFrameIfNeeded(elapsed());
349 }
350 
updateAnimationsAndScheduleFrameIfNeeded(SMILTime elapsed,bool seekToTime)351 void SMILTimeContainer::updateAnimationsAndScheduleFrameIfNeeded(SMILTime elapsed, bool seekToTime)
352 {
353 #if !ENABLE(OILPAN)
354     DiscardScope discardScope(m_ownerSVGElement);
355 #endif
356     SMILTime earliestFireTime = updateAnimations(elapsed, seekToTime);
357     // If updateAnimations() ended up triggering a synchronization (most likely
358     // via syncbases), then give that priority.
359     if (hasPendingSynchronization())
360         return;
361 
362     if (!isTimelineRunning())
363         return;
364 
365     if (!earliestFireTime.isFinite())
366         return;
367 
368     scheduleAnimationFrame(earliestFireTime);
369 }
370 
updateAnimations(SMILTime elapsed,bool seekToTime)371 SMILTime SMILTimeContainer::updateAnimations(SMILTime elapsed, bool seekToTime)
372 {
373     SMILTime earliestFireTime = SMILTime::unresolved();
374 
375 #if ENABLE(ASSERT)
376     // This boolean will catch any attempts to schedule/unschedule scheduledAnimations during this critical section.
377     // Similarly, any elements removed will unschedule themselves, so this will catch modification of animationsToApply.
378     m_preventScheduledAnimationsChanges = true;
379 #endif
380 
381     if (m_documentOrderIndexesDirty)
382         updateDocumentOrderIndexes();
383 
384     WillBeHeapHashSet<ElementAttributePair> invalidKeys;
385     typedef WillBeHeapVector<RefPtrWillBeMember<SVGSMILElement> > AnimationsVector;
386     AnimationsVector animationsToApply;
387     for (GroupedAnimationsMap::iterator it = m_scheduledAnimations.begin(), end = m_scheduledAnimations.end(); it != end; ++it) {
388         if (!it->key.first || it->value->isEmpty()) {
389             invalidKeys.add(it->key);
390             continue;
391         }
392 
393         AnimationsLinkedHashSet* scheduled = it->value.get();
394 
395         // Sort according to priority. Elements with later begin time have higher priority.
396         // In case of a tie, document order decides.
397         // FIXME: This should also consider timing relationships between the elements. Dependents
398         // have higher priority.
399         AnimationsVector scheduledAnimations;
400         copyToVector(*scheduled, scheduledAnimations);
401         std::sort(scheduledAnimations.begin(), scheduledAnimations.end(), PriorityCompare(elapsed));
402 
403         SVGSMILElement* resultElement = 0;
404         for (AnimationsVector::const_iterator itAnimation = scheduledAnimations.begin(), itAnimationEnd = scheduledAnimations.end(); itAnimation != itAnimationEnd; ++itAnimation) {
405             SVGSMILElement* animation = itAnimation->get();
406             ASSERT(animation->timeContainer() == this);
407             ASSERT(animation->targetElement());
408             ASSERT(animation->hasValidAttributeName());
409 
410             // Results are accumulated to the first animation that animates and contributes to a particular element/attribute pair.
411             // FIXME: we should ensure that resultElement is of an appropriate type.
412             if (!resultElement) {
413                 if (!animation->hasValidAttributeType())
414                     continue;
415                 resultElement = animation;
416             }
417 
418             // This will calculate the contribution from the animation and add it to the resultsElement.
419             if (!animation->progress(elapsed, resultElement, seekToTime) && resultElement == animation)
420                 resultElement = 0;
421 
422             SMILTime nextFireTime = animation->nextProgressTime();
423             if (nextFireTime.isFinite())
424                 earliestFireTime = std::min(nextFireTime, earliestFireTime);
425         }
426 
427         if (resultElement)
428             animationsToApply.append(resultElement);
429     }
430     m_scheduledAnimations.removeAll(invalidKeys);
431 
432     std::sort(animationsToApply.begin(), animationsToApply.end(), PriorityCompare(elapsed));
433 
434     unsigned animationsToApplySize = animationsToApply.size();
435     if (!animationsToApplySize) {
436 #if ENABLE(ASSERT)
437         m_preventScheduledAnimationsChanges = false;
438 #endif
439         return earliestFireTime;
440     }
441 
442     // Apply results to target elements.
443     for (unsigned i = 0; i < animationsToApplySize; ++i)
444         animationsToApply[i]->applyResultsToTarget();
445 
446 #if ENABLE(ASSERT)
447     m_preventScheduledAnimationsChanges = false;
448 #endif
449 
450     for (unsigned i = 0; i < animationsToApplySize; ++i) {
451         if (animationsToApply[i]->inDocument() && animationsToApply[i]->isSVGDiscardElement()) {
452             RefPtrWillBeRawPtr<SVGSMILElement> animDiscard = animationsToApply[i];
453             RefPtrWillBeRawPtr<SVGElement> targetElement = animDiscard->targetElement();
454             if (targetElement && targetElement->inDocument()) {
455                 targetElement->remove(IGNORE_EXCEPTION);
456                 ASSERT(!targetElement->inDocument());
457             }
458 
459             if (animDiscard->inDocument()) {
460                 animDiscard->remove(IGNORE_EXCEPTION);
461                 ASSERT(!animDiscard->inDocument());
462             }
463         }
464     }
465     return earliestFireTime;
466 }
467 
trace(Visitor * visitor)468 void SMILTimeContainer::trace(Visitor* visitor)
469 {
470 #if ENABLE(OILPAN)
471     visitor->trace(m_scheduledAnimations);
472 #endif
473 }
474 
475 }
476