/*
Copyright (C) 2012-22 Miranda NG team (https://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"
#define IDC_ICO 12344
#define IDC_ENTER 2000 // Pseudo control to handle enter in the main window
HIMAGELIST hIml;
long main_dialog_open = 0;
HWND hwndMain = nullptr;
// array where the contacts are put into
struct c_struct
{
wchar_t szname[120];
wchar_t szgroup[50];
MCONTACT hcontact;
wchar_t proto[20];
c_struct()
{
szname[0] = 0;
szgroup[0] = 0;
hcontact = 0;
proto[0] = 0;
}
};
LIST contacts(200);
long max_proto_width;
// Get the name the contact has in list
// This was not made to be called by more than one thread!
wchar_t tmp_list_name[120];
wchar_t *GetListName(c_struct *cs)
{
if (opts.group_append && cs->szgroup[0] != '\0') {
mir_snwprintf(tmp_list_name, L"%s (%s)", cs->szname, cs->szgroup);
return tmp_list_name;
}
else {
return cs->szname;
}
}
int lstreq(wchar_t *a, wchar_t *b, size_t len = -1)
{
a = CharLower(wcsdup(a));
b = CharLower(wcsdup(b));
int ret;
if (len > 0)
ret = wcsncmp(a, b, len);
else
ret = mir_wstrcmp(a, b);
free(a);
free(b);
return ret;
}
// simple sorting function to have
// the contact array in alphabetical order
void SortArray(void)
{
int loop, doop;
c_struct *cs_temp;
SortedList *sl = (SortedList *)&contacts;
for (loop = 0; loop < contacts.getCount(); loop++) {
for (doop = loop + 1; doop < contacts.getCount(); doop++) {
int cmp = lstreq(contacts[loop]->szname, contacts[doop]->szname);
if (cmp > 0) {
cs_temp = contacts[loop];
sl->items[loop] = contacts[doop];
sl->items[doop] = cs_temp;
}
else if (cmp == 0) {
if (lstreq(contacts[loop]->proto, contacts[doop]->proto) > 0) {
cs_temp = contacts[loop];
sl->items[loop] = contacts[doop];
sl->items[doop] = cs_temp;
}
}
}
}
}
int GetStatus(MCONTACT hContact, char *proto = nullptr)
{
if (proto == nullptr)
proto = Proto_GetBaseAccountName(hContact);
if (proto == nullptr)
return ID_STATUS_OFFLINE;
return db_get_w(hContact, proto, "Status", ID_STATUS_OFFLINE);
}
void FreeContacts()
{
for (auto &it : contacts)
delete it;
contacts.destroy();
}
void LoadContacts(HWND hwndDlg, BOOL show_all)
{
BOOL metacontactsEnabled = db_mc_isEnabled();
// Read last-sent-to contact from db and set handle as window-userdata
HANDLE hlastsent = (HANDLE)g_plugin.getDword("LastSentTo", -1);
SetWindowLongPtr(hwndMain, GWLP_USERDATA, (LONG_PTR)hlastsent);
// enumerate all contacts and write them to the array
// item data of listbox-strings is the array position
FreeContacts();
for (auto &hContact : Contacts()) {
char *pszProto = Proto_GetBaseAccountName(hContact);
if (pszProto == nullptr)
continue;
// Get meta
MCONTACT hMeta = NULL;
if (metacontactsEnabled) {
if ((!show_all && opts.hide_subcontacts) || opts.group_append)
hMeta = db_mc_getMeta(hContact);
}
else if (!mir_strcmp(META_PROTO, pszProto))
continue;
if (!show_all) {
// Check if is offline and have to show
if (GetStatus(hContact, pszProto) <= ID_STATUS_OFFLINE) {
// See if has to show
char setting[128];
mir_snprintf(setting, "ShowOffline%s", pszProto);
if (!g_plugin.getByte(setting, FALSE))
continue;
// Check if proto offline
else if (opts.hide_from_offline_proto && Proto_GetStatus(pszProto) <= ID_STATUS_OFFLINE)
continue;
}
// Check if is subcontact
if (opts.hide_subcontacts && hMeta != NULL) {
if (!opts.keep_subcontacts_from_offline)
continue;
if (GetStatus(hMeta, META_PROTO) > ID_STATUS_OFFLINE)
continue;
char setting[128];
mir_snprintf(setting, "ShowOffline%s", META_PROTO);
if (g_plugin.getByte(setting, FALSE))
continue;
}
}
// Add to list
// Get group
c_struct *contact = new c_struct();
if (opts.group_append) {
ptrW wszGroup(Clist_GetGroup(hMeta == NULL ? hContact : hMeta));
if (wszGroup)
wcsncpy_s(contact->szgroup, wszGroup, _TRUNCATE);
}
// Make contact name
wchar_t *tmp = (wchar_t *)Clist_GetContactDisplayName(hContact);
mir_wstrncpy(contact->szname, tmp, _countof(contact->szname));
PROTOACCOUNT *acc = Proto_GetAccount(pszProto);
if (acc != nullptr)
mir_wstrncpy(contact->proto, acc->tszAccountName, _countof(contact->proto));
contact->hcontact = hContact;
contacts.insert(contact);
}
SortArray();
SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_RESETCONTENT, 0, 0);
for (int loop = 0; loop < contacts.getCount(); loop++)
SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_SETITEMDATA, SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_ADDSTRING, 0, (LPARAM)GetListName(contacts[loop])), loop);
}
// Enable buttons for the selected contact
void EnableButtons(HWND hwndDlg, MCONTACT hContact)
{
if (hContact == NULL) {
EnableWindow(GetDlgItem(hwndDlg, IDC_MESSAGE), FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_FILE), FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_USERINFO), FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_HISTORY), FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_MENU), FALSE);
SendDlgItemMessage(hwndDlg, IDC_ICO, STM_SETICON, 0, 0);
}
else {
// Is a meta?
MCONTACT hSub = db_mc_getMostOnline(hContact);
if (hSub != NULL)
hContact = hSub;
// Get caps
INT_PTR caps = 0;
char *pszProto = Proto_GetBaseAccountName(hContact);
if (pszProto != nullptr)
caps = CallProtoService(pszProto, PS_GETCAPS, PFLAGNUM_1, 0);
EnableWindow(GetDlgItem(hwndDlg, IDC_MESSAGE), caps & PF1_IMSEND ? TRUE : FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_FILE), caps & PF1_FILESEND ? TRUE : FALSE);
EnableWindow(GetDlgItem(hwndDlg, IDC_USERINFO), TRUE);
EnableWindow(GetDlgItem(hwndDlg, IDC_HISTORY), TRUE);
EnableWindow(GetDlgItem(hwndDlg, IDC_MENU), TRUE);
HICON ico = ImageList_GetIcon(hIml, Clist_GetContactIcon(hContact), ILD_IMAGE);
SendDlgItemMessage(hwndDlg, IDC_ICO, STM_SETICON, (WPARAM)ico, 0);
}
}
// check if the char(s) entered appears in a contacts name
int CheckText(HWND hdlg, wchar_t *sztext, BOOL only_enable = FALSE)
{
EnableButtons(hwndMain, NULL);
if (sztext == nullptr || sztext[0] == '\0')
return 0;
size_t len = mir_wstrlen(sztext);
if (only_enable) {
for (auto &it : contacts) {
if (lstreq(sztext, it->szname) == 0 || lstreq(sztext, GetListName(it)) == 0) {
EnableButtons(hwndMain, it->hcontact);
return 0;
}
}
}
else {
for (auto &it : contacts) {
if (lstreq(sztext, GetListName(it), len) == 0) {
SetWindowText(hdlg, GetListName(it));
SendMessage(hdlg, EM_SETSEL, (WPARAM)len, (LPARAM)-1);
EnableButtons(hwndMain, it->hcontact);
return 0;
}
}
}
EnableButtons(hwndMain, NULL);
return 0;
}
MCONTACT GetSelectedContact(HWND hwndDlg)
{
// First try selection
int sel = SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_GETCURSEL, 0, 0);
if (sel != CB_ERR) {
int pos = SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_GETITEMDATA, sel, 0);
if (pos != CB_ERR)
return contacts[pos]->hcontact;
}
// Now try the name
wchar_t cname[120] = L"";
GetDlgItemText(hwndDlg, IDC_USERNAME, cname, _countof(cname));
for (auto &it : contacts)
if (!mir_wstrcmpi(cname, GetListName(it)))
return it->hcontact;
return NULL;
}
// get array position from handle
int GetItemPos(MCONTACT hcontact)
{
for (auto &it : contacts)
if (hcontact == it->hcontact)
return contacts.indexOf(&it);
return -1;
}
// callback function for edit-box of the listbox
// without this the autofill function isn't possible
// this was done like ie does it..as far as spy++ could tell ;)
LRESULT CALLBACK EditProc(HWND hdlg, UINT msg, WPARAM wparam, LPARAM lparam)
{
switch (msg) {
case WM_CHAR:
{
if (wparam < 32 && wparam != VK_BACK)
break;
wchar_t sztext[120] = L"";
uint32_t start;
uint32_t end;
int ret = SendMessage(hdlg, EM_GETSEL, (WPARAM)&start, (LPARAM)&end);
GetWindowText(hdlg, sztext, _countof(sztext));
BOOL at_end = (mir_wstrlen(sztext) == (int)end);
if (ret != -1) {
if (wparam == VK_BACK) {
if (start > 0)
SendMessage(hdlg, EM_SETSEL, (WPARAM)start - 1, (LPARAM)end);
sztext[0] = 0;
}
else {
sztext[0] = wparam;
sztext[1] = 0;
}
SendMessage(hdlg, EM_REPLACESEL, 0, (LPARAM)sztext);
GetWindowText(hdlg, sztext, _countof(sztext));
}
CheckText(hdlg, sztext, !at_end);
return 1;
}
case WM_KEYUP:
{
wchar_t sztext[120] = L"";
if (wparam == VK_RETURN) {
switch (SendMessage(GetParent(hdlg), CB_GETDROPPEDSTATE, 0, 0)) {
case FALSE:
SendMessage(GetParent(GetParent(hdlg)), WM_COMMAND, MAKEWPARAM(IDC_ENTER, STN_CLICKED), 0);
break;
case TRUE:
SendMessage(GetParent(hdlg), CB_SHOWDROPDOWN, FALSE, 0);
break;
}
}
else if (wparam == VK_DELETE) {
GetWindowText(hdlg, sztext, _countof(sztext));
CheckText(hdlg, sztext, TRUE);
}
return 0;
}
case WM_GETDLGCODE:
return DLGC_WANTCHARS | DLGC_WANTARROWS;
}
return mir_callNextSubclass(hdlg, EditProc, msg, wparam, lparam);
}
HACCEL hAcct;
HHOOK hHook;
// This function filters the message queue and translates
// the keyboard accelerators
LRESULT CALLBACK HookProc(int code, WPARAM, LPARAM lparam)
{
if (code != MSGF_DIALOGBOX)
return 0;
MSG *msg = (MSG*)lparam;
int action = Hotkey_Check(msg, "Quick Contacts");
if (action != 0) {
SendMessage(hwndMain, WM_COMMAND, action, 0);
return 1;
}
if (msg->message == WM_KEYDOWN && msg->wParam == VK_ESCAPE) {
switch (SendDlgItemMessage(hwndMain, IDC_USERNAME, CB_GETDROPPEDSTATE, 0, 0)) {
case FALSE:
SendMessage(hwndMain, WM_CLOSE, 0, 0);
break;
case TRUE:
SendDlgItemMessage(hwndMain, IDC_USERNAME, CB_SHOWDROPDOWN, FALSE, 0);
break;
}
}
return 0;
}
BOOL ScreenToClient(HWND hWnd, LPRECT lpRect)
{
BOOL ret;
POINT pt;
pt.x = lpRect->left;
pt.y = lpRect->top;
ret = ScreenToClient(hWnd, &pt);
if (!ret) return ret;
lpRect->left = pt.x;
lpRect->top = pt.y;
pt.x = lpRect->right;
pt.y = lpRect->bottom;
ret = ScreenToClient(hWnd, &pt);
lpRect->right = pt.x;
lpRect->bottom = pt.y;
return ret;
}
BOOL MoveWindow(HWND hWnd, const RECT &rect, BOOL bRepaint)
{
return MoveWindow(hWnd, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, bRepaint);
}
static void FillButton(HWND hwndDlg, int dlgItem, wchar_t *name, wchar_t *key, HICON icon)
{
wchar_t tmp[256];
wchar_t *full = tmp;
if (key == nullptr)
full = TranslateW(name);
else
mir_snwprintf(tmp, L"%s (%s)", TranslateW(name), key);
SendDlgItemMessage(hwndDlg, dlgItem, BUTTONSETASFLATBTN, 0, 0);
SendDlgItemMessage(hwndDlg, dlgItem, BUTTONADDTOOLTIP, (LPARAM)full, BATF_UNICODE);
SendDlgItemMessage(hwndDlg, dlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)icon);
}
static void FillCheckbox(HWND hwndDlg, int dlgItem, wchar_t *name, wchar_t *key)
{
wchar_t tmp[256];
wchar_t *full = tmp;
if (key == nullptr)
full = TranslateW(name);
else
mir_snwprintf(tmp, L"%s (%s)", TranslateW(name), key);
SetDlgItemText(hwndDlg, dlgItem, full);
}
static INT_PTR CALLBACK MainDlgProc(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
{
switch (msg) {
case WM_INITDIALOG:
TranslateDialogDefault(hwndDlg);
{
RECT rc;
GetWindowRect(GetDlgItem(hwndDlg, IDC_USERNAME), &rc);
ScreenToClient(hwndDlg, &rc);
CreateWindow(L"STATIC", L"", WS_CHILD | WS_VISIBLE | SS_ICON | SS_CENTERIMAGE,
rc.left - 20, rc.top + (rc.bottom - rc.top - 16) / 2, 16, 16, hwndDlg, (HMENU)IDC_ICO,
g_plugin.getInst(), nullptr);
hHook = SetWindowsHookEx(WH_MSGFILTER, HookProc, nullptr, GetCurrentThreadId());
// Combo
SendDlgItemMessage(hwndDlg, IDC_USERNAME, EM_LIMITTEXT, (WPARAM)119, 0);
mir_subclassWindow(GetWindow(GetDlgItem(hwndDlg, IDC_USERNAME), GW_CHILD), EditProc);
// Buttons
FillCheckbox(hwndDlg, IDC_SHOW_ALL_CONTACTS, LPGENW("Show all contacts"), NULL);
FillButton(hwndDlg, IDC_MESSAGE, LPGENW("Send message"), nullptr, Skin_LoadIcon(SKINICON_EVENT_MESSAGE));
FillButton(hwndDlg, IDC_FILE, LPGENW("Send file"), NULL, Skin_LoadIcon(SKINICON_EVENT_FILE));
FillButton(hwndDlg, IDC_USERINFO, LPGENW("Open user info"), NULL, Skin_LoadIcon(SKINICON_OTHER_USERDETAILS));
FillButton(hwndDlg, IDC_HISTORY, LPGENW("Open history"), NULL, Skin_LoadIcon(SKINICON_OTHER_HISTORY));
FillButton(hwndDlg, IDC_MENU, LPGENW("Open contact menu"), NULL, Skin_LoadIcon(SKINICON_OTHER_DOWNARROW));
SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_SETEXTENDEDUI, TRUE, 0);
Utils_RestoreWindowPositionNoSize(hwndDlg, NULL, MODULENAME, "window");
LoadContacts(hwndDlg, FALSE);
EnableButtons(hwndDlg, NULL);
if (g_plugin.getByte("EnableLastSentTo", 0)) {
int pos = GetItemPos(g_plugin.getDword("LastSentTo", -1));
if (pos != -1) {
SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_SETCURSEL, (WPARAM)pos, 0);
EnableButtons(hwndDlg, contacts[pos]->hcontact);
}
}
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
}
return TRUE;
case WM_COMMAND:
switch (LOWORD(wParam)) {
case IDC_USERNAME:
if (HIWORD(wParam) == CBN_SELCHANGE) {
int pos = SendDlgItemMessage(hwndDlg, IDC_USERNAME, CB_GETCURSEL, 0, 0);
EnableButtons(hwndDlg, pos < contacts.getCount() ? contacts[pos]->hcontact : NULL);
}
break;
case IDC_ENTER:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL)
break;
Clist_ContactDoubleClicked(hContact);
g_plugin.setDword("LastSentTo", hContact);
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
}
break;
case IDC_MESSAGE:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL) {
SetDlgItemText(hwndDlg, IDC_USERNAME, L"");
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
break;
}
// Is button enabled?
if (!IsWindowEnabled(GetDlgItem(hwndDlg, IDC_MESSAGE)))
break;
CallService(MS_MSG_SENDMESSAGEW, hContact, 0);
g_plugin.setDword("LastSentTo", hContact);
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
break;
}
case HOTKEY_FILE:
case IDC_FILE:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL) {
SetDlgItemText(hwndDlg, IDC_USERNAME, L"");
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
break;
}
// Is button enabled?
if (!IsWindowEnabled(GetDlgItem(hwndDlg, IDC_FILE)))
break;
CallService(MS_FILE_SENDFILE, hContact, 0);
g_plugin.setDword("LastSentTo", hContact);
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
}
break;
case HOTKEY_INFO:
case IDC_USERINFO:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL) {
SetDlgItemText(hwndDlg, IDC_USERNAME, L"");
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
break;
}
// Is button enabled?
if (!IsWindowEnabled(GetDlgItem(hwndDlg, IDC_USERINFO)))
break;
CallService(MS_USERINFO_SHOWDIALOG, hContact, 0);
g_plugin.setDword("LastSentTo", hContact);
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
}
break;
case HOTKEY_HISTORY:
case IDC_HISTORY:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL) {
SetDlgItemText(hwndDlg, IDC_USERNAME, L"");
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
break;
}
// Is button enabled?
if (!IsWindowEnabled(GetDlgItem(hwndDlg, IDC_HISTORY)))
break;
CallService(MS_HISTORY_SHOWCONTACTHISTORY, hContact, 0);
g_plugin.setDword("LastSentTo", hContact);
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
}
break;
case HOTKEY_MENU:
case IDC_MENU:
{
MCONTACT hContact = GetSelectedContact(hwndDlg);
if (hContact == NULL) {
SetDlgItemText(hwndDlg, IDC_USERNAME, L"");
SetFocus(GetDlgItem(hwndDlg, IDC_USERNAME));
break;
}
// Is button enabled?
if (!IsWindowEnabled(GetDlgItem(hwndDlg, IDC_MENU)))
break;
RECT rc;
GetWindowRect(GetDlgItem(hwndDlg, IDC_MENU), &rc);
HMENU hMenu = Menu_BuildContactMenu(hContact);
int ret = TrackPopupMenu(hMenu, TPM_TOPALIGN | TPM_RIGHTBUTTON | TPM_RETURNCMD, rc.left, rc.bottom, 0, hwndDlg, nullptr);
DestroyMenu(hMenu);
if (ret) {
SendMessage(hwndDlg, WM_CLOSE, 0, 0);
Clist_MenuProcessCommand(LOWORD(ret), MPCF_CONTACTMENU, hContact);
}
g_plugin.setDword("LastSentTo", (uint32_t)hContact);
}
break;
case HOTKEY_ALL_CONTACTS:
case IDC_SHOW_ALL_CONTACTS:
{
// Get old text
HWND hEdit = GetWindow(GetWindow(hwndDlg, GW_CHILD), GW_CHILD);
wchar_t sztext[120] = L"";
if (SendMessage(hEdit, EM_GETSEL, 0, 0) != -1)
SendMessage(hEdit, EM_REPLACESEL, 0, (LPARAM)L"");
GetWindowText(hEdit, sztext, _countof(sztext));
// Fill combo
BOOL all = IsDlgButtonChecked(hwndDlg, IDC_SHOW_ALL_CONTACTS);
if (LOWORD(wParam) == HOTKEY_ALL_CONTACTS) {
// Toggle checkbox
all = !all;
CheckDlgButton(hwndDlg, IDC_SHOW_ALL_CONTACTS, all ? BST_CHECKED : BST_UNCHECKED);
}
LoadContacts(hwndDlg, all);
// Return selection
CheckText(hEdit, sztext);
}
}
break;
case WM_CLOSE:
Utils_SaveWindowPosition(hwndDlg, NULL, MODULENAME, "window");
DestroyWindow(hwndDlg);
break;
case WM_DESTROY:
UnhookWindowsHookEx(hHook);
hwndMain = nullptr;
FreeContacts();
InterlockedExchange(&main_dialog_open, 0);
break;
case WM_DRAWITEM:
{
// add icons and protocol to listbox
LPDRAWITEMSTRUCT lpdis = (LPDRAWITEMSTRUCT)lParam;
// Handle contact menu
if (lpdis->CtlID != IDC_USERNAME) {
if (lpdis->CtlType == ODT_MENU)
return Menu_DrawItem(lParam);
break;
}
// Handle combo
if (lpdis->itemID == -1)
break;
TEXTMETRIC tm;
int icon_width = 0, icon_height = 0;
RECT rc;
GetTextMetrics(lpdis->hDC, &tm);
ImageList_GetIconSize(hIml, &icon_width, &icon_height);
COLORREF clrfore = SetTextColor(lpdis->hDC, GetSysColor(lpdis->itemState & ODS_SELECTED ? COLOR_HIGHLIGHTTEXT : COLOR_WINDOWTEXT));
COLORREF clrback = SetBkColor(lpdis->hDC, GetSysColor(lpdis->itemState & ODS_SELECTED ? COLOR_HIGHLIGHT : COLOR_WINDOW));
FillRect(lpdis->hDC, &lpdis->rcItem, GetSysColorBrush(lpdis->itemState & ODS_SELECTED ? COLOR_HIGHLIGHT : COLOR_WINDOW));
// Draw icon
rc.left = lpdis->rcItem.left + 5;
rc.top = (lpdis->rcItem.bottom + lpdis->rcItem.top - icon_height) / 2;
ImageList_Draw(hIml, Clist_GetContactIcon(contacts[lpdis->itemData]->hcontact), lpdis->hDC, rc.left, rc.top, ILD_NORMAL);
// Make rect for text
rc.left += icon_width + 5;
rc.right = lpdis->rcItem.right - 1;
rc.top = (lpdis->rcItem.bottom + lpdis->rcItem.top - tm.tmHeight) / 2;
rc.bottom = rc.top + tm.tmHeight;
// Draw Protocol
if (opts.num_protos > 1) {
if (max_proto_width == 0) {
// Has to be done, else the DC isnt the right one
// Dont ask me why
for (auto &it : contacts) {
RECT rcc = { 0, 0, 0x7FFF, 0x7FFF };
DrawText(lpdis->hDC, it->proto, -1, &rcc, DT_END_ELLIPSIS | DT_NOPREFIX | DT_SINGLELINE | DT_CALCRECT);
max_proto_width = max(max_proto_width, rcc.right - rcc.left);
}
// Fix max_proto_width
if (opts.group_append && opts.group_column)
max_proto_width = min(max_proto_width, (rc.right - rc.left) / 5);
else if (opts.group_append)
max_proto_width = min(max_proto_width, (rc.right - rc.left) / 4);
else
max_proto_width = min(max_proto_width, (rc.right - rc.left) / 3);
}
RECT rc_tmp = rc;
rc_tmp.left = rc_tmp.right - max_proto_width;
DrawText(lpdis->hDC, contacts[lpdis->itemData]->proto, -1, &rc_tmp, DT_END_ELLIPSIS | DT_NOPREFIX | DT_SINGLELINE);
rc.right = rc_tmp.left - 5;
}
// Draw group
if (opts.group_append && opts.group_column) {
RECT rc_tmp = rc;
if (opts.group_column_left) {
rc_tmp.right = rc_tmp.left + (rc.right - rc.left) / 3;
rc.left = rc_tmp.right + 5;
}
else {
rc_tmp.left = rc_tmp.right - (rc.right - rc.left) / 3;
rc.right = rc_tmp.left - 5;
}
DrawText(lpdis->hDC, contacts[lpdis->itemData]->szgroup, -1, &rc_tmp, DT_END_ELLIPSIS | DT_NOPREFIX | DT_SINGLELINE);
}
// Draw text
wchar_t *name;
if (opts.group_append && !opts.group_column)
name = GetListName(contacts[lpdis->itemData]);
else
name = contacts[lpdis->itemData]->szname;
DrawText(lpdis->hDC, name, -1, &rc, DT_END_ELLIPSIS | DT_NOPREFIX | DT_SINGLELINE);
// Restore old colors
SetTextColor(lpdis->hDC, clrfore);
SetBkColor(lpdis->hDC, clrback);
}
return TRUE;
case WM_MEASUREITEM:
{
LPMEASUREITEMSTRUCT lpmis = (LPMEASUREITEMSTRUCT)lParam;
// Handle contact menu
if (lpmis->CtlID != IDC_USERNAME) {
if (lpmis->CtlType == ODT_MENU)
return Menu_MeasureItem(lParam);
break;
}
// Handle combo
TEXTMETRIC tm;
int icon_width = 0, icon_height = 0;
GetTextMetrics(GetDC(hwndDlg), &tm);
ImageList_GetIconSize(hIml, &icon_width, &icon_height);
lpmis->itemHeight = max(icon_height, tm.tmHeight);
return TRUE;
}
}
return FALSE;
}
// Show the main dialog
INT_PTR ShowDialog(WPARAM, LPARAM)
{
// Get the icons for the listbox
hIml = Clist_GetImageList();
if (!main_dialog_open) {
InterlockedExchange(&main_dialog_open, 1);
hwndMain = CreateDialog(g_plugin.getInst(), MAKEINTRESOURCE(IDD_MAIN), nullptr, MainDlgProc);
}
// Show it
SetForegroundWindow(hwndMain);
SetFocus(hwndMain);
ShowWindow(hwndMain, SW_SHOW);
return 0;
}