#include "stdafx.h" mir_cs csNotifications; OBJLIST lstNotifications(1); std::map mp_Classes; wchar_t wszTempDir[MAX_PATH]; void __stdcall ShowToastNotification(void* p) { std::unique_ptr td((ToastData*)p); if (!db_get_b(0, "Popup", "ModuleIsEnabled", 1)) return; ptrT imagePath; if (td->hContact != NULL && td->hContact != INVALID_CONTACT_ID) { const char* szProto = GetContactProto(td->hContact); if (ProtoServiceExists(szProto, PS_GETAVATARINFO)) { PROTO_AVATAR_INFORMATION pai = { 0 }; pai.hContact = td->hContact; if (CallProtoService(szProto, PS_GETAVATARINFO, 0, (LPARAM)&pai) == GAIR_SUCCESS) { imagePath = mir_tstrdup(pai.filename); } } if (imagePath == NULL) { if (td->iType == 1 && td->hBitmap && !szProto) { imagePath = ToasterImage(td->hBitmap); } else if (td->iType == 2 && td->hIcon && !szProto) { imagePath = ToasterImage(td->hIcon); } else if (szProto) { imagePath = ToasterImage(szProto); } } } else { if (td->hIcon) { imagePath = ToasterImage(td->hIcon); } } ToastNotification *notification = new ToastNotification(td->tszText, td->tszTitle, imagePath); HRESULT hr = notification->Initialize(); if (SUCCEEDED(hr)) { ToastHandlerData *thd = new ToastHandlerData(); thd->hContact = td->hContact; thd->vPopupData = td->vPopupData; thd->pPopupProc = td->pPopupProc; thd->tstNotification = notification; notification->Show(new ToastEventHandler(thd)); lstNotifications.insert(notification); } else { delete notification; } } static INT_PTR GetPopupData(WPARAM wParam, LPARAM) { return (INT_PTR)((ToastEventHandler*)wParam)->_thd->vPopupData; } static INT_PTR GetPopupContact(WPARAM wParam, LPARAM) { return (INT_PTR)((ToastEventHandler*)wParam)->_thd->hContact; } static INT_PTR CreatePopup(WPARAM wParam, LPARAM) { POPUPDATA *ppd = (POPUPDATA*)wParam; ptrW text(mir_a2u(ppd->lpzText)); ptrW contactName(mir_a2u(ppd->lpzContactName)); ToastData *td = new ToastData(ppd->lchContact, contactName, text, ppd->lchIcon); td->vPopupData = ppd->PluginData; td->pPopupProc = ppd->PluginWindowProc; CallFunctionAsync(&ShowToastNotification, td); return 0; } static INT_PTR CreatePopupW(WPARAM wParam, LPARAM) { POPUPDATAW *ppd = (POPUPDATAW*)wParam; ToastData *td = new ToastData(ppd->lchContact, ppd->lpwzContactName, ppd->lpwzText, ppd->lchIcon); td->vPopupData = ppd->PluginData; td->pPopupProc = ppd->PluginWindowProc; CallFunctionAsync(&ShowToastNotification, td); return 0; } static INT_PTR CreatePopup2(WPARAM wParam, LPARAM) { POPUPDATA2 *ppd = (POPUPDATA2*)wParam; ptrW text, title; if (ppd->flags & PU2_UNICODE) { text = mir_wstrdup(ppd->lpwzText); title = mir_wstrdup(ppd->lpwzTitle); } else { text = mir_a2u(ppd->lpzText); title = mir_a2u(ppd->lpzTitle); } ToastData *td = NULL; if (ppd->hbmAvatar) td = new ToastData(ppd->lchContact, title, text, ppd->hbmAvatar); else td = new ToastData(ppd->lchContact, title, text, ppd->lchIcon); td->vPopupData = ppd->PluginData; td->pPopupProc = ppd->PluginWindowProc; CallFunctionAsync(&ShowToastNotification, td); return 0; } static INT_PTR RegisterClass(WPARAM, LPARAM lParam) { POPUPCLASS *pc = (POPUPCLASS*)lParam; ClassData *cd = new ClassData(pc->flags, pc->hIcon); cd->pPopupProc = pc->PluginWindowProc; mp_Classes[pc->pszName] = cd; return (INT_PTR)cd->handle; } static INT_PTR CreateClassPopup(WPARAM, LPARAM lParam) { POPUPDATACLASS *ppc = (POPUPDATACLASS*)lParam; auto it = mp_Classes.find(ppc->pszClassName); if (it != mp_Classes.end()) { ToastData *td = NULL; if (it->second->iFlags & PCF_TCHAR) { td = new ToastData(ppc->hContact, ppc->ptszTitle, ppc->ptszText, it->second->hIcon); } else { td = new ToastData(ppc->hContact, mir_utf8decodeT(ppc->pszTitle), mir_utf8decodeT(ppc->pszText), it->second->hIcon); } td->vPopupData = ppc->PluginData; td->pPopupProc = it->second->pPopupProc; CallFunctionAsync(&ShowToastNotification, td); } return 0; } static INT_PTR UnRegisterClass(WPARAM, LPARAM lParam) { HANDLE h = (HANDLE)lParam; for (auto it = mp_Classes.begin(); it != mp_Classes.end(); it++) { if (it->second->handle == h) { delete it->second; mp_Classes.erase(it); break; } } return 0; } void CleanupClasses() { for (auto it = mp_Classes.begin(); it != mp_Classes.end(); ++it) { delete it->second; } mp_Classes.clear(); } static INT_PTR PopupQuery(WPARAM wParam, LPARAM) { switch (wParam) { case PUQS_ENABLEPOPUPS: { bool enabled = db_get_b(0, "Popup", "ModuleIsEnabled", 1) != 0; if (!enabled) db_set_b(0, "Popup", "ModuleIsEnabled", 1); return !enabled; } break; case PUQS_DISABLEPOPUPS: { bool enabled = db_get_b(0, "Popup", "ModuleIsEnabled", 1) != 0; if (enabled) db_set_b(0, "Popup", "ModuleIsEnabled", 0); CallFunctionAsync(&HideAllToasts, NULL); return enabled; } break; case PUQS_GETSTATUS: return db_get_b(0, "Popup", "ModuleIsEnabled", 1); default: return 1; } } static INT_PTR ShowMessage(WPARAM wParam, LPARAM) { ptrT tszText(mir_utf8decodeT((char*)wParam)); ToastData *td = new ToastData(NULL, NULL, tszText, HICON(0)); CallFunctionAsync(&ShowToastNotification, td); return 0; } static INT_PTR ShowMessageW(WPARAM wParam, LPARAM) { ToastData *td = new ToastData(NULL, NULL, (wchar_t*)wParam, HICON(0)); CallFunctionAsync(&ShowToastNotification, td); return 0; } void __stdcall HideAllToasts(void*) { mir_cslock lck(csNotifications); while (lstNotifications.getCount()) { lstNotifications[0].Hide(); lstNotifications.remove(0); } } void InitServices() { CreateServiceFunction(MS_POPUP_SHOWMESSAGE, ShowMessage); CreateServiceFunction(MS_POPUP_SHOWMESSAGEW, ShowMessageW); CreateServiceFunction(MS_POPUP_ADDPOPUP, CreatePopup); CreateServiceFunction(MS_POPUP_ADDPOPUPW, CreatePopupW); CreateServiceFunction(MS_POPUP_ADDPOPUP2, CreatePopup2); CreateServiceFunction(MS_POPUP_QUERY, PopupQuery); CreateServiceFunction(MS_POPUP_ADDPOPUPCLASS, CreateClassPopup); CreateServiceFunction(MS_POPUP_REGISTERCLASS, RegisterClass); CreateServiceFunction(MS_POPUP_UNREGISTERCLASS, UnRegisterClass); CreateServiceFunction(MS_POPUP_GETPLUGINDATA, GetPopupData); CreateServiceFunction(MS_POPUP_GETCONTACT, GetPopupContact); }