blob: 05e09a4ca48b9b9877118a3151132cf402a5f9fd [file] [log] [blame]
/*
* Copyright (C) 2011 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:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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 "core/dom/ScriptedAnimationController.h"
#include "core/css/MediaQueryListListener.h"
#include "core/dom/Document.h"
#include "core/dom/FrameRequestCallback.h"
#include "core/events/Event.h"
#include "core/frame/FrameView.h"
#include "core/frame/LocalDOMWindow.h"
#include "core/inspector/InspectorInstrumentation.h"
#include "core/inspector/InspectorTraceEvents.h"
#include "core/loader/DocumentLoader.h"
namespace blink {
std::pair<EventTarget*, StringImpl*> eventTargetKey(const Event* event) {
return std::make_pair(event->target(), event->type().impl());
}
ScriptedAnimationController::ScriptedAnimationController(Document* document)
: m_document(document), m_callbackCollection(document), m_suspendCount(0) {}
DEFINE_TRACE(ScriptedAnimationController) {
visitor->trace(m_document);
visitor->trace(m_callbackCollection);
visitor->trace(m_eventQueue);
visitor->trace(m_mediaQueryListListeners);
visitor->trace(m_perFrameEvents);
}
void ScriptedAnimationController::suspend() {
++m_suspendCount;
}
void ScriptedAnimationController::resume() {
// It would be nice to put an DCHECK(m_suspendCount > 0) here, but in WK1
// resume() can be called even when suspend hasn't (if a tab was created in
// the background).
if (m_suspendCount > 0)
--m_suspendCount;
scheduleAnimationIfNeeded();
}
void ScriptedAnimationController::dispatchEventsAndCallbacksForPrinting() {
dispatchEvents(EventNames::MediaQueryListEvent);
callMediaQueryListListeners();
}
ScriptedAnimationController::CallbackId
ScriptedAnimationController::registerCallback(FrameRequestCallback* callback) {
CallbackId id = m_callbackCollection.registerCallback(callback);
scheduleAnimationIfNeeded();
return id;
}
void ScriptedAnimationController::cancelCallback(CallbackId id) {
m_callbackCollection.cancelCallback(id);
}
void ScriptedAnimationController::dispatchEvents(
const AtomicString& eventInterfaceFilter) {
HeapVector<Member<Event>> events;
if (eventInterfaceFilter.isEmpty()) {
events.swap(m_eventQueue);
m_perFrameEvents.clear();
} else {
HeapVector<Member<Event>> remaining;
for (auto& event : m_eventQueue) {
if (event && event->interfaceName() == eventInterfaceFilter) {
m_perFrameEvents.remove(eventTargetKey(event.get()));
events.append(event.release());
} else {
remaining.append(event.release());
}
}
remaining.swap(m_eventQueue);
}
for (size_t i = 0; i < events.size(); ++i) {
EventTarget* eventTarget = events[i]->target();
// FIXME: we should figure out how to make dispatchEvent properly virtual to
// avoid special casting window.
// FIXME: We should not fire events for nodes that are no longer in the
// tree.
InspectorInstrumentation::AsyncTask asyncTask(
eventTarget->getExecutionContext(), events[i]);
if (LocalDOMWindow* window = eventTarget->toLocalDOMWindow())
window->dispatchEvent(events[i], nullptr);
else
eventTarget->dispatchEvent(events[i]);
}
}
void ScriptedAnimationController::executeCallbacks(double monotonicTimeNow) {
// dispatchEvents() runs script which can cause the document to be destroyed.
if (!m_document)
return;
double highResNowMs =
1000.0 *
m_document->loader()->timing().monotonicTimeToZeroBasedDocumentTime(
monotonicTimeNow);
double legacyHighResNowMs =
1000.0 *
m_document->loader()->timing().monotonicTimeToPseudoWallTime(
monotonicTimeNow);
m_callbackCollection.executeCallbacks(highResNowMs, legacyHighResNowMs);
}
void ScriptedAnimationController::callMediaQueryListListeners() {
MediaQueryListListeners listeners;
listeners.swap(m_mediaQueryListListeners);
for (const auto& listener : listeners) {
listener->notifyMediaQueryChanged();
}
}
bool ScriptedAnimationController::hasScheduledItems() const {
if (m_suspendCount)
return false;
return !m_callbackCollection.isEmpty() || !m_eventQueue.isEmpty() ||
!m_mediaQueryListListeners.isEmpty();
}
void ScriptedAnimationController::serviceScriptedAnimations(
double monotonicTimeNow) {
if (!hasScheduledItems())
return;
callMediaQueryListListeners();
dispatchEvents();
executeCallbacks(monotonicTimeNow);
scheduleAnimationIfNeeded();
}
void ScriptedAnimationController::enqueueEvent(Event* event) {
InspectorInstrumentation::asyncTaskScheduled(
event->target()->getExecutionContext(), event->type(), event);
m_eventQueue.append(event);
scheduleAnimationIfNeeded();
}
void ScriptedAnimationController::enqueuePerFrameEvent(Event* event) {
if (!m_perFrameEvents.add(eventTargetKey(event)).isNewEntry)
return;
enqueueEvent(event);
}
void ScriptedAnimationController::enqueueMediaQueryChangeListeners(
HeapVector<Member<MediaQueryListListener>>& listeners) {
for (size_t i = 0; i < listeners.size(); ++i) {
m_mediaQueryListListeners.add(listeners[i]);
}
scheduleAnimationIfNeeded();
}
void ScriptedAnimationController::scheduleAnimationIfNeeded() {
if (!hasScheduledItems())
return;
if (!m_document)
return;
if (FrameView* frameView = m_document->view())
frameView->scheduleAnimation();
}
} // namespace blink