//-------------------------------------------------------------
BOOL APIENTRY AddTool(HWND hTip, HWND hWnd, RECT* pr, UINT nIDTool, LPCTSTR szText) {
TOOLINFO ti;
RECT r = {0,0,0,0};
FillInToolInfo(&ti, hWnd, nIDTool);
ti.hinst = (HINSTANCE)GetModuleHandle(NULL);
ti.uFlags |= TTF_SUBCLASS | TTF_TRANSPARENT;
ti.lpszText = LPSTR(szText ? szText : LPSTR_TEXTCALLBACK);
if (!(ti.uFlags & TTF_IDISHWND)) {
if (!pr) {
pr = &r;
GetClientRect(hWnd, pr);
}
memcpy(&ti.rect, pr, sizeof(RECT));
}
BOOL res = SendMessage(hTip, TTM_ADDTOOL, 0, (LPARAM)&ti);
return res;
}
После того, как область зарегистрирована, можно управлять ее текстом посредством UpdateTipText(). Можно заметить, что в ней может быть использован тот же механизм обратного вызова текста подсказки, что и в AddTool(). Т.е. в том случае, если указатель lpszText будет установлен в NULL, то будет задействован механизм обратного вызова текста подсказки. А как же поступить в случае, если нужно просто прекратить вывод какой-либо одной подсказки, если установка lpszText в NULL задействует альтернативный способ? В этом случае нужно, чтобы lpszText указывал на пустую строку "".
//-------------------------------------------------------------
void APIENTRY UpdateTipText(HWND hTip, HWND hWnd, UINT nIDTool, LPCTSTR lpszText) {
TOOLINFO ti;
FillInToolInfo(&ti, hWnd, nIDTool);
ti.lpszText = LPSTR(lpszText ? lpszText : LPSTR_TEXTCALLBACK);
SendMessage(hTip, TTM_UPDATETIPTEXT, 0, (LPARAM)&ti);
}
Получить текст конкретной подсказки можно посредством GetTipText().
//-------------------------------------------------------------
void APIENTRY GetTipText(HWND hTip, HWND hWnd, UINT nIDTool, LPSTR szText) {
TOOLINFO ti;
if (!szText) return;
*szText = 0;
FillInToolInfo(&ti, hWnd, nIDTool);
ti.lpszText = szText;
SendMessage(hTip, TTM_GETTEXT, 0, (LPARAM)&ti);
}
Включить/выключить вывод всех подсказок, зарегистрированных данным tooltip-контролом, можно функцией EnableToolTip().
//-------------------------------------------------------------
void APIENTRY EnableToolTip(HWND hTip, BOOL activate) {
SendMessage(hTip, TTM_ACTIVATE, activate, 0);
}
ПРИМЕЧАНИЕ
Необходимо отметить, что в данной реализации способа работы с областями подсказки имеется одно ограничение – если программист явным образом задает идентификаторы областей подсказки (флаг TTF_IDISHWND в этом случае не установлен), то механизм обратного вызова текста подсказки не работает, поскольку нотификационные сообщения обратного вызова приходят не диалогу, а окну-носителю области подсказки, которое не умеет их обрабатывать (в данной реализации).
MFC
В MFC для работы с всплывающими подсказками предназначен класс CToolTipCtrl. Рассмотрим, как им пользоваться.
Первым делом необходимо добавить объект класса CToolTipCtrl в класс диалогового окна, которое вы хотите снабдить всплывающими подсказками. Тем самым мы гарантируем, что этот объект будет существовать ровно столько, сколько сам диалог. Например:
class CMFCTipsDlg : public CDialog {
…
protected:
CToolTipCtrl m_tt;
…
};
Хотя большую часть времени всплывающая подсказка не видна на экране, это обыкновенное окно, и прежде чем работать с ним, его необходимо создать и связать с уже имеющимся у нас объектом m_tt. Для этого используется функция CToolTipCtrl::Create, которая получает указатель на объект родительского окна и стиль подсказки, например:
BOOL CMFCTipsDlg::OnInitDialog() {
…
m_tt.Create(this);
…
}
Следующая наша задача – сообщить всплывающей подсказке, над какими контролами она должна появляться и какой текст при этом выдавать. Для этого нужно зарегистрировать каждый контрол в подсказке. Это выполняется с помощью функции CToolTipCtrl::AddTool.
BOOL AddTool(CWnd* pWnd, LPCTSTR lpszText = LPSTR_TEXTCALLBACK, LPCRECT lpRectTool = NULL, UINT nIDTool = 0);
Параметры pWnd и lpRectTool задают окно и прямоугольную область внутри этого окна, над которой будет появляться подсказка, а в nIDTool записывается уникальный идентификатор этой области. Если задать lpRectTool равным NULL, создаётся область, занимающая окно целиком. Именно это нам и требуется, поскольку мы хотим добавить подсказки для контролов в диалоге. В этом случае nIDTool должен быть равен нулю (значение по умолчанию). Параметр lpszText содержит указатель на текст подсказки. Если передать вместо текста значение LPSTR_TEXTCALLBACK, подсказка будет запрашивать его непосредственно перед отображением, посылая окну, содержащему контрол (или прямоугольную область), сообщение TTN_GETDISPINFO. О том, как обрабатывать это сообщение, мы поговорим немного позже.
Обычно подсказки для контролов также назначают в обработчике WM_INITDIALOG. Поступим так и мы. Например:
BOOL CMFCTipsDlg::OnInitDialog() {
Читать дальше