#include "common.h" #include "version.h" #include "mirandaInterface.h" // Miranda - handle from DllMain CMPlugin g_plugin; //Miranda - Langpack int &hLangpack(g_plugin.m_hLang); //Miranda - HANDLE to hooked protocols ACK HANDLE hProtoAck; //popup classes handles HANDLE hPopupNotify; HANDLE hPopupError; //main add-on's object CMirfoxMiranda mirfoxMiranda; ///////////////////////////////////////////////////////////////////////////////////////// PLUGININFOEX pluginInfoEx = { sizeof(PLUGININFOEX), __PLUGIN_NAME, PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM), __DESCRIPTION, __AUTHOR, __COPYRIGHT, __AUTHORWEB, UNICODE_AWARE, { 0xe99a09b2, 0xe05b, 0x4633, { 0xaa, 0x3a, 0x5c, 0x83, 0xef, 0x1c, 0xba, 0xb6 } } }; CMPlugin::CMPlugin() : PLUGIN(MODULENAME, pluginInfoEx) {} extern "C" __declspec (dllexport) PLUGININFOEX* MirandaPluginInfoEx(DWORD) { return &pluginInfoEx; } ///////////////////////////////////////////////////////////////////////////////////////// static int __cdecl onAccListChanged(WPARAM wParam, LPARAM lParam) { if (mirfoxMiranda.getMirfoxData().Plugin_Terminated) return 0; mirfoxMiranda.onAccListChanged(wParam, lParam); return 0; } static int __cdecl onContactAdded(WPARAM wParam, LPARAM) { if (mirfoxMiranda.getMirfoxData().Plugin_Terminated) return 0; OnContactAsyncThreadArgStruct* onContactAsyncThreadArgStruct = new(OnContactAsyncThreadArgStruct); onContactAsyncThreadArgStruct->hContact = wParam; onContactAsyncThreadArgStruct->mirfoxMiranda = &mirfoxMiranda; mir_forkthread(CMirfoxMiranda::onContactAdded_async, onContactAsyncThreadArgStruct); return 0; } static int __cdecl onContactDeleted(WPARAM wParam, LPARAM) { if (mirfoxMiranda.getMirfoxData().Plugin_Terminated) return 0; mirfoxMiranda.onContactDeleted(wParam); return 0; } static int __cdecl onContactSettingChanged(WPARAM hContact, LPARAM lParam) { if (mirfoxMiranda.getMirfoxData().Plugin_Terminated) return 0; if (hContact == NULL || lParam == NULL) return 0; DBCONTACTWRITESETTING* cws = (DBCONTACTWRITESETTING*)lParam; if (!strcmp(cws->szModule, "CList")) { if (!strcmp(cws->szSetting, "Hidden")) { mirfoxMiranda.onContactSettingChanged(hContact, lParam); } if (!strcmp(cws->szSetting, "MyHandle")) { OnContactAsyncThreadArgStruct* onContactAsyncThreadArgStruct = new(OnContactAsyncThreadArgStruct); onContactAsyncThreadArgStruct->hContact = hContact; onContactAsyncThreadArgStruct->mirfoxMiranda = &mirfoxMiranda; mir_forkthread(CMirfoxMiranda::onContactSettingChanged_async, onContactAsyncThreadArgStruct); } } return 0; } ///////////////////////////////////////////////////////////////////////////////////////// // hook on ME_SYSTEM_MODULESLOADED at Load() static int onModulesLoaded(WPARAM, LPARAM) { //init popup classes POPUPCLASS puc = { 0 }; puc.cbSize = sizeof(puc); puc.flags = PCF_TCHAR; puc.pszName = "MirFox_Notify"; puc.pwszDescription = TranslateT("MirFox/Notification"); puc.colorBack = RGB(173, 206, 247); //light blue puc.colorText = GetSysColor(COLOR_WINDOWTEXT); puc.iSeconds = 3; puc.hIcon = LoadIcon(g_plugin.getInst(), MAKEINTRESOURCE(IDI_ICON_PN)); hPopupNotify = Popup_RegisterClass(&puc); puc.pszName = "MirFox_Error"; puc.pwszDescription = TranslateT("MirFox/Error"); puc.colorBack = RGB(255, 128, 128); //light red puc.colorText = GetSysColor(COLOR_WINDOWTEXT); puc.iSeconds = 20; puc.hIcon = LoadIcon(g_plugin.getInst(), MAKEINTRESOURCE(IDI_ICON_PE)); hPopupError = Popup_RegisterClass(&puc); //init refresh hooks HookEvent(ME_PROTO_ACCLISTCHANGED, onAccListChanged); HookEvent(ME_DB_CONTACT_ADDED, onContactAdded); HookEvent(ME_DB_CONTACT_DELETED, onContactDeleted); HookEvent(ME_DB_CONTACT_SETTINGCHANGED, onContactSettingChanged); return 0; } static int OnShutdown(WPARAM, LPARAM) { Popup_UnregisterClass(hPopupError); Popup_UnregisterClass(hPopupNotify); return 0; } ///////////////////////////////////////////////////////////////////////////////////////// extern "C" int __declspec(dllexport) Load(void) { HookEvent(ME_SYSTEM_MODULESLOADED, onModulesLoaded); HookEvent(ME_SYSTEM_SHUTDOWN, OnShutdown); hProtoAck = HookEvent(ME_PROTO_ACK, MirandaUtils::onProtoAck); // Ensure that the common control DLL is loaded. needed to use ICC_LISTVIEW_CLASSES control in options TODO move to InitOptions();? INITCOMMONCONTROLSEX icex = { 0 }; icex.dwSize = sizeof(icex); icex.dwICC = ICC_LISTVIEW_CLASSES; InitCommonControlsEx(&icex); mirfoxMiranda.getMirfoxData().Plugin_Terminated = false; mirfoxMiranda.getMirfoxData().setPluginState(MFENUM_PLUGIN_STATE_INIT); mirfoxMiranda.onMirandaInterfaceLoad(); InitOptions(); if (mirfoxMiranda.getMirfoxData().getPluginState() != MFENUM_PLUGIN_STATE_ERROR) mirfoxMiranda.getMirfoxData().setPluginState(MFENUM_PLUGIN_STATE_WORK); return 0; } ///////////////////////////////////////////////////////////////////////////////////////// extern "C" __declspec (dllexport) int Unload(void) { MFLogger::getInstance()->log(L"Unload: start"); // wait for csmThread, msgQueueThread and userActionThread's end mirfoxMiranda.getMirfoxData().Plugin_Terminated = true; UnhookEvent(hProtoAck); int counter = 0; const int UNLOAD_WAIT_TIME = 50; //[ms] const int MAX_UNLOAD_WAIT_COUNTER = 10; //10 * 50ms = 0,5s while (mirfoxMiranda.getMirfoxData().workerThreadsCount > 0) { MFLogger::getInstance()->log_p(L"Unload: workerThreadsCount=%d > 0, waiting. counter=%d", mirfoxMiranda.getMirfoxData().workerThreadsCount, counter); SleepEx(UNLOAD_WAIT_TIME, TRUE); counter++; } MFLogger::getInstance()->log_p(L"Unload: workerThreadsCount=%d. counter=%d", mirfoxMiranda.getMirfoxData().workerThreadsCount, counter); MFLogger::getInstance()->log(L"Unload: last log"); mirfoxMiranda.onMirandaInterfaceUnload(); DeinitOptions(); return 0; }