EventDispatcher.cpp [plain text]
#include "config.h"
#include "EventDispatcher.h"
#include "CompositionEvent.h"
#include "EventContext.h"
#include "EventPath.h"
#include "Frame.h"
#include "FrameLoader.h"
#include "FrameView.h"
#include "HTMLInputElement.h"
#include "InputEvent.h"
#include "KeyboardEvent.h"
#include "Logging.h"
#include "MouseEvent.h"
#include "ScopedEventQueue.h"
#include "ScriptDisallowedScope.h"
#include "ShadowRoot.h"
#include "TextEvent.h"
#include "TouchEvent.h"
namespace WebCore {
void EventDispatcher::dispatchScopedEvent(Node& node, Event& event)
{
event.setTarget(EventPath::eventTargetRespectingTargetRules(node));
ScopedEventQueue::singleton().enqueueEvent(event);
}
static void callDefaultEventHandlersInBubblingOrder(Event& event, const EventPath& path)
{
if (path.isEmpty())
return;
path.contextAt(0).node()->defaultEventHandler(event);
ASSERT(!event.defaultPrevented());
if (event.defaultHandled() || !event.bubbles())
return;
size_t size = path.size();
for (size_t i = 1; i < size; ++i) {
path.contextAt(i).node()->defaultEventHandler(event);
ASSERT(!event.defaultPrevented());
if (event.defaultHandled())
return;
}
}
static void dispatchEventInDOM(Event& event, const EventPath& path)
{
event.setEventPhase(Event::CAPTURING_PHASE);
for (size_t i = path.size() - 1; i > 0; --i) {
const EventContext& eventContext = path.contextAt(i);
if (eventContext.currentTarget() == eventContext.target())
continue;
eventContext.handleLocalEvents(event);
if (event.propagationStopped())
return;
}
event.setEventPhase(Event::AT_TARGET);
path.contextAt(0).handleLocalEvents(event);
if (event.propagationStopped())
return;
size_t size = path.size();
for (size_t i = 1; i < size; ++i) {
const EventContext& eventContext = path.contextAt(i);
if (eventContext.currentTarget() == eventContext.target())
event.setEventPhase(Event::AT_TARGET);
else if (event.bubbles())
event.setEventPhase(Event::BUBBLING_PHASE);
else
continue;
eventContext.handleLocalEvents(event);
if (event.propagationStopped())
return;
}
}
static bool shouldSuppressEventDispatchInDOM(Node& node, Event& event)
{
if (!event.isTrusted())
return false;
auto frame = node.document().frame();
if (!frame)
return false;
if (!frame->mainFrame().loader().shouldSuppressTextInputFromEditing())
return false;
if (is<TextEvent>(event)) {
auto& textEvent = downcast<TextEvent>(event);
return textEvent.isKeyboard() || textEvent.isComposition();
}
return is<CompositionEvent>(event) || is<InputEvent>(event) || is<KeyboardEvent>(event);
}
void EventDispatcher::dispatchEvent(Node& node, Event& event)
{
ASSERT_WITH_SECURITY_IMPLICATION(ScriptDisallowedScope::InMainThread::isEventDispatchAllowedInSubtree(node));
LOG(Events, "EventDispatcher::dispatchEvent %s on node %s", event.type().string().utf8().data(), node.nodeName().utf8().data());
auto protectedNode = makeRef(node);
auto protectedView = makeRefPtr(node.document().view());
EventPath eventPath { node, event };
ChildNodesLazySnapshot::takeChildNodesLazySnapshot();
event.resetBeforeDispatch();
event.setTarget(EventPath::eventTargetRespectingTargetRules(node));
if (!event.target())
return;
InputElementClickState clickHandlingState;
if (is<HTMLInputElement>(node))
downcast<HTMLInputElement>(node).willDispatchEvent(event, clickHandlingState);
if (shouldSuppressEventDispatchInDOM(node, event))
event.stopPropagation();
if (!event.propagationStopped() && !eventPath.isEmpty()) {
event.setEventPath(eventPath);
dispatchEventInDOM(event, eventPath);
}
event.resetAfterDispatch();
if (clickHandlingState.stateful)
downcast<HTMLInputElement>(node).didDispatchClickEvent(event, clickHandlingState);
if (!event.defaultPrevented() && !event.defaultHandled() && !event.isDefaultEventHandlerIgnored()) {
auto* finalTarget = event.target();
event.setTarget(EventPath::eventTargetRespectingTargetRules(node));
callDefaultEventHandlersInBubblingOrder(event, eventPath);
event.setTarget(finalTarget);
}
}
template<typename T>
static void dispatchEventWithType(const Vector<T*>& targets, Event& event)
{
ASSERT(targets.size() >= 1);
ASSERT(*targets.begin());
EventPath eventPath { targets };
event.setTarget(*targets.begin());
event.setEventPath(eventPath);
event.resetBeforeDispatch();
dispatchEventInDOM(event, eventPath);
event.resetAfterDispatch();
}
void EventDispatcher::dispatchEvent(const Vector<EventTarget*>& targets, Event& event)
{
dispatchEventWithType<EventTarget>(targets, event);
}
void EventDispatcher::dispatchEvent(const Vector<Element*>& targets, Event& event)
{
dispatchEventWithType<Element>(targets, event);
}
}