X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fcommon%2Feventmanager.cpp;h=6c539e72c660188b6962ed48cf371c52352e7043;hp=552765a243dccf9c83f9ad06b51e64863228f858;hb=5e9a649d94d1832e0b4f3f296e8a2df2d8741d9b;hpb=80cd03dc284fecbac2b7db1ede215f82023b72d9 diff --git a/src/common/eventmanager.cpp b/src/common/eventmanager.cpp index 552765a2..6c539e72 100644 --- a/src/common/eventmanager.cpp +++ b/src/common/eventmanager.cpp @@ -20,14 +20,22 @@ #include "eventmanager.h" +#include +#include #include #include "event.h" +#include "ircevent.h" EventManager::EventManager(QObject *parent) : QObject(parent) { } +EventManager::~EventManager() { + // pending events won't be delivered anymore, but we do need to delete them + qDeleteAll(_eventQueue); +} + QMetaEnum EventManager::eventEnum() const { if(!_enum.isValid()) { int eventEnumIndex = metaObject()->indexOfEnumerator("EventType"); @@ -66,32 +74,72 @@ QString EventManager::enumName(EventType type) const { with the actual class. Possibly (optionally) using rtti... */ -void EventManager::registerObject(QObject *object, Priority priority, const QString &methodPrefix) { - for(int i = object->metaObject()->methodOffset(); i < object->metaObject()->methodCount(); i++) { - QString methodSignature(object->metaObject()->method(i).signature()); +int EventManager::findEventType(const QString &methodSignature_, const QString &methodPrefix) const { + if(!methodSignature_.startsWith(methodPrefix)) + return -1; - if(!methodSignature.startsWith(methodPrefix)) - continue; + QString methodSignature = methodSignature_; - methodSignature = methodSignature.section('(',0,0); // chop the attribute list - methodSignature = methodSignature.mid(methodPrefix.length()); // strip prefix + methodSignature = methodSignature.section('(',0,0); // chop the attribute list + methodSignature = methodSignature.mid(methodPrefix.length()); // strip prefix - int eventType = eventEnum().keyToValue(methodSignature.toAscii()); - if(eventType < 0) { - qWarning() << Q_FUNC_INFO << QString("Could not find EventType %1").arg(methodSignature); - continue; + int eventType = -1; + + // special handling for numeric IrcEvents: IrcEvent042 gets mapped to IrcEventNumeric + 42 + if(methodSignature.length() == 8+3 && methodSignature.startsWith("IrcEvent")) { + int num = methodSignature.right(3).toUInt(); + if(num > 0) { + QString numericSig = methodSignature.left(methodSignature.length() - 3) + "Numeric"; + eventType = eventEnum().keyToValue(numericSig.toAscii()); + if(eventType < 0) { + qWarning() << Q_FUNC_INFO << "Could not find EventType" << numericSig << "for handling" << methodSignature; + return -1; + } + eventType += num; + } + } + + if(eventType < 0) + eventType = eventEnum().keyToValue(methodSignature.toAscii()); + if(eventType < 0) { + qWarning() << Q_FUNC_INFO << "Could not find EventType" << methodSignature; + return -1; + } + return eventType; +} + +void EventManager::registerObject(QObject *object, Priority priority, const QString &methodPrefix, const QString &filterPrefix) { + for(int i = object->metaObject()->methodOffset(); i < object->metaObject()->methodCount(); i++) { + QString methodSignature(object->metaObject()->method(i).signature()); + + int eventType = findEventType(methodSignature, methodPrefix); + if(eventType > 0) { + Handler handler(object, i, priority); + registeredHandlers()[eventType].append(handler); + qDebug() << "Registered event handler for" << methodSignature << "in" << object; + } + eventType = findEventType(methodSignature, filterPrefix); + if(eventType > 0) { + Handler handler(object, i, priority); + registeredFilters()[eventType].append(handler); + qDebug() << "Registered event filterer for" << methodSignature << "in" << object; } - Handler handler(object, i, priority); - registeredHandlers()[static_cast(eventType)].append(handler); - qDebug() << "Registered event handler for" << methodSignature << "in" << object; } } -void EventManager::registerEventHandler(EventType event, QObject *object, const char *slot, Priority priority) { - registerEventHandler(QList() << event, object, slot, priority); +void EventManager::registerEventFilter(EventType event, QObject *object, const char *slot) { + registerEventHandler(QList() << event, object, slot, NormalPriority, true); +} + +void EventManager::registerEventFilter(QList events, QObject *object, const char *slot) { + registerEventHandler(events, object, slot, NormalPriority, true); +} + +void EventManager::registerEventHandler(EventType event, QObject *object, const char *slot, Priority priority, bool isFilter) { + registerEventHandler(QList() << event, object, slot, priority, isFilter); } -void EventManager::registerEventHandler(QList events, QObject *object, const char *slot, Priority priority) { +void EventManager::registerEventHandler(QList events, QObject *object, const char *slot, Priority priority, bool isFilter) { int methodIndex = object->metaObject()->indexOfMethod(slot); if(methodIndex < 0) { qWarning() << Q_FUNC_INFO << QString("Slot %1 not found in object %2").arg(slot).arg(object->objectName()); @@ -99,47 +147,107 @@ void EventManager::registerEventHandler(QList events, QObject *object } Handler handler(object, methodIndex, priority); foreach(EventType event, events) { - registeredHandlers()[event].append(handler); - qDebug() << "Registered event handler for" << event << "in" << object; + if(isFilter) { + registeredFilters()[event].append(handler); + qDebug() << "Registered event filter for" << event << "in" << object; + } else { + registeredHandlers()[event].append(handler); + qDebug() << "Registered event handler for" << event << "in" << object; + } } } // not threadsafe! if we should want that, we need to add a mutexed queue somewhere in this general area. void EventManager::sendEvent(Event *event) { // qDebug() << "Sending" << event; - dispatchEvent(event); + _eventQueue.append(event); + if(_eventQueue.count() == 1) // we're not currently processing another event + processEvents(); +} + +void EventManager::customEvent(QEvent *event) { + if(event->type() == QEvent::User) { + processEvents(); + event->accept(); + } +} + +void EventManager::processEvents() { + // we only process one event at a time for now, and let Qt's own event processing come in between + if(_eventQueue.isEmpty()) + return; + dispatchEvent(_eventQueue.first()); + _eventQueue.removeFirst(); + if(_eventQueue.count()) + QCoreApplication::postEvent(this, new QEvent(QEvent::User)); } void EventManager::dispatchEvent(Event *event) { + //qDebug() << "Dispatching" << event; + // we try handlers from specialized to generic by masking the enum // build a list sorted by priorities that contains all eligible handlers QList handlers; - EventType type = event->type(); + QHash filters; + QSet ignored; + uint type = event->type(); + + // special handling for numeric IrcEvents + if((type & ~IrcEventNumericMask) == IrcEventNumeric) { + ::IrcEventNumeric *numEvent = static_cast< ::IrcEventNumeric *>(event); + if(!numEvent) + qWarning() << "Invalid event type for IrcEventNumeric!"; + else { + int num = numEvent->number(); + if(num > 0) { + insertHandlers(registeredHandlers().value(type + num), handlers); + insertFilters(registeredFilters().value(type + num), filters); + } + } + } + + // exact type insertHandlers(registeredHandlers().value(type), handlers); + insertFilters(registeredFilters().value(type), filters); // check if we have a generic handler for the event group - if((type & EventGroupMask) != type) + if((type & EventGroupMask) != type) { insertHandlers(registeredHandlers().value(type & EventGroupMask), handlers); + insertFilters(registeredFilters().value(type & EventGroupMask), filters); + } // now dispatch the event QList::const_iterator it = handlers.begin(); - while(it != handlers.end()) { + while(it != handlers.end() && !event->isStopped()) { + QObject *obj = it->object; + + if(ignored.contains(obj)) // we only deliver an event once to any given object + continue; - // TODO: check event flags here! + if(filters.contains(obj)) { // we have a filter, so let's check if we want to deliver the event + Handler filter = filters.value(obj); + 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) + continue; // mmmh, event filter told us to not accept + } + // finally, deliverance! void *param[] = {0, Q_ARG(Event *, event).data() }; - it->object->qt_metacall(QMetaObject::InvokeMetaMethod, it->methodIndex, param); + obj->qt_metacall(QMetaObject::InvokeMetaMethod, it->methodIndex, param); ++it; } - // finally, delete it + // that's it delete event; } void EventManager::insertHandlers(const QList &newHandlers, QList &existing) { - foreach(Handler handler, newHandlers) { + foreach(const Handler &handler, newHandlers) { if(existing.isEmpty()) existing.append(handler); else { @@ -154,3 +262,12 @@ void EventManager::insertHandlers(const QList &newHandlers, QList &newFilters, QHash &existing) { + foreach(const Handler &filter, newFilters) { + if(!existing.contains(filter.object)) + existing[filter.object] = filter; + } +}