| 1 | /*
 | 
|---|
| 2 |  * Hotkey control
 | 
|---|
| 3 |  *
 | 
|---|
| 4 |  * Copyright 1998, 1999 Eric Kohl
 | 
|---|
| 5 |  * Copyright 2002 Gyorgy 'Nog' Jeney
 | 
|---|
| 6 |  *
 | 
|---|
| 7 |  * This library is free software; you can redistribute it and/or
 | 
|---|
| 8 |  * modify it under the terms of the GNU Lesser General Public
 | 
|---|
| 9 |  * License as published by the Free Software Foundation; either
 | 
|---|
| 10 |  * version 2.1 of the License, or (at your option) any later version.
 | 
|---|
| 11 |  *
 | 
|---|
| 12 |  * This library is distributed in the hope that it will be useful,
 | 
|---|
| 13 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
|---|
| 14 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | 
|---|
| 15 |  * Lesser General Public License for more details.
 | 
|---|
| 16 |  *
 | 
|---|
| 17 |  * You should have received a copy of the GNU Lesser General Public
 | 
|---|
| 18 |  * License along with this library; if not, write to the Free Software
 | 
|---|
| 19 |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 | 
|---|
| 20 |  *
 | 
|---|
| 21 |  * TODO:
 | 
|---|
| 22 |  *   - What are we meant to do with the WM_CHAR message?
 | 
|---|
| 23 |  */
 | 
|---|
| 24 | 
 | 
|---|
| 25 | #include <string.h>
 | 
|---|
| 26 | #include "winbase.h"
 | 
|---|
| 27 | #include "commctrl.h"
 | 
|---|
| 28 | #include "comctl32.h"
 | 
|---|
| 29 | #include "wine/debug.h"
 | 
|---|
| 30 | 
 | 
|---|
| 31 | WINE_DEFAULT_DEBUG_CHANNEL(hotkey);
 | 
|---|
| 32 | 
 | 
|---|
| 33 | typedef struct tagHOTKEY_INFO
 | 
|---|
| 34 | {
 | 
|---|
| 35 |     HWND  hwndSelf;
 | 
|---|
| 36 |     HFONT hFont;
 | 
|---|
| 37 |     BOOL  bFocus;
 | 
|---|
| 38 |     INT   nHeight;
 | 
|---|
| 39 |     WORD  HotKey;
 | 
|---|
| 40 |     WORD  InvComb;
 | 
|---|
| 41 |     WORD  InvMod;
 | 
|---|
| 42 |     BYTE  CurrMod;
 | 
|---|
| 43 |     INT   CaretPos;
 | 
|---|
| 44 |     DWORD ScanCode;
 | 
|---|
| 45 |     WCHAR strNone[15]; /* hope its long enough ... */
 | 
|---|
| 46 | } HOTKEY_INFO;
 | 
|---|
| 47 | 
 | 
|---|
| 48 | #define HOTKEY_GetInfoPtr(hwnd) ((HOTKEY_INFO *)GetWindowLongA (hwnd, 0))
 | 
|---|
| 49 | 
 | 
|---|
| 50 | static const WCHAR HOTKEY_plussep[] = { ' ', '+', ' ' };
 | 
|---|
| 51 | 
 | 
|---|
| 52 | #define IsOnlySet(flags) (infoPtr->CurrMod == (flags))
 | 
|---|
| 53 | 
 | 
|---|
| 54 | static BOOL
 | 
|---|
| 55 | HOTKEY_IsCombInv(HOTKEY_INFO *infoPtr)
 | 
|---|
| 56 | {
 | 
|---|
| 57 |     TRACE("(infoPtr=%p)\n", infoPtr);
 | 
|---|
| 58 |     if((infoPtr->InvComb & HKCOMB_NONE) && !infoPtr->CurrMod)
 | 
|---|
| 59 |         return TRUE;
 | 
|---|
| 60 |     if((infoPtr->InvComb & HKCOMB_S) && IsOnlySet(HOTKEYF_SHIFT))
 | 
|---|
| 61 |         return TRUE;
 | 
|---|
| 62 |     if((infoPtr->InvComb & HKCOMB_C) && IsOnlySet(HOTKEYF_CONTROL))
 | 
|---|
| 63 |         return TRUE;
 | 
|---|
| 64 |     if((infoPtr->InvComb & HKCOMB_A) && IsOnlySet(HOTKEYF_ALT))
 | 
|---|
| 65 |         return TRUE;
 | 
|---|
| 66 |     if((infoPtr->InvComb & HKCOMB_SC) && 
 | 
|---|
| 67 |        IsOnlySet(HOTKEYF_SHIFT | HOTKEYF_CONTROL))
 | 
|---|
| 68 |         return TRUE;
 | 
|---|
| 69 |     if((infoPtr->InvComb & HKCOMB_SA) && IsOnlySet(HOTKEYF_SHIFT | HOTKEYF_ALT))
 | 
|---|
| 70 |         return TRUE;
 | 
|---|
| 71 |     if((infoPtr->InvComb & HKCOMB_CA) && 
 | 
|---|
| 72 |        IsOnlySet(HOTKEYF_CONTROL | HOTKEYF_ALT))
 | 
|---|
| 73 |         return TRUE;
 | 
|---|
| 74 |     if((infoPtr->InvComb & HKCOMB_SCA) && 
 | 
|---|
| 75 |        IsOnlySet(HOTKEYF_SHIFT | HOTKEYF_CONTROL | HOTKEYF_ALT))
 | 
|---|
| 76 |         return TRUE;
 | 
|---|
| 77 | 
 | 
|---|
| 78 |     TRACE("() Modifiers are valid\n");
 | 
|---|
| 79 |     return FALSE;
 | 
|---|
| 80 | }
 | 
|---|
| 81 | #undef IsOnlySet
 | 
|---|
| 82 | 
 | 
|---|
| 83 | static void
 | 
|---|
| 84 | HOTKEY_DrawHotKey(HOTKEY_INFO *infoPtr, LPCWSTR KeyName, WORD NameLen, 
 | 
|---|
| 85 |                   LPRECT rc, HDC hdc)
 | 
|---|
| 86 | {
 | 
|---|
| 87 |     SIZE TextSize;
 | 
|---|
| 88 |     DWORD dwExStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_EXSTYLE);
 | 
|---|
| 89 | 
 | 
|---|
| 90 |     /* We have to allow some space for the frame to be drawn */
 | 
|---|
| 91 |     rc->left += 2;
 | 
|---|
| 92 |     rc->top++;
 | 
|---|
| 93 |     DrawTextW(hdc, KeyName, NameLen, rc, DT_LEFT | DT_VCENTER);
 | 
|---|
| 94 |     rc->left -= 2;
 | 
|---|
| 95 |     rc->top--;
 | 
|---|
| 96 |     if(dwExStyle & WS_EX_CLIENTEDGE)
 | 
|---|
| 97 |         DrawEdge(hdc, rc, EDGE_SUNKEN, BF_RECT | BF_ADJUST);
 | 
|---|
| 98 | 
 | 
|---|
| 99 |     /* Get the text size and position the caret accordingly */
 | 
|---|
| 100 |     GetTextExtentPoint32W (hdc, KeyName, NameLen, &TextSize);
 | 
|---|
| 101 |     infoPtr->CaretPos = TextSize.cx + 2;
 | 
|---|
| 102 |     SetCaretPos(infoPtr->CaretPos, 3);
 | 
|---|
| 103 | }
 | 
|---|
| 104 | 
 | 
|---|
| 105 | /* Draw the names of the keys in the control */
 | 
|---|
| 106 | static void 
 | 
|---|
| 107 | HOTKEY_Refresh(HOTKEY_INFO *infoPtr, HDC hdc)
 | 
|---|
| 108 | {
 | 
|---|
| 109 |     WCHAR KeyName[sizeof(WCHAR) * 64];
 | 
|---|
| 110 |     WORD NameLen = 0;
 | 
|---|
| 111 |     BYTE Modifier;
 | 
|---|
| 112 |     RECT rc;
 | 
|---|
| 113 | 
 | 
|---|
| 114 |     GetClientRect(infoPtr->hwndSelf, &rc);
 | 
|---|
| 115 | 
 | 
|---|
| 116 |     TRACE("(infoPtr=%p hdc=%p)\n", infoPtr, hdc);
 | 
|---|
| 117 | 
 | 
|---|
| 118 |     if(!infoPtr->CurrMod && !infoPtr->HotKey) {
 | 
|---|
| 119 |         HOTKEY_DrawHotKey (infoPtr, infoPtr->strNone, 4, &rc, hdc);
 | 
|---|
| 120 |         return;
 | 
|---|
| 121 |     }
 | 
|---|
| 122 |         
 | 
|---|
| 123 |     if(infoPtr->HotKey)
 | 
|---|
| 124 |         Modifier = HIBYTE(infoPtr->HotKey);
 | 
|---|
| 125 |     else if(HOTKEY_IsCombInv(infoPtr)) 
 | 
|---|
| 126 |         Modifier = infoPtr->InvMod;
 | 
|---|
| 127 |     else
 | 
|---|
| 128 |         Modifier = infoPtr->CurrMod;
 | 
|---|
| 129 | 
 | 
|---|
| 130 |     if(Modifier & HOTKEYF_CONTROL) {
 | 
|---|
| 131 |         GetKeyNameTextW(MAKELPARAM(0, MapVirtualKeyW(VK_CONTROL, 0)),
 | 
|---|
| 132 |                                   KeyName, 64);
 | 
|---|
| 133 |         NameLen = lstrlenW(KeyName);
 | 
|---|
| 134 |         memcpy(&KeyName[NameLen], HOTKEY_plussep, sizeof(HOTKEY_plussep));
 | 
|---|
| 135 |         NameLen += 3;
 | 
|---|
| 136 |     }
 | 
|---|
| 137 |     if(Modifier & HOTKEYF_SHIFT) {
 | 
|---|
| 138 |         GetKeyNameTextW(MAKELPARAM(0, MapVirtualKeyW(VK_SHIFT, 0)),
 | 
|---|
| 139 |                                    &KeyName[NameLen], 64 - NameLen);
 | 
|---|
| 140 |         NameLen = lstrlenW(KeyName);
 | 
|---|
| 141 |         memcpy(&KeyName[NameLen], HOTKEY_plussep, sizeof(HOTKEY_plussep));
 | 
|---|
| 142 |         NameLen += 3;
 | 
|---|
| 143 |     }
 | 
|---|
| 144 |     if(Modifier & HOTKEYF_ALT) {
 | 
|---|
| 145 |         GetKeyNameTextW(MAKELPARAM(0, MapVirtualKeyW(VK_MENU, 0)),
 | 
|---|
| 146 |                                    &KeyName[NameLen], 64 - NameLen);
 | 
|---|
| 147 |         NameLen = lstrlenW(KeyName);
 | 
|---|
| 148 |         memcpy(&KeyName[NameLen], HOTKEY_plussep, sizeof(HOTKEY_plussep));
 | 
|---|
| 149 |         NameLen += 3;
 | 
|---|
| 150 |     }
 | 
|---|
| 151 | 
 | 
|---|
| 152 |     if(infoPtr->HotKey) {
 | 
|---|
| 153 |         GetKeyNameTextW(infoPtr->ScanCode, &KeyName[NameLen], 64 - NameLen);
 | 
|---|
| 154 |         NameLen = lstrlenW(KeyName);
 | 
|---|
| 155 |     }
 | 
|---|
| 156 |     else
 | 
|---|
| 157 |         KeyName[NameLen] = 0;
 | 
|---|
| 158 | 
 | 
|---|
| 159 |     HOTKEY_DrawHotKey (infoPtr, KeyName, NameLen, &rc, hdc);
 | 
|---|
| 160 | }
 | 
|---|
| 161 | 
 | 
|---|
| 162 | static void
 | 
|---|
| 163 | HOTKEY_Paint(HOTKEY_INFO *infoPtr, HDC hdc)
 | 
|---|
| 164 | {
 | 
|---|
| 165 |     if (hdc)
 | 
|---|
| 166 |         HOTKEY_Refresh(infoPtr, hdc);
 | 
|---|
| 167 |     else {
 | 
|---|
| 168 |         PAINTSTRUCT ps;
 | 
|---|
| 169 |         hdc = BeginPaint (infoPtr->hwndSelf, &ps);
 | 
|---|
| 170 |         HOTKEY_Refresh (infoPtr, hdc);
 | 
|---|
| 171 |         EndPaint (infoPtr->hwndSelf, &ps);
 | 
|---|
| 172 |     }
 | 
|---|
| 173 | }
 | 
|---|
| 174 | 
 | 
|---|
| 175 | static LRESULT
 | 
|---|
| 176 | HOTKEY_GetHotKey(HOTKEY_INFO *infoPtr)
 | 
|---|
| 177 | {
 | 
|---|
| 178 |     TRACE("(infoPtr=%p) Modifiers: 0x%x, Virtual Key: %d\n", infoPtr, 
 | 
|---|
| 179 |           HIBYTE(infoPtr->HotKey), LOBYTE(infoPtr->HotKey));
 | 
|---|
| 180 |     return (LRESULT)infoPtr->HotKey;
 | 
|---|
| 181 | }
 | 
|---|
| 182 | 
 | 
|---|
| 183 | static void
 | 
|---|
| 184 | HOTKEY_SetHotKey(HOTKEY_INFO *infoPtr, WPARAM wParam)
 | 
|---|
| 185 | {
 | 
|---|
| 186 |     infoPtr->HotKey = (WORD)wParam;
 | 
|---|
| 187 |     infoPtr->ScanCode = 
 | 
|---|
| 188 |         MAKELPARAM(0, MapVirtualKeyW(LOBYTE(infoPtr->HotKey), 0));
 | 
|---|
| 189 |     TRACE("(infoPtr=%p wParam=%x) Modifiers: 0x%x, Virtual Key: %d\n", infoPtr,
 | 
|---|
| 190 |           wParam, HIBYTE(infoPtr->HotKey), LOBYTE(infoPtr->HotKey));
 | 
|---|
| 191 |     InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
 | 
|---|
| 192 | }
 | 
|---|
| 193 | 
 | 
|---|
| 194 | static void 
 | 
|---|
| 195 | HOTKEY_SetRules(HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 196 | {
 | 
|---|
| 197 |     infoPtr->InvComb = (WORD)wParam;
 | 
|---|
| 198 |     infoPtr->InvMod = (WORD)lParam;
 | 
|---|
| 199 |     TRACE("(infoPtr=%p) Invalid Modifers: 0x%x, If Invalid: 0x%x\n", infoPtr,
 | 
|---|
| 200 |           infoPtr->InvComb, infoPtr->InvMod);
 | 
|---|
| 201 | }
 | 
|---|
| 202 | 
 | 
|---|
| 203 | /* << HOTKEY_Char >> */
 | 
|---|
| 204 | 
 | 
|---|
| 205 | static LRESULT
 | 
|---|
| 206 | HOTKEY_Create (HWND hwnd, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 207 | {
 | 
|---|
| 208 |     HOTKEY_INFO *infoPtr;
 | 
|---|
| 209 |     TEXTMETRICW tm;
 | 
|---|
| 210 |     HDC hdc;
 | 
|---|
| 211 | 
 | 
|---|
| 212 |     /* allocate memory for info structure */
 | 
|---|
| 213 |     infoPtr = (HOTKEY_INFO *)COMCTL32_Alloc (sizeof(HOTKEY_INFO));
 | 
|---|
| 214 |     SetWindowLongW (hwnd, 0, (DWORD)infoPtr);
 | 
|---|
| 215 | 
 | 
|---|
| 216 |     /* initialize info structure */
 | 
|---|
| 217 |     infoPtr->HotKey = infoPtr->InvComb = infoPtr->InvMod = infoPtr->CurrMod = 0;
 | 
|---|
| 218 |     infoPtr->CaretPos = 2;
 | 
|---|
| 219 |     infoPtr->hwndSelf = hwnd;
 | 
|---|
| 220 |     LoadStringW(COMCTL32_hModule, HKY_NONE, infoPtr->strNone, 15);
 | 
|---|
| 221 | 
 | 
|---|
| 222 |     /* get default font height */
 | 
|---|
| 223 |     hdc = GetDC (hwnd);
 | 
|---|
| 224 |     GetTextMetricsW (hdc, &tm);
 | 
|---|
| 225 |     infoPtr->nHeight = tm.tmHeight;
 | 
|---|
| 226 |     ReleaseDC (hwnd, hdc);
 | 
|---|
| 227 | 
 | 
|---|
| 228 |     return 0;
 | 
|---|
| 229 | }
 | 
|---|
| 230 | 
 | 
|---|
| 231 | 
 | 
|---|
| 232 | static LRESULT
 | 
|---|
| 233 | HOTKEY_Destroy (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 234 | {
 | 
|---|
| 235 |     HWND hwnd = infoPtr->hwndSelf;
 | 
|---|
| 236 |     /* free hotkey info data */
 | 
|---|
| 237 |     COMCTL32_Free (infoPtr);
 | 
|---|
| 238 |     SetWindowLongW (hwnd, 0, 0);
 | 
|---|
| 239 |     return 0;
 | 
|---|
| 240 | }
 | 
|---|
| 241 | 
 | 
|---|
| 242 | 
 | 
|---|
| 243 | static LRESULT
 | 
|---|
| 244 | HOTKEY_EraseBackground (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 245 | {
 | 
|---|
| 246 |     HBRUSH hBrush;
 | 
|---|
| 247 |     RECT   rc;
 | 
|---|
| 248 | 
 | 
|---|
| 249 |     hBrush =
 | 
|---|
| 250 |         (HBRUSH)SendMessageW (GetParent (infoPtr->hwndSelf), WM_CTLCOLOREDIT,
 | 
|---|
| 251 |                                 wParam, (LPARAM)infoPtr->hwndSelf);
 | 
|---|
| 252 |     if (hBrush)
 | 
|---|
| 253 |         hBrush = (HBRUSH)GetStockObject (WHITE_BRUSH);
 | 
|---|
| 254 |     GetClientRect (infoPtr->hwndSelf, &rc);
 | 
|---|
| 255 | 
 | 
|---|
| 256 |     FillRect ((HDC)wParam, &rc, hBrush);
 | 
|---|
| 257 | 
 | 
|---|
| 258 |     return -1;
 | 
|---|
| 259 | }
 | 
|---|
| 260 | 
 | 
|---|
| 261 | 
 | 
|---|
| 262 | inline static LRESULT
 | 
|---|
| 263 | HOTKEY_GetFont (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 264 | {
 | 
|---|
| 265 |     return (LRESULT)infoPtr->hFont;
 | 
|---|
| 266 | }
 | 
|---|
| 267 | 
 | 
|---|
| 268 | static LRESULT
 | 
|---|
| 269 | HOTKEY_KeyDown (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 270 | {
 | 
|---|
| 271 |     TRACE("() Key: %d\n", wParam);
 | 
|---|
| 272 |     /* If any key is Pressed, we have to reset the hotkey in the control */
 | 
|---|
| 273 |     infoPtr->HotKey = 0;
 | 
|---|
| 274 | 
 | 
|---|
| 275 |     switch (wParam) {
 | 
|---|
| 276 |         case VK_RETURN:
 | 
|---|
| 277 |         case VK_TAB:
 | 
|---|
| 278 |         case VK_SPACE:
 | 
|---|
| 279 |         case VK_DELETE:
 | 
|---|
| 280 |         case VK_ESCAPE:
 | 
|---|
| 281 |         case VK_BACK:
 | 
|---|
| 282 |             InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
 | 
|---|
| 283 |             return DefWindowProcW (infoPtr->hwndSelf, WM_KEYDOWN, wParam, 
 | 
|---|
| 284 |                                    lParam);
 | 
|---|
| 285 | 
 | 
|---|
| 286 |         case VK_SHIFT:
 | 
|---|
| 287 |             infoPtr->CurrMod |= HOTKEYF_SHIFT;
 | 
|---|
| 288 |             break;
 | 
|---|
| 289 |         case VK_CONTROL:
 | 
|---|
| 290 |             infoPtr->CurrMod |= HOTKEYF_CONTROL;
 | 
|---|
| 291 |             break;
 | 
|---|
| 292 |         case VK_MENU:
 | 
|---|
| 293 |             infoPtr->CurrMod |= HOTKEYF_ALT;
 | 
|---|
| 294 |             break;
 | 
|---|
| 295 | 
 | 
|---|
| 296 |         default:
 | 
|---|
| 297 |             if(HOTKEY_IsCombInv(infoPtr))
 | 
|---|
| 298 |                 infoPtr->HotKey = MAKEWORD(wParam, infoPtr->InvMod);
 | 
|---|
| 299 |             else
 | 
|---|
| 300 |                 infoPtr->HotKey = MAKEWORD(wParam, infoPtr->CurrMod);
 | 
|---|
| 301 |             infoPtr->ScanCode = lParam;
 | 
|---|
| 302 |             break;
 | 
|---|
| 303 |     }
 | 
|---|
| 304 | 
 | 
|---|
| 305 |     InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
 | 
|---|
| 306 |     return 0;
 | 
|---|
| 307 | }
 | 
|---|
| 308 | 
 | 
|---|
| 309 | 
 | 
|---|
| 310 | static LRESULT
 | 
|---|
| 311 | HOTKEY_KeyUp (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 312 | {
 | 
|---|
| 313 |     TRACE("() Key: %d\n", wParam);
 | 
|---|
| 314 |     switch (wParam) {
 | 
|---|
| 315 |         case VK_SHIFT:
 | 
|---|
| 316 |             infoPtr->CurrMod &= ~HOTKEYF_SHIFT;
 | 
|---|
| 317 |             break;
 | 
|---|
| 318 |         case VK_CONTROL:
 | 
|---|
| 319 |             infoPtr->CurrMod &= ~HOTKEYF_CONTROL;
 | 
|---|
| 320 |             break;
 | 
|---|
| 321 |         case VK_MENU:
 | 
|---|
| 322 |             infoPtr->CurrMod &= ~HOTKEYF_ALT;
 | 
|---|
| 323 |             break;
 | 
|---|
| 324 |         default:
 | 
|---|
| 325 |             return 1;
 | 
|---|
| 326 |     }
 | 
|---|
| 327 | 
 | 
|---|
| 328 |     InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
 | 
|---|
| 329 | 
 | 
|---|
| 330 |     return 0;
 | 
|---|
| 331 | }
 | 
|---|
| 332 | 
 | 
|---|
| 333 | 
 | 
|---|
| 334 | static LRESULT
 | 
|---|
| 335 | HOTKEY_KillFocus (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 336 | {
 | 
|---|
| 337 |     infoPtr->bFocus = FALSE;
 | 
|---|
| 338 |     DestroyCaret ();
 | 
|---|
| 339 | 
 | 
|---|
| 340 |     return 0;
 | 
|---|
| 341 | }
 | 
|---|
| 342 | 
 | 
|---|
| 343 | 
 | 
|---|
| 344 | static LRESULT
 | 
|---|
| 345 | HOTKEY_LButtonDown (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 346 | {
 | 
|---|
| 347 |     SetFocus (infoPtr->hwndSelf);
 | 
|---|
| 348 | 
 | 
|---|
| 349 |     return 0;
 | 
|---|
| 350 | }
 | 
|---|
| 351 | 
 | 
|---|
| 352 | 
 | 
|---|
| 353 | inline static LRESULT
 | 
|---|
| 354 | HOTKEY_NCCreate (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 355 | {
 | 
|---|
| 356 |     DWORD dwExStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_EXSTYLE);
 | 
|---|
| 357 |     SetWindowLongW (infoPtr->hwndSelf, GWL_EXSTYLE, 
 | 
|---|
| 358 |                     dwExStyle | WS_EX_CLIENTEDGE);
 | 
|---|
| 359 |     return DefWindowProcW (infoPtr->hwndSelf, WM_NCCREATE, wParam, lParam);
 | 
|---|
| 360 | }
 | 
|---|
| 361 | 
 | 
|---|
| 362 | static LRESULT
 | 
|---|
| 363 | HOTKEY_SetFocus (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 364 | {
 | 
|---|
| 365 |     infoPtr->bFocus = TRUE;
 | 
|---|
| 366 | 
 | 
|---|
| 367 | 
 | 
|---|
| 368 |     CreateCaret (infoPtr->hwndSelf, NULL, 1, infoPtr->nHeight - 2);
 | 
|---|
| 369 | 
 | 
|---|
| 370 |     SetCaretPos (infoPtr->CaretPos, 3);
 | 
|---|
| 371 | 
 | 
|---|
| 372 |     ShowCaret (infoPtr->hwndSelf);
 | 
|---|
| 373 | 
 | 
|---|
| 374 | 
 | 
|---|
| 375 |     return 0;
 | 
|---|
| 376 | }
 | 
|---|
| 377 | 
 | 
|---|
| 378 | 
 | 
|---|
| 379 | inline static LRESULT
 | 
|---|
| 380 | HOTKEY_SetFont (HOTKEY_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 381 | {
 | 
|---|
| 382 |     TEXTMETRICW tm;
 | 
|---|
| 383 |     HDC hdc;
 | 
|---|
| 384 |     HFONT hOldFont = 0;
 | 
|---|
| 385 | 
 | 
|---|
| 386 |     infoPtr->hFont = (HFONT)wParam;
 | 
|---|
| 387 | 
 | 
|---|
| 388 |     hdc = GetDC (infoPtr->hwndSelf);
 | 
|---|
| 389 |     if (infoPtr->hFont)
 | 
|---|
| 390 |         hOldFont = SelectObject (hdc, infoPtr->hFont);
 | 
|---|
| 391 | 
 | 
|---|
| 392 |     GetTextMetricsW (hdc, &tm);
 | 
|---|
| 393 |     infoPtr->nHeight = tm.tmHeight;
 | 
|---|
| 394 | 
 | 
|---|
| 395 |     if (infoPtr->hFont)
 | 
|---|
| 396 |         SelectObject (hdc, hOldFont);
 | 
|---|
| 397 |     ReleaseDC (infoPtr->hwndSelf, hdc);
 | 
|---|
| 398 | 
 | 
|---|
| 399 |     if (LOWORD(lParam))
 | 
|---|
| 400 |         InvalidateRect (infoPtr->hwndSelf, NULL, TRUE);
 | 
|---|
| 401 | 
 | 
|---|
| 402 |     return 0;
 | 
|---|
| 403 | }
 | 
|---|
| 404 | 
 | 
|---|
| 405 | static LRESULT WINAPI
 | 
|---|
| 406 | HOTKEY_WindowProc (HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
 | 
|---|
| 407 | {
 | 
|---|
| 408 |     HOTKEY_INFO *infoPtr = HOTKEY_GetInfoPtr (hwnd);
 | 
|---|
| 409 |     TRACE("hwnd=%p msg=%x wparam=%x lparam=%lx\n", hwnd, uMsg, wParam, lParam);
 | 
|---|
| 410 |     if (!infoPtr && (uMsg != WM_CREATE))
 | 
|---|
| 411 |         return DefWindowProcW (hwnd, uMsg, wParam, lParam);
 | 
|---|
| 412 |     switch (uMsg)
 | 
|---|
| 413 |     {
 | 
|---|
| 414 |         case HKM_GETHOTKEY:
 | 
|---|
| 415 |             return HOTKEY_GetHotKey (infoPtr);
 | 
|---|
| 416 |         case HKM_SETHOTKEY:
 | 
|---|
| 417 |             HOTKEY_SetHotKey (infoPtr, wParam);
 | 
|---|
| 418 |             break;
 | 
|---|
| 419 |         case HKM_SETRULES:
 | 
|---|
| 420 |             HOTKEY_SetRules (infoPtr, wParam, lParam);
 | 
|---|
| 421 |             break;
 | 
|---|
| 422 | 
 | 
|---|
| 423 | /*      case WM_CHAR: */
 | 
|---|
| 424 | 
 | 
|---|
| 425 |         case WM_CREATE:
 | 
|---|
| 426 |             return HOTKEY_Create (hwnd, wParam, lParam);
 | 
|---|
| 427 | 
 | 
|---|
| 428 |         case WM_DESTROY:
 | 
|---|
| 429 |             return HOTKEY_Destroy (infoPtr, wParam, lParam);
 | 
|---|
| 430 | 
 | 
|---|
| 431 |         case WM_ERASEBKGND:
 | 
|---|
| 432 |             return HOTKEY_EraseBackground (infoPtr, wParam, lParam);
 | 
|---|
| 433 | 
 | 
|---|
| 434 |         case WM_GETDLGCODE:
 | 
|---|
| 435 |             return DLGC_WANTCHARS | DLGC_WANTARROWS;
 | 
|---|
| 436 | 
 | 
|---|
| 437 |         case WM_GETFONT:
 | 
|---|
| 438 |             return HOTKEY_GetFont (infoPtr, wParam, lParam);
 | 
|---|
| 439 | 
 | 
|---|
| 440 |         case WM_KEYDOWN:
 | 
|---|
| 441 |         case WM_SYSKEYDOWN:
 | 
|---|
| 442 |             return HOTKEY_KeyDown (infoPtr, wParam, lParam);
 | 
|---|
| 443 | 
 | 
|---|
| 444 |         case WM_KEYUP:
 | 
|---|
| 445 |         case WM_SYSKEYUP:
 | 
|---|
| 446 |             return HOTKEY_KeyUp (infoPtr, wParam, lParam);
 | 
|---|
| 447 | 
 | 
|---|
| 448 |         case WM_KILLFOCUS:
 | 
|---|
| 449 |             return HOTKEY_KillFocus (infoPtr, wParam, lParam);
 | 
|---|
| 450 | 
 | 
|---|
| 451 |         case WM_LBUTTONDOWN:
 | 
|---|
| 452 |             return HOTKEY_LButtonDown (infoPtr, wParam, lParam);
 | 
|---|
| 453 | 
 | 
|---|
| 454 |         case WM_NCCREATE:
 | 
|---|
| 455 |             return HOTKEY_NCCreate (infoPtr, wParam, lParam);
 | 
|---|
| 456 | 
 | 
|---|
| 457 |         case WM_PAINT:
 | 
|---|
| 458 |             HOTKEY_Paint(infoPtr, (HDC)wParam);
 | 
|---|
| 459 |             return 0;
 | 
|---|
| 460 | 
 | 
|---|
| 461 |         case WM_SETFOCUS:
 | 
|---|
| 462 |             return HOTKEY_SetFocus (infoPtr, wParam, lParam);
 | 
|---|
| 463 | 
 | 
|---|
| 464 |         case WM_SETFONT:
 | 
|---|
| 465 |             return HOTKEY_SetFont (infoPtr, wParam, lParam);
 | 
|---|
| 466 | 
 | 
|---|
| 467 | /*      case WM_SYSCHAR: */
 | 
|---|
| 468 | 
 | 
|---|
| 469 |         default:
 | 
|---|
| 470 |             if ((uMsg >= WM_USER) && (uMsg < WM_APP))
 | 
|---|
| 471 |                 ERR("unknown msg %04x wp=%08x lp=%08lx\n",
 | 
|---|
| 472 |                      uMsg, wParam, lParam);
 | 
|---|
| 473 |             return DefWindowProcW (hwnd, uMsg, wParam, lParam);
 | 
|---|
| 474 |     }
 | 
|---|
| 475 |     return 0;
 | 
|---|
| 476 | }
 | 
|---|
| 477 | 
 | 
|---|
| 478 | 
 | 
|---|
| 479 | void
 | 
|---|
| 480 | HOTKEY_Register (void)
 | 
|---|
| 481 | {
 | 
|---|
| 482 |     WNDCLASSW wndClass;
 | 
|---|
| 483 | 
 | 
|---|
| 484 |     ZeroMemory (&wndClass, sizeof(WNDCLASSW));
 | 
|---|
| 485 |     wndClass.style         = CS_GLOBALCLASS;
 | 
|---|
| 486 |     wndClass.lpfnWndProc   = (WNDPROC)HOTKEY_WindowProc;
 | 
|---|
| 487 |     wndClass.cbClsExtra    = 0;
 | 
|---|
| 488 |     wndClass.cbWndExtra    = sizeof(HOTKEY_INFO *);
 | 
|---|
| 489 |     wndClass.hCursor       = 0;
 | 
|---|
| 490 |     wndClass.hbrBackground = 0;
 | 
|---|
| 491 |     wndClass.lpszClassName = HOTKEY_CLASSW;
 | 
|---|
| 492 | 
 | 
|---|
| 493 |     RegisterClassW (&wndClass);
 | 
|---|
| 494 | }
 | 
|---|
| 495 | 
 | 
|---|
| 496 | 
 | 
|---|
| 497 | void
 | 
|---|
| 498 | HOTKEY_Unregister (void)
 | 
|---|
| 499 | {
 | 
|---|
| 500 |     UnregisterClassW (HOTKEY_CLASSW, NULL);
 | 
|---|
| 501 | }
 | 
|---|