/*
Facebook plugin for Miranda Instant Messenger
_____________________________________________
Copyright © 2011-17 Robert Pösel, 2017-18 Miranda NG team
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"
#include
#include
enum ChatMenuItems
{
IDM_INVITE = 10, IDM_EXIT, IDM_DESTROY,
IDM_DETAILS = 20, IDM_HISTORY
};
static const struct gc_item LogMenuItems[] =
{
{ LPGENW("&Invite user..."), IDM_INVITE, MENU_ITEM, FALSE },
{ LPGENW("E&xit chat session"), IDM_EXIT, MENU_ITEM, FALSE },
{ LPGENW("&Destroy chat session"), IDM_DESTROY, MENU_ITEM, FALSE },
};
static const struct gc_item NickMenuItems[] =
{
{ LPGENW("User &details"), IDM_DETAILS, MENU_ITEM, FALSE },
{ LPGENW("User &history"), IDM_HISTORY, MENU_ITEM, FALSE },
};
void FacebookProto::UpdateChat(const char *chat_id, const char *id, const char *name, const char *message, DWORD timestamp, bool is_old)
{
// replace % to %% to not interfere with chat color codes
std::string smessage = message;
utils::text::replace_all(&smessage, "%", "%%");
ptrW tid(mir_a2u(id));
ptrW tnick(mir_a2u_cp(name, CP_UTF8));
ptrW ttext(mir_a2u_cp(smessage.c_str(), CP_UTF8));
ptrW tchat_id(mir_a2u(chat_id));
GCEVENT gce = { m_szModuleName, tchat_id, GC_EVENT_MESSAGE };
gce.ptszText = ttext;
gce.time = timestamp ? timestamp : ::time(0);
if (id != nullptr)
gce.bIsMe = !mir_strcmp(id, facy.self_.user_id.c_str());
gce.dwFlags |= GCEF_ADDTOLOG;
if (is_old) {
gce.dwFlags |= GCEF_NOTNOTIFY;
gce.dwFlags &= ~GCEF_ADDTOLOG;
}
gce.ptszNick = tnick;
gce.ptszUID = tid;
Chat_Event(&gce);
facy.erase_reader(ChatIDToHContact(chat_id));
}
void FacebookProto::RenameChat(const char *chat_id, const char *name)
{
ptrW tchat_id(mir_a2u(chat_id));
ptrW tname(mir_a2u_cp(name, CP_UTF8));
Chat_ChangeSessionName(m_szModuleName, tchat_id, tname);
}
int FacebookProto::OnGCEvent(WPARAM, LPARAM lParam)
{
GCHOOK *hook = reinterpret_cast(lParam);
if (mir_strcmp(hook->pszModule, m_szModuleName))
return 0;
// Ignore for special chatrooms
if (!mir_wstrcmp(hook->ptszID, _A2W(FACEBOOK_NOTIFICATIONS_CHATROOM)))
return 0;
switch (hook->iType) {
case GC_USER_MESSAGE:
if (isOnline()) {
debugLogA(" > Chat - Outgoing message");
std::string msg = _T2A(hook->ptszText, CP_UTF8);
std::string chat_id = _T2A(hook->ptszID, CP_UTF8);
ForkThread(&FacebookProto::SendChatMsgWorker, new send_chat(chat_id, msg));
}
break;
case GC_USER_PRIVMESS:
{
facebook_user fbu;
fbu.user_id = _T2A(hook->ptszUID, CP_UTF8);
// Find this contact in list or add new temporary contact
MCONTACT hContact = AddToContactList(&fbu, false, true);
if (!hContact)
break;
CallService(MS_MSG_SENDMESSAGEW, hContact);
}
break;
case GC_USER_LOGMENU:
switch (hook->dwData) {
case IDM_INVITE:
break;
case IDM_EXIT:
{
std::string thread_id = _T2A(hook->ptszID, CP_UTF8);
auto it = facy.chat_rooms.find(thread_id);
if (it != facy.chat_rooms.end())
facy.sendRequest(facy.exitThreadRequest(it->second));
}
break;
case IDM_DESTROY:
{
std::string thread_id = _T2A(hook->ptszID, CP_UTF8);
auto it = facy.chat_rooms.find(thread_id);
if (it != facy.chat_rooms.end())
if (IDOK == MessageBoxW(nullptr, TranslateT("Delete conversation"), TranslateT("This will permanently delete the conversation history"), MB_OKCANCEL))
facy.sendRequest(facy.destroyThreadRequest(it->second));
}
}
break;
case GC_USER_NICKLISTMENU:
MCONTACT hContact = 0;
if (hook->dwData == 10 || hook->dwData == 20) {
facebook_user fbu;
fbu.user_id = _T2A(hook->ptszUID, CP_UTF8);
// Find this contact in list or add new temporary contact
hContact = AddToContactList(&fbu, false, true);
if (!hContact)
break;
}
switch (hook->dwData) {
case IDM_DETAILS:
CallService(MS_USERINFO_SHOWDIALOG, hContact);
break;
case IDM_HISTORY:
CallService(MS_HISTORY_SHOWCONTACTHISTORY, hContact);
break;
}
break;
}
return 0;
}
void FacebookProto::AddChatContact(const char *chat_id, const chatroom_participant &user, bool addToLog)
{
// Don't add user if it's already there
if (IsChatContact(chat_id, user.user_id.c_str()))
return;
ptrW tchat_id(mir_a2u(chat_id));
ptrW tnick(mir_a2u_cp(user.nick.c_str(), CP_UTF8));
ptrW tid(mir_a2u(user.user_id.c_str()));
GCEVENT gce = { m_szModuleName, tchat_id, GC_EVENT_JOIN };
gce.dwFlags = addToLog ? GCEF_ADDTOLOG : 0;
gce.ptszNick = tnick;
gce.ptszUID = tid;
gce.time = ::time(0);
gce.bIsMe = (user.role == ROLE_ME);
if (user.is_former) {
gce.ptszStatus = TranslateT("Former");
}
else {
switch (user.role) {
case ROLE_ME:
gce.ptszStatus = TranslateT("Myself");
break;
case ROLE_FRIEND:
gce.ptszStatus = TranslateT("Friend");
break;
case ROLE_NONE:
gce.ptszStatus = TranslateT("User");
break;
}
}
Chat_Event(&gce);
}
void FacebookProto::RemoveChatContact(const char *chat_id, const char *id, const char *name)
{
ptrW tchat_id(mir_a2u(chat_id));
ptrW tnick(mir_a2u_cp(name, CP_UTF8));
ptrW tid(mir_a2u(id));
GCEVENT gce = { m_szModuleName, tchat_id, GC_EVENT_PART };
gce.dwFlags = GCEF_ADDTOLOG;
gce.ptszNick = tnick;
gce.ptszUID = tid;
gce.time = ::time(0);
gce.bIsMe = false;
Chat_Event(&gce);
}
/** Caller must free result */
char *FacebookProto::GetChatUsers(const char *chat_id)
{
ptrW ptszChatID(mir_a2u(chat_id));
GC_INFO gci = { 0 };
gci.Flags = GCF_USERS;
gci.pszModule = m_szModuleName;
gci.pszID = ptszChatID;
Chat_GetInfo(&gci);
// mir_free(gci.pszUsers);
return gci.pszUsers;
}
bool FacebookProto::IsChatContact(const char *chat_id, const char *id)
{
ptrA users(GetChatUsers(chat_id));
return (users != nullptr && strstr(users, id) != nullptr);
}
void FacebookProto::AddChat(const char *id, const wchar_t *tname)
{
ptrW tid(mir_a2u(id));
// Create the group chat session
Chat_NewSession(GCW_PRIVMESS, m_szModuleName, tid, tname);
// Send setting events
Chat_AddGroup(m_szModuleName, tid, TranslateT("Myself"));
Chat_AddGroup(m_szModuleName, tid, TranslateT("Friend"));
Chat_AddGroup(m_szModuleName, tid, TranslateT("User"));
Chat_AddGroup(m_szModuleName, tid, TranslateT("Former"));
// Finish initialization
bool hideChats = getBool(FACEBOOK_KEY_HIDE_CHATS, DEFAULT_HIDE_CHATS);
Chat_Control(m_szModuleName, tid, (hideChats ? WINDOW_HIDDEN : SESSION_INITDONE));
Chat_Control(m_szModuleName, tid, SESSION_ONLINE);
}
INT_PTR FacebookProto::OnJoinChat(WPARAM hContact, LPARAM)
{
if (isOffline() || facy.dtsg_.empty() || !m_enableChat || IsSpecialChatRoom(hContact))
return 0;
ptrW idT(getWStringA(hContact, "ChatRoomID"));
ptrA threadId(getStringA(hContact, FACEBOOK_KEY_TID));
if (!idT || !threadId)
return 0;
facebook_chatroom *fbc;
std::string thread_id = threadId;
auto it = facy.chat_rooms.find(thread_id);
if (it != facy.chat_rooms.end())
fbc = it->second;
else {
// We don't have this chat loaded in memory yet, lets load some info (name, list of users)
fbc = new facebook_chatroom(thread_id);
LoadChatInfo(fbc);
facy.chat_rooms.insert(std::make_pair(thread_id, fbc));
// Update loaded info about this chat
setByte(hContact, FACEBOOK_KEY_CHAT_CAN_REPLY, fbc->can_reply);
setByte(hContact, FACEBOOK_KEY_CHAT_READ_ONLY, fbc->read_only);
setByte(hContact, FACEBOOK_KEY_CHAT_IS_ARCHIVED, fbc->is_archived);
setByte(hContact, FACEBOOK_KEY_CHAT_IS_SUBSCRIBED, fbc->is_subscribed);
}
// RM TODO: better use check if chatroom exists/is in db/is online... no?
// like: if (ChatIDToHContact(thread_id) == nullptr) {
ptrA users(GetChatUsers(thread_id.c_str()));
if (users == nullptr) {
// Add chatroom
AddChat(fbc->thread_id.c_str(), fbc->chat_name.c_str());
// Add chat contacts
for (auto &jt : fbc->participants)
AddChatContact(fbc->thread_id.c_str(), jt.second, false);
// Load last messages
delSetting(hContact, FACEBOOK_KEY_MESSAGE_ID); // We're creating new chatroom so we want load all recent messages
ForkThread(&FacebookProto::LoadLastMessages, new MCONTACT(hContact));
}
return 0;
}
INT_PTR FacebookProto::OnLeaveChat(WPARAM wParam, LPARAM)
{
ptrW idT(wParam ? getWStringA(wParam, "ChatRoomID") : nullptr);
Chat_Control(m_szModuleName, idT, SESSION_OFFLINE);
Chat_Terminate(m_szModuleName, idT);
if (!wParam)
facy.clear_chatrooms();
else if (!IsSpecialChatRoom(wParam)) {
ptrA threadId(getStringA(wParam, FACEBOOK_KEY_TID));
if (!threadId)
return 0;
auto it = facy.chat_rooms.find(std::string(threadId));
if (it != facy.chat_rooms.end()) {
delete it->second;
facy.chat_rooms.erase(it);
}
}
return 0;
}
int FacebookProto::OnGCMenuHook(WPARAM, LPARAM lParam)
{
GCMENUITEMS *gcmi = (GCMENUITEMS*)lParam;
if (mir_strcmp(gcmi->pszModule, m_szModuleName))
return 0;
if (gcmi->Type == MENU_ON_LOG)
Chat_AddMenuItems(gcmi->hMenu, _countof(LogMenuItems), LogMenuItems);
else if (gcmi->Type == MENU_ON_NICKLIST)
Chat_AddMenuItems(gcmi->hMenu, _countof(NickMenuItems), NickMenuItems);
return 0;
}
bool FacebookProto::IsSpecialChatRoom(MCONTACT hContact)
{
if (!isChatRoom(hContact))
return false;
ptrA id(getStringA(hContact, "ChatRoomID"));
return id && !mir_strcmp(id, FACEBOOK_NOTIFICATIONS_CHATROOM);
}
void FacebookProto::PrepareNotificationsChatRoom()
{
if (!getBool(FACEBOOK_KEY_NOTIFICATIONS_CHATROOM, DEFAULT_NOTIFICATIONS_CHATROOM))
return;
// Prepare notifications chatroom if not exists
MCONTACT hNotificationsChatRoom = ChatIDToHContact(FACEBOOK_NOTIFICATIONS_CHATROOM);
if (hNotificationsChatRoom == 0 || getDword(hNotificationsChatRoom, "Status", ID_STATUS_OFFLINE) != ID_STATUS_ONLINE) {
wchar_t nameT[200];
mir_snwprintf(nameT, L"%s: %s", m_tszUserName, TranslateT("Notifications"));
// Create the group chat session
Chat_NewSession(GCW_PRIVMESS, m_szModuleName, _A2W(FACEBOOK_NOTIFICATIONS_CHATROOM), nameT);
// Send setting events
Chat_Control(m_szModuleName, _A2W(FACEBOOK_NOTIFICATIONS_CHATROOM), WINDOW_HIDDEN);
Chat_Control(m_szModuleName, _A2W(FACEBOOK_NOTIFICATIONS_CHATROOM), SESSION_ONLINE);
}
}
void FacebookProto::UpdateNotificationsChatRoom(facebook_notification *notification)
{
if (!getBool(FACEBOOK_KEY_NOTIFICATIONS_CHATROOM, DEFAULT_NOTIFICATIONS_CHATROOM))
return;
std::stringstream text;
text << notification->text << "\n\n" << PrepareUrl(notification->link);
std::string message = text.str();
utils::text::replace_all(&message, "%", "%%");
ptrW idT(mir_wstrdup(_A2W(FACEBOOK_NOTIFICATIONS_CHATROOM)));
ptrW messageT(mir_a2u_cp(message.c_str(), CP_UTF8));
GCEVENT gce = { m_szModuleName, _A2W(FACEBOOK_NOTIFICATIONS_CHATROOM), GC_EVENT_MESSAGE };
gce.ptszText = messageT;
gce.time = notification->time ? notification->time : ::time(0);
gce.bIsMe = false;
gce.dwFlags |= GCEF_ADDTOLOG;
gce.ptszNick = TranslateT("Notifications");
gce.ptszUID = idT;
Chat_Event(&gce);
}
std::string FacebookProto::GenerateChatName(facebook_chatroom *fbc)
{
std::string name = "";
unsigned int namesUsed = 0;
for (auto &it : fbc->participants) {
std::string participant = it.second.nick;
// Ignore self contact, empty and numeric only participant names
if (it.second.role == ROLE_ME || participant.empty() || participant.find_first_not_of("0123456789") == std::string::npos)
continue;
if (namesUsed > 0)
name += ", ";
name += utils::text::prepare_name(participant, false);
if (++namesUsed >= FACEBOOK_CHATROOM_NAMES_COUNT)
break;
}
// Participants.size()-1 because we ignore self contact
if (fbc->participants.size() - 1 > namesUsed) {
wchar_t more[200];
mir_snwprintf(more, TranslateT("%s and more (%d)"), fbc->chat_name.c_str(), fbc->participants.size() - 1 - namesUsed); // -1 because we ignore self contact
fbc->chat_name = more;
}
// If there are no participants to create a name from, use just thread_id
if (name.empty())
name = fbc->thread_id.c_str();
return name;
}
void FacebookProto::LoadParticipantsNames(facebook_chatroom *fbc)
{
std::vector namelessIds;
// TODO: We could load all names from server at once by skipping this for cycle and using namelessIds as all in participants list, but we would lost our local names of our contacts. But maybe that's not a problem?
for (auto &it : fbc->participants) {
const char *id = it.first.c_str();
chatroom_participant &user = it.second;
if (!user.loaded) {
if (!mir_strcmp(id, facy.self_.user_id.c_str())) {
user.nick = facy.self_.real_name;
user.role = ROLE_ME;
user.loaded = true;
}
else {
MCONTACT hContact = ContactIDToHContact(id);
if (hContact != 0) {
DBVARIANT dbv;
if (!getStringUtf(hContact, FACEBOOK_KEY_NICK, &dbv)) {
user.nick = dbv.pszVal;
db_free(&dbv);
}
if (user.role == ROLE_NONE) {
int type = getByte(hContact, FACEBOOK_KEY_CONTACT_TYPE);
if (type == CONTACT_FRIEND)
user.role = ROLE_FRIEND;
else
user.role = ROLE_NONE;
}
user.loaded = true;
}
if (!user.loaded)
namelessIds.push_back(id);
}
}
}
if (!namelessIds.empty()) {
// we have some contacts without name, let's load them all from the server
LIST userIds(1);
for (std::string::size_type i = 0; i < namelessIds.size(); i++)
userIds.insert(mir_strdup(namelessIds.at(i).c_str()));
http::response resp = facy.sendRequest(facy.userInfoRequest(userIds));
FreeList(userIds);
userIds.destroy();
if (resp.code == HTTP_CODE_OK) {
try {
// TODO: We can cache these results and next time (e.g. for different chatroom) we can use that already cached names
ParseChatParticipants(&resp.data, &fbc->participants);
debugLogA("*** Participant names processed");
}
catch (const std::exception &e) {
debugLogA("*** Error processing participant names: %s", e.what());
}
}
}
}
void FacebookProto::JoinChatrooms()
{
for (auto &hContact : AccContacts()) {
if (!isChatRoom(hContact))
continue;
// Ignore archived and unsubscribed chats
if (getBool(hContact, FACEBOOK_KEY_CHAT_IS_ARCHIVED, false) || !getBool(hContact, FACEBOOK_KEY_CHAT_IS_SUBSCRIBED, true))
continue;
OnJoinChat(hContact, 0);
}
}
void FacebookProto::LoadChatInfo(facebook_chatroom *fbc)
{
if (isOffline())
return;
// request info about chat thread
http::response resp = facy.sendRequest(facy.threadInfoRequest(true, fbc->thread_id.c_str()));
if (resp.code != HTTP_CODE_OK) {
facy.handle_error("LoadChatInfo");
return;
}
try {
ParseChatInfo(&resp.data, fbc);
// Load missing participants names
LoadParticipantsNames(fbc);
// If chat has no name, create name from participants list
if (fbc->chat_name.empty()) {
std::string newName = GenerateChatName(fbc);
fbc->chat_name = _A2T(newName.c_str(), CP_UTF8);
}
debugLogA("*** Chat thread info processed");
}
catch (const std::exception &e) {
debugLogA("*** Error processing chat thread info: %s", e.what());
}
facy.handle_success("LoadChatInfo");
}
int FacebookProto::ParseChatInfo(std::string *data, facebook_chatroom* fbc)
{
size_t len = data->find("\r\n");
if (len != data->npos)
data->erase(len);
JSONNode root = JSONNode::parse(data->c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &thread = root["o0"]["data"]["message_thread"];
if (!thread)
return EXIT_FAILURE;
const JSONNode &thread_fbid_ = thread["thread_key"]["thread_fbid"];
const JSONNode &name_ = thread["name"];
if (!thread_fbid_)
return EXIT_FAILURE;
std::string tid = "id." + thread_fbid_.as_string();
if (fbc->thread_id != tid)
return EXIT_FAILURE;
chatroom_participant user;
user.is_former = false;
user.role = ROLE_NONE;
const JSONNode &participants = thread["all_participants"]["nodes"];
for (auto &jt : participants) {
user.user_id = jt["messaging_actor"]["id"].as_string();
fbc->participants.insert(std::make_pair(user.user_id, user));
}
fbc->can_reply = thread["can_reply"].as_bool();
fbc->is_archived = thread["has_viewer_archived"].as_bool();
fbc->is_subscribed = thread["is_viewer_subscribed"].as_bool();
fbc->read_only = thread["read_only"].as_bool();
fbc->chat_name = std::wstring(ptrW(mir_utf8decodeW(name_.as_string().c_str())));
return EXIT_SUCCESS;
}