/* Copyright (C) 2012-23 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" using namespace CWarning; static MWindowList hWindowList; class CWarningImpl { ptrW m_szTitle, m_szText; UINT m_uId; HFONT m_hFontCaption = nullptr; uint32_t m_dwFlags; HWND m_hwnd = nullptr; bool m_fIsModal; public: CWarningImpl(const wchar_t *tszTitle, const wchar_t *tszText, const UINT uId, const uint32_t dwFlags) : m_szTitle(mir_wstrdup(tszTitle)), m_szText(mir_wstrdup(tszText)) { m_uId = uId; m_dwFlags = dwFlags; m_fIsModal = ((m_dwFlags & MB_YESNO || m_dwFlags & MB_YESNOCANCEL) ? true : false); } ~CWarningImpl() { if (m_hFontCaption) ::DeleteObject(m_hFontCaption); } // static function to construct and show the dialog, returns the user's choice LRESULT ShowDialog() const { if (!m_fIsModal) { ::CreateDialogParam(g_plugin.getInst(), MAKEINTRESOURCE(IDD_WARNING), nullptr, stubDlgProc, LPARAM(this)); return 0; } return ::DialogBoxParam(g_plugin.getInst(), MAKEINTRESOURCE(IDD_WARNING), nullptr, stubDlgProc, LPARAM(this)); } ///////////////////////////////////////////////////////////////////////////////////////// // stub dlg procedure.Just register the object pointer in WM_INITDIALOG static INT_PTR CALLBACK stubDlgProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { CWarningImpl *w = reinterpret_cast(::GetWindowLongPtr(hwnd, GWLP_USERDATA)); if (w) return(w->dlgProc(hwnd, msg, wParam, lParam)); switch (msg) { case WM_INITDIALOG: w = reinterpret_cast(lParam); if (w) { ::SetWindowLongPtr(hwnd, GWLP_USERDATA, lParam); return(w->dlgProc(hwnd, msg, wParam, lParam)); } break; } return FALSE; } ///////////////////////////////////////////////////////////////////////////////////////// // dialog procedure for the warning dialog box INT_PTR CALLBACK dlgProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch (msg) { case WM_INITDIALOG: m_hwnd = hwnd; ::SetWindowTextW(hwnd, TranslateT("TabSRMM warning message")); ::Window_SetSkinIcon_IcoLib(hwnd, SKINICON_OTHER_MIRANDA); ::SendDlgItemMessage(hwnd, IDC_WARNTEXT, EM_AUTOURLDETECT, TRUE, 0); ::SendDlgItemMessage(hwnd, IDC_WARNTEXT, EM_SETEVENTMASK, 0, ENM_LINK); TranslateDialogDefault(hwnd); { CMStringW str(FORMAT, RTF_DEFAULT_HEADER, 0, 0, 0, 30 * 15); str.Append(m_szText); str.Append(L"}"); str.Replace(L"\n", L"\\line "); SETTEXTEX stx = {ST_SELECTION, CP_UTF8}; ::SendDlgItemMessage(hwnd, IDC_WARNTEXT, EM_SETTEXTEX, (WPARAM)&stx, T2Utf(str)); ::SetDlgItemTextW(hwnd, IDC_CAPTION, m_szTitle); if (m_dwFlags & CWF_NOALLOWHIDE) Utils::showDlgControl(hwnd, IDC_DONTSHOWAGAIN, SW_HIDE); if (m_dwFlags & MB_YESNO || m_dwFlags & MB_YESNOCANCEL) { Utils::showDlgControl(hwnd, IDOK, SW_HIDE); ::SetFocus(::GetDlgItem(hwnd, IDCANCEL)); } else { Utils::showDlgControl(hwnd, IDCANCEL, SW_HIDE); Utils::showDlgControl(hwnd, IDYES, SW_HIDE); Utils::showDlgControl(hwnd, IDNO, SW_HIDE); ::SetFocus(::GetDlgItem(hwnd, IDOK)); } UINT uResId = 0; if ((m_dwFlags & MB_ICONERROR) || (m_dwFlags & MB_ICONHAND)) uResId = 32513; else if ((m_dwFlags & MB_ICONEXCLAMATION) || (m_dwFlags & MB_ICONWARNING)) uResId = 32515; else if ((m_dwFlags & MB_ICONASTERISK) || (m_dwFlags & MB_ICONINFORMATION)) uResId = 32516; else if (m_dwFlags & MB_ICONQUESTION) uResId = 32514; HICON hIcon; if (uResId) hIcon = reinterpret_cast(::LoadImage(nullptr, MAKEINTRESOURCE(uResId), IMAGE_ICON, 0, 0, LR_SHARED | LR_DEFAULTSIZE)); else hIcon = ::Skin_LoadIcon(SKINICON_EVENT_MESSAGE, true); ::SendDlgItemMessageW(hwnd, IDC_WARNICON, STM_SETICON, reinterpret_cast(hIcon), 0); if (!(m_dwFlags & MB_YESNO || m_dwFlags & MB_YESNOCANCEL)) ::ShowWindow(hwnd, SW_SHOWNORMAL); WindowList_Add(hWindowList, hwnd, (UINT_PTR)hwnd); } return TRUE; case WM_CTLCOLORSTATIC: { HWND hwndChild = reinterpret_cast(lParam); UINT id = ::GetDlgCtrlID(hwndChild); if (nullptr == m_hFontCaption) { HFONT hFont = reinterpret_cast(::SendDlgItemMessage(hwnd, IDC_CAPTION, WM_GETFONT, 0, 0)); LOGFONT lf = {0}; ::GetObject(hFont, sizeof(lf), &lf); lf.lfHeight = (int)((double)lf.lfHeight * 1.7f); m_hFontCaption = ::CreateFontIndirect(&lf); ::SendDlgItemMessage(hwnd, IDC_CAPTION, WM_SETFONT, (WPARAM)m_hFontCaption, FALSE); } if (IDC_CAPTION == id) { ::SetTextColor(reinterpret_cast(wParam), ::GetSysColor(COLOR_HIGHLIGHT)); ::SendMessage(hwndChild, WM_SETFONT, (WPARAM)m_hFontCaption, FALSE); } if (IDC_WARNGROUP != id && IDC_DONTSHOWAGAIN != id) { ::SetBkColor((HDC)wParam, ::GetSysColor(COLOR_WINDOW)); return reinterpret_cast(::GetSysColorBrush(COLOR_WINDOW)); } } break; case WM_COMMAND: switch (LOWORD(wParam)) { case IDOK: case IDYES: case IDNO: if (::IsDlgButtonChecked(hwnd, IDC_DONTSHOWAGAIN)) { uint32_t newVal = M.GetDword("cWarningsL", 0) | ((uint32_t)1L << m_uId); db_set_dw(0, SRMSGMOD_T, "cWarningsL", newVal); if (LOWORD(wParam) != IDNO) { newVal = M.GetDword("cWarningsV", 0) | ((uint32_t)1L << m_uId); db_set_dw(0, SRMSGMOD_T, "cWarningsV", newVal); } } __fallthrough; case IDCANCEL: if (!m_fIsModal && (IDOK == LOWORD(wParam) || IDCANCEL == LOWORD(wParam))) // modeless dialogs can receive a IDCANCEL from destroyAll() ::DestroyWindow(hwnd); else ::EndDialog(hwnd, LOWORD(wParam)); break; } break; case WM_NOTIFY: switch (((NMHDR *)lParam)->code) { case EN_LINK: switch (((ENLINK *)lParam)->msg) { case WM_LBUTTONUP: ENLINK *e = reinterpret_cast(lParam); const wchar_t *wszUrl = Utils::extractURLFromRichEdit(e, ::GetDlgItem(hwnd, IDC_WARNTEXT)); if (wszUrl) { Utils_OpenUrlW(wszUrl); mir_free(const_cast(wszUrl)); } } } break; case WM_DESTROY: ::SetWindowLongPtr(hwnd, GWLP_USERDATA, 0); delete this; WindowList_Remove(hWindowList, hwnd); Window_FreeIcon_IcoLib(hwnd); break; } return FALSE; } }; ///////////////////////////////////////////////////////////////////////////////////////// // implementation of the CWarningImpl class // // IMPORTANT note to translators for translation of the warning dialogs: // // Make sure to NOT remove the pipe character ( | ) from the strings. This separates the // warning title from the actual warning text. // // Also, do NOT insert multiple | characters in the translated string. Not well-formatted // warnings cannot be translated and the plugin will show the untranslated versions. // // strings marked with a NOT TRANSLATABLE comment cannot be translated at all. This // will be used for important and critical error messages only. // // some strings are empty, this is intentional and used for error messages that share // the message with other possible error notifications (popups, tool tips etc.) // // Entries that do not use the LPGENW() macro are NOT TRANSLATABLE, so don't bother translating them. static wchar_t *Warnings[] = { nullptr, LPGENW("Save file|Unable to save temporary file"), // WARN_SAVEFILE LPGENW("Edit user notes|You are editing the user notes. Click the button again or use the hotkey (default: Alt+N) to save the notes and return to normal messaging mode"), /* WARN_EDITUSERNOTES */ LPGENW("Missing component|The icon pack is missing. Please install it to the default icons folder.\n\nNo icons will be available"), /* WARN_ICONPACKMISSING */ LPGENW("Aero peek warning|You have enabled Aero Peek features and loaded a custom container window skin\n\nThis can result in minor visual anomalies in the live preview feature."), /* WARN_AEROPEEKSKIN */ LPGENW("File transfer problem|Sending the image by file transfer failed.\n\nPossible reasons: File transfers not supported, either you or the target contact is offline, or you are invisible and the target contact is not on your visibility list."), /* WARN_IMGSVC_MISSING */ LPGENW("Settings problem|The option \\b1 History -> Imitate IEView API\\b0 is enabled and the History++ plugin is active. This can cause problems when using IEView as message log viewer.\n\nShould I correct the option (a restart is required)?"), /* WARN_HPP_APICHECK */ LPGENW("Configuration issue|The unattended send feature is disabled. The \\b1 send later\\b0 and \\b1 send to multiple contacts\\b0 features depend on it.\n\nYou must enable it under \\b1Options -> Message sessions -> Advanced tweaks\\b0. Changing this option requires a restart."), /* WARN_NO_SENDLATER */ LPGENW("Closing Window|You are about to close a window with multiple tabs open.\n\nProceed?"), /* WARN_CLOSEWINDOW */ LPGENW("Closing options dialog|To reflect the changes done by importing a theme in the options dialog, the dialog must be closed after loading a theme \\b1 and unsaved changes might be lost\\b0 .\n\nDo you want to continue?"), /* WARN_OPTION_CLOSE */ LPGENW("Loading a theme|Loading a color and font theme can overwrite the settings defined by your skin.\n\nDo you want to continue?"), /* WARN_THEME_OVERWRITE */ }; ///////////////////////////////////////////////////////////////////////////////////////// // send cancel message to all open warning dialogs so they are destroyed // before TabSRMM is unloaded. // // called by the OkToExit handler in globals.cpp void CWarning::destroyAll() { if (hWindowList) WindowList_Broadcast(hWindowList, WM_COMMAND, MAKEWPARAM(IDCANCEL, 0), 0); } ///////////////////////////////////////////////////////////////////////////////////////// // show a warning dialog using the id value. Check whether the user has chosen to // not show this message again. This has room for 64 different warning dialogs, which // should be enough in the first place. Extending it should not be too hard though. LRESULT CWarning::show(const int uId, uint32_t dwFlags, const wchar_t *tszTxt) { if (hWindowList == nullptr) hWindowList = WindowList_Create(); // don't open new warnings when shutdown was initiated (modal ones will otherwise // block the shutdown) if (CMimAPI::m_shutDown) return -1; wchar_t *_s = nullptr; if (tszTxt) _s = const_cast(tszTxt); else { if (uId == -1) return -1; if (dwFlags & CWF_UNTRANSLATED) _s = TranslateW(Warnings[uId]); else { // revert to untranslated warning when the translated message // is not well-formatted. _s = TranslateW(Warnings[uId]); if (mir_wstrlen(_s) < 3 || nullptr == wcschr(_s, '|')) _s = TranslateW(Warnings[uId]); } } if (mir_wstrlen(_s) > 3 && wcschr(_s, '|') != nullptr) { if (uId >= 0 && !(dwFlags & CWF_NOALLOWHIDE)) { uint32_t val = M.GetDword("cWarningsL", 0); uint32_t mask = ((__int64)1L) << uId; if (mask & val) { bool bResult = (M.GetDword("cWarningsV", 0) & mask) != 0; if (dwFlags & MB_YESNO || dwFlags & MB_YESNOCANCEL) return (bResult) ? IDYES : IDNO; return IDOK; } } ptrW s(mir_wstrdup(_s)); wchar_t *separator_pos = wcschr(s, '|'); if (separator_pos) { *separator_pos = 0; CWarningImpl *w = new CWarningImpl(s, separator_pos + 1, uId, dwFlags); if (dwFlags & MB_YESNO || dwFlags & MB_YESNOCANCEL) return w->ShowDialog(); w->ShowDialog(); } } return -1; }