X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fcommon%2Feventmanager.cpp;h=fd3cf9bdbc655a44e5e8bd2b8fed03eb40570b84;hp=36682b06510304eaa9665127a71682185135c41c;hb=d9588ab4fff449eeb77ebb03a6cb1c5c91d1449d;hpb=bd6311ec1d07e4daf082b5f752ef6b46d7808430 diff --git a/src/common/eventmanager.cpp b/src/common/eventmanager.cpp index 36682b06..fd3cf9bd 100644 --- a/src/common/eventmanager.cpp +++ b/src/common/eventmanager.cpp @@ -193,6 +193,8 @@ void EventManager::dispatchEvent(Event *event) { QSet ignored; uint type = event->type(); + bool checkDupes = false; + // special handling for numeric IrcEvents if((type & ~IrcEventNumericMask) == IrcEventNumeric) { ::IrcEventNumeric *numEvent = static_cast< ::IrcEventNumeric *>(event); @@ -201,28 +203,29 @@ void EventManager::dispatchEvent(Event *event) { else { int num = numEvent->number(); if(num > 0) { - insertHandlers(registeredHandlers().value(type + num), handlers); + insertHandlers(registeredHandlers().value(type + num), handlers, false); insertFilters(registeredFilters().value(type + num), filters); + checkDupes = true; } } } // exact type - insertHandlers(registeredHandlers().value(type), handlers); + insertHandlers(registeredHandlers().value(type), handlers, checkDupes); insertFilters(registeredFilters().value(type), filters); // check if we have a generic handler for the event group if((type & EventGroupMask) != type) { - insertHandlers(registeredHandlers().value(type & EventGroupMask), handlers); + insertHandlers(registeredHandlers().value(type & EventGroupMask), handlers, true); insertFilters(registeredFilters().value(type & EventGroupMask), filters); } // now dispatch the event - QList::const_iterator it = handlers.begin(); - while(it != handlers.end() && !event->isStopped()) { + QList::const_iterator it; + for(it = handlers.begin(); it != handlers.end() && !event->isStopped(); ++it) { QObject *obj = it->object; - if(ignored.contains(obj)) // we only deliver an event once to any given object + if(ignored.contains(obj)) // object has filtered the event continue; if(filters.contains(obj)) { // we have a filter, so let's check if we want to deliver the event @@ -230,35 +233,42 @@ void EventManager::dispatchEvent(Event *event) { bool result = false; void *param[] = {Q_RETURN_ARG(bool, result).data(), Q_ARG(Event *, event).data() }; obj->qt_metacall(QMetaObject::InvokeMetaMethod, filter.methodIndex, param); - ignored.insert(obj); // don't try to deliver the event again either way - if(!result) + if(!result) { + ignored.insert(obj); continue; // mmmh, event filter told us to not accept + } } // finally, deliverance! void *param[] = {0, Q_ARG(Event *, event).data() }; obj->qt_metacall(QMetaObject::InvokeMetaMethod, it->methodIndex, param); - - ++it; } // that's it delete event; } -void EventManager::insertHandlers(const QList &newHandlers, QList &existing) { +void EventManager::insertHandlers(const QList &newHandlers, QList &existing, bool checkDupes) { foreach(const Handler &handler, newHandlers) { if(existing.isEmpty()) existing.append(handler); else { - // need to insert it at the proper position + // need to insert it at the proper position, but only if we don't yet have a handler for this event and object! + bool insert = true; + QList::iterator insertpos = existing.end(); QList::iterator it = existing.begin(); while(it != existing.end()) { - if(handler.priority > it->priority) + if(checkDupes && handler.object == it->object) { + insert = false; break; + } + if(insertpos == existing.end() && handler.priority > it->priority) + insertpos = it; + ++it; } - existing.insert(it, handler); + if(insert) + existing.insert(it, handler); } } }