| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "config.h" | 5 #include "config.h" |
| 6 #include "core/frame/EventHandlerRegistry.h" | 6 #include "core/frame/EventHandlerRegistry.h" |
| 7 | 7 |
| 8 #include "core/events/ThreadLocalEventNames.h" | 8 #include "core/events/ThreadLocalEventNames.h" |
| 9 #include "core/frame/DOMWindow.h" | 9 #include "core/frame/DOMWindow.h" |
| 10 #include "core/frame/LocalFrame.h" | 10 #include "core/frame/LocalFrame.h" |
| 11 #include "core/html/HTMLFrameOwnerElement.h" | 11 #include "core/html/HTMLFrameOwnerElement.h" |
| 12 #include "core/page/Page.h" | 12 #include "core/page/Page.h" |
| 13 #include "core/page/scrolling/ScrollingCoordinator.h" | 13 #include "core/page/scrolling/ScrollingCoordinator.h" |
| 14 | 14 |
| 15 namespace WebCore { | 15 namespace WebCore { |
| 16 | 16 |
| 17 EventHandlerRegistry::HandlerState::HandlerState() |
| 18 : externalHandlerCount(0) |
| 19 { |
| 20 } |
| 21 |
| 22 EventHandlerRegistry::HandlerState::~HandlerState() |
| 23 { |
| 24 } |
| 25 |
| 17 EventHandlerRegistry::EventHandlerRegistry(FrameHost& frameHost) | 26 EventHandlerRegistry::EventHandlerRegistry(FrameHost& frameHost) |
| 18 : m_frameHost(frameHost) | 27 : m_frameHost(frameHost) |
| 19 { | 28 { |
| 20 } | 29 } |
| 21 | 30 |
| 22 EventHandlerRegistry::~EventHandlerRegistry() | 31 EventHandlerRegistry::~EventHandlerRegistry() |
| 23 { | 32 { |
| 24 checkConsistency(); | 33 checkConsistency(); |
| 25 } | 34 } |
| 26 | 35 |
| 27 bool EventHandlerRegistry::eventTypeToClass(const AtomicString& eventType, Event
HandlerClass* result) | 36 bool EventHandlerRegistry::eventTypeToClass(const AtomicString& eventType, Event
HandlerClass* result) |
| 28 { | 37 { |
| 29 if (eventType == EventTypeNames::scroll) { | 38 if (eventType == EventTypeNames::scroll) { |
| 30 *result = ScrollEvent; | 39 *result = ScrollEvent; |
| 40 } else if (eventType == EventTypeNames::wheel || eventType == EventTypeNames
::mousewheel) { |
| 41 *result = WheelEvent; |
| 31 #if ASSERT_ENABLED | 42 #if ASSERT_ENABLED |
| 32 } else if (eventType == EventTypeNames::load || eventType == EventTypeNames:
:mousemove || eventType == EventTypeNames::touchstart) { | 43 } else if (eventType == EventTypeNames::load || eventType == EventTypeNames:
:mousemove || eventType == EventTypeNames::touchstart) { |
| 33 *result = EventsForTesting; | 44 *result = EventsForTesting; |
| 34 #endif | 45 #endif |
| 35 } else { | 46 } else { |
| 36 return false; | 47 return false; |
| 37 } | 48 } |
| 38 return true; | 49 return true; |
| 39 } | 50 } |
| 40 | 51 |
| 41 const EventTargetSet* EventHandlerRegistry::eventHandlerTargets(EventHandlerClas
s handlerClass) const | 52 const EventTargetSet* EventHandlerRegistry::eventHandlerTargets(EventHandlerClas
s handlerClass) const |
| 42 { | 53 { |
| 43 checkConsistency(); | 54 checkConsistency(); |
| 44 return &m_targets[handlerClass]; | 55 return &m_handlers[handlerClass].targets; |
| 56 } |
| 57 |
| 58 unsigned EventHandlerRegistry::externalEventHandlerCount(EventHandlerClass handl
erClass) const |
| 59 { |
| 60 return m_handlers[handlerClass].externalHandlerCount; |
| 45 } | 61 } |
| 46 | 62 |
| 47 bool EventHandlerRegistry::hasEventHandlers(EventHandlerClass handlerClass) cons
t | 63 bool EventHandlerRegistry::hasEventHandlers(EventHandlerClass handlerClass) cons
t |
| 48 { | 64 { |
| 49 return m_targets[handlerClass].size(); | 65 const HandlerState& state = m_handlers[handlerClass]; |
| 66 return state.targets.size() || state.externalHandlerCount; |
| 67 } |
| 68 |
| 69 void EventHandlerRegistry::updateExternalHandlerCount(ChangeOperation op, EventH
andlerClass handlerClass) |
| 70 { |
| 71 if (op == Add) { |
| 72 m_handlers[handlerClass].externalHandlerCount++; |
| 73 } else { |
| 74 ASSERT(op == Remove); |
| 75 ASSERT(m_handlers[handlerClass].externalHandlerCount > 0); |
| 76 m_handlers[handlerClass].externalHandlerCount--; |
| 77 } |
| 50 } | 78 } |
| 51 | 79 |
| 52 bool EventHandlerRegistry::updateEventHandlerTargets(ChangeOperation op, EventHa
ndlerClass handlerClass, EventTarget* target) | 80 bool EventHandlerRegistry::updateEventHandlerTargets(ChangeOperation op, EventHa
ndlerClass handlerClass, EventTarget* target) |
| 53 { | 81 { |
| 54 EventTargetSet* targets = &m_targets[handlerClass]; | 82 EventTargetSet* targets = &m_handlers[handlerClass].targets; |
| 55 if (op == Add) { | 83 if (op == Add) { |
| 56 if (!targets->add(target).isNewEntry) { | 84 if (!targets->add(target).isNewEntry) { |
| 57 // Just incremented refcount, no real change. | 85 // Just incremented refcount, no real change. |
| 58 return false; | 86 return false; |
| 59 } | 87 } |
| 60 } else { | 88 } else { |
| 61 ASSERT(op == Remove || op == RemoveAll); | 89 ASSERT(op == Remove || op == RemoveAll); |
| 62 ASSERT(op == RemoveAll || targets->contains(target)); | 90 ASSERT(op == RemoveAll || targets->contains(target)); |
| 63 | 91 |
| 64 if (op == RemoveAll) { | 92 if (op == RemoveAll) { |
| 65 if (!targets->contains(target)) | 93 if (!targets->contains(target)) |
| 66 return false; | 94 return false; |
| 67 targets->removeAll(target); | 95 targets->removeAll(target); |
| 68 } else { | 96 } else { |
| 69 if (!targets->remove(target)) { | 97 if (!targets->remove(target)) { |
| 70 // Just decremented refcount, no real update. | 98 // Just decremented refcount, no real update. |
| 71 return false; | 99 return false; |
| 72 } | 100 } |
| 73 } | 101 } |
| 74 } | 102 } |
| 75 return true; | 103 return true; |
| 76 } | 104 } |
| 77 | 105 |
| 78 void EventHandlerRegistry::updateEventHandlerInternal(ChangeOperation op, EventH
andlerClass handlerClass, EventTarget* target) | 106 void EventHandlerRegistry::updateEventHandlerInternal(ChangeOperation op, EventH
andlerClass handlerClass, EventTarget* target) |
| 79 { | 107 { |
| 80 bool hadHandlers = hasEventHandlers(handlerClass); | 108 bool hadHandlers = hasEventHandlers(handlerClass); |
| 81 updateEventHandlerTargets(op, handlerClass, target); | 109 if (target) { |
| 110 updateEventHandlerTargets(op, handlerClass, target); |
| 111 } else { |
| 112 updateExternalHandlerCount(op, handlerClass); |
| 113 } |
| 82 bool hasHandlers = hasEventHandlers(handlerClass); | 114 bool hasHandlers = hasEventHandlers(handlerClass); |
| 83 | 115 |
| 84 if (hadHandlers != hasHandlers) { | 116 if (hadHandlers != hasHandlers) { |
| 85 notifyHasHandlersChanged(handlerClass, hasHandlers); | 117 notifyHasHandlersChanged(handlerClass, hasHandlers); |
| 86 } | 118 } |
| 87 checkConsistency(); | 119 checkConsistency(); |
| 88 } | 120 } |
| 89 | 121 |
| 90 void EventHandlerRegistry::updateEventHandlerOfType(ChangeOperation op, const At
omicString& eventType, EventTarget* target) | 122 void EventHandlerRegistry::updateEventHandlerOfType(ChangeOperation op, const At
omicString& eventType, EventTarget* target) |
| 91 { | 123 { |
| 92 EventHandlerClass handlerClass; | 124 EventHandlerClass handlerClass; |
| 93 if (!eventTypeToClass(eventType, &handlerClass)) | 125 if (!eventTypeToClass(eventType, &handlerClass)) |
| 94 return; | 126 return; |
| 95 updateEventHandlerInternal(op, handlerClass, target); | 127 updateEventHandlerInternal(op, handlerClass, target); |
| 96 } | 128 } |
| 97 | 129 |
| 130 void EventHandlerRegistry::didAddExternalEventHandler(const AtomicString& eventT
ype) |
| 131 { |
| 132 updateEventHandlerOfType(Add, eventType, 0); |
| 133 } |
| 134 |
| 135 void EventHandlerRegistry::didRemoveExternalEventHandler(const AtomicString& eve
ntType) |
| 136 { |
| 137 updateEventHandlerOfType(Remove, eventType, 0); |
| 138 } |
| 139 |
| 98 void EventHandlerRegistry::didAddEventHandler(EventTarget& target, const AtomicS
tring& eventType) | 140 void EventHandlerRegistry::didAddEventHandler(EventTarget& target, const AtomicS
tring& eventType) |
| 99 { | 141 { |
| 100 updateEventHandlerOfType(Add, eventType, &target); | 142 updateEventHandlerOfType(Add, eventType, &target); |
| 101 } | 143 } |
| 102 | 144 |
| 103 void EventHandlerRegistry::didRemoveEventHandler(EventTarget& target, const Atom
icString& eventType) | 145 void EventHandlerRegistry::didRemoveEventHandler(EventTarget& target, const Atom
icString& eventType) |
| 104 { | 146 { |
| 105 updateEventHandlerOfType(Remove, eventType, &target); | 147 updateEventHandlerOfType(Remove, eventType, &target); |
| 106 } | 148 } |
| 107 | 149 |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 148 continue; | 190 continue; |
| 149 } | 191 } |
| 150 for (unsigned count = target.getEventListeners(eventTypes[i]).size(); co
unt > 0; --count) | 192 for (unsigned count = target.getEventListeners(eventTypes[i]).size(); co
unt > 0; --count) |
| 151 updateEventHandlerInternal(op, handlerClass, &target); | 193 updateEventHandlerInternal(op, handlerClass, &target); |
| 152 } | 194 } |
| 153 } | 195 } |
| 154 | 196 |
| 155 void EventHandlerRegistry::notifyHasHandlersChanged(EventHandlerClass handlerCla
ss, bool hasActiveHandlers) | 197 void EventHandlerRegistry::notifyHasHandlersChanged(EventHandlerClass handlerCla
ss, bool hasActiveHandlers) |
| 156 { | 198 { |
| 157 ScrollingCoordinator* scrollingCoordinator = m_frameHost.page().scrollingCoo
rdinator(); | 199 ScrollingCoordinator* scrollingCoordinator = m_frameHost.page().scrollingCoo
rdinator(); |
| 200 LocalFrame* mainFrame = m_frameHost.page().mainFrame(); |
| 158 | 201 |
| 159 switch (handlerClass) { | 202 switch (handlerClass) { |
| 160 case ScrollEvent: | 203 case ScrollEvent: |
| 161 if (scrollingCoordinator) | 204 if (scrollingCoordinator) |
| 162 scrollingCoordinator->updateHaveScrollEventHandlers(); | 205 scrollingCoordinator->updateHaveScrollEventHandlers(); |
| 163 break; | 206 break; |
| 207 case WheelEvent: |
| 208 if (mainFrame) { |
| 209 // TODO(skyostil): This notification is wired up to a black hole, so
remove it. |
| 210 mainFrame->notifyChromeClientWheelEventHandlerCountChanged(); |
| 211 } |
| 212 if (scrollingCoordinator) |
| 213 scrollingCoordinator->updateHaveWheelEventHandlers(); |
| 214 break; |
| 164 #if ASSERT_ENABLED | 215 #if ASSERT_ENABLED |
| 165 case EventsForTesting: | 216 case EventsForTesting: |
| 166 break; | 217 break; |
| 167 #endif | 218 #endif |
| 168 default: | 219 default: |
| 169 ASSERT_NOT_REACHED(); | 220 ASSERT_NOT_REACHED(); |
| 170 break; | 221 break; |
| 171 } | 222 } |
| 172 } | 223 } |
| 173 | 224 |
| 174 void EventHandlerRegistry::trace(Visitor* visitor) | 225 void EventHandlerRegistry::trace(Visitor* visitor) |
| 175 { | 226 { |
| 176 visitor->registerWeakMembers<EventHandlerRegistry, &EventHandlerRegistry::cl
earWeakMembers>(this); | 227 visitor->registerWeakMembers<EventHandlerRegistry, &EventHandlerRegistry::cl
earWeakMembers>(this); |
| 177 } | 228 } |
| 178 | 229 |
| 179 void EventHandlerRegistry::clearWeakMembers(Visitor* visitor) | 230 void EventHandlerRegistry::clearWeakMembers(Visitor* visitor) |
| 180 { | 231 { |
| 181 Vector<EventTarget*> deadTargets; | 232 Vector<EventTarget*> deadTargets; |
| 182 for (size_t i = 0; i < EventHandlerClassCount; ++i) { | 233 for (size_t i = 0; i < EventHandlerClassCount; ++i) { |
| 183 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(i); | 234 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(i); |
| 184 const EventTargetSet* targets = &m_targets[handlerClass]; | 235 const EventTargetSet* targets = &m_handlers[handlerClass].targets; |
| 185 for (EventTargetSet::const_iterator it = targets->begin(); it != targets
->end(); ++it) { | 236 for (EventTargetSet::const_iterator it = targets->begin(); it != targets
->end(); ++it) { |
| 186 Node* node = it->key->toNode(); | 237 Node* node = it->key->toNode(); |
| 187 DOMWindow* window = it->key->toDOMWindow(); | 238 DOMWindow* window = it->key->toDOMWindow(); |
| 188 if (node && !visitor->isAlive(node)) { | 239 if (node && !visitor->isAlive(node)) { |
| 189 deadTargets.append(node); | 240 deadTargets.append(node); |
| 190 } else if (window && !visitor->isAlive(window)) { | 241 } else if (window && !visitor->isAlive(window)) { |
| 191 deadTargets.append(window); | 242 deadTargets.append(window); |
| 192 } | 243 } |
| 193 } | 244 } |
| 194 } | 245 } |
| 195 for (size_t i = 0; i < deadTargets.size(); ++i) | 246 for (size_t i = 0; i < deadTargets.size(); ++i) |
| 196 didRemoveAllEventHandlers(*deadTargets[i]); | 247 didRemoveAllEventHandlers(*deadTargets[i]); |
| 197 } | 248 } |
| 198 | 249 |
| 199 void EventHandlerRegistry::documentDetached(Document& document) | 250 void EventHandlerRegistry::documentDetached(Document& document) |
| 200 { | 251 { |
| 201 // Remove all event targets under the detached document. | 252 // Remove all event targets under the detached document. |
| 202 for (size_t handlerClassIndex = 0; handlerClassIndex < EventHandlerClassCoun
t; ++handlerClassIndex) { | 253 for (size_t handlerClassIndex = 0; handlerClassIndex < EventHandlerClassCoun
t; ++handlerClassIndex) { |
| 203 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(handlerC
lassIndex); | 254 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(handlerC
lassIndex); |
| 204 Vector<EventTarget*> targetsToRemove; | 255 Vector<EventTarget*> targetsToRemove; |
| 205 const EventTargetSet* targets = &m_targets[handlerClass]; | 256 const EventTargetSet* targets = &m_handlers[handlerClass].targets; |
| 206 for (EventTargetSet::const_iterator iter = targets->begin(); iter != tar
gets->end(); ++iter) { | 257 for (EventTargetSet::const_iterator iter = targets->begin(); iter != tar
gets->end(); ++iter) { |
| 207 if (Node* node = iter->key->toNode()) { | 258 if (Node* node = iter->key->toNode()) { |
| 208 for (Document* doc = &node->document(); doc; doc = doc->ownerEle
ment() ? &doc->ownerElement()->document() : 0) { | 259 for (Document* doc = &node->document(); doc; doc = doc->ownerEle
ment() ? &doc->ownerElement()->document() : 0) { |
| 209 if (doc == &document) { | 260 if (doc == &document) { |
| 210 targetsToRemove.append(iter->key); | 261 targetsToRemove.append(iter->key); |
| 211 break; | 262 break; |
| 212 } | 263 } |
| 213 } | 264 } |
| 214 } else if (iter->key->toDOMWindow()) { | 265 } else if (iter->key->toDOMWindow()) { |
| 215 // DOMWindows may outlive their documents, so we shouldn't remov
e their handlers | 266 // DOMWindows may outlive their documents, so we shouldn't remov
e their handlers |
| 216 // here. | 267 // here. |
| 217 } else { | 268 } else { |
| 218 ASSERT_NOT_REACHED(); | 269 ASSERT_NOT_REACHED(); |
| 219 } | 270 } |
| 220 } | 271 } |
| 221 for (size_t i = 0; i < targetsToRemove.size(); ++i) | 272 for (size_t i = 0; i < targetsToRemove.size(); ++i) |
| 222 updateEventHandlerInternal(RemoveAll, handlerClass, targetsToRemove[
i]); | 273 updateEventHandlerInternal(RemoveAll, handlerClass, targetsToRemove[
i]); |
| 223 } | 274 } |
| 224 } | 275 } |
| 225 | 276 |
| 226 void EventHandlerRegistry::checkConsistency() const | 277 void EventHandlerRegistry::checkConsistency() const |
| 227 { | 278 { |
| 228 #if ASSERT_ENABLED | 279 #if ASSERT_ENABLED |
| 229 for (size_t i = 0; i < EventHandlerClassCount; ++i) { | 280 for (size_t i = 0; i < EventHandlerClassCount; ++i) { |
| 230 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(i); | 281 EventHandlerClass handlerClass = static_cast<EventHandlerClass>(i); |
| 231 const EventTargetSet* targets = &m_targets[handlerClass]; | 282 const EventTargetSet* targets = &m_handlers[handlerClass].targets; |
| 232 for (EventTargetSet::const_iterator iter = targets->begin(); iter != tar
gets->end(); ++iter) { | 283 for (EventTargetSet::const_iterator iter = targets->begin(); iter != tar
gets->end(); ++iter) { |
| 233 if (Node* node = iter->key->toNode()) { | 284 if (Node* node = iter->key->toNode()) { |
| 234 // See the comment for |documentDetached| if either of these ass
ertions fails. | 285 // See the comment for |documentDetached| if either of these ass
ertions fails. |
| 235 ASSERT(node->document().frameHost()); | 286 ASSERT(node->document().frameHost()); |
| 236 ASSERT(node->document().frameHost() == &m_frameHost); | 287 ASSERT(node->document().frameHost() == &m_frameHost); |
| 237 } else if (DOMWindow* window = iter->key->toDOMWindow()) { | 288 } else if (DOMWindow* window = iter->key->toDOMWindow()) { |
| 238 // If any of these assertions fail, DOMWindow failed to unregist
er its handlers | 289 // If any of these assertions fail, DOMWindow failed to unregist
er its handlers |
| 239 // properly. | 290 // properly. |
| 240 ASSERT(window->frame()); | 291 ASSERT(window->frame()); |
| 241 ASSERT(window->frame()->host()); | 292 ASSERT(window->frame()->host()); |
| 242 ASSERT(window->frame()->host() == &m_frameHost); | 293 ASSERT(window->frame()->host() == &m_frameHost); |
| 243 } | 294 } |
| 244 } | 295 } |
| 245 } | 296 } |
| 246 #endif // ASSERT_ENABLED | 297 #endif // ASSERT_ENABLED |
| 247 } | 298 } |
| 248 | 299 |
| 249 } // namespace WebCore | 300 } // namespace WebCore |
| OLD | NEW |