//////////////////////////////////////////////////////////////////////////////// // Gadu-Gadu Plugin for Miranda IM // // Copyright (c) 2009-2012 Bartosz BiaƂek // // 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 "gg.h" #define GGS_CONCUR_SESS "/ConcurSess" static void gg_clearsessionslist(HWND hwndDlg) { HWND hList = GetDlgItem(hwndDlg, IDC_SESSIONS); if (!hList) return; ListView_DeleteAllItems(hList); while (ListView_DeleteColumn(hList, 0)); LV_COLUMN column = { 0 }; column.mask = LVCF_TEXT; column.cx = 500; column.pszText = TranslateT("Client Name"); ListView_InsertColumn(hList, 1, &column); column.pszText = TranslateT("IP Address"); ListView_InsertColumn(hList, 2, &column); column.pszText = TranslateT("Login Time"); ListView_InsertColumn(hList, 3, &column); column.pszText = TranslateT("Action"); ListView_InsertColumn(hList, 4, &column); RECT rc; GetClientRect(hList, &rc); int iWidth = rc.right - rc.left; ListView_SetColumnWidth(hList, 0, iWidth * 45 / 100); ListView_SetColumnWidth(hList, 1, iWidth * 20 / 100); ListView_SetColumnWidth(hList, 2, iWidth * 20 / 100); ListView_SetColumnWidth(hList, 3, LVSCW_AUTOSIZE_USEHEADER); } static void ListView_SetItemTextA(HWND hwndLV, int i, int iSubItem, char* pszText) { LV_ITEMA _ms_lvi; _ms_lvi.iSubItem = iSubItem; _ms_lvi.pszText = pszText; SendMessageA(hwndLV, LVM_SETITEMTEXTA, i, (LPARAM)&_ms_lvi); } static int gg_insertlistitem(HWND hList, gg_multilogon_id_t* id, const char* clientName, const char* ip, const char* loginTime) { LVITEM item = { 0 }; item.iItem = ListView_GetItemCount(hList); item.mask = LVIF_PARAM; item.lParam = (LPARAM)id; int index = ListView_InsertItem(hList, &item); if (index < 0) return index; ListView_SetItemTextA(hList, index, 0, (char*)clientName); ListView_SetItemTextA(hList, index, 1, (char*)ip); ListView_SetItemTextA(hList, index, 2, (char*)loginTime); ListView_SetItemText(hList, index, 3, TranslateT("sign out")); return index; } static void gg_listsessions(GaduProto* gg, HWND hwndDlg) { HWND hList = GetDlgItem(hwndDlg, IDC_SESSIONS); if (!hList) return; list_t l; gg->gg_EnterCriticalSection(&gg->sessions_mutex, "gg_listsessions", 73, "sessions_mutex", 1); for (l = gg->sessions; l; l = l->next) { struct gg_multilogon_session* sess = (struct gg_multilogon_session*)l->data; struct in_addr ia; char* ip; char loginTime[20]; ia.S_un.S_addr = sess->remote_addr; ip = inet_ntoa(ia); strftime(loginTime, sizeof(loginTime), "%d-%m-%Y %H:%M:%S", localtime(&sess->logon_time)); gg_insertlistitem(hList, &sess->id, sess->name, ip, loginTime); } gg->gg_LeaveCriticalSection(&gg->sessions_mutex, "gg_listsessions", 73, 1, "sessions_mutex", 1); EnableWindow(GetDlgItem(hwndDlg, IDC_SIGNOUTALL), ListView_GetItemCount(hList) > 0); } static int sttSessionsDlgResizer(HWND, LPARAM, UTILRESIZECONTROL* urc) { switch (urc->wId) { case IDC_HEADERBAR: return RD_ANCHORX_LEFT | RD_ANCHORY_TOP | RD_ANCHORX_WIDTH; case IDC_SESSIONS: return RD_ANCHORX_LEFT | RD_ANCHORY_TOP | RD_ANCHORY_HEIGHT | RD_ANCHORX_WIDTH; case IDC_SIGNOUTALL: return RD_ANCHORX_RIGHT | RD_ANCHORY_BOTTOM; } return RD_ANCHORX_LEFT | RD_ANCHORY_TOP; } static BOOL IsOverAction(HWND hwndDlg) { HWND hList = GetDlgItem(hwndDlg, IDC_SESSIONS); LVHITTESTINFO hti; GetCursorPos(&hti.pt); ScreenToClient(hList, &hti.pt); RECT rc; GetClientRect(hList, &rc); if (!PtInRect(&rc, hti.pt) || ListView_SubItemHitTest(hList, &hti) == -1 || hti.iSubItem != 3 || !(hti.flags & LVHT_ONITEMLABEL)) return FALSE; ListView_GetSubItemRect(hList, hti.iItem, hti.iSubItem, LVIR_LABEL, &rc); wchar_t szText[256]; szText[0] = 0; ListView_GetItemText(hList, hti.iItem, hti.iSubItem, szText, _countof(szText)); HDC hdc = GetDC(hList); SIZE textSize; GetTextExtentPoint32(hdc, szText, (int)mir_wstrlen(szText), &textSize); ReleaseDC(hList, hdc); int textPosX = rc.left + (((rc.right - rc.left) - textSize.cx) / 2); return (hti.pt.x > textPosX && hti.pt.x < textPosX + textSize.cx); } static HCURSOR hHandCursor = nullptr; #define WM_MULTILOGONINFO (WM_USER + 10) #define HM_PROTOACK (WM_USER + 11) static INT_PTR CALLBACK gg_sessions_viewdlg(HWND hwndDlg, UINT message, WPARAM wParam, LPARAM lParam) { GaduProto* gg = (GaduProto*)GetWindowLongPtr(hwndDlg, DWLP_USER); switch (message) { case WM_INITDIALOG: TranslateDialogDefault(hwndDlg); gg = (GaduProto*)lParam; gg->hwndSessionsDlg = hwndDlg; SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)lParam); { wchar_t oldTitle[256], newTitle[256]; GetWindowText(hwndDlg, oldTitle, _countof(oldTitle)); mir_snwprintf(newTitle, oldTitle, gg->m_tszUserName); SetWindowText(hwndDlg, newTitle); Window_SetIcon_IcoLib(hwndDlg, GetIconHandle(IDI_SESSIONS)); if (hHandCursor == nullptr) hHandCursor = LoadCursor(nullptr, IDC_HAND); HANDLE hProtoAckEvent = HookEventMessage(ME_PROTO_ACK, hwndDlg, HM_PROTOACK); SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)hProtoAckEvent); ListView_SetExtendedListViewStyle(GetDlgItem(hwndDlg, IDC_SESSIONS), LVS_EX_FULLROWSELECT); SendMessage(hwndDlg, WM_MULTILOGONINFO, 0, 0); return TRUE; } case HM_PROTOACK: { ACKDATA* ack = (ACKDATA*)lParam; if (!mir_strcmp(ack->szModule, gg->m_szModuleName) && !ack->hContact && ack->type == ACKTYPE_STATUS && ack->result == ACKRESULT_SUCCESS && (ack->lParam == ID_STATUS_OFFLINE || (ack->hProcess == (HANDLE)ID_STATUS_CONNECTING && ack->lParam != ID_STATUS_OFFLINE && !ListView_GetItemCount(GetDlgItem(hwndDlg, IDC_SESSIONS))))) { gg_clearsessionslist(hwndDlg); EnableWindow(GetDlgItem(hwndDlg, IDC_SIGNOUTALL), FALSE); } break; } case WM_MULTILOGONINFO: gg_clearsessionslist(hwndDlg); gg_listsessions(gg, hwndDlg); break; case WM_COMMAND: switch (LOWORD(wParam)) { case IDC_SIGNOUTALL: { HWND hList = GetDlgItem(hwndDlg, IDC_SESSIONS); LVITEM lvi = { 0 }; int iCount = ListView_GetItemCount(hList); lvi.mask = LVIF_PARAM; for (int i = 0; i < iCount; i++) { lvi.iItem = i; ListView_GetItem(hList, &lvi); gg->gg_EnterCriticalSection(&gg->sess_mutex, "gg_sessions_viewdlg", 74, "sess_mutex", 1); gg_multilogon_disconnect(gg->m_sess, *((gg_multilogon_id_t*)lvi.lParam)); gg->gg_LeaveCriticalSection(&gg->sess_mutex, "gg_sessions_viewdlg", 74, 1, "sess_mutex", 1); } break; } } break; case WM_NOTIFY: if (((LPNMHDR)lParam)->idFrom == IDC_SESSIONS) { switch (((LPNMHDR)lParam)->code) { case NM_CUSTOMDRAW: { LPNMLVCUSTOMDRAW nm = (LPNMLVCUSTOMDRAW)lParam; switch (nm->nmcd.dwDrawStage) { case CDDS_PREPAINT: if (ListView_GetItemCount(nm->nmcd.hdr.hwndFrom) == 0) { const LPCTSTR szText = gg->isonline() ? TranslateT("There are no active concurrent sessions for this account.") : TranslateT("You have to be logged in to view concurrent sessions."); RECT rc; HWND hwndHeader = ListView_GetHeader(nm->nmcd.hdr.hwndFrom); GetClientRect(nm->nmcd.hdr.hwndFrom, &rc); if (hwndHeader != nullptr) { RECT rcHeader; GetClientRect(hwndHeader, &rcHeader); rc.top += rcHeader.bottom; } int cbLen = (int)mir_wstrlen(szText); SIZE textSize; GetTextExtentPoint32(nm->nmcd.hdc, szText, cbLen, &textSize); int textPosX = rc.left + (((rc.right - rc.left) - textSize.cx) / 2); ExtTextOut(nm->nmcd.hdc, textPosX, rc.top + textSize.cy, ETO_OPAQUE, &rc, szText, cbLen, nullptr); } // FALL THROUGH case CDDS_ITEMPREPAINT: SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, CDRF_NOTIFYSUBITEMDRAW); return TRUE; case CDDS_SUBITEM | CDDS_ITEMPREPAINT: { RECT rc; ListView_GetSubItemRect(nm->nmcd.hdr.hwndFrom, nm->nmcd.dwItemSpec, nm->iSubItem, LVIR_LABEL, &rc); if (nm->nmcd.hdr.idFrom == IDC_SESSIONS && nm->iSubItem == 3) { wchar_t szText[256]; szText[0] = 0; ListView_GetItemText(nm->nmcd.hdr.hwndFrom, nm->nmcd.dwItemSpec, nm->iSubItem, szText, _countof(szText)); FillRect(nm->nmcd.hdc, &rc, GetSysColorBrush(COLOR_WINDOW)); SetTextColor(nm->nmcd.hdc, RGB(0, 0, 255)); DrawText(nm->nmcd.hdc, szText, -1, &rc, DT_END_ELLIPSIS | DT_CENTER | DT_NOPREFIX | DT_SINGLELINE | DT_TOP); SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, CDRF_SKIPDEFAULT); return TRUE; } break; } } break; } case NM_CLICK: if (IsOverAction(hwndDlg)) { LPNMITEMACTIVATE nm = (LPNMITEMACTIVATE)lParam; LVITEM lvi = { 0 }; lvi.mask = LVIF_PARAM; lvi.iItem = nm->iItem; ListView_GetItem(nm->hdr.hwndFrom, &lvi); gg->gg_EnterCriticalSection(&gg->sess_mutex, "gg_sessions_viewdlg", 75, "sess_mutex", 1); gg_multilogon_disconnect(gg->m_sess, *((gg_multilogon_id_t*)lvi.lParam)); gg->gg_LeaveCriticalSection(&gg->sess_mutex, "gg_sessions_viewdlg", 75, 1, "sess_mutex", 1); } break; } } break; case WM_CONTEXTMENU: { HWND hList = GetDlgItem(hwndDlg, IDC_SESSIONS); POINT pt = { (short)LOWORD(lParam), (short)HIWORD(lParam) }, ptDlg = pt; LVHITTESTINFO lvhti = { 0 }; ScreenToClient(hwndDlg, &ptDlg); if (ChildWindowFromPoint(hwndDlg, ptDlg) == hList) { HMENU hMenu; lvhti.pt = pt; ScreenToClient(hList, &lvhti.pt); if (ListView_HitTest(hList, &lvhti) == -1) break; hMenu = CreatePopupMenu(); AppendMenu(hMenu, MFT_STRING, 10001, TranslateT("Copy Text")); AppendMenu(hMenu, MFT_STRING, 10002, TranslateT("Whois")); int iSelection = TrackPopupMenu(hMenu, TPM_RIGHTBUTTON | TPM_RETURNCMD, pt.x, pt.y, 0, hwndDlg, nullptr); switch (iSelection) { case 10001: { wchar_t szText[512], szClientName[256], szIP[64], szLoginTime[64]; HGLOBAL hData; if (!OpenClipboard(hwndDlg)) break; EmptyClipboard(); szClientName[0] = szIP[0] = szLoginTime[0] = 0; ListView_GetItemText(hList, lvhti.iItem, 0, szClientName, _countof(szClientName)); ListView_GetItemText(hList, lvhti.iItem, 1, szIP, _countof(szIP)); ListView_GetItemText(hList, lvhti.iItem, 2, szLoginTime, _countof(szLoginTime)); mir_snwprintf(szText, L"%s\t%s\t%s", szClientName, szIP, szLoginTime); if ((hData = GlobalAlloc(GMEM_MOVEABLE, mir_wstrlen(szText) + 1)) != nullptr) { mir_wstrcpy((wchar_t*)GlobalLock(hData), szText); GlobalUnlock(hData); SetClipboardData(CF_TEXT, hData); } CloseClipboard(); break; } case 10002: { wchar_t szUrl[256], szIP[64]; szIP[0] = 0; ListView_GetItemText(hList, lvhti.iItem, 1, szIP, _countof(szIP)); mir_snwprintf(szUrl, L"http://whois.domaintools.com/%s", szIP); Utils_OpenUrlW(szUrl); break; } } DestroyMenu(hMenu); } break; } case WM_GETMINMAXINFO: ((LPMINMAXINFO)lParam)->ptMinTrackSize.x = 620; ((LPMINMAXINFO)lParam)->ptMinTrackSize.y = 220; return 0; case WM_SIZE: Utils_ResizeDialog(hwndDlg, g_plugin.getInst(), MAKEINTRESOURCEA(IDD_SESSIONS), sttSessionsDlgResizer); return 0; case WM_SETCURSOR: if (LOWORD(lParam) == HTCLIENT && IsOverAction(hwndDlg)) { SetCursor(hHandCursor); SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, TRUE); return TRUE; } break; case WM_CLOSE: DestroyWindow(hwndDlg); break; case WM_DESTROY: { HANDLE hProtoAckEvent = (HANDLE)GetWindowLongPtr(hwndDlg, GWLP_USERDATA); if (hProtoAckEvent) UnhookEvent(hProtoAckEvent); gg->hwndSessionsDlg = nullptr; Window_FreeIcon_IcoLib(hwndDlg); break; } } return FALSE; } INT_PTR GaduProto::sessions_view(WPARAM, LPARAM) { if (hwndSessionsDlg && IsWindow(hwndSessionsDlg)) { ShowWindow(hwndSessionsDlg, SW_SHOWNORMAL); SetForegroundWindow(hwndSessionsDlg); SetFocus(hwndSessionsDlg); } else CreateDialogParam(g_plugin.getInst(), MAKEINTRESOURCE(IDD_SESSIONS), nullptr, gg_sessions_viewdlg, (LPARAM)this); return 0; } void GaduProto::sessions_updatedlg() { if (hwndSessionsDlg && IsWindow(hwndSessionsDlg)) SendMessage(hwndSessionsDlg, WM_MULTILOGONINFO, 0, 0); } BOOL GaduProto::sessions_closedlg() { if (hwndSessionsDlg && IsWindow(hwndSessionsDlg)) return PostMessage(hwndSessionsDlg, WM_CLOSE, 0, 0); return FALSE; } void GaduProto::sessions_menus_init(HGENMENU hRoot) { CMenuItem mi(&g_plugin); mi.flags = CMIF_UNICODE; mi.root = hRoot; mi.pszService = GGS_CONCUR_SESS; CreateProtoService(mi.pszService, &GaduProto::sessions_view); mi.position = 200003; mi.hIcolibItem = iconList[16].hIcolib; mi.name.w = LPGENW("Concurrent &sessions"); Menu_AddProtoMenuItem(&mi, m_szModuleName); }