Как использовать Windows ToolTip Control без ограничения инструмента
Я хочу использовать собственный инструмент подсказки tooltip для Windows (чистый API Win32, без материала MFC).
Я прочитал документ, кажется, что мне нужно отправить сообщение TTM_ADDTOOL, чтобы связать инструмент с элементом управления всплывающей подсказкой. Только после этого я могу отправить TTM_TRACKACTIVATE и TTM_TRACKPOSITION, чтобы показать всплывающую подсказку.
Но я хочу отображать всплывающую подсказку в любом месте, где бы я ни находился. Например, когда мышь нависает над областью моего окна. Этот регион не является инструментом в глазах Windows, это просто регион в моем окне.
Возможно, я могу связать окно с элементом управления всплывающей подсказкой, но не означает ли это, что мне нужно связать каждое окно, которое я создал, с элементом управления всплывающей подсказкой?
Есть ли простое решение, так что мне не нужно отправлять сообщения TTM_ADDTOOL для каждого окна?
Я на самом деле написал код, но всплывающая подсказка просто не появляется. Ответ Андерса фактически решает некоторые вопросы. И после того, как я подшутил над своим кодом, я заставляю его работать.
Если кто-то хочет знать, как это работает:
HWND toolTipWnd = ::CreateWindowExW(WS_EX_TOPMOST,
TOOLTIPS_CLASSW,0,WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP,
CW_USEDEFAULT, CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,
0,0,appHandle,0);
TOOLINFOW ti = {};
ti.cbSize = sizeof(TOOLINFOW);
ti.uFlags = TTF_ABSOLUTE | TTF_IDISHWND /* | TTF_TRACK */; // Don't specify TTF_TRACK here. Otherwise the tooltip won't show up.
ti.hwnd = toolTipWnd; // By doing this, you don't have to create another window.
ti.hinst = NULL;
ti.uId = (UINT)toolTipWnd;
ti.lpszText = L"";
::SendMessageW(toolTipWnd, TTM_ADDTOOLW, 0, (LPARAM)&ti);
::SendMessageW(toolTipWnd, TTM_SETMAXTIPWIDTH,0, (LPARAM)350);
Это создаст окно всплывающей подсказки, которое не привязано ни к какому другому окну.
Поэтому, когда вы хотите показать всплывающую подсказку (например, в ответ на сообщение WM_MOUSEHOVER), вызовите это:
TOOLINFOW ti = {};
ti.cbSize = sizeof(TOOLINFOW);
ti.hwnd = toolTipWnd;
ti.uId = (UINT)toolTipWnd;
ti.lpszText = L"Sample Tip Text";
::SendMessageW(toolTipWnd,TTM_UPDATETIPTEXTW,0,(LPARAM)&ti); // This will update the tooltip content.
::SendMessageW(toolTipWnd,TTM_TRACKACTIVATE,(WPARAM)TRUE,(LPARAM)&ti);
::SendMessageW(toolTipWnd, TTM_TRACKPOSITION,0,(LPARAM)MAKELONG(x,y)); // Update the position of your tooltip. Screen coordinate.
//::SendMessageW(toolTipWnd,TTM_POPUP,0,0); // TTM_POPUP not working.. Don't know why.
Ответы
Ответ 1
Вам нужно вызывать TTM_ADDTOOL хотя бы один раз, вы не можете вызвать TTM_SETTOOLINFO или получить TTN_GETDISPINFO без него AFAIK.
Если ваша цель XP +, вы можете уйти с использованием TTM_POPUP, чтобы отображать подсказку в любой позиции и в любое время (но вам нужно самостоятельно обрабатывать начальную задержку, если вы не хотите использовать подсказку для отслеживания)
Обычно вы вызываете TTM_ADDTOOL и связываете его с прямоугольником (TOOLINFO.rect) или дочерним окном, или вы можете установить текст в LPSTR_TEXTCALLBACK и обрабатывать TTN_GETDISPINFO, если все имеет подсказку. MSDN имеет несколько пример кода, вы должны взглянуть на...
Ответ 2
Дополнение Windows 10 (Visual Studio 2015, консольное приложение Win32)
#include "Commctrl.h"
#pragma comment (lib,"comctl32.lib")
#pragma comment(linker,"\"/manifestdependency:type='win32' \
name='Microsoft.Windows.Common-Controls' version='6.0.0.0' \
processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"")
TOOLINFOW ti = {};
ti.cbSize = sizeof(TOOLINFOW);
ti.uFlags = TTF_ABSOLUTE | TTF_IDISHWND | TTF_TRACK ; // WITH TTF_TRACK! Otherwise the tooltip doesn't follow TTM_TRACKPOSITION message!
ti.hwnd = toolTipWnd;
ti.hinst = 0;
ti.uId = (UINT)toolTipWnd;
ti.lpszText = L"";
LRESULT result;
int error;
if (!SendMessageW(toolTipWnd, TTM_ADDTOOLW, 0, (LPARAM)&ti)) {
MessageBox(NULL, L"Couldn't create the ToolTip control.", L"Error", MB_OK);
error = 0;
error = GetLastError();
}
if (!SendMessageW(toolTipWnd, TTM_SETMAXTIPWIDTH, 0, (LPARAM)350)) {
MessageBox(NULL, L"Couldn't create the ToolTip control.", L"Error", MB_OK);
error = 0;
error = GetLastError();
}