/* Miranda NG: the free IM client for Microsoft* Windows* Copyright (C) 2012-18 Miranda NG team, all portions of this codebase are copyrighted to the people listed in contributors.txt. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "stdafx.h" static int CompareProtos(const PROTOCOLDESCRIPTOR *p1, const PROTOCOLDESCRIPTOR *p2) { return strcmp(p1->szName, p2->szName); } LIST protos(10, CompareProtos); extern HANDLE hAckEvent; ///////////////////////////////////////////////////////////////////////////////////////// MIR_APP_DLL(PROTOCOLDESCRIPTOR*) Proto_IsProtocolLoaded(const char *szProtoName) { if (szProtoName == nullptr) return nullptr; PROTOCOLDESCRIPTOR tmp; tmp.szName = (char*)szProtoName; return protos.find(&tmp); } ///////////////////////////////////////////////////////////////////////////////////////// MIR_APP_DLL(void) Proto_EnumProtocols(int *nProtos, PROTOCOLDESCRIPTOR ***pProtos) { if (nProtos) *nProtos = protos.getCount(); if (pProtos) *pProtos = protos.getArray(); } ///////////////////////////////////////////////////////////////////////////////////////// MIR_APP_DLL(INT_PTR) ProtoBroadcastAck(const char *szModule, MCONTACT hContact, int type, int result, HANDLE hProcess, LPARAM lParam) { ACKDATA ack = { sizeof(ACKDATA), szModule, hContact, type, result, hProcess, lParam }; return NotifyEventHooks(hAckEvent, 0, (LPARAM)&ack); } ///////////////////////////////////////////////////////////////////////////////////////// void PROTO_INTERFACE::setAllContactStatuses(int iStatus, bool bSkipChats) { for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (isChatRoom(hContact)) { if (!bSkipChats && iStatus == ID_STATUS_OFFLINE) { ptrW wszRoom(getWStringA(hContact, "ChatRoomID")); if (wszRoom != nullptr) Chat_Control(m_szModuleName, wszRoom, SESSION_OFFLINE); } } else setWord(hContact, "Status", iStatus); } } ///////////////////////////////////////////////////////////////////////////////////////// // protocol constructor & destructor PROTO_INTERFACE::PROTO_INTERFACE(const char *pszModuleName, const wchar_t *ptszUserName) { m_iVersion = 2; m_iStatus = m_iDesiredStatus = ID_STATUS_OFFLINE; m_szModuleName = mir_strdup(pszModuleName); m_hProtoIcon = IcoLib_IsManaged(Skin_LoadProtoIcon(pszModuleName, ID_STATUS_ONLINE)); m_tszUserName = mir_wstrdup(ptszUserName); db_set_resident(m_szModuleName, "Status"); } PROTO_INTERFACE::~PROTO_INTERFACE() { mir_free(m_szModuleName); mir_free(m_tszUserName); WindowList_Destroy(m_hWindowList); } ///////////////////////////////////////////////////////////////////////////////////////// // protocol services MIR_APP_DLL(void) ProtoCreateService(PROTO_INTERFACE *pThis, const char* szService, ProtoServiceFunc serviceProc) { char str[MAXMODULELABELLENGTH * 2]; strncpy_s(str, pThis->m_szModuleName, _TRUNCATE); strncat_s(str, szService, _TRUNCATE); ::CreateServiceFunctionObj(str, (MIRANDASERVICEOBJ)*(void**)&serviceProc, pThis); } MIR_APP_DLL(void) ProtoCreateServiceParam(PROTO_INTERFACE *pThis, const char* szService, ProtoServiceFuncParam serviceProc, LPARAM lParam) { char str[MAXMODULELABELLENGTH * 2]; strncpy_s(str, pThis->m_szModuleName, _TRUNCATE); strncat_s(str, szService, _TRUNCATE); ::CreateServiceFunctionObjParam(str, (MIRANDASERVICEOBJPARAM)*(void**)&serviceProc, pThis, lParam); } ///////////////////////////////////////////////////////////////////////////////////////// // protocol events MIR_APP_DLL(void) ProtoHookEvent(PROTO_INTERFACE *pThis, const char* szEvent, ProtoEventFunc handler) { ::HookEventObj(szEvent, (MIRANDAHOOKOBJ)*(void**)&handler, pThis); } MIR_APP_DLL(HANDLE) ProtoCreateHookableEvent(PROTO_INTERFACE *pThis, const char* szName) { char str[MAXMODULELABELLENGTH * 2]; strncpy_s(str, pThis->m_szModuleName, _TRUNCATE); strncat_s(str, szName, _TRUNCATE); return CreateHookableEvent(str); } ///////////////////////////////////////////////////////////////////////////////////////// // protocol threads MIR_APP_DLL(void) ProtoForkThread(PROTO_INTERFACE *pThis, ProtoThreadFunc pFunc, void *param) { UINT threadID; CloseHandle((HANDLE)::mir_forkthreadowner((pThreadFuncOwner)*(void**)&pFunc, pThis, param, &threadID)); } MIR_APP_DLL(HANDLE) ProtoForkThreadEx(PROTO_INTERFACE *pThis, ProtoThreadFunc pFunc, void *param, UINT* threadID) { UINT lthreadID; return (HANDLE)::mir_forkthreadowner((pThreadFuncOwner)*(void**)&pFunc, pThis, param, threadID ? threadID : <hreadID); } ///////////////////////////////////////////////////////////////////////////////////////// // protocol windows void PROTO_INTERFACE::WindowSubscribe(HWND hwnd) { if (m_hWindowList == nullptr) m_hWindowList = WindowList_Create(); WindowList_Add(m_hWindowList, hwnd, 0); } void PROTO_INTERFACE::WindowUnsubscribe(HWND hwnd) { WindowList_Remove(m_hWindowList, hwnd); } ///////////////////////////////////////////////////////////////////////////////////////// // avatar support MIR_APP_DLL(LPCTSTR) ProtoGetAvatarExtension(int format) { if (format == PA_FORMAT_PNG) return L".png"; if (format == PA_FORMAT_JPEG) return L".jpg"; if (format == PA_FORMAT_ICON) return L".ico"; if (format == PA_FORMAT_BMP) return L".bmp"; if (format == PA_FORMAT_GIF) return L".gif"; if (format == PA_FORMAT_SWF) return L".swf"; if (format == PA_FORMAT_XML) return L".xml"; return L""; } MIR_APP_DLL(int) ProtoGetAvatarFormat(const wchar_t *ptszFileName) { if (ptszFileName == nullptr) return PA_FORMAT_UNKNOWN; const wchar_t *ptszExt = wcsrchr(ptszFileName, '.'); if (ptszExt == nullptr) return PA_FORMAT_UNKNOWN; if (!wcsicmp(ptszExt, L".png")) return PA_FORMAT_PNG; if (!wcsicmp(ptszExt, L".jpg") || !wcsicmp(ptszExt, L".jpeg")) return PA_FORMAT_JPEG; if (!wcsicmp(ptszExt, L".ico")) return PA_FORMAT_ICON; if (!wcsicmp(ptszExt, L".bmp") || !wcsicmp(ptszExt, L".rle")) return PA_FORMAT_BMP; if (!wcsicmp(ptszExt, L".gif")) return PA_FORMAT_GIF; if (!wcsicmp(ptszExt, L".swf")) return PA_FORMAT_SWF; if (!wcsicmp(ptszExt, L".xml")) return PA_FORMAT_XML; return PA_FORMAT_UNKNOWN; } MIR_APP_DLL(int) ProtoGetBufferFormat(const void *pBuffer, const wchar_t **ptszExtension) { if (!memcmp(pBuffer, "\x89PNG", 4)) { if (ptszExtension) *ptszExtension = L".png"; return PA_FORMAT_PNG; } if (!memcmp(pBuffer, "GIF8", 4)) { if (ptszExtension) *ptszExtension = L".gif"; return PA_FORMAT_GIF; } if (!memicmp(pBuffer, "= 0 && iFileType < _countof(wszMimeTypes)) return wszMimeTypes[iFileType]; return nullptr; } MIR_APP_DLL(int) ProtoGetAvatarFormatByMimeType(const wchar_t *pwszMimeType) { for (int i = 0; i < _countof(wszMimeTypes); i++) if (!mir_wstrcmp(pwszMimeType, wszMimeTypes[i])) return i; return PA_FORMAT_UNKNOWN; } ///////////////////////////////////////////////////////////////////////////////////////// // default PROTO_INTERFACE method implementations MCONTACT PROTO_INTERFACE::AddToList(int, PROTOSEARCHRESULT*) { return 0; // error } MCONTACT PROTO_INTERFACE::AddToListByEvent(int, int, MEVENT) { return 0; // error } int PROTO_INTERFACE::Authorize(MEVENT) { return 1; // error } int PROTO_INTERFACE::AuthDeny(MEVENT, const wchar_t*) { return 1; // error } int PROTO_INTERFACE::AuthRecv(MCONTACT, PROTORECVEVENT*) { return 1; // error } int PROTO_INTERFACE::AuthRequest(MCONTACT, const wchar_t*) { return 1; // error } HANDLE PROTO_INTERFACE::FileAllow(MCONTACT, HANDLE, const wchar_t*) { return nullptr; // error } int PROTO_INTERFACE::FileCancel(MCONTACT, HANDLE) { return 1; // error } int PROTO_INTERFACE::FileDeny(MCONTACT, HANDLE, const wchar_t*) { return 1; // error } int PROTO_INTERFACE::FileResume(HANDLE, int*, const wchar_t**) { return 1; // error } DWORD_PTR PROTO_INTERFACE::GetCaps(int, MCONTACT) { return 0; // empty value } int PROTO_INTERFACE::GetInfo(MCONTACT, int) { return 1; // error } HANDLE PROTO_INTERFACE::SearchBasic(const wchar_t*) { return nullptr; // error } HANDLE PROTO_INTERFACE::SearchByEmail(const wchar_t*) { return nullptr; // error } HANDLE PROTO_INTERFACE::SearchByName(const wchar_t*, const wchar_t*, const wchar_t*) { return nullptr; // error } HWND PROTO_INTERFACE::SearchAdvanced(HWND) { return nullptr; // error } HWND PROTO_INTERFACE::CreateExtendedSearchUI(HWND) { return nullptr; // error } int PROTO_INTERFACE::RecvContacts(MCONTACT, PROTORECVEVENT*) { return 1; // error } int PROTO_INTERFACE::RecvFile(MCONTACT hContact, PROTORECVFILET *evt) { return ::Proto_RecvFile(hContact, evt); // default file receiver } int PROTO_INTERFACE::RecvMsg(MCONTACT hContact, PROTORECVEVENT *evt) { ::Proto_RecvMessage(hContact, evt); // default message receiver return 0; } int PROTO_INTERFACE::RecvUrl(MCONTACT, PROTORECVEVENT*) { return 1; // error } int PROTO_INTERFACE::SendContacts(MCONTACT, int, int, MCONTACT*) { return 1; // error } HANDLE PROTO_INTERFACE::SendFile(MCONTACT, const wchar_t*, wchar_t**) { return nullptr; // error } int PROTO_INTERFACE::SendMsg(MCONTACT, int, const char*) { return 0; // error } int PROTO_INTERFACE::SendUrl(MCONTACT, int, const char*) { return 1; // error } int PROTO_INTERFACE::SetApparentMode(MCONTACT, int) { return 1; // error } int PROTO_INTERFACE::SetStatus(int) { return 1; // you better declare it } HANDLE PROTO_INTERFACE::GetAwayMsg(MCONTACT) { return nullptr; // no away message } int PROTO_INTERFACE::RecvAwayMsg(MCONTACT, int, PROTORECVEVENT*) { return 1; // error } int PROTO_INTERFACE::SetAwayMsg(int, const wchar_t*) { return 1; // error } int PROTO_INTERFACE::UserIsTyping(MCONTACT, int) { return 1; // error } int PROTO_INTERFACE::OnEvent(PROTOEVENTTYPE, WPARAM, LPARAM) { return 1; // not an error, vitally important }