Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

297 lines
8.6 KiB

  1. #include "shellprv.h"
  2. #include "EBalloon.h"
  3. class CErrorBalloon
  4. {
  5. public:
  6. CErrorBalloon();
  7. ~CErrorBalloon();
  8. HRESULT ShowToolTip(HINSTANCE hInstance, HWND hwndTarget, const POINT *ppt, LPTSTR pszTitle, LPTSTR pszMessage, DWORD dwIconIndex, int iTimeout);
  9. void HideToolTip(BOOL fDestroy);
  10. protected:
  11. HWND _CreateToolTipWindow(HWND hwnd);
  12. static LRESULT CALLBACK _SubclassTipProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam, UINT_PTR uID, ULONG_PTR dwRefData);
  13. HWND _hwndTarget; // the targeted control hwnd
  14. HWND _hwndToolTip; // the tooltip control
  15. UINT_PTR _uTimerID; // the timer id
  16. };
  17. #define ERRORBALLOONTIMERID 1000
  18. #define EB_WARNINGBELOW 0x00000000 // default value. Balloon tooltips will be shown below the window by default.
  19. #define EB_WARNINGABOVE 0x00000004 // Ballon tooltips will be shown above the window by default.
  20. #define EB_WARNINGCENTERED 0x00000008 // Ballon tooltips will be shown pointing to the center of the window.
  21. CErrorBalloon::CErrorBalloon()
  22. {
  23. // our allocation function should have zeroed our memory. Check to make sure:
  24. ASSERT(0==_hwndToolTip);
  25. ASSERT(0==_uTimerID);
  26. }
  27. CErrorBalloon::~CErrorBalloon()
  28. {
  29. ASSERT(0==_hwndToolTip);
  30. ASSERT(0==_hwndTarget);
  31. }
  32. LRESULT CALLBACK CErrorBalloon::_SubclassTipProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam, UINT_PTR uID, ULONG_PTR dwRefData)
  33. {
  34. UNREFERENCED_PARAMETER(uID);
  35. CErrorBalloon * pthis = (CErrorBalloon*)dwRefData;
  36. switch (uMsg)
  37. {
  38. case WM_MOUSEACTIVATE: // Never activate tooltip
  39. pthis->HideToolTip(FALSE);
  40. return MA_NOACTIVATEANDEAT;
  41. case WM_DESTROY:
  42. pthis->HideToolTip(TRUE);
  43. delete pthis;
  44. break;
  45. case WM_TIMER:
  46. if (wParam == ERRORBALLOONTIMERID)
  47. {
  48. pthis->HideToolTip(FALSE);
  49. return 0;
  50. }
  51. break;
  52. default:
  53. break;
  54. }
  55. return DefSubclassProc(hwnd, uMsg, wParam, lParam);
  56. }
  57. HRESULT CErrorBalloon::ShowToolTip(HINSTANCE hinst, HWND hwndTarget, const POINT *ppt, LPTSTR pszTitle, LPTSTR pszMessage, DWORD dwIconIndex, int iTimeout)
  58. {
  59. if (_hwndToolTip)
  60. {
  61. HideToolTip(FALSE);
  62. }
  63. HWND hwnd = _CreateToolTipWindow(hwndTarget);
  64. if (hwnd)
  65. {
  66. int x, y;
  67. x = ppt->x;
  68. y = ppt->y;
  69. SendMessage(hwnd, TTM_TRACKPOSITION, 0, MAKELONG(x,y));
  70. if (pszTitle)
  71. {
  72. SendMessage(hwnd, TTM_SETTITLE, (WPARAM)dwIconIndex, (LPARAM)pszTitle);
  73. }
  74. TOOLINFO ti = {0};
  75. ti.cbSize = TTTOOLINFOW_V2_SIZE;
  76. ti.hwnd = hwnd;
  77. ti.uId = 1;
  78. ti.lpszText = pszMessage;
  79. SendMessage(hwnd, TTM_UPDATETIPTEXT, 0, (LPARAM)&ti);
  80. // Show the tooltip
  81. SendMessage(hwnd, TTM_TRACKACTIVATE, TRUE, (LPARAM)&ti);
  82. _uTimerID = SetTimer(hwnd, ERRORBALLOONTIMERID, iTimeout, NULL);
  83. if (SetWindowSubclass(hwnd, CErrorBalloon::_SubclassTipProc, (UINT_PTR)this, (LONG_PTR)this))
  84. {
  85. _hwndToolTip = hwnd;
  86. return S_OK;
  87. }
  88. // we blew the subclassing
  89. DestroyWindow(hwnd);
  90. }
  91. return E_FAIL;
  92. }
  93. // CreateToolTipWindow
  94. //
  95. // Creates our tooltip control. We share this one tooltip control and use it for all invalid
  96. // input messages. The control is hiden when not in use and then shown when needed.
  97. //
  98. HWND CErrorBalloon::_CreateToolTipWindow(HWND hwndTarget)
  99. {
  100. HWND hwnd = CreateWindow(
  101. TOOLTIPS_CLASS,
  102. NULL,
  103. WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP | TTS_BALLOON,
  104. CW_USEDEFAULT,
  105. CW_USEDEFAULT,
  106. CW_USEDEFAULT,
  107. CW_USEDEFAULT,
  108. hwndTarget,
  109. NULL,
  110. GetModuleHandle(NULL),
  111. NULL);
  112. ASSERT(!_hwndToolTip);
  113. ASSERT(!_hwndTarget);
  114. if (hwnd)
  115. {
  116. TOOLINFO ti = {0};
  117. ti.cbSize = TTTOOLINFOW_V2_SIZE;
  118. ti.uFlags = TTF_TRACK;
  119. ti.hwnd = hwnd;
  120. ti.uId = 1;
  121. // set the version so we can have non buggy mouse event forwarding
  122. SendMessage(hwnd, CCM_SETVERSION, COMCTL32_VERSION, 0);
  123. SendMessage(hwnd, TTM_ADDTOOL, 0, (LPARAM)&ti);
  124. SendMessage(hwnd, TTM_SETMAXTIPWIDTH, 0, 300);
  125. // set tink-tink?
  126. }
  127. return hwnd;
  128. }
  129. void CErrorBalloon::HideToolTip(BOOL fOnDestroy)
  130. {
  131. // When the timer fires we hide the tooltip window
  132. if (fOnDestroy)
  133. {
  134. // we need to tear everything down
  135. if (_uTimerID)
  136. {
  137. KillTimer(_hwndTarget, ERRORBALLOONTIMERID);
  138. _uTimerID = 0;
  139. }
  140. if (_hwndTarget)
  141. {
  142. // RemoveWindowSubclass(_hwndTarget, CErrorBalloon::_SubclassTargetProc, (UINT_PTR)this);
  143. RemoveProp(_hwndTarget, L"ShellConditionalBalloon");
  144. _hwndTarget = NULL;
  145. }
  146. if (_hwndToolTip)
  147. {
  148. RemoveWindowSubclass(_hwndToolTip, CErrorBalloon::_SubclassTipProc, (UINT_PTR)this);
  149. SendMessage(_hwndToolTip, TTM_TRACKACTIVATE, FALSE, 0);
  150. _hwndToolTip = NULL;
  151. }
  152. }
  153. else
  154. DestroyWindow(_hwndToolTip);
  155. }
  156. #if 0
  157. _GetTipPoint()
  158. {
  159. RECT rc;
  160. GetWindowRect(hwndTarget, &rc);
  161. POINT pt;
  162. pt.x = (rc.left+rc.right)/2;
  163. if ( EB_WARNINGABOVE & dwFlags )
  164. {
  165. pt.y = rc.top;
  166. }
  167. else if ( EB_WARNINGCENTERED & dwFlags )
  168. {
  169. pt.y = (rc.top+rc.bottom)/2;
  170. }
  171. else
  172. {
  173. pt.y = rc.bottom;
  174. }
  175. }
  176. #endif
  177. STDAPI SHShowConditionalBalloon(HWND hwnd, CBSHOW show, CONDITIONALBALLOON *pscb)
  178. {
  179. HRESULT hr = E_OUTOFMEMORY;
  180. if (hwnd)
  181. {
  182. CErrorBalloon *peb = (CErrorBalloon *) GetProp(hwnd, L"ShellConditionalBalloon");
  183. if (show != CBSHOW_HIDE && pscb)
  184. {
  185. DWORD dw = 0;
  186. BOOL fShow = TRUE;
  187. HKEY hkSession = NULL;
  188. if (SUCCEEDED(SHCreateSessionKey(MAXIMUM_ALLOWED, &hkSession)))
  189. {
  190. fShow = (ERROR_SUCCESS != SHGetValue(hkSession, NULL, pscb->pszValue, NULL, NULL, NULL));
  191. }
  192. // check cLimit
  193. if (fShow && pscb->cLimit)
  194. {
  195. ASSERT(pscb->hKey);
  196. DWORD cb = sizeof(dw);
  197. SHGetValue(pscb->hKey, pscb->pszSubKey, pscb->pszValue, NULL, &dw, &cb);
  198. fShow = dw < pscb->cLimit;
  199. }
  200. if (fShow)
  201. {
  202. // we need to show something
  203. if (!peb)
  204. {
  205. peb = new CErrorBalloon();
  206. if (peb && !SetProp(hwnd, L"ShellConditionalBalloon", peb))
  207. {
  208. delete peb;
  209. peb = NULL;
  210. }
  211. }
  212. if (peb)
  213. {
  214. TCHAR szTitle[MAX_PATH];
  215. TCHAR szMessage[INFOTIPSIZE];
  216. LoadString(pscb->hinst, pscb->idsTitle, szTitle, ARRAYSIZE(szTitle));
  217. LoadString(pscb->hinst, pscb->idsMessage, szMessage, ARRAYSIZE(szMessage));
  218. // Set the tooltip display point
  219. //if (pscb->pt.x == -1 && pscb->pt.y == -1)
  220. // _GetTipPoint(hwndTarget, &pscb->pt);
  221. DWORD dwMSecs = pscb->dwMSecs;
  222. if (dwMSecs == 0)
  223. {
  224. // default to 1 sec / 10 chars;
  225. dwMSecs = lstrlen(szMessage) * 100;
  226. if (dw == 0)
  227. dwMSecs *= 5; // first time put it up for a while
  228. }
  229. hr = peb->ShowToolTip(pscb->hinst, hwnd, &pscb->pt, szTitle, szMessage, pscb->ttiIcon, dwMSecs);
  230. if (FAILED(hr))
  231. {
  232. RemoveProp(hwnd, L"ShellConditionalBalloon");
  233. delete peb;
  234. }
  235. if (pscb->cLimit)
  236. {
  237. dw++;
  238. SHSetValueW(pscb->hKey, pscb->pszSubKey, pscb->pszValue, REG_DWORD, &dw, sizeof(dw));
  239. }
  240. }
  241. }
  242. else
  243. hr = S_FALSE;
  244. if (hkSession)
  245. {
  246. SHSetValueW(hkSession, NULL, pscb->pszValue, REG_NONE, NULL, NULL);
  247. RegCloseKey(hkSession);
  248. }
  249. }
  250. else if (peb)
  251. {
  252. peb->HideToolTip(FALSE);
  253. // we delete ourselves during WM_DESTROY
  254. }
  255. }
  256. return hr;
  257. }