/* Copyright (c) 2005 Victor Pavlychko (nullbyte@sotline.net.ua) Copyright (C) 2012-24 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 AVERAGE_ITEM_HEIGHT 100 HANDLE htuLog = 0; void InitHotkeys() { HOTKEYDESC hkd = {}; hkd.szSection.a = MODULENAME; hkd.szDescription.a = LPGEN("Toggle bookmark"); hkd.pszName = "ns_bookmark"; hkd.DefHotKey = HOTKEYCODE(HOTKEYF_CONTROL, 'B') | HKF_MIRANDA_LOCAL; hkd.lParam = HOTKEY_BOOKMARK; g_plugin.addHotkey(&hkd); hkd.szDescription.a = LPGEN("Search"); hkd.pszName = "ns_search"; hkd.DefHotKey = HOTKEYCODE(HOTKEYF_CONTROL, 'F') | HKF_MIRANDA_LOCAL; hkd.lParam = HOTKEY_SEARCH; g_plugin.addHotkey(&hkd); hkd.szDescription.a = LPGEN("Search forward"); hkd.pszName = "ns_seek_forward"; hkd.DefHotKey = HOTKEYCODE(0, VK_F3) | HKF_MIRANDA_LOCAL; hkd.lParam = HOTKEY_SEEK_FORWARD; g_plugin.addHotkey(&hkd); hkd.szDescription.a = LPGEN("Search backward"); hkd.pszName = "ns_seek_back"; hkd.DefHotKey = HOTKEYCODE(HOTKEYF_SHIFT, VK_F3) | HKF_MIRANDA_LOCAL; hkd.lParam = HOTKEY_SEEK_BACK; g_plugin.addHotkey(&hkd); } ///////////////////////////////////////////////////////////////////////////////////////// // Control utilities, types and constants NewstoryListData::NewstoryListData(HWND _1) : m_hwnd(_1), redrawTimer(Miranda_GetSystemWindow(), LPARAM(this)) { items.setOwner(_1); bSortAscending = g_plugin.bSortAscending; redrawTimer.OnEvent = Callback(this, &NewstoryListData::onTimer_Draw); } void NewstoryListData::onTimer_Draw(CTimer *pTimer) { pTimer->Stop(); if (bWasAtBottom) EnsureVisible(totalCount - 1); InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::OnContextMenu(int index, POINT pt) { HMENU hMenu = NSMenu_Build(this, (index == -1) ? 0 : LoadItem(index)); if (pMsgDlg != nullptr && pMsgDlg->isChat()) Chat_CreateMenu(hMenu, pMsgDlg->getChat(), nullptr); TrackPopupMenu(hMenu, TPM_TOPALIGN | TPM_LEFTALIGN | TPM_LEFTBUTTON, pt.x, pt.y, 0, m_hwnd, nullptr); Menu_DestroyNestedMenu(hMenu); } void NewstoryListData::OnResize(int newWidth, int newHeight) { bool bDraw = false; if (newWidth != cachedWindowWidth) { cachedWindowWidth = newWidth; for (int i = 0; i < totalCount; i++) GetItem(i)->savedHeight = -1; bDraw = true; } if (newHeight != cachedWindowHeight) { cachedWindowHeight = newHeight; FixScrollPosition(true); bDraw = true; } if (bDraw) InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::AddChatEvent(SESSION_INFO *si, const LOGINFO *lin) { ScheduleDraw(); items.addChatEvent(si, lin); totalCount++; } ///////////////////////////////////////////////////////////////////////////////////////// void NewstoryListData::AddEvent(MCONTACT hContact, MEVENT hFirstEvent, int iCount) { ScheduleDraw(); items.addEvent(hContact, hFirstEvent, iCount); totalCount = items.getCount(); } ///////////////////////////////////////////////////////////////////////////////////////// void NewstoryListData::AddResults(const OBJLIST &results) { ScheduleDraw(); items.addResults(results); totalCount = items.getCount(); } void NewstoryListData::AddSelection(int iFirst, int iLast) { int start = min(totalCount - 1, iFirst); int end = min(totalCount - 1, max(0, iLast)); if (start > end) std::swap(start, end); for (int i = start; i <= end; ++i) if (auto *p = GetItem(i)) p->m_bSelected = true; InvalidateRect(m_hwnd, 0, FALSE); } bool NewstoryListData::AtBottom(void) const { if (scrollTopItem > cachedMaxTopItem) return true; if (scrollTopItem == cachedMaxTopItem && cachedMaxTopPixel >= scrollTopPixel) return true; return false; } bool NewstoryListData::AtTop(void) const { if (scrollTopItem < 0) return true; if (scrollTopItem == 0 && scrollTopPixel == 0) return true; return false; } ///////////////////////////////////////////////////////////////////////////////////////// // Edit box window procedure static LRESULT CALLBACK HistoryEditWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { auto *pData = (NewstoryListData *)GetWindowLongPtr(GetParent(hwnd), 0); switch (msg) { case WM_KEYDOWN: switch (wParam) { case VK_RETURN: pData->EndEditItem(true); return 0; case VK_ESCAPE: pData->EndEditItem(false); return 0; } break; case WM_GETDLGCODE: if (lParam) { MSG *msg2 = (MSG *)lParam; if (msg2->message == WM_KEYDOWN && msg2->wParam == VK_TAB) return 0; if (msg2->message == WM_CHAR && msg2->wParam == '\t') return 0; } return DLGC_WANTMESSAGE; case WM_KILLFOCUS: pData->EndEditItem(false); return 0; } return mir_callNextSubclass(hwnd, HistoryEditWndProc, msg, wParam, lParam); } void NewstoryListData::BeginEditItem() { if (hwndEditBox) EndEditItem(false); if (scrollTopItem > caret) return; ItemData *item = LoadItem(caret); if (item->dbe.eventType != EVENTTYPE_MESSAGE) return; RECT rc; GetClientRect(m_hwnd, &rc); int height = rc.bottom - rc.top; int top = scrollTopPixel; int idx = scrollTopItem; int itemHeight = GetItemHeight(idx); while (top < height) { if (idx == caret) break; top += itemHeight; idx++; itemHeight = GetItemHeight(idx); } int fontid, colorid; item->getFontColor(fontid, colorid); // #4012 make sure that both single & double CRLF are now double CMStringW wszText(item->getWBuf()); wszText.Replace(L"\r\n", L"\n"); wszText.Replace(L"\n", L"\r\n"); uint32_t dwStyle = WS_CHILD | WS_BORDER | WS_VSCROLL | ES_MULTILINE | ES_AUTOVSCROLL; hwndEditBox = CreateWindow(L"EDIT", wszText, dwStyle, 0, top, rc.right - rc.left, itemHeight, m_hwnd, NULL, g_plugin.getInst(), NULL); mir_subclassWindow(hwndEditBox, HistoryEditWndProc); SendMessage(hwndEditBox, WM_SETFONT, (WPARAM)g_fontTable[fontid].hfnt, 0); SendMessage(hwndEditBox, EM_SETMARGINS, EC_RIGHTMARGIN, 100); ShowWindow(hwndEditBox, SW_SHOW); SetFocus(hwndEditBox); SetForegroundWindow(hwndEditBox); } ///////////////////////////////////////////////////////////////////////////////////////// void NewstoryListData::CalcBottom() { int maxTopItem = totalCount, tmp = 0; while (maxTopItem > 0 && tmp < cachedWindowHeight) tmp += GetItemHeight(--maxTopItem); cachedMaxTopItem = maxTopItem; cachedMaxTopPixel = (cachedWindowHeight < tmp) ? cachedWindowHeight - tmp : 0; } void NewstoryListData::Clear() { items.clear(); totalCount = 0; InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ClearSelection(int iFirst, int iLast) { int start = min(0, iFirst); int end = (iLast <= 0) ? totalCount - 1 : iLast; if (start > end) std::swap(start, end); for (int i = start; i <= end; ++i) if (auto *pItem = GetItem(i)) pItem->m_bSelected = false; InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::Copy(bool bTextOnly) { Utils_ClipboardCopy(GatherSelected(bTextOnly)); } void NewstoryListData::CopyPath() { if (auto *pItem = GetItem(caret)) if (pItem->completed()) { DB::EventInfo dbei(pItem->hEvent); DB::FILE_BLOB blob(dbei); Utils_ClipboardCopy(blob.getLocalName()); } } void NewstoryListData::CopyUrl() { if (auto *pItem = GetItem(caret)) Srmm_DownloadOfflineFile(pItem->hContact, pItem->hEvent, OFD_COPYURL); } ///////////////////////////////////////////////////////////////////////////////////////// // Delete events dialog class CDeleteEventsDlg : public CDlgBase { MCONTACT m_hContact; CCtrlCheck chkDelHistory, chkForEveryone; public: bool bDelHistory = false, bForEveryone = false; CDeleteEventsDlg(MCONTACT hContact) : CDlgBase(g_plugin, IDD_EMPTYHISTORY), chkDelHistory(this, IDC_DELSERVERHISTORY), chkForEveryone(this, IDC_BOTH) { if (char *szProto = Proto_GetBaseAccountName(hContact)) { bDelHistory = ProtoServiceExists(szProto, PS_EMPTY_SRV_HISTORY); bForEveryone = (CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_4, 0) & PF4_DELETEFORALL) != 0; } } bool OnInitDialog() override { chkDelHistory.SetState(bDelHistory); chkDelHistory.Enable(bDelHistory); bool bEnabled = bDelHistory && bForEveryone; chkForEveryone.SetState(!bEnabled); chkForEveryone.Enable(bEnabled); LOGFONT lf; HFONT hFont = (HFONT)SendDlgItemMessage(m_hwnd, IDOK, WM_GETFONT, 0, 0); GetObject(hFont, sizeof(lf), &lf); lf.lfWeight = FW_BOLD; SendDlgItemMessage(m_hwnd, IDC_TOPLINE, WM_SETFONT, (WPARAM)CreateFontIndirect(&lf), 0); wchar_t szFormat[256], szFinal[256]; GetDlgItemText(m_hwnd, IDC_TOPLINE, szFormat, _countof(szFormat)); mir_snwprintf(szFinal, szFormat, Clist_GetContactDisplayName(m_hContact)); SetDlgItemText(m_hwnd, IDC_TOPLINE, szFinal); SetFocus(GetDlgItem(m_hwnd, IDNO)); SetWindowPos(m_hwnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE); return true; } bool OnApply() override { bDelHistory = chkDelHistory.IsChecked(); bForEveryone = chkForEveryone.IsChecked(); return true; } void OnDestroy() override { DeleteObject((HFONT)SendDlgItemMessage(m_hwnd, IDC_TOPLINE, WM_GETFONT, 0, 0)); } }; void NewstoryListData::DeleteItems(void) { CDeleteEventsDlg dlg(m_hContact); if (IDOK != dlg.DoModal()) return; g_plugin.bDisableDelete = true; int firstSel = -1, flags = 0; if (dlg.bDelHistory) flags |= CDF_DEL_HISTORY; if (dlg.bForEveryone) flags |= CDF_FOR_EVERYONE; for (int i = totalCount - 1; i >= 0; i--) { auto *p = GetItem(i); if (!p->m_bSelected) continue; if (p->hEvent) db_event_delete(p->hEvent, flags); items.remove(i); totalCount--; firstSel = i; } g_plugin.bDisableDelete = false; if (firstSel != -1) { SetCaret(firstSel, false); SetSelection(firstSel, firstSel); FixScrollPosition(true); } } ///////////////////////////////////////////////////////////////////////////////////////// void NewstoryListData::Download(int options) { if (auto *p = LoadItem(caret)) Srmm_DownloadOfflineFile(p->hContact, p->hEvent, options); } void NewstoryListData::EndEditItem(bool bAccept) { if (hwndEditBox == nullptr) return; if (bAccept) { if ((GetWindowLong(hwndEditBox, GWL_STYLE) & ES_READONLY) == 0) { auto *pItem = GetItem(caret); int iTextLen = GetWindowTextLengthW(hwndEditBox); mir_free(pItem->wtext); pItem->wtext = (wchar_t *)mir_alloc((iTextLen + 1) * sizeof(wchar_t)); GetWindowTextW(hwndEditBox, pItem->wtext, iTextLen+1); pItem->wtext[iTextLen] = 0; if (pItem->hContact && pItem->hEvent) { DBEVENTINFO dbei = pItem->dbe; ptrA szUtf(mir_utf8encodeW(pItem->wtext)); dbei.cbBlob = (int)mir_strlen(szUtf) + 1; dbei.pBlob = szUtf.get(); db_event_edit(pItem->hEvent, &dbei); } MTextDestroy(pItem->data); pItem->data = 0; pItem->savedHeight = -1; pItem->checkCreate(m_hwnd); } } DestroyWindow(hwndEditBox); hwndEditBox = nullptr; } void NewstoryListData::EnsureVisible(int item) { if (scrollTopItem >= item) { scrollTopItem = item; scrollTopPixel = 0; } else { RECT rc; GetClientRect(m_hwnd, &rc); int height = rc.bottom - rc.top; int idx = scrollTopItem; int itemHeight = GetItemHeight(idx); int top = itemHeight + scrollTopPixel; bool found = false; while (top < height) { if (idx == item) { itemHeight = GetItemHeight(idx); found = true; break; } top += itemHeight; idx++; itemHeight = GetItemHeight(idx); } if (!found) { scrollTopItem = item; scrollTopPixel = 0; } } FixScrollPosition(); InvalidateRect(m_hwnd, 0, FALSE); } int NewstoryListData::FindNext(const wchar_t *pwszText) { int idx = items.FindNext(caret, Filter(Filter::EVENTONLY, pwszText)); if (idx == -1 && caret > 0) idx = items.FindNext(-1, Filter(Filter::EVENTONLY, pwszText)); if (idx >= 0) { SetSelection(idx, idx); SetCaret(idx); } return idx; } int NewstoryListData::FindPrev(const wchar_t *pwszText) { int idx = items.FindPrev(caret, Filter(Filter::EVENTONLY, pwszText)); if (idx == -1 && caret != totalCount - 1) idx = items.FindPrev(totalCount, Filter(Filter::EVENTONLY, pwszText)); if (idx >= 0) { SetSelection(idx, idx); SetCaret(idx); } return idx; } void NewstoryListData::FixScrollPosition(bool bForce) { EndEditItem(false); if (bForce || cachedMaxTopItem != scrollTopItem) CalcBottom(); if (scrollTopItem < 0) scrollTopItem = 0; if (bForce || scrollTopItem > cachedMaxTopItem || (scrollTopItem == cachedMaxTopItem && scrollTopPixel < cachedMaxTopPixel)) { scrollTopItem = cachedMaxTopItem; scrollTopPixel = cachedMaxTopPixel; } } CMStringW NewstoryListData::GatherSelected(bool bTextOnly) { CMStringW ret; int eventCount = totalCount; for (int i = 0; i < eventCount; i++) { ItemData *p = GetItem(i); if (!p->m_bSelected) continue; CMStringW wszText(bTextOnly ? p->wtext : p->formatString()); RemoveBbcodes(wszText); ret.Append(wszText); ret.Append(L"\r\n"); } return ret; } ItemData* NewstoryListData::GetItem(int idx) const { if (totalCount == 0) return nullptr; return (bSortAscending) ? items.get(idx, false) : items.get(totalCount - 1 - idx, false); } int NewstoryListData::GetItemFromPixel(int yPos) { if (!totalCount) return -1; RECT rc; GetClientRect(m_hwnd, &rc); int height = rc.bottom - rc.top; int current = scrollTopItem; int top = scrollTopPixel; int bottom = top + GetItemHeight(current); while (top <= height) { if (yPos >= top && yPos <= bottom) return current; if (++current >= totalCount) break; top = bottom; bottom = top + GetItemHeight(current); } return -1; } int NewstoryListData::GetItemHeight(int index) { if (auto *pItem = LoadItem(index)) return GetItemHeight(pItem); return 0; } int NewstoryListData::GetItemHeight(ItemData *pItem) { if (pItem->savedHeight == -1) { HDC hdc = GetDC(m_hwnd); pItem->savedHeight = PaintItem(hdc, pItem, 0, cachedWindowWidth, false); ReleaseDC(m_hwnd, hdc); } return pItem->savedHeight; } bool NewstoryListData::HasSelection() const { for (int i = 0; i < totalCount; i++) if (auto *p = GetItem(i)) if (p->m_bSelected) return true; return false; } void NewstoryListData::HitTotal(int yCurr, int yTotal) { int i = 0, y = yCurr; while (i < totalCount && y > 0) { auto *pItem = GetItem(i++); if (!pItem->m_bLoaded) { i = totalCount * (double(yCurr) / double(yTotal)); y = 0; break; } else y -= GetItemHeight(pItem); } scrollTopItem = i; scrollTopPixel = y; FixScrollPosition(); } ItemData* NewstoryListData::LoadItem(int idx) { if (totalCount == 0) return nullptr; mir_cslock lck(m_csItems); return (bSortAscending) ? items.get(idx, true) : items.get(totalCount - 1 - idx, true); } void NewstoryListData::OpenFolder() { if (auto *pItem = GetItem(caret)) { if (pItem->completed()) { DB::EventInfo dbei(pItem->hEvent); DB::FILE_BLOB blob(dbei); CMStringW wszFile(blob.getLocalName()); int idx = wszFile.ReverseFind('\\'); if (idx != -1) wszFile.Truncate(idx); ::ShellExecute(nullptr, L"open", wszFile, nullptr, nullptr, SW_SHOWNORMAL); } } } int NewstoryListData::PaintItem(HDC hdc, ItemData *pItem, int top, int width, bool bDraw) { // remove any selections that might be created by the BBCodes parser MTextSendMessage(m_hwnd, pItem->data, EM_SETSEL, 0, 0); // LOGFONT lfText; COLORREF clText, clBack, clLine; int fontid, colorid; pItem->getFontColor(fontid, colorid); clText = g_fontTable[fontid].cl; if (pItem->m_bHighlighted) { clText = g_fontTable[FONT_HIGHLIGHT].cl; clBack = g_colorTable[COLOR_HIGHLIGHT_BACK].cl; clLine = g_colorTable[COLOR_FRAME].cl; } else if (pItem->m_bSelected) { clText = g_colorTable[COLOR_SELTEXT].cl; clBack = g_colorTable[COLOR_SELBACK].cl; clLine = g_colorTable[COLOR_SELFRAME].cl; } else { clLine = g_colorTable[COLOR_FRAME].cl; clBack = g_colorTable[colorid].cl; } pItem->checkCreate(m_hwnd); SIZE sz; sz.cx = width - 2; POINT pos; pos.x = 2; pos.y = top + 2; if (g_plugin.bShowType) // Message type icon pos.x += 18; if (g_plugin.bShowDirecction) // Message direction icon pos.x += 18; if (pItem->dbe.flags & DBEF_BOOKMARK) // Bookmark icon pos.x += 18; sz.cx -= pos.x; if (pItem->m_bOfflineDownloaded != 0) // Download completed icon sz.cx -= 18; HFONT hfnt = (HFONT)SelectObject(hdc, g_fontTable[fontid].hfnt); MTextMeasure(hdc, &sz, pItem->data); SelectObject(hdc, hfnt); int height = sz.cy + 5; if (!bDraw) return height; HBRUSH hbr = CreateSolidBrush(clBack); RECT rc = { 0, top, width, top + height }; FillRect(hdc, &rc, hbr); DeleteObject(hbr); SetTextColor(hdc, clText); SetBkMode(hdc, TRANSPARENT); pos.x = 2; HICON hIcon; // Message type icon if (g_plugin.bShowType) { switch (pItem->dbe.eventType) { case EVENTTYPE_MESSAGE: hIcon = g_plugin.getIcon(IDI_SENDMSG); break; case EVENTTYPE_FILE: hIcon = Skin_LoadIcon(SKINICON_EVENT_FILE); break; case EVENTTYPE_STATUSCHANGE: hIcon = g_plugin.getIcon(IDI_SIGNIN); break; default: hIcon = g_plugin.getIcon(IDI_UNKNOWN); break; } DrawIconEx(hdc, pos.x, pos.y, hIcon, 16, 16, 0, 0, DI_NORMAL); pos.x += 18; } // Direction icon if (g_plugin.bShowDirecction) { if (pItem->dbe.flags & DBEF_SENT) hIcon = g_plugin.getIcon(IDI_MSGOUT); else hIcon = g_plugin.getIcon(IDI_MSGIN); DrawIconEx(hdc, pos.x, pos.y, hIcon, 16, 16, 0, 0, DI_NORMAL); pos.x += 18; } // Bookmark icon if (pItem->dbe.flags & DBEF_BOOKMARK) { DrawIconEx(hdc, pos.x, pos.y, g_plugin.getIcon(IDI_BOOKMARK), 16, 16, 0, 0, DI_NORMAL); pos.x += 18; } // Finished icon if (pItem->m_bOfflineDownloaded != 0) { if (pItem->completed()) DrawIconEx(hdc, width - 20, pos.y, g_plugin.getIcon(IDI_OK), 16, 16, 0, 0, DI_NORMAL); else { HPEN hpn = (HPEN)SelectObject(hdc, CreatePen(PS_SOLID, 4, g_colorTable[COLOR_PROGRESS].cl)); MoveToEx(hdc, rc.left, rc.bottom - 4, 0); LineTo(hdc, rc.left + (rc.right - rc.left) * int(pItem->m_bOfflineDownloaded) / 100, rc.bottom - 4); DeleteObject(SelectObject(hdc, hpn)); } } hfnt = (HFONT)SelectObject(hdc, g_fontTable[fontid].hfnt); MTextDisplay(hdc, pos, sz, pItem->data); SelectObject(hdc, hfnt); HPEN hpn = (HPEN)SelectObject(hdc, CreatePen(PS_SOLID, 1, clLine)); MoveToEx(hdc, rc.left, rc.bottom - 1, 0); LineTo(hdc, rc.right, rc.bottom - 1); DeleteObject(SelectObject(hdc, hpn)); return height; } void NewstoryListData::RecalcScrollBar() { if (totalCount == 0) return; int yTotal = 0, yTop = 0, numRec = 0; for (int i = 0; i < totalCount; i++) { if (i == scrollTopItem) yTop = yTotal - scrollTopPixel; auto *pItem = GetItem(i); if (pItem->m_bLoaded) { yTotal += GetItemHeight(pItem); numRec++; } } if (numRec != totalCount) { double averageH = double(yTotal) / double(numRec); yTotal = totalCount * averageH; yTop = scrollTopItem * averageH; } SCROLLINFO si = {}; si.cbSize = sizeof(si); si.fMask = SIF_ALL; si.nMin = 0; si.nMax = yTotal; si.nPage = cachedWindowHeight; si.nPos = yTop; if (si.nPos != cachedScrollbarPos || si.nMax != cachedScrollbarMax) { cachedScrollbarPos = si.nPos; cachedScrollbarMax = si.nMax; SetScrollInfo(m_hwnd, SB_VERT, &si, TRUE); } } void NewstoryListData::Quote() { if (pMsgDlg) { CMStringW wszText(GatherSelected(true)); RemoveBbcodes(wszText); pMsgDlg->SetMessageText(Srmm_Quote(wszText)); SetFocus(pMsgDlg->GetInput()); } } void NewstoryListData::Reply() { if (pMsgDlg) if (auto *pItem = GetItem(caret)) pMsgDlg->SetQuoteEvent(pItem->hEvent); } void NewstoryListData::ScheduleDraw() { bWasAtBottom = AtBottom(); redrawTimer.Stop(); redrawTimer.Start(30); } void NewstoryListData::SetCaret(int idx, bool bEnsureVisible) { if (idx < totalCount) { caret = idx; if (bEnsureVisible) EnsureVisible(idx); } } void NewstoryListData::SetContact(MCONTACT hContact) { m_hContact = hContact; WindowList_Add(g_hNewstoryLogs, m_hwnd, hContact); } void NewstoryListData::SetDialog(CSrmmBaseDialog *pDlg) { if (pMsgDlg = pDlg) SetContact(pDlg->m_hContact); } void NewstoryListData::SetPos(int pos) { SetSelection((selStart == -1) ? pos : selStart, pos); SetCaret(pos); } void NewstoryListData::SetSelection(int iFirst, int iLast) { int start = min(totalCount - 1, iFirst); int end = min(totalCount - 1, max(0, iLast)); if (start > end) std::swap(start, end); int count = totalCount; for (int i = 0; i < count; ++i) { auto *p = GetItem(i); if (i >= start && i <= end) p->m_bSelected = true; else p->m_bSelected = false; } InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ToggleBookmark() { int eventCount = totalCount; for (int i = 0; i < eventCount; i++) { ItemData *p = GetItem(i); if (!p->m_bSelected) continue; if (p->dbe.flags & DBEF_BOOKMARK) p->dbe.flags &= ~DBEF_BOOKMARK; else p->dbe.flags |= DBEF_BOOKMARK; db_event_edit(p->hEvent, &p->dbe); p->setText(m_hwnd); } InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ToggleSelection(int iFirst, int iLast) { int start = min(totalCount - 1, iFirst); int end = min(totalCount - 1, max(0, iLast)); if (start > end) std::swap(start, end); for (int i = start; i <= end; ++i) { auto *p = GetItem(i); p->m_bSelected = !p->m_bSelected; } InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::TryUp(int iCount) { if (totalCount == 0) return; auto *pTop = GetItem(0); MCONTACT hContact = pTop->hContact; if (pTop->hEvent == 0 || hContact == 0) return; int i; for (i = 0; i < iCount; i++) { MEVENT hPrev = db_event_prev(hContact, pTop->hEvent); if (hPrev == 0) break; auto *p = items.insert(0); p->hContact = hContact; p->hEvent = hPrev; totalCount++; } ItemData *pPrev = nullptr; for (int j = 0; j < i + 1; j++) { auto *pItem = GetItem(j); pPrev = pItem->checkNext(pPrev, m_hwnd); } caret = 0; CalcBottom(); FixScrollPosition(); InvalidateRect(m_hwnd, 0, FALSE); } ///////////////////////////////////////////////////////////////////////////////////////// // Navigation by coordinates void NewstoryListData::LineUp() { if (AtTop()) TryUp(1); else ScrollUp(10); } void NewstoryListData::LineDown() { if (!AtBottom()) ScrollDown(10); } void NewstoryListData::PageUp() { if (AtTop()) TryUp(10); else ScrollUp(cachedWindowHeight); } void NewstoryListData::PageDown() { if (!AtBottom()) ScrollDown(cachedWindowHeight); } ///////////////////////////////////////////////////////////////////////////////////////// // Navigation by events void NewstoryListData::EventUp() { if (caret == 0) TryUp(1); else SetPos(caret - 1); } void NewstoryListData::EventDown() { if (caret < totalCount-1) SetPos(caret + 1); } void NewstoryListData::EventPageUp() { if (caret >= 10) SetPos(caret - 10); else TryUp(caret == 10 ? 1 : 10 - caret); } void NewstoryListData::EventPageDown() { if (caret < totalCount - 10) SetPos(caret + 10); else SetPos(totalCount - 1); } ///////////////////////////////////////////////////////////////////////////////////////// // Common navigation functions void NewstoryListData::ScrollBottom() { if (!totalCount) return; scrollTopItem = cachedMaxTopItem; scrollTopPixel = cachedMaxTopPixel; FixScrollPosition(true); InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ScrollDown(int deltaY) { int iHeight = GetItemHeight(scrollTopItem) + scrollTopPixel; if (iHeight > deltaY) scrollTopPixel -= deltaY; else { deltaY -= iHeight; bool bFound = false; for (int i = scrollTopItem + 1; i < totalCount; i++) { iHeight = GetItemHeight(i); if (iHeight > deltaY) { scrollTopPixel = -deltaY; scrollTopItem = i; bFound = true; break; } deltaY -= iHeight; } if (!bFound) scrollTopItem = scrollTopPixel = 0; } FixScrollPosition(); InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ScrollTop() { scrollTopItem = scrollTopPixel = 0; FixScrollPosition(); InvalidateRect(m_hwnd, 0, FALSE); } void NewstoryListData::ScrollUp(int deltaY) { int reserveY = -scrollTopPixel; // distance in pixels between the top event beginning and the window top coordinate if (reserveY >= deltaY) scrollTopPixel += deltaY; // stay on the same event, just move up else { deltaY -= reserveY; // move to the appropriate event first, then calculate the gap bool bFound = false; for (int i = scrollTopItem - 1; i >= 0; i--) { int iHeight = GetItemHeight(i); if (iHeight > deltaY) { scrollTopPixel = deltaY - iHeight; scrollTopItem = i; bFound = true; break; } deltaY -= iHeight; } if (!bFound) scrollTopItem = scrollTopPixel = 0; } FixScrollPosition(); InvalidateRect(m_hwnd, 0, FALSE); } ///////////////////////////////////////////////////////////////////////////////////////// // NewStory history control window procedure LRESULT CALLBACK NewstoryListWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { int idx; POINT pt; NewstoryListData *data = (NewstoryListData *)GetWindowLongPtr(hwnd, 0); MSG message = { hwnd, msg, wParam, lParam }; switch (Hotkey_Check(&message, MODULENAME)) { case HOTKEY_SEEK_FORWARD: PostMessage(GetParent(hwnd), WM_COMMAND, MAKELONG(IDOK, BN_CLICKED), 1); break; case HOTKEY_SEEK_BACK: PostMessage(GetParent(hwnd), WM_COMMAND, MAKELONG(IDC_FINDPREV, BN_CLICKED), 1); break; case HOTKEY_SEARCH: PostMessage(GetParent(hwnd), WM_COMMAND, MAKELONG(IDC_SEARCH, BN_CLICKED), 1); break; case HOTKEY_BOOKMARK: data->ToggleBookmark(); return 0; } switch (msg) { case WM_CREATE: data = new NewstoryListData(hwnd); SetWindowLongPtr(hwnd, 0, (LONG_PTR)data); if (!g_plugin.bOptVScroll) SetWindowLong(hwnd, GWL_STYLE, GetWindowLong(hwnd, GWL_STYLE) & ~WS_VSCROLL); break; // History list control messages case NSM_SELECTITEMS: data->AddSelection(wParam, lParam); return 0; case NSM_SEEKTIME: { int eventCount = data->totalCount; for (int i = 0; i < eventCount; i++) { auto *p = data->GetItem(i); if (p->dbe.timestamp >= wParam) { data->SetSelection(i, i); data->SetCaret(i); break; } if (i == eventCount - 1) { data->SetSelection(i, i); data->SetCaret(i); } } } return TRUE; case NSM_ADDEVENT: data->AddEvent(wParam, lParam, 1); break; case NSM_SET_OPTIONS: data->bSortAscending = g_plugin.bSortAscending; data->scrollTopPixel = 0; data->FixScrollPosition(true); InvalidateRect(hwnd, 0, FALSE); break; case UM_ADDEVENT: if (data->pMsgDlg == nullptr) data->AddEvent(wParam, lParam, 1); break; case UM_EDITEVENT: idx = data->items.find(lParam); if (idx != -1) { auto *p = data->GetItem(idx); p->load(true); p->setText(data->m_hwnd); InvalidateRect(hwnd, 0, FALSE); } break; case UM_REMOVEEVENT: idx = data->items.find(lParam); if (idx != -1) { data->items.remove(idx); data->totalCount--; data->FixScrollPosition(true); InvalidateRect(hwnd, 0, FALSE); } break; case WM_SIZE: data->OnResize(LOWORD(lParam), HIWORD(lParam)); break; case WM_COMMAND: if (NSMenu_Process(LOWORD(wParam), data)) return 1; break; case WM_ERASEBKGND: return 1; case WM_PAINT: /* we get so many InvalidateRect()'s that there is no point painting, Windows in theory shouldn't queue up WM_PAINTs in this case but it does so we'll just ignore them */ if (IsWindowVisible(hwnd)) { PAINTSTRUCT ps; HDC hdcWindow = BeginPaint(hwnd, &ps); RECT rc; GetClientRect(hwnd, &rc); HDC hdc = CreateCompatibleDC(hdcWindow); HBITMAP hbmSave = (HBITMAP)SelectObject(hdc, CreateCompatibleBitmap(hdcWindow, rc.right - rc.left, rc.bottom - rc.top)); int height = rc.bottom - rc.top; int width = rc.right - rc.left; int top = data->scrollTopPixel; for (idx = data->scrollTopItem; top < height && idx < data->totalCount; idx++) top += data->PaintItem(hdc, data->LoadItem(idx), top, width, !data->hwndEditBox || data->caret != idx); data->cachedMaxDrawnItem = idx; if (top <= height) { RECT rc2; SetRect(&rc2, 0, top, width, height); HBRUSH hbr = CreateSolidBrush(g_colorTable[COLOR_BACK].cl); FillRect(hdc, &rc2, hbr); DeleteObject(hbr); } if (g_plugin.bOptVScroll) data->RecalcScrollBar(); if (g_plugin.bDrawEdge) DrawEdge(hdc, &rc, BDR_SUNKENOUTER, BF_RECT); BitBlt(hdcWindow, 0, 0, rc.right, rc.bottom, hdc, 0, 0, SRCCOPY); DeleteObject(SelectObject(hdc, hbmSave)); DeleteDC(hdc); EndPaint(hwnd, &ps); } break; case WM_CONTEXTMENU: pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; if (pt.x == -1 && pt.y == -1) GetCursorPos(&pt); POINT pt2 = pt; ScreenToClient(hwnd, &pt2); idx = data->GetItemFromPixel(pt2.y); if (idx != -1) { if (data->caret != idx) data->EndEditItem(false); data->SetCaret(idx); if (!data->HasSelection()) data->SetSelection(idx, idx); } data->OnContextMenu(idx, pt); break; case WM_KILLFOCUS: if (wParam && (HWND)wParam != data->hwndEditBox) data->EndEditItem(false); if (data->pMsgDlg && ((HWND)wParam == data->pMsgDlg->GetInput() || (HWND)wParam == data->pMsgDlg->GetHwnd())) data->ClearSelection(0, -1); return 0; case WM_SETFOCUS: return 0; case WM_GETDLGCODE: if (lParam) { MSG *msg2 = (MSG *)lParam; if (msg2->message == WM_KEYDOWN) { if (msg2->wParam == VK_TAB) return 0; if (msg2->wParam == VK_ESCAPE && !data->hwndEditBox) return 0; } else if (msg2->message == WM_CHAR) { if (msg2->wParam == '\t') return 0; if (msg2->wParam == 27 && !data->hwndEditBox) return 0; } } return DLGC_WANTMESSAGE; case WM_KEYDOWN: { bool isShift = (GetKeyState(VK_SHIFT) & 0x80) != 0; bool isCtrl = (GetKeyState(VK_CONTROL) & 0x80) != 0; if (!data->bWasShift && isShift) data->selStart = data->caret; else if (data->bWasShift && !isShift) data->selStart = -1; data->bWasShift = isShift; switch (wParam) { case VK_UP: if (g_plugin.bHppCompat) data->EventUp(); else data->LineUp(); break; case VK_DOWN: if (g_plugin.bHppCompat) data->EventDown(); else data->LineDown(); break; case VK_PRIOR: if (isCtrl) data->ScrollTop(); else if (g_plugin.bHppCompat) data->EventPageUp(); else data->PageUp(); break; case VK_NEXT: if (isCtrl) data->ScrollBottom(); else if (g_plugin.bHppCompat) data->EventPageDown(); else data->PageDown(); break; case VK_HOME: data->ScrollTop(); break; case VK_END: data->ScrollBottom(); break; case VK_F2: data->BeginEditItem(); break; case VK_ESCAPE: if (data->hwndEditBox) data->EndEditItem(false); break; case VK_DELETE: data->DeleteItems(); break; case VK_INSERT: case 'C': if (isCtrl) data->Copy(); break; case 'A': if (isCtrl) data->AddSelection(0, data->totalCount); break; } } break; case WM_LBUTTONDOWN: pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; idx = data->GetItemFromPixel(pt.y); if (idx >= 0) { if (data->caret != idx) data->EndEditItem(false); auto *pItem = data->LoadItem(idx); if (wParam & MK_CONTROL) { data->ToggleSelection(idx, idx); data->SetCaret(idx); } else if (wParam & MK_SHIFT) { data->AddSelection(data->caret, idx); data->SetCaret(idx); } else { pt.y -= pItem->savedTop; CMStringW wszUrl; if (pItem->isLink(hwnd, pt, &wszUrl)) { Utils_OpenUrlW(wszUrl); return 0; } data->selStart = idx; data->SetSelection(idx, idx); data->SetCaret(idx); } } SetFocus(hwnd); return 0; case WM_LBUTTONUP: pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; idx = data->GetItemFromPixel(pt.y); if (idx >= 0) data->selStart = -1; break; case WM_LBUTTONDBLCLK: pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; idx = data->GetItemFromPixel(pt.y); if (idx >= 0) { if (data->caret != idx) data->EndEditItem(false); auto *pItem = data->LoadItem(idx); pt.y -= pItem->savedTop; if (pItem->m_bOfflineFile) { Srmm_DownloadOfflineFile(pItem->hContact, pItem->hEvent, OFD_DOWNLOAD | OFD_RUN); return 0; } if (data->caret == idx) { data->BeginEditItem(); return 0; } } SetFocus(hwnd); return 0; case WM_MOUSEMOVE: pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; idx = data->GetItemFromPixel(pt.y); if (idx >= 0) { auto *pItem = data->LoadItem(idx); MTextSendMessage(hwnd, pItem->data, msg, wParam, lParam); HCURSOR hOldCursor = GetCursor(); HCURSOR hNewCursor = LoadCursor(0, (pItem->isLink(hwnd, pt) || pItem->m_bOfflineFile) ? IDC_HAND : IDC_ARROW); if (hOldCursor != hNewCursor) SetCursor(hNewCursor); if (data->selStart != -1) { data->SetSelection(data->selStart, idx); InvalidateRect(hwnd, 0, FALSE); } } break; case WM_MOUSEWHEEL: if ((short)HIWORD(wParam) < 0) data->LineDown(); else data->LineUp(); return TRUE; case WM_VSCROLL: { int s_scrollTopItem = data->scrollTopItem; int s_scrollTopPixel = data->scrollTopPixel; switch (LOWORD(wParam)) { case SB_LINEUP: if (g_plugin.bHppCompat) data->EventUp(); else data->LineUp(); break; case SB_LINEDOWN: if (g_plugin.bHppCompat) data->EventDown(); else data->LineDown(); break; case SB_PAGEUP: if (g_plugin.bHppCompat) data->EventPageUp(); else data->PageUp(); break; case SB_PAGEDOWN: if (g_plugin.bHppCompat) data->EventPageDown(); else data->PageDown(); break; case SB_BOTTOM: data->ScrollBottom(); break; case SB_TOP: data->ScrollTop(); break; case SB_THUMBTRACK: SCROLLINFO si; si.cbSize = sizeof(si); si.fMask = SIF_ALL; GetScrollInfo(hwnd, SB_VERT, &si); data->HitTotal(si.nTrackPos, si.nMax); break; default: return 0; } if (s_scrollTopItem != data->scrollTopItem || s_scrollTopPixel != data->scrollTopPixel) InvalidateRect(hwnd, 0, FALSE); } break; case WM_CTLCOLORSTATIC: case WM_CTLCOLOREDIT: if (lParam == INT_PTR(data->hwndEditBox)) { SetBkColor((HDC)wParam, g_colorTable[COLOR_SELBACK].cl); return (LRESULT)g_plugin.hBackBrush; } break; case WM_DESTROY: WindowList_Add(g_hNewstoryLogs, hwnd); delete data; SetWindowLongPtr(hwnd, 0, 0); break; } return DefWindowProc(hwnd, msg, wParam, lParam); } void InitNewstoryControl() { htuLog = MTextRegister("Newstory", MTEXT_FANCY_DEFAULT | MTEXT_SYSTEM_HICONS | MTEXT_FANCY_SMILEYS); WNDCLASS wndclass = {}; wndclass.style = /*CS_HREDRAW | CS_VREDRAW | */CS_DBLCLKS | CS_GLOBALCLASS; wndclass.lpfnWndProc = NewstoryListWndProc; wndclass.cbWndExtra = sizeof(void *); wndclass.hInstance = g_plugin.getInst(); wndclass.hCursor = LoadCursor(NULL, IDC_ARROW); wndclass.lpszClassName = _T(NEWSTORYLIST_CLASS); RegisterClass(&wndclass); }