/*
Plugin of Miranda IM for communicating with users of the AIM protocol.
Copyright (c) 2008-2010 Boris Krasnovskiy
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, see .
*/
#include "stdafx.h"
extern OBJLIST g_Instances;
static int SingleHexToDecimal(wchar_t c)
{
if (c >= '0' && c <= '9') return c - '0';
if (c >= 'a' && c <= 'f') return c - 'a' + 10;
if (c >= 'A' && c <= 'F') return c - 'A' + 10;
return -1;
}
static wchar_t* url_decode(wchar_t* str)
{
wchar_t* s = str, *d = str;
while (*s) {
if (*s == '%') {
int digit1 = SingleHexToDecimal(s[1]);
if (digit1 != -1) {
int digit2 = SingleHexToDecimal(s[2]);
if (digit2 != -1) {
s += 3;
*d++ = (wchar_t)((digit1 << 4) | digit2);
continue;
}
}
}
*d++ = *s++;
}
*d = 0;
return str;
}
static INT_PTR ServiceParseAimLink(WPARAM, LPARAM lParam)
{
if (lParam == 0) return 1; /* sanity check */
wchar_t *arg = (wchar_t*)lParam;
/* skip leading prefix */
arg = wcschr(arg, ':');
if (arg == nullptr) return 1; /* parse failed */
for (++arg; *arg == '/'; ++arg);
arg = NEWWSTR_ALLOCA(arg);
if (g_Instances.getCount() == 0) return 0;
CAimProto *proto = &g_Instances[0];
for (int i = 0; i < g_Instances.getCount(); ++i) {
if (g_Instances[i].m_iStatus != ID_STATUS_OFFLINE && !IsStatusConnecting(g_Instances[i].m_iStatus)) {
proto = &g_Instances[i];
break;
}
}
if (proto == nullptr) return 1;
/*
add user: aim:addbuddy?screenname=NICK&groupname=GROUP
send message: aim:goim?screenname=NICK&message=MSG
open chatroom: aim:gochat?roomname=ROOM&exchange=NUM
*/
/* add a contact to the list */
if (!wcsnicmp(arg, L"addbuddy?", 9)) {
wchar_t *tok, *tok2;
char *sn = nullptr, *group = nullptr;
for (tok = arg + 8; tok != nullptr; tok = tok2) {
tok2 = wcschr(++tok, '&'); /* first token */
if (tok2) *tok2 = 0;
if (!wcsnicmp(tok, L"screenname=", 11) && *(tok + 11) != 0)
sn = mir_u2a(url_decode(tok + 11));
if (!wcsnicmp(tok, L"groupname=", 10) && *(tok + 10) != 0)
group = mir_utf8encodeW(url_decode(tok + 10)); /* group is currently ignored */
}
if (sn == nullptr) {
mir_free(group);
return 1;
}
if (!proto->contact_from_sn(sn)) /* does not yet check if sn is current user */
{
MCONTACT hContact = proto->contact_from_sn(sn, true);
proto->add_contact_to_group(hContact, group && group[0] ? group : AIM_DEFAULT_GROUP);
}
mir_free(group);
mir_free(sn);
return 0;
}
/* send a message to a contact */
else if (!wcsnicmp(arg, L"goim?", 5)) {
wchar_t *tok, *tok2, *msg = nullptr;
char *sn = nullptr;
for (tok = arg + 4; tok != nullptr; tok = tok2) {
tok2 = wcschr(++tok, '&'); /* first token */
if (tok2) *tok2 = 0;
if (!wcsnicmp(tok, L"screenname=", 11) && *(tok + 11) != 0)
sn = mir_u2a(url_decode(tok + 11));
if (!wcsnicmp(tok, L"message=", 8) && *(tok + 8) != 0)
msg = url_decode(tok + 8);
}
if (sn == nullptr) return 1; /* parse failed */
MCONTACT hContact = proto->contact_from_sn(sn, true, true);
if (hContact)
CallService(MS_MSG_SENDMESSAGEW, hContact, (LPARAM)msg);
mir_free(sn);
return 0;
}
/* open a chatroom */
else if (!wcsnicmp(arg, L"gochat?", 7)) {
wchar_t *tok, *tok2;
char *rm = nullptr;
int exchange = 0;
for (tok = arg + 6; tok != nullptr; tok = tok2) {
tok2 = wcschr(++tok, '&'); /* first token */
if (tok2) *tok2 = 0;
if (!wcsnicmp(tok, L"roomname=", 9) && *(tok + 9) != 0) {
rm = mir_u2a(url_decode(tok + 9));
for (char *ch = rm; *ch; ++ch)
if (*ch == '+') *ch = ' ';
}
if (!wcsnicmp(tok, L"exchange=", 9))
exchange = _wtoi(tok + 9);
}
if (rm == nullptr || exchange <= 0) {
mir_free(rm);
return 1;
}
chatnav_param* par = new chatnav_param(rm, (unsigned short)exchange);
proto->ForkThread(&CAimProto::chatnav_request_thread, par);
mir_free(rm);
return 0;
}
return 1; /* parse failed */
}
void aim_links_init(void)
{
static const char szService[] = "AIM/ParseAimLink";
CreateServiceFunction(szService, ServiceParseAimLink);
AssocMgr_AddNewUrlTypeT("aim:", TranslateT("AIM link protocol"), hInstance, IDI_AOL, szService, 0);
}