Обработка событий в неуправляемом C++
Для обработки собственных событий C++ необходимо настроить источник и приемник событий с помощью атрибутов event_source и event_receiver, соответственно, указав type=native. Эти атрибуты позволяют классам, к которым они применены, порождать и обрабатывать события в собственном контексте, не связанном с моделью COM.
Объявление событий
В классе источника событий используйте в объявлении метода ключевое слово __event, чтобы объявить метод как событие. Метод должен только объявляться, но не определяться; в противном случае возникнет ошибка компилятора, поскольку компилятор определяет этот метод неявно при его преобразовании в событие. Собственные события могут быть методами с нулевым или большим количеством параметров. Возвращаемый тип может быть пустым или любым целочисленным типом.
Определение обработчиков событий
В классе приемника событий необходимо указать обработчики событий, которые представляют собой методы с подписями (возвращаемые типы, соглашения о вызовах и аргументы), соответствующие событию, которое они будут обрабатывать.
Прикрепление обработчиков событий к событиям
Также в классе приемника событий следует использовать встроенную функцию __hook, чтобы связать события с обработчиками, и функцию __unhook, чтобы разъединить их. Можно прикрепить несколько событий к обработчику событий либо несколько обработчиков событий к одному событию.
Запуск событий
Для порождения события просто вызовите метод, объявленный как событие в классе источника события. Если обработчики прикреплены к событию, они будут вызваны.
Код собственного события C++
В следующем примере демонстрируется порождение события в собственном коде C++. Для компиляции и выполнения примера см. комментарии в коде.
Пример
Код
// evh_native.cpp
#include <stdio.h>
[event_source(native)]
class CSource {
public:
__event void MyEvent(int nValue);
};
[event_receiver(native)]
class CReceiver {
public:
void MyHandler1(int nValue) {
printf_s("MyHandler1 was called with value %d.\n", nValue);
}
void MyHandler2(int nValue) {
printf_s("MyHandler2 was called with value %d.\n", nValue);
}
void hookEvent(CSource* pSource) {
__hook(&CSource::MyEvent, pSource, &CReceiver::MyHandler1);
__hook(&CSource::MyEvent, pSource, &CReceiver::MyHandler2);
}
void unhookEvent(CSource* pSource) {
__unhook(&CSource::MyEvent, pSource, &CReceiver::MyHandler1);
__unhook(&CSource::MyEvent, pSource, &CReceiver::MyHandler2);
}
};
int main() {
CSource source;
CReceiver receiver;
receiver.hookEvent(&source);
__raise source.MyEvent(123);
receiver.unhookEvent(&source);
}
Output
MyHandler2 was called with value 123.
MyHandler1 was called with value 123.