Как использовать 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();
}