/* Copyright (C) 2013 Miranda NG Project (http://miranda-ng.org) 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 version 2 of the License. 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, see . */ #include "stdafx.h" CVkProto::CVkProto(const char *szModuleName, const TCHAR *ptszUserName) : PROTO(szModuleName, ptszUserName), m_arRequestsQueue(10), m_sendIds(3, PtrKeySortT), m_msgId(1) { InitQueue(); CreateProtoService(PS_CREATEACCMGRUI, &CVkProto::SvcCreateAccMgrUI); CreateProtoService(PS_GETAVATARINFOT, &CVkProto::SvcGetAvatarInfo); CreateProtoService(PS_GETAVATARCAPS, &CVkProto::SvcGetAvatarCaps); TCHAR descr[512]; mir_sntprintf(descr, SIZEOF(descr), TranslateT("%s server connection"), m_tszUserName); NETLIBUSER nlu = {sizeof(nlu)}; nlu.flags = NUF_INCOMING | NUF_OUTGOING | NUF_HTTPCONNS | NUF_TCHAR; nlu.szSettingsModule = m_szModuleName; nlu.szSettingsModule = m_szModuleName; nlu.ptszDescriptiveName = descr; m_hNetlibUser = (HANDLE)CallService(MS_NETLIB_REGISTERUSER, 0, (LPARAM)&nlu); m_defaultGroup = getTStringA("ProtoGroup"); if (m_defaultGroup == NULL) m_defaultGroup = mir_tstrdup( TranslateT("VKontakte")); Clist_CreateGroup(NULL, m_defaultGroup); db_set_resident(m_szModuleName, "Status"); // Set all contacts offline -- in case we crashed SetAllContactStatuses(ID_STATUS_OFFLINE); } CVkProto::~CVkProto() { UninitQueue(); } int CVkProto::OnModulesLoaded(WPARAM wParam, LPARAM lParam) { return 0; } int CVkProto::OnPreShutdown(WPARAM wParam, LPARAM lParam) { m_bTerminated = true; SetEvent(m_evRequestsQueue); return 0; } ////////////////////////////////////////////////////////////////////////////// DWORD_PTR CVkProto::GetCaps(int type, HANDLE hContact) { switch(type) { case PFLAGNUM_1: return PF1_IM | PF1_CHAT | PF1_SERVERCLIST | PF1_AUTHREQ | PF1_BASICSEARCH | PF1_SEARCHBYEMAIL | PF1_SEARCHBYNAME | PF1_MODEMSG; case PFLAGNUM_2: return PF2_ONLINE | PF2_INVISIBLE | PF2_ONTHEPHONE | PF2_IDLE; // | PF2_SHORTAWAY; case PFLAGNUM_3: return PF2_ONLINE; // | PF2_SHORTAWAY; case PFLAGNUM_4: return PF4_NOCUSTOMAUTH | PF4_FORCEADDED | PF4_IMSENDUTF | PF4_AVATARS | PF4_SUPPORTTYPING | PF4_NOAUTHDENYREASON | PF4_IMSENDOFFLINE; case PFLAGNUM_5: return PF2_ONTHEPHONE; case PFLAG_MAXLENOFMESSAGE: return 2000; case PFLAG_UNIQUEIDTEXT: return (DWORD_PTR)"VK ID"; case PFLAG_UNIQUEIDSETTING: return (DWORD_PTR)"ID"; } return 0; } ////////////////////////////////////////////////////////////////////////////// int CVkProto::RecvMsg(HANDLE hContact, PROTORECVEVENT *pre) { Proto_RecvMessage(hContact, pre); return 0; } ////////////////////////////////////////////////////////////////////////////// struct TFakeAckParams { __inline TFakeAckParams(HANDLE _hContact, int _msgid) : hContact(_hContact), msgid(_msgid) {} HANDLE hContact; int msgid; }; void CVkProto::SendMsgAck(void *param) { TFakeAckParams *ack = (TFakeAckParams*)param; Sleep(100); ProtoBroadcastAck(ack->hContact, ACKTYPE_MESSAGE, ACKRESULT_SUCCESS, (HANDLE)ack->msgid, 0); delete ack; } int CVkProto::SendMsg(HANDLE hContact, int flags, const char *msg) { LONG userID = getDword(hContact, "ID", -1); if (userID == -1) return 0; ptrA szMsg; if (flags & PREF_UTF) szMsg = mir_strdup(msg); else if (flags & PREF_UNICODE) msg = mir_utf8encodeW((wchar_t*)&msg[strlen(msg)+1]); else msg = mir_utf8encode(msg); char szID[40]; _itoa(userID, szID, 10); HttpParam params[] = { { "type", "0" }, { "uid", szID }, { "message", szMsg }, { "access_token", m_szAccessToken } }; PushAsyncHttpRequest(REQUEST_GET, "/method/messages.send.json", true, &CVkProto::OnSendMessage, SIZEOF(params), params); ULONG msgId = ::InterlockedIncrement(&m_msgId); ForkThread(&CVkProto::SendMsgAck, new TFakeAckParams(hContact, msgId)); return msgId; } void CVkProto::OnSendMessage(NETLIBHTTPREQUEST *reply, void*) { debugLogA("CVkProto::OnSendMessage %d", reply->resultCode); if (reply->resultCode != 200) return; JSONROOT pRoot(reply->pData); if ( !CheckJsonResult(pRoot)) return; JSONNODE *pResponse = json_get(pRoot, "response"); if (pResponse != NULL) m_sendIds.insert((HANDLE)json_as_int(pResponse)); } ////////////////////////////////////////////////////////////////////////////// int CVkProto::SetStatus(int iNewStatus) { if (m_iDesiredStatus == iNewStatus || iNewStatus == ID_STATUS_IDLE) return 0; int oldStatus = m_iStatus; m_iDesiredStatus = iNewStatus; if (iNewStatus == ID_STATUS_OFFLINE) { if ( IsOnline()) ShutdownSession(); m_iStatus = m_iDesiredStatus = ID_STATUS_OFFLINE; ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus); } else if (m_hWorkerThread == NULL && !(m_iStatus >= ID_STATUS_CONNECTING && m_iStatus < ID_STATUS_CONNECTING + MAX_CONNECT_RETRIES)) { m_iStatus = ID_STATUS_CONNECTING; ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus); m_hWorkerThread = ForkThreadEx(&CVkProto::WorkerThread, 0, NULL); } else { if ( IsOnline()) SetServerStatus(iNewStatus); m_iStatus = iNewStatus; ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus); } return 0; } ////////////////////////////////////////////////////////////////////////////// int CVkProto::OnEvent(PROTOEVENTTYPE event, WPARAM wParam, LPARAM lParam) { switch(event) { case EV_PROTO_ONLOAD: return OnModulesLoaded(wParam,lParam); case EV_PROTO_ONEXIT: return OnPreShutdown(wParam,lParam); case EV_PROTO_ONOPTIONS: return OnOptionsInit(wParam,lParam); } return 1; } ////////////////////////////////////////////////////////////////////////////// HANDLE CVkProto::SearchBasic(const PROTOCHAR* id) { return 0; } HANDLE CVkProto::SearchByEmail(const PROTOCHAR* email) { return 0; } HANDLE CVkProto::SearchByName(const PROTOCHAR* nick, const PROTOCHAR* firstName, const PROTOCHAR* lastName) { return 0; } HANDLE CVkProto::AddToList(int flags, PROTOSEARCHRESULT* psr) { return NULL; } int CVkProto::AuthRequest(HANDLE hContact,const PROTOCHAR *message) { return 0; } int CVkProto::Authorize(HANDLE hDbEvent) { //if (!hDbEvent) return 1; } int CVkProto::AuthDeny(HANDLE hDbEvent, const PROTOCHAR *reason) { //if (!hDbEvent || isOffline()) return 1; } int CVkProto::UserIsTyping(HANDLE hContact, int type) { return 0; } HANDLE CVkProto::AddToListByEvent(int flags,int iContact,HANDLE hDbEvent) { return NULL; } int CVkProto::AuthRecv(HANDLE hContact,PROTORECVEVENT *) { return 1; } HANDLE CVkProto::ChangeInfo(int type,void *info_data) { MessageBoxA(0,"ChangeInfo","",0); return NULL; } HANDLE CVkProto::FileAllow(HANDLE hContact,HANDLE hTransfer,const PROTOCHAR *path) { return NULL; } int CVkProto::FileCancel(HANDLE hContact,HANDLE hTransfer) { return 1; } int CVkProto::FileDeny(HANDLE hContact,HANDLE hTransfer,const PROTOCHAR *reason) { return 1; } int CVkProto::FileResume(HANDLE hTransfer,int *action,const PROTOCHAR **filename) { return 1; } int CVkProto::GetInfo(HANDLE hContact, int infoType) { // TODO: Most probably some ProtoAck should be here instead return 1; } HWND CVkProto::SearchAdvanced(HWND owner) { return NULL; } HWND CVkProto::CreateExtendedSearchUI(HWND owner) { return NULL; } int CVkProto::RecvContacts(HANDLE hContact,PROTORECVEVENT *) { return 1; } int CVkProto::RecvFile(HANDLE hContact,PROTORECVFILET *) { return 1; } int CVkProto::RecvUrl(HANDLE hContact,PROTORECVEVENT *) { return 1; } int CVkProto::SendContacts(HANDLE hContact,int flags,int nContacts,HANDLE *hContactsList) { return 1; } HANDLE CVkProto::SendFile(HANDLE hContact,const PROTOCHAR *desc, PROTOCHAR **files) { return NULL; } int CVkProto::SendUrl(HANDLE hContact,int flags,const char *url) { return 1; } int CVkProto::SetApparentMode(HANDLE hContact,int mode) { return 1; } int CVkProto::RecvAwayMsg(HANDLE hContact,int mode,PROTORECVEVENT *evt) { return 1; } HANDLE CVkProto::GetAwayMsg(HANDLE hContact) { return 0; // Status messages are disabled } int CVkProto::SetAwayMsg(int status, const PROTOCHAR *msg) { return 0; // Status messages are disabled }