user32: Fix input tests failing on WinXP+.
[wine] / dlls / user32 / tests / input.c
1 /* Test Key event to Key message translation
2  *
3  * Copyright 2003 Rein Klazes
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  */
19
20 /* test whether the right type of messages:
21  * WM_KEYUP/DOWN vs WM_SYSKEYUP/DOWN  are sent in case of combined
22  * keystrokes.
23  *
24  * For instance <ALT>-X can be accomplished by
25  * the sequence ALT-KEY-DOWN, X-KEY-DOWN, ALT-KEY-UP, X-KEY-UP
26  * but also X-KEY-DOWN, ALT-KEY-DOWN, X-KEY-UP, ALT-KEY-UP
27  * Whether a KEY or a SYSKEY message is sent is not always clear, it is
28  * also not the same in WINNT as in WIN9X */
29
30 /* NOTE that there will be test failures under WIN9X
31  * No applications are known to me that rely on this
32  * so I don't fix it */
33
34 /* TODO:
35  * 1. extend it to the wm_command and wm_syscommand notifications
36  * 2. add some more tests with special cases like dead keys or right (alt) key
37  * 3. there is some adapted code from input.c in here. Should really
38  *    make that code exactly the same.
39  * 4. resolve the win9x case when there is a need or the testing frame work
40  *    offers a nice way.
41  * 5. The test app creates a window, the user should not take the focus
42  *    away during its short existence. I could do something to prevent that
43  *    if it is a problem.
44  *
45  */
46
47 #define _WIN32_WINNT 0x401
48
49 #include <stdarg.h>
50 #include <assert.h>
51
52 #include "windef.h"
53 #include "winbase.h"
54 #include "winuser.h"
55
56 #include "wine/test.h"
57
58 /* globals */
59 static HWND hWndTest;
60 static long timetag = 0x10000000;
61
62 static UINT (WINAPI *pSendInput) (UINT, INPUT*, size_t);
63 static int (WINAPI *pGetMouseMovePointsEx) (UINT, LPMOUSEMOVEPOINT, LPMOUSEMOVEPOINT, int, DWORD);
64
65 #define MAXKEYEVENTS 6
66 #define MAXKEYMESSAGES MAXKEYEVENTS /* assuming a key event generates one
67                                        and only one message */
68
69 /* keyboard message names, sorted as their value */
70 static const char *MSGNAME[]={"WM_KEYDOWN", "WM_KEYUP", "WM_CHAR","WM_DEADCHAR",
71     "WM_SYSKEYDOWN", "WM_SYSKEYUP", "WM_SYSCHAR", "WM_SYSDEADCHAR" ,"WM_KEYLAST"};
72
73 /* keyevents, add more as needed */
74 typedef enum KEVtag
75 {  ALTDOWN = 1, ALTUP, XDOWN, XUP, SHIFTDOWN, SHIFTUP, CTRLDOWN, CTRLUP } KEV;
76 /* matching VK's */
77 static const int GETVKEY[]={0, VK_MENU, VK_MENU, 'X', 'X', VK_SHIFT, VK_SHIFT, VK_CONTROL, VK_CONTROL};
78 /* matching scan codes */
79 static const int GETSCAN[]={0, 0x38, 0x38, 0x2D, 0x2D, 0x2A, 0x2A, 0x1D, 0x1D };
80 /* matching updown events */
81 static const int GETFLAGS[]={0, 0, KEYEVENTF_KEYUP, 0, KEYEVENTF_KEYUP, 0, KEYEVENTF_KEYUP, 0, KEYEVENTF_KEYUP};
82 /* matching descriptions */
83 static const char *getdesc[]={"", "+alt","-alt","+X","-X","+shift","-shift","+ctrl","-ctrl"};
84
85 /* The MSVC headers ignore our NONAMELESSUNION requests so we have to define our own type */
86 typedef struct
87 {
88     DWORD type;
89     union
90     {
91         MOUSEINPUT      mi;
92         KEYBDINPUT      ki;
93         HARDWAREINPUT   hi;
94     } u;
95 } TEST_INPUT;
96
97 #define ADDTOINPUTS(kev) \
98 inputs[evtctr].type = INPUT_KEYBOARD; \
99     ((TEST_INPUT*)inputs)[evtctr].u.ki.wVk = GETVKEY[ kev]; \
100     ((TEST_INPUT*)inputs)[evtctr].u.ki.wScan = GETSCAN[ kev]; \
101     ((TEST_INPUT*)inputs)[evtctr].u.ki.dwFlags = GETFLAGS[ kev]; \
102     ((TEST_INPUT*)inputs)[evtctr].u.ki.dwExtraInfo = 0; \
103     ((TEST_INPUT*)inputs)[evtctr].u.ki.time = ++timetag; \
104     if( kev) evtctr++;
105
106 typedef struct {
107     UINT    message;
108     WPARAM  wParam;
109     LPARAM  lParam;
110 } KMSG;
111
112 /*******************************************
113  * add new test sets here
114  * the software will make all combinations of the
115  * keyevent defined here
116  */
117 static const struct {
118     int nrkev;
119     KEV keydwn[MAXKEYEVENTS];
120     KEV keyup[MAXKEYEVENTS];
121 } testkeyset[]= {
122     { 2, { ALTDOWN, XDOWN }, { ALTUP, XUP}},
123     { 3, { ALTDOWN, XDOWN , SHIFTDOWN}, { ALTUP, XUP, SHIFTUP}},
124     { 3, { ALTDOWN, XDOWN , CTRLDOWN}, { ALTUP, XUP, CTRLUP}},
125     { 3, { SHIFTDOWN, XDOWN , CTRLDOWN}, { SHIFTUP, XUP, CTRLUP}},
126     { 0 } /* mark the end */
127 };
128
129 /**********************adapted from input.c **********************************/
130
131 static BYTE InputKeyStateTable[256];
132 static BYTE AsyncKeyStateTable[256];
133 static BYTE TrackSysKey = 0; /* determine whether ALT key up will cause a WM_SYSKEYUP
134                          or a WM_KEYUP message */
135
136 static void init_function_pointers(void)
137 {
138     HMODULE hdll = GetModuleHandleA("user32");
139
140 #define GET_PROC(func) \
141     p ## func = (void*)GetProcAddress(hdll, #func); \
142     if(!p ## func) \
143       trace("GetProcAddress(%s) failed\n", #func);
144
145     GET_PROC(SendInput)
146     GET_PROC(GetMouseMovePointsEx)
147
148 #undef GET_PROC
149 }
150
151 static int KbdMessage( KEV kev, WPARAM *pwParam, LPARAM *plParam )
152 {
153     UINT message;
154     int VKey = GETVKEY[kev];
155     WORD flags;
156
157     flags = LOBYTE(GETSCAN[kev]);
158     if (GETFLAGS[kev] & KEYEVENTF_EXTENDEDKEY) flags |= KF_EXTENDED;
159
160     if (GETFLAGS[kev] & KEYEVENTF_KEYUP )
161     {
162         message = WM_KEYUP;
163         if( (InputKeyStateTable[VK_MENU] & 0x80) && (
164                 (VKey == VK_MENU) || (VKey == VK_CONTROL) ||
165                  !(InputKeyStateTable[VK_CONTROL] & 0x80))) {
166             if(  TrackSysKey == VK_MENU || /* <ALT>-down/<ALT>-up sequence */
167                     (VKey != VK_MENU)) /* <ALT>-down...<something else>-up */
168                 message = WM_SYSKEYUP;
169                 TrackSysKey = 0;
170         }
171         InputKeyStateTable[VKey] &= ~0x80;
172         flags |= KF_REPEAT | KF_UP;
173     }
174     else
175     {
176         if (InputKeyStateTable[VKey] & 0x80) flags |= KF_REPEAT;
177         if (!(InputKeyStateTable[VKey] & 0x80)) InputKeyStateTable[VKey] ^= 0x01;
178         InputKeyStateTable[VKey] |= 0x80;
179         AsyncKeyStateTable[VKey] |= 0x80;
180
181         message = WM_KEYDOWN;
182         if( (InputKeyStateTable[VK_MENU] & 0x80) &&
183                 !(InputKeyStateTable[VK_CONTROL] & 0x80)) {
184             message = WM_SYSKEYDOWN;
185             TrackSysKey = VKey;
186         }
187     }
188
189     if (InputKeyStateTable[VK_MENU] & 0x80) flags |= KF_ALTDOWN;
190
191     if( plParam) *plParam = MAKELPARAM( 1, flags );
192     if( pwParam) *pwParam = VKey;
193     return message;
194 }
195
196 /****************************** end copy input.c ****************************/
197
198 /*
199  * . prepare the keyevents for SendInputs
200  * . calculate the "expected" messages
201  * . Send the events to our window
202  * . retrieve the messages from the input queue
203  * . verify
204  */
205 static void do_test( HWND hwnd, int seqnr, const KEV td[] )
206 {
207     INPUT inputs[MAXKEYEVENTS];
208     KMSG expmsg[MAXKEYEVENTS];
209     MSG msg;
210     char buf[100];
211     UINT evtctr=0;
212     int kmctr, i;
213
214     buf[0]='\0';
215     TrackSysKey=0; /* see input.c */
216     for( i = 0; i < MAXKEYEVENTS; i++) {
217         ADDTOINPUTS(td[i])
218         strcat(buf, getdesc[td[i]]);
219         if(td[i])
220             expmsg[i].message = KbdMessage(td[i], &(expmsg[i].wParam), &(expmsg[i].lParam)); /* see queue_kbd_event() */
221         else
222             expmsg[i].message = 0;
223     }
224     for( kmctr = 0; kmctr < MAXKEYEVENTS && expmsg[kmctr].message; kmctr++)
225         ;
226     assert( evtctr <= MAXKEYEVENTS );
227     assert( evtctr == pSendInput(evtctr, &inputs[0], sizeof(INPUT)));
228     i = 0;
229     if (winetest_debug > 1)
230         trace("======== key stroke sequence #%d: %s =============\n",
231             seqnr + 1, buf);
232     while( PeekMessage(&msg,hwnd,WM_KEYFIRST,WM_KEYLAST,PM_REMOVE) ) {
233         if (winetest_debug > 1)
234             trace("message[%d] %-15s wParam %04lx lParam %08lx time %x\n", i,
235                   MSGNAME[msg.message - WM_KEYFIRST], msg.wParam, msg.lParam, msg.time);
236         if( i < kmctr ) {
237             ok( msg.message == expmsg[i].message &&
238                     msg.wParam == expmsg[i].wParam &&
239                     msg.lParam == expmsg[i].lParam,
240                     "wrong message! expected:\n"
241                     "message[%d] %-15s wParam %04lx lParam %08lx\n",i,
242                     MSGNAME[(expmsg[i]).message - WM_KEYFIRST],
243                     expmsg[i].wParam, expmsg[i].lParam );
244         }
245         i++;
246     }
247     if (winetest_debug > 1)
248         trace("%d messages retrieved\n", i);
249     ok( i == kmctr, "message count is wrong: got %d expected: %d\n", i, kmctr);
250 }
251
252 /* test all combinations of the specified key events */
253 static void TestASet( HWND hWnd, int nrkev, const KEV kevdwn[], const KEV kevup[] )
254 {
255     int i,j,k,l,m,n;
256     static int count=0;
257     KEV kbuf[MAXKEYEVENTS];
258     assert( nrkev==2 || nrkev==3);
259     for(i=0;i<MAXKEYEVENTS;i++) kbuf[i]=0;
260     /* two keys involved gives 4 test cases */
261     if(nrkev==2) {
262         for(i=0;i<nrkev;i++) {
263             for(j=0;j<nrkev;j++) {
264                 kbuf[0] = kevdwn[i];
265                 kbuf[1] = kevdwn[1-i];
266                 kbuf[2] = kevup[j];
267                 kbuf[3] = kevup[1-j];
268                 do_test( hWnd, count++, kbuf);
269             }
270         }
271     }
272     /* three keys involved gives 36 test cases */
273     if(nrkev==3){
274         for(i=0;i<nrkev;i++){
275             for(j=0;j<nrkev;j++){
276                 if(j==i) continue;
277                 for(k=0;k<nrkev;k++){
278                     if(k==i || k==j) continue;
279                     for(l=0;l<nrkev;l++){
280                         for(m=0;m<nrkev;m++){
281                             if(m==l) continue;
282                             for(n=0;n<nrkev;n++){
283                                 if(n==l ||n==m) continue;
284                                 kbuf[0] = kevdwn[i];
285                                 kbuf[1] = kevdwn[j];
286                                 kbuf[2] = kevdwn[k];
287                                 kbuf[3] = kevup[l];
288                                 kbuf[4] = kevup[m];
289                                 kbuf[5] = kevup[n];
290                                 do_test( hWnd, count++, kbuf);
291                             }
292                         }
293                     }
294                 }
295             }
296         }
297     }
298 }
299
300 /* test each set specified in the global testkeyset array */
301 static void TestSysKeys( HWND hWnd)
302 {
303     int i;
304     for(i=0; testkeyset[i].nrkev;i++)
305         TestASet( hWnd, testkeyset[i].nrkev, testkeyset[i].keydwn,
306                 testkeyset[i].keyup);
307 }
308
309 static LRESULT CALLBACK WndProc( HWND hWnd, UINT msg, WPARAM wParam,
310         LPARAM lParam )
311 {
312     switch (msg) {
313         case WM_USER:
314             SetFocus(hWnd);
315             /* window has focus, now do the test */
316             if( hWnd == hWndTest) TestSysKeys( hWnd);
317             /* finished :-) */
318             break;
319
320         case WM_DESTROY:
321             PostQuitMessage( 0 );
322             break;
323
324         default:
325             return( DefWindowProcA( hWnd, msg, wParam, lParam ) );
326     }
327
328     return 0;
329 }
330
331 static void test_Input_whitebox(void)
332 {
333     MSG msg;
334     WNDCLASSA  wclass;
335     HANDLE hInstance = GetModuleHandleA( NULL );
336
337     wclass.lpszClassName = "InputSysKeyTestClass";
338     wclass.style         = CS_HREDRAW | CS_VREDRAW;
339     wclass.lpfnWndProc   = WndProc;
340     wclass.hInstance     = hInstance;
341     wclass.hIcon         = LoadIconA( 0, (LPSTR)IDI_APPLICATION );
342     wclass.hCursor       = LoadCursorA( NULL, IDC_ARROW);
343     wclass.hbrBackground = (HBRUSH)( COLOR_WINDOW + 1);
344     wclass.lpszMenuName = 0;
345     wclass.cbClsExtra    = 0;
346     wclass.cbWndExtra    = 0;
347     assert (RegisterClassA( &wclass ));
348     /* create the test window that will receive the keystrokes */
349     assert ( hWndTest = CreateWindowA( wclass.lpszClassName, "InputSysKeyTest",
350                 WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, 100, 100,
351                 NULL, NULL, hInstance, NULL) );
352     ShowWindow( hWndTest, SW_SHOW);
353     UpdateWindow( hWndTest);
354
355     /* flush pending messages */
356     while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessageA( &msg );
357
358     SendMessageA(hWndTest, WM_USER, 0, 0);
359     DestroyWindow(hWndTest);
360 }
361
362 static void empty_message_queue(void) {
363     MSG msg;
364     while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
365         TranslateMessage(&msg);
366         DispatchMessage(&msg);
367     }
368 }
369
370 struct transition_s {
371     WORD wVk;
372     BYTE before_state;
373 };
374
375 typedef enum {
376     sent=0x1,
377     posted=0x2,
378     parent=0x4,
379     wparam=0x8,
380     lparam=0x10,
381     defwinproc=0x20,
382     beginpaint=0x40,
383     optional=0x80,
384     hook=0x100,
385     winevent_hook=0x200
386 } msg_flags_t;
387
388 struct message {
389     UINT message;          /* the WM_* code */
390     msg_flags_t flags;     /* message props */
391     WPARAM wParam;         /* expected value of wParam */
392     LPARAM lParam;         /* expected value of lParam */
393 };
394
395 struct sendinput_test_s {
396     WORD wVk;
397     DWORD dwFlags;
398     BOOL _todo_wine;
399     struct transition_s expected_transitions[MAXKEYEVENTS+1];
400     struct message expected_messages[MAXKEYMESSAGES+1];
401 } sendinput_test[] = {
402     /* test ALT+F */
403     {VK_LMENU, 0, 0, {{VK_MENU, 0x00}, {VK_LMENU, 0x00}, {0}},
404         {{WM_SYSKEYDOWN, hook|wparam, VK_LMENU}, {WM_SYSKEYDOWN}, {0}}},
405     {'F', 0, 0, {{'F', 0x00}, {0}},
406         {{WM_SYSKEYDOWN, hook}, {WM_SYSKEYDOWN},
407         {WM_SYSCHAR},
408         {WM_SYSCOMMAND}, {0}}},
409     {'F', KEYEVENTF_KEYUP, 0, {{'F', 0x80}, {0}},
410         {{WM_SYSKEYUP, hook}, {WM_SYSKEYUP}, {0}}},
411     {VK_LMENU, KEYEVENTF_KEYUP, 0, {{VK_MENU, 0x80}, {VK_LMENU, 0x80}, {0}},
412         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
413
414     /* test CTRL+O */
415     {VK_LCONTROL, 0, 0, {{VK_CONTROL, 0x00}, {VK_LCONTROL, 0x00}, {0}},
416         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {0}}},
417     {'O', 0, 0, {{'O', 0x00}, {0}},
418         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {WM_CHAR}, {0}}},
419     {'O', KEYEVENTF_KEYUP, 0, {{'O', 0x80}, {0}},
420         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
421     {VK_LCONTROL, KEYEVENTF_KEYUP, 0, {{VK_CONTROL, 0x80}, {VK_LCONTROL, 0x80}, {0}},
422         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
423
424     /* test ALT+CTRL+X */
425     {VK_LMENU, 0, 0, {{VK_MENU, 0x00}, {VK_LMENU, 0x00}, {0}},
426         {{WM_SYSKEYDOWN, hook}, {WM_SYSKEYDOWN}, {0}}},
427     {VK_LCONTROL, 0, 0, {{VK_CONTROL, 0x00}, {VK_LCONTROL, 0x00}, {0}},
428         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {0}}},
429     {'X', 0, 0, {{'X', 0x00}, {0}},
430         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {0}}},
431     {'X', KEYEVENTF_KEYUP, 0, {{'X', 0x80}, {0}},
432         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
433     {VK_LCONTROL, KEYEVENTF_KEYUP, 0, {{VK_CONTROL, 0x80}, {VK_LCONTROL, 0x80}, {0}},
434         {{WM_SYSKEYUP, hook}, {WM_SYSKEYUP}, {0}}},
435     {VK_LMENU, KEYEVENTF_KEYUP, 0, {{VK_MENU, 0x80}, {VK_LMENU, 0x80}, {0}},
436         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
437
438     /* test SHIFT+A */
439     {VK_LSHIFT, 0, 0, {{VK_SHIFT, 0x00}, {VK_LSHIFT, 0x00}, {0}},
440         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {0}}},
441     {'A', 0, 0, {{'A', 0x00}, {0}},
442         {{WM_KEYDOWN, hook}, {WM_KEYDOWN}, {WM_CHAR}, {0}}},
443     {'A', KEYEVENTF_KEYUP, 0, {{'A', 0x80}, {0}},
444         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
445     {VK_LSHIFT, KEYEVENTF_KEYUP, 0, {{VK_SHIFT, 0x80}, {VK_LSHIFT, 0x80}, {0}},
446         {{WM_KEYUP, hook}, {WM_KEYUP}, {0}}},
447     /* test L-SHIFT & R-SHIFT: */
448     /* RSHIFT == LSHIFT */
449     {VK_RSHIFT, 0, 0,
450         {{VK_SHIFT, 0x00}, {VK_LSHIFT, 0x00}, {0}},
451         {{WM_KEYDOWN, hook|wparam, VK_RSHIFT},
452         {WM_KEYDOWN}, {0}}},
453     {VK_RSHIFT, KEYEVENTF_KEYUP, 0,
454         {{VK_SHIFT, 0x80}, {VK_LSHIFT, 0x80}, {0}},
455         {{WM_KEYUP, hook, hook|wparam, VK_RSHIFT},
456         {WM_KEYUP}, {0}}},
457
458     /* LSHIFT | KEYEVENTF_EXTENDEDKEY == RSHIFT */
459     {VK_LSHIFT, KEYEVENTF_EXTENDEDKEY, 0,
460         {{VK_SHIFT, 0x00}, {VK_RSHIFT, 0x00}, {0}},
461         {{WM_KEYDOWN, hook|wparam|lparam, VK_LSHIFT, LLKHF_EXTENDED},
462         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
463     {VK_LSHIFT, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
464         {{VK_SHIFT, 0x80}, {VK_RSHIFT, 0x80}, {0}},
465         {{WM_KEYUP, hook|wparam|lparam, VK_LSHIFT, LLKHF_UP|LLKHF_EXTENDED},
466         {WM_KEYUP, wparam|lparam, VK_SHIFT, KF_UP}, {0}}},
467     /* RSHIFT | KEYEVENTF_EXTENDEDKEY == RSHIFT */
468     {VK_RSHIFT, KEYEVENTF_EXTENDEDKEY, 0,
469         {{VK_SHIFT, 0x00}, {VK_RSHIFT, 0x00}, {0}},
470         {{WM_KEYDOWN, hook|wparam|lparam, VK_RSHIFT, LLKHF_EXTENDED},
471         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
472     {VK_RSHIFT, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
473         {{VK_SHIFT, 0x80}, {VK_RSHIFT, 0x80}, {0}},
474         {{WM_KEYUP, hook|wparam|lparam, VK_RSHIFT, LLKHF_UP|LLKHF_EXTENDED},
475         {WM_KEYUP, wparam|lparam, VK_SHIFT, KF_UP}, {0}}},
476
477     /* Note about wparam for hook with generic key (VK_SHIFT, VK_CONTROL, VK_MENU):
478        win2k  - sends to hook whatever we generated here
479        winXP+ - Attempts to convert key to L/R key but not always correct
480     */
481     /* SHIFT == LSHIFT */
482     {VK_SHIFT, 0, 0,
483         {{VK_SHIFT, 0x00}, {VK_LSHIFT, 0x00}, {0}},
484         {{WM_KEYDOWN, hook/* |wparam */|lparam, VK_SHIFT, 0},
485         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
486     {VK_SHIFT, KEYEVENTF_KEYUP, 0,
487         {{VK_SHIFT, 0x80}, {VK_LSHIFT, 0x80}, {0}},
488         {{WM_KEYUP, hook/*|wparam*/|lparam, VK_SHIFT, LLKHF_UP},
489         {WM_KEYUP, wparam|lparam, VK_SHIFT, KF_UP}, {0}}},
490     /* SHIFT | KEYEVENTF_EXTENDEDKEY == RSHIFT */
491     {VK_SHIFT, KEYEVENTF_EXTENDEDKEY, 0,
492         {{VK_SHIFT, 0x00}, {VK_RSHIFT, 0x00}, {0}},
493         {{WM_KEYDOWN, hook/*|wparam*/|lparam, VK_SHIFT, LLKHF_EXTENDED},
494         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
495     {VK_SHIFT, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
496         {{VK_SHIFT, 0x80}, {VK_RSHIFT, 0x80}, {0}},
497         {{WM_KEYUP, hook/*|wparam*/|lparam, VK_SHIFT, LLKHF_UP|LLKHF_EXTENDED},
498         {WM_KEYUP, wparam|lparam, VK_SHIFT, KF_UP}, {0}}},
499
500     /* test L-CONTROL & R-CONTROL: */
501     /* RCONTROL == LCONTROL */
502     {VK_RCONTROL, 0, 0,
503         {{VK_CONTROL, 0x00}, {VK_LCONTROL, 0x00}, {0}},
504         {{WM_KEYDOWN, hook|wparam, VK_RCONTROL},
505         {WM_KEYDOWN, wparam|lparam, VK_CONTROL, 0}, {0}}},
506     {VK_RCONTROL, KEYEVENTF_KEYUP, 0,
507         {{VK_CONTROL, 0x80}, {VK_LCONTROL, 0x80}, {0}},
508         {{WM_KEYUP, hook|wparam, VK_RCONTROL},
509         {WM_KEYUP, wparam|lparam, VK_CONTROL, KF_UP}, {0}}},
510     /* LCONTROL | KEYEVENTF_EXTENDEDKEY == RCONTROL */
511     {VK_LCONTROL, KEYEVENTF_EXTENDEDKEY, 0,
512         {{VK_CONTROL, 0x00}, {VK_RCONTROL, 0x00}, {0}},
513         {{WM_KEYDOWN, hook|wparam|lparam, VK_LCONTROL, LLKHF_EXTENDED},
514         {WM_KEYDOWN, wparam|lparam, VK_CONTROL, KF_EXTENDED}, {0}}},
515     {VK_LCONTROL, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
516         {{VK_CONTROL, 0x80}, {VK_RCONTROL, 0x80}, {0}},
517         {{WM_KEYUP, hook|wparam|lparam, VK_LCONTROL, LLKHF_UP|LLKHF_EXTENDED},
518         {WM_KEYUP, wparam|lparam, VK_CONTROL, KF_UP|KF_EXTENDED}, {0}}},
519     /* RCONTROL | KEYEVENTF_EXTENDEDKEY == RCONTROL */
520     {VK_RCONTROL, KEYEVENTF_EXTENDEDKEY, 0,
521         {{VK_CONTROL, 0x00}, {VK_RCONTROL, 0x00}, {0}},
522         {{WM_KEYDOWN, hook|wparam|lparam, VK_RCONTROL, LLKHF_EXTENDED},
523         {WM_KEYDOWN, wparam|lparam, VK_CONTROL, KF_EXTENDED}, {0}}},
524     {VK_RCONTROL, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
525         {{VK_CONTROL, 0x80}, {VK_RCONTROL, 0x80}, {0}},
526         {{WM_KEYUP, hook|wparam|lparam, VK_RCONTROL, LLKHF_UP|LLKHF_EXTENDED},
527         {WM_KEYUP, wparam|lparam, VK_CONTROL, KF_UP|KF_EXTENDED}, {0}}},
528     /* CONTROL == LCONTROL */
529     {VK_CONTROL, 0, 0,
530         {{VK_CONTROL, 0x00}, {VK_LCONTROL, 0x00}, {0}},
531         {{WM_KEYDOWN, hook/*|wparam, VK_CONTROL*/},
532         {WM_KEYDOWN, wparam|lparam, VK_CONTROL, 0}, {0}}},
533     {VK_CONTROL, KEYEVENTF_KEYUP, 0,
534         {{VK_CONTROL, 0x80}, {VK_LCONTROL, 0x80}, {0}},
535         {{WM_KEYUP, hook/*|wparam, VK_CONTROL*/},
536         {WM_KEYUP, wparam|lparam, VK_CONTROL, KF_UP}, {0}}},
537     /* CONTROL | KEYEVENTF_EXTENDEDKEY == RCONTROL */
538     {VK_CONTROL, KEYEVENTF_EXTENDEDKEY, 0,
539         {{VK_CONTROL, 0x00}, {VK_RCONTROL, 0x00}, {0}},
540         {{WM_KEYDOWN, hook/*|wparam*/|lparam, VK_CONTROL, LLKHF_EXTENDED},
541         {WM_KEYDOWN, wparam|lparam, VK_CONTROL, KF_EXTENDED}, {0}}},
542     {VK_CONTROL, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
543         {{VK_CONTROL, 0x80}, {VK_RCONTROL, 0x80}, {0}},
544         {{WM_KEYUP, hook/*|wparam*/|lparam, VK_CONTROL, LLKHF_UP|LLKHF_EXTENDED},
545         {WM_KEYUP, wparam|lparam, VK_CONTROL, KF_UP|KF_EXTENDED}, {0}}},
546
547     /* test L-MENU & R-MENU: */
548     /* RMENU == LMENU */
549     {VK_RMENU, 0, 0,
550         {{VK_MENU, 0x00}, {VK_LMENU, 0x00}, {0}},
551         {{WM_SYSKEYDOWN, hook|wparam, VK_RMENU},
552         {WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0}, {0}}},
553     {VK_RMENU, KEYEVENTF_KEYUP, 1,
554         {{VK_MENU, 0x80}, {VK_LMENU, 0x80}, {0}},
555         {{WM_KEYUP, hook|wparam, VK_RMENU},
556         {WM_SYSKEYUP, wparam|lparam, VK_MENU, KF_UP},
557         {WM_SYSCOMMAND}, {0}}},
558     /* LMENU | KEYEVENTF_EXTENDEDKEY == RMENU */
559     {VK_LMENU, KEYEVENTF_EXTENDEDKEY, 0,
560         {{VK_MENU, 0x00}, {VK_RMENU, 0x00}, {0}},
561         {{WM_SYSKEYDOWN, hook|wparam|lparam, VK_LMENU, LLKHF_EXTENDED},
562         {WM_SYSKEYDOWN, wparam|lparam, VK_MENU, KF_EXTENDED}, {0}}},
563     {VK_LMENU, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 1,
564         {{VK_MENU, 0x80}, {VK_RMENU, 0x80}, {0}},
565         {{WM_KEYUP, hook|wparam|lparam, VK_LMENU, LLKHF_UP|LLKHF_EXTENDED},
566         {WM_SYSKEYUP, wparam|lparam, VK_MENU, KF_UP|KF_EXTENDED},
567         {WM_SYSCOMMAND}, {0}}},
568     /* RMENU | KEYEVENTF_EXTENDEDKEY == RMENU */
569     {VK_RMENU, KEYEVENTF_EXTENDEDKEY, 0,
570         {{VK_MENU, 0x00}, {VK_RMENU, 0x00}, {0}},
571         {{WM_SYSKEYDOWN, hook|wparam|lparam, VK_RMENU, LLKHF_EXTENDED},
572         {WM_SYSKEYDOWN, wparam|lparam, VK_MENU, KF_EXTENDED}, {0}}},
573     {VK_RMENU, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 1,
574         {{VK_MENU, 0x80}, {VK_RMENU, 0x80}, {0}},
575         {{WM_KEYUP, hook|wparam|lparam, VK_RMENU, LLKHF_UP|LLKHF_EXTENDED},
576         {WM_SYSKEYUP, wparam|lparam, VK_MENU, KF_UP|KF_EXTENDED},
577         {WM_SYSCOMMAND}, {0}}},
578     /* MENU == LMENU */
579     {VK_MENU, 0, 0,
580         {{VK_MENU, 0x00}, {VK_LMENU, 0x00}, {0}},
581         {{WM_SYSKEYDOWN, hook/*|wparam, VK_MENU*/},
582         {WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0}, {0}}},
583     {VK_MENU, KEYEVENTF_KEYUP, 1,
584         {{VK_MENU, 0x80}, {VK_LMENU, 0x80}, {0}},
585         {{WM_KEYUP, hook/*|wparam, VK_MENU*/},
586         {WM_SYSKEYUP, wparam|lparam, VK_MENU, KF_UP},
587         {WM_SYSCOMMAND}, {0}}},
588     /* MENU | KEYEVENTF_EXTENDEDKEY == RMENU */
589     {VK_MENU, KEYEVENTF_EXTENDEDKEY, 0,
590         {{VK_MENU, 0x00}, {VK_RMENU, 0x00}, {0}},
591         {{WM_SYSKEYDOWN, hook/*|wparam*/|lparam, VK_MENU, LLKHF_EXTENDED},
592         {WM_SYSKEYDOWN, wparam|lparam, VK_MENU, KF_EXTENDED}, {0}}},
593     {VK_MENU, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 1,
594         {{VK_MENU, 0x80}, {VK_RMENU, 0x80}, {0}},
595         {{WM_KEYUP, hook/*|wparam*/|lparam, VK_MENU, LLKHF_UP|LLKHF_EXTENDED},
596         {WM_SYSKEYUP, wparam|lparam, VK_MENU, KF_UP|KF_EXTENDED},
597         {WM_SYSCOMMAND}, {0}}},
598
599     /* test LSHIFT & RSHIFT */
600     {VK_LSHIFT, 0, 0,
601         {{VK_SHIFT, 0x00}, {VK_LSHIFT, 0x00}, {0}},
602         {{WM_KEYDOWN, hook|wparam|lparam, VK_LSHIFT, 0},
603         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
604     {VK_RSHIFT, KEYEVENTF_EXTENDEDKEY, 0,
605         {{VK_RSHIFT, 0x00}, {0}},
606         {{WM_KEYDOWN, hook|wparam|lparam, VK_RSHIFT, LLKHF_EXTENDED},
607         {WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0}, {0}}},
608     {VK_RSHIFT, KEYEVENTF_KEYUP | KEYEVENTF_EXTENDEDKEY, 0,
609         {{VK_RSHIFT, 0x80}, {0}},
610         {{WM_KEYUP, hook|wparam|lparam, VK_RSHIFT, LLKHF_UP|LLKHF_EXTENDED},
611         {WM_KEYUP, optional}, {0}}},
612     {VK_LSHIFT, KEYEVENTF_KEYUP, 0,
613         {{VK_SHIFT, 0x80}, {VK_LSHIFT, 0x80}, {0}},
614         {{WM_KEYUP, hook|wparam, VK_LSHIFT},
615         {WM_KEYUP, wparam|lparam, VK_SHIFT, KF_UP}, {0}}},
616
617     {0, 0, 0, {{0}}, {{0}}} /* end */
618 };
619
620 static struct message sent_messages[MAXKEYMESSAGES];
621 static UINT sent_messages_cnt;
622
623 /* Verify that only specified key state transitions occur */
624 static void compare_and_check(int id, BYTE *ks1, BYTE *ks2, struct sendinput_test_s *test)
625 {
626     int i, failcount = 0;
627     struct transition_s *t = test->expected_transitions;
628     UINT actual_cnt = 0;
629     const struct message *expected = test->expected_messages;
630
631     while (t->wVk) {
632         int matched = ((ks1[t->wVk]&0x80) == (t->before_state&0x80)
633                        && (ks2[t->wVk]&0x80) == (~t->before_state&0x80));
634
635         if (!matched && test->_todo_wine)
636         {
637             failcount++;
638             todo_wine {
639                 ok(matched, "%02d: %02x from %02x -> %02x "
640                    "instead of %02x -> %02x\n", id, t->wVk,
641                    ks1[t->wVk]&0x80, ks2[t->wVk]&0x80, t->before_state,
642                    ~t->before_state&0x80);
643             }
644         } else {
645             ok(matched, "%02d: %02x from %02x -> %02x "
646                "instead of %02x -> %02x\n", id, t->wVk,
647                ks1[t->wVk]&0x80, ks2[t->wVk]&0x80, t->before_state,
648                ~t->before_state&0x80);
649         }
650         ks2[t->wVk] = ks1[t->wVk]; /* clear the match */
651         t++;
652     }
653     for (i = 0; i < 256; i++)
654         if (ks2[i] != ks1[i] && test->_todo_wine)
655         {
656             failcount++;
657             todo_wine
658                 ok(FALSE, "%02d: %02x from %02x -> %02x unexpected\n", id, i, ks1[i], ks2[i]);
659         }
660         else
661             ok(ks2[i] == ks1[i], "%02d: %02x from %02x -> %02x unexpected\n",
662                id, i, ks1[i], ks2[i]);
663
664     while (expected->message && actual_cnt < sent_messages_cnt)
665     {
666         const struct message *actual = &sent_messages[actual_cnt];
667
668         if (expected->message == actual->message)
669         {
670             ok((expected->flags & hook) == (actual->flags & hook),
671                "%x/%x: the msg 0x%04x should have been sent by a hook\n",
672                test->wVk, test->dwFlags, expected->message);
673
674             if (expected->flags & wparam)
675             {
676                 if (expected->wParam != actual->wParam && test->_todo_wine)
677                 {
678                     failcount++;
679                     todo_wine
680                         ok(FALSE, "%x/%x: in msg 0x%04x expecting wParam 0x%lx got 0x%lx\n",
681                            test->wVk, test->dwFlags, expected->message, expected->wParam, actual->wParam);
682                 }
683                 else
684                     ok(expected->wParam == actual->wParam,
685                        "%x/%x: in msg 0x%04x expecting wParam 0x%lx got 0x%lx\n",
686                        test->wVk, test->dwFlags, expected->message, expected->wParam, actual->wParam);
687             }
688             if (expected->flags & lparam)
689             {
690                 if (expected->lParam != actual->lParam && test->_todo_wine)
691                 {
692                     failcount++;
693                     todo_wine
694                         ok(FALSE, "%x/%x: in msg 0x%04x expecting lParam 0x%lx got 0x%lx\n",
695                            test->wVk, test->dwFlags, expected->message, expected->lParam, actual->lParam);
696                 }
697                 else
698                     ok(expected->lParam == actual->lParam,
699                        "%x/%x: in msg 0x%04x expecting lParam 0x%lx got 0x%lx\n",
700                        test->wVk, test->dwFlags, expected->message, expected->lParam, actual->lParam);
701             }
702         }
703         else if (expected->flags & optional)
704         {
705             expected++;
706             continue;
707         }
708         else if (test->_todo_wine)
709         {
710             failcount++;
711             todo_wine
712             ok(FALSE,
713                "%x/%x: the msg 0x%04x was expected, but got msg 0x%04x instead\n",
714                test->wVk, test->dwFlags, expected->message, actual->message);
715         }
716         else
717             ok(FALSE,
718                "%x/%x: the msg 0x%04x was expected, but got msg 0x%04x instead\n",
719                test->wVk, test->dwFlags, expected->message, actual->message);
720
721         actual_cnt++;
722         expected++;
723     }
724     /* skip all optional trailing messages */
725     while (expected->message && (expected->flags & optional))
726         expected++;
727
728
729     if (expected->message || actual_cnt < sent_messages_cnt)
730     {
731         if (test->_todo_wine)
732         {
733             failcount++;
734             todo_wine
735                 ok(FALSE, "%x/%x: the msg sequence is not complete: expected %04x - actual %04x\n",
736                    test->wVk, test->dwFlags, expected->message, sent_messages[actual_cnt].message);
737         }
738         else
739             ok(FALSE, "%x/%x: the msg sequence is not complete: expected %04x - actual %04x\n",
740                test->wVk, test->dwFlags, expected->message, sent_messages[actual_cnt].message);
741     }
742
743     if( test->_todo_wine && !failcount) /* succeeded yet marked todo */
744         todo_wine
745             ok(TRUE, "%x/%x: marked \"todo_wine\" but succeeds\n", test->wVk, test->dwFlags);
746
747     sent_messages_cnt = 0;
748 }
749
750 /* WndProc2 checks that we get at least the messages specified */
751 static LRESULT CALLBACK WndProc2(HWND hWnd, UINT Msg, WPARAM wParam,
752                                    LPARAM lParam)
753 {
754     if (winetest_debug > 1) trace("MSG:  %8x W:%8lx L:%8lx\n", Msg, wParam, lParam);
755
756     if (Msg != WM_PAINT &&
757         Msg != WM_NCPAINT &&
758         Msg != WM_SYNCPAINT &&
759         Msg != WM_ERASEBKGND &&
760         Msg != WM_NCHITTEST &&
761         Msg != WM_GETTEXT &&
762         Msg != WM_GETICON &&
763         Msg != WM_DEVICECHANGE)
764     {
765         sent_messages[sent_messages_cnt].message = Msg;
766         sent_messages[sent_messages_cnt].flags = 0;
767         sent_messages[sent_messages_cnt].wParam = wParam;
768         sent_messages[sent_messages_cnt++].lParam = HIWORD(lParam) & (KF_UP|KF_EXTENDED);
769     }
770     return DefWindowProc(hWnd, Msg, wParam, lParam);
771 }
772
773 static LRESULT CALLBACK hook_proc(int code, WPARAM wparam, LPARAM lparam)
774 {
775     KBDLLHOOKSTRUCT *hook_info = (KBDLLHOOKSTRUCT *)lparam;
776
777     if (code == HC_ACTION)
778     {
779         sent_messages[sent_messages_cnt].message = wparam;
780         sent_messages[sent_messages_cnt].flags = hook;
781         sent_messages[sent_messages_cnt].wParam = hook_info->vkCode;
782         sent_messages[sent_messages_cnt++].lParam = hook_info->flags & (LLKHF_UP|LLKHF_EXTENDED);
783
784 if(0) /* For some reason not stable on Wine */
785 {
786         if (wparam == WM_KEYDOWN || wparam == WM_SYSKEYDOWN)
787             ok(!(GetAsyncKeyState(hook_info->vkCode) & 0x8000), "key %x should be up\n", hook_info->vkCode);
788         else if (wparam == WM_KEYUP || wparam == WM_SYSKEYUP)
789             ok(GetAsyncKeyState(hook_info->vkCode) & 0x8000, "key %x should be down\n", hook_info->vkCode);
790 }
791
792         if (winetest_debug > 1)
793             trace("Hook:   w=%lx vk:%8x sc:%8x fl:%8x %lx\n", wparam,
794                   hook_info->vkCode, hook_info->scanCode, hook_info->flags, hook_info->dwExtraInfo);
795     }
796     return CallNextHookEx( 0, code, wparam, lparam );
797 }
798 static void test_Input_blackbox(void)
799 {
800     TEST_INPUT i;
801     int ii;
802     BYTE ks1[256], ks2[256];
803     LONG_PTR prevWndProc;
804     HWND window;
805     HHOOK hook;
806
807     if (!pSendInput)
808     {
809         skip("SendInput is not available\n");
810         return;
811     }
812
813     window = CreateWindow("Static", NULL, WS_POPUP|WS_HSCROLL|WS_VSCROLL
814         |WS_VISIBLE, 0, 0, 200, 60, NULL, NULL,
815         NULL, NULL);
816     ok(window != NULL, "error: %d\n", (int) GetLastError());
817
818     hook = SetWindowsHookExA(WH_KEYBOARD_LL, hook_proc, GetModuleHandleA( NULL ), 0);
819
820     /* must process all initial messages, otherwise X11DRV_KeymapNotify unsets
821      * key state set by SendInput(). */
822     empty_message_queue();
823
824     prevWndProc = SetWindowLongPtr(window, GWLP_WNDPROC, (LONG_PTR) WndProc2);
825     ok(prevWndProc != 0 || (prevWndProc == 0 && GetLastError() == 0),
826        "error: %d\n", (int) GetLastError());
827
828     i.type = INPUT_KEYBOARD;
829     i.u.ki.time = 0;
830     i.u.ki.dwExtraInfo = 0;
831
832     for (ii = 0; ii < sizeof(sendinput_test)/sizeof(struct sendinput_test_s)-1;
833          ii++) {
834         GetKeyboardState(ks1);
835         i.u.ki.wScan = ii+1 /* useful for debugging */;
836         i.u.ki.dwFlags = sendinput_test[ii].dwFlags;
837         i.u.ki.wVk = sendinput_test[ii].wVk;
838         pSendInput(1, (INPUT*)&i, sizeof(TEST_INPUT));
839         empty_message_queue();
840         GetKeyboardState(ks2);
841         compare_and_check(ii, ks1, ks2, &sendinput_test[ii]);
842     }
843
844     empty_message_queue();
845     DestroyWindow(window);
846     UnhookWindowsHookEx(hook);
847 }
848
849 static void test_keynames(void)
850 {
851     int i, len;
852     char buff[256];
853
854     for (i = 0; i < 512; i++)
855     {
856         strcpy(buff, "----");
857         len = GetKeyNameTextA(i << 16, buff, sizeof(buff));
858         ok(len || !buff[0], "%d: Buffer is not zeroed\n", i);
859     }
860 }
861
862 static POINT pt_old, pt_new;
863 static BOOL clipped;
864 #define STEP 20
865
866 static LRESULT CALLBACK hook_proc1( int code, WPARAM wparam, LPARAM lparam )
867 {
868     MSLLHOOKSTRUCT *hook = (MSLLHOOKSTRUCT *)lparam;
869     POINT pt, pt1;
870
871     if (code == HC_ACTION)
872     {
873         /* This is our new cursor position */
874         pt_new = hook->pt;
875         /* Should return previous position */
876         GetCursorPos(&pt);
877         ok(pt.x == pt_old.x && pt.y == pt_old.y, "GetCursorPos: (%d,%d)\n", pt.x, pt.y);
878
879         /* Should set new position until hook chain is finished. */
880         pt.x = pt_old.x + STEP;
881         pt.y = pt_old.y + STEP;
882         SetCursorPos(pt.x, pt.y);
883         GetCursorPos(&pt1);
884         if (clipped)
885             ok(pt1.x == pt_old.x && pt1.y == pt_old.y, "Wrong set pos: (%d,%d)\n", pt1.x, pt1.y);
886         else
887             ok(pt1.x == pt.x && pt1.y == pt.y, "Wrong set pos: (%d,%d)\n", pt1.x, pt1.y);
888     }
889     return CallNextHookEx( 0, code, wparam, lparam );
890 }
891
892 static LRESULT CALLBACK hook_proc2( int code, WPARAM wparam, LPARAM lparam )
893 {
894     MSLLHOOKSTRUCT *hook = (MSLLHOOKSTRUCT *)lparam;
895     POINT pt;
896
897     if (code == HC_ACTION)
898     {
899         ok(hook->pt.x == pt_new.x && hook->pt.y == pt_new.y,
900            "Wrong hook coords: (%d %d) != (%d,%d)\n", hook->pt.x, hook->pt.y, pt_new.x, pt_new.y);
901
902         /* Should match position set above */
903         GetCursorPos(&pt);
904         if (clipped)
905             ok(pt.x == pt_old.x && pt.y == pt_old.y, "GetCursorPos: (%d,%d)\n", pt.x, pt.y);
906         else
907             ok(pt.x == pt_old.x +STEP && pt.y == pt_old.y +STEP, "GetCursorPos: (%d,%d)\n", pt.x, pt.y);
908     }
909     return CallNextHookEx( 0, code, wparam, lparam );
910 }
911
912 static void test_mouse_ll_hook(void)
913 {
914     HWND hwnd;
915     HHOOK hook1, hook2;
916     POINT pt_org, pt;
917     RECT rc;
918
919     GetCursorPos(&pt_org);
920     hwnd = CreateWindow("static", "Title", WS_OVERLAPPEDWINDOW | WS_VISIBLE,
921                         10, 10, 200, 200, NULL, NULL, NULL, NULL);
922     SetCursorPos(100, 100);
923
924     hook2 = SetWindowsHookExA(WH_MOUSE_LL, hook_proc2, GetModuleHandleA(0), 0);
925     hook1 = SetWindowsHookExA(WH_MOUSE_LL, hook_proc1, GetModuleHandleA(0), 0);
926
927     GetCursorPos(&pt_old);
928     mouse_event(MOUSEEVENTF_MOVE, -STEP,  0, 0, 0);
929     GetCursorPos(&pt_old);
930     ok(pt_old.x == pt_new.x && pt_old.y == pt_new.y, "Wrong new pos: (%d,%d)\n", pt_old.x, pt_old.y);
931     mouse_event(MOUSEEVENTF_MOVE, +STEP,  0, 0, 0);
932     GetCursorPos(&pt_old);
933     ok(pt_old.x == pt_new.x && pt_old.y == pt_new.y, "Wrong new pos: (%d,%d)\n", pt_old.x, pt_old.y);
934     mouse_event(MOUSEEVENTF_MOVE,  0, -STEP, 0, 0);
935     GetCursorPos(&pt_old);
936     ok(pt_old.x == pt_new.x && pt_old.y == pt_new.y, "Wrong new pos: (%d,%d)\n", pt_old.x, pt_old.y);
937     mouse_event(MOUSEEVENTF_MOVE,  0, +STEP, 0, 0);
938     GetCursorPos(&pt_old);
939     ok(pt_old.x == pt_new.x && pt_old.y == pt_new.y, "Wrong new pos: (%d,%d)\n", pt_old.x, pt_old.y);
940
941     SetRect(&rc, 50, 50, 151, 151);
942     ClipCursor(&rc);
943     clipped = TRUE;
944
945     SetCursorPos(40, 40);
946     GetCursorPos(&pt_old);
947     ok(pt_old.x == 50 && pt_old.y == 50, "Wrong new pos: (%d,%d)\n", pt_new.x, pt_new.y);
948     SetCursorPos(160, 160);
949     GetCursorPos(&pt_old);
950     ok(pt_old.x == 150 && pt_old.y == 150, "Wrong new pos: (%d,%d)\n", pt_new.x, pt_new.y);
951     mouse_event(MOUSEEVENTF_MOVE, +STEP, +STEP, 0, 0);
952     GetCursorPos(&pt_old);
953     ok(pt_old.x == 150 && pt_old.y == 150, "Wrong new pos: (%d,%d)\n", pt_new.x, pt_new.y);
954
955     clipped = FALSE;
956     pt_new.x = pt_new.y = 150;
957     ClipCursor(NULL);
958     UnhookWindowsHookEx(hook1);
959
960     /* Now check that mouse buttons do not change mouse position
961        if we don't have MOUSEEVENTF_MOVE flag specified. */
962
963     /* We reusing the same hook callback, so make it happy */
964     pt_old.x = pt_new.x - STEP;
965     pt_old.y = pt_new.y - STEP;
966     mouse_event(MOUSEEVENTF_LEFTUP, 123, 456, 0, 0);
967     GetCursorPos(&pt);
968     ok(pt.x == pt_new.x && pt.y == pt_new.y, "Position changed: (%d,%d)\n", pt.x, pt.y);
969     mouse_event(MOUSEEVENTF_RIGHTUP, 456, 123, 0, 0);
970     GetCursorPos(&pt);
971     ok(pt.x == pt_new.x && pt.y == pt_new.y, "Position changed: (%d,%d)\n", pt.x, pt.y);
972
973     mouse_event(MOUSEEVENTF_LEFTUP | MOUSEEVENTF_ABSOLUTE, 123, 456, 0, 0);
974     GetCursorPos(&pt);
975     ok(pt.x == pt_new.x && pt.y == pt_new.y, "Position changed: (%d,%d)\n", pt.x, pt.y);
976     mouse_event(MOUSEEVENTF_RIGHTUP | MOUSEEVENTF_ABSOLUTE, 456, 123, 0, 0);
977     GetCursorPos(&pt);
978     ok(pt.x == pt_new.x && pt.y == pt_new.y, "Position changed: (%d,%d)\n", pt.x, pt.y);
979
980     UnhookWindowsHookEx(hook2);
981     DestroyWindow(hwnd);
982     SetCursorPos(pt_org.x, pt_org.y);
983 }
984
985 static void test_GetMouseMovePointsEx(void)
986 {
987 #define BUFLIM  64
988 #define MYERROR 0xdeadbeef
989     int count, retval;
990     MOUSEMOVEPOINT in;
991     MOUSEMOVEPOINT out[200];
992     POINT point;
993
994     /* Get a valid content for the input struct */
995     if(!GetCursorPos(&point)) {
996         skip("GetCursorPos() failed with error %u\n", GetLastError());
997         return;
998     }
999     memset(&in, 0, sizeof(MOUSEMOVEPOINT));
1000     in.x = point.x;
1001     in.y = point.y;
1002
1003     /* test first parameter
1004      * everything different than sizeof(MOUSEMOVEPOINT)
1005      * is expected to fail with ERROR_INVALID_PARAMETER
1006      */
1007     SetLastError(MYERROR);
1008     retval = pGetMouseMovePointsEx(0, &in, out, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1009     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1010     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1011        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1012
1013     SetLastError(MYERROR);
1014     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT)-1, &in, out, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1015     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1016     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1017        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1018
1019     SetLastError(MYERROR);
1020     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT)+1, &in, out, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1021     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1022     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1023        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1024
1025     /* test second and third parameter
1026      */
1027     SetLastError(MYERROR);
1028     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), NULL, out, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1029     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1030     ok(ERROR_NOACCESS == GetLastError(),
1031        "expected error ERROR_NOACCESS, got %u\n", GetLastError());
1032
1033     SetLastError(MYERROR);
1034     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, NULL, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1035     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1036     ok(ERROR_NOACCESS == GetLastError(),
1037        "expected error ERROR_NOACCESS, got %u\n", GetLastError());
1038
1039     SetLastError(MYERROR);
1040     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), NULL, NULL, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1041     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1042     ok(ERROR_NOACCESS == GetLastError(),
1043        "expected error ERROR_NOACCESS, got %u\n", GetLastError());
1044
1045     SetLastError(MYERROR);
1046     count = 0;
1047     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, NULL, count, GMMP_USE_DISPLAY_POINTS);
1048     todo_wine {
1049     ok(retval == count, "expected GetMouseMovePointsEx to succeed, got %d\n", retval);
1050     ok(MYERROR == GetLastError(),
1051        "expected error %d, got %u\n", MYERROR, GetLastError());
1052     }
1053
1054     /* test fourth parameter
1055      * a value higher than 64 is expected to fail with ERROR_INVALID_PARAMETER
1056      */
1057     SetLastError(MYERROR);
1058     count = -1;
1059     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, out, count, GMMP_USE_DISPLAY_POINTS);
1060     ok(retval == count, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1061     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1062        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1063
1064     SetLastError(MYERROR);
1065     count = 0;
1066     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, out, count, GMMP_USE_DISPLAY_POINTS);
1067     todo_wine {
1068     ok(retval == count, "expected GetMouseMovePointsEx to succeed, got %d\n", retval);
1069     ok(MYERROR == GetLastError(),
1070        "expected error %d, got %u\n", MYERROR, GetLastError());
1071     }
1072
1073     SetLastError(MYERROR);
1074     count = BUFLIM;
1075     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, out, count, GMMP_USE_DISPLAY_POINTS);
1076     todo_wine {
1077     ok((0 <= retval) && (retval <= count), "expected GetMouseMovePointsEx to succeed, got %d\n", retval);
1078     ok(MYERROR == GetLastError(),
1079        "expected error %d, got %u\n", MYERROR, GetLastError());
1080     }
1081
1082     SetLastError(MYERROR);
1083     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, out, BUFLIM+1, GMMP_USE_DISPLAY_POINTS);
1084     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1085     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1086        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1087
1088     /* it was not possible to force an error with the fifth parameter on win2k */
1089
1090     /* test combinations of wrong parameters to see which error wins */
1091     SetLastError(MYERROR);
1092     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT)-1, NULL, out, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1093     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1094     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1095        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1096
1097     SetLastError(MYERROR);
1098     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT)-1, &in, NULL, BUFLIM, GMMP_USE_DISPLAY_POINTS);
1099     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1100     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1101        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1102
1103     SetLastError(MYERROR);
1104     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), NULL, out, BUFLIM+1, GMMP_USE_DISPLAY_POINTS);
1105     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1106     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1107        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1108
1109     SetLastError(MYERROR);
1110     retval = pGetMouseMovePointsEx(sizeof(MOUSEMOVEPOINT), &in, NULL, BUFLIM+1, GMMP_USE_DISPLAY_POINTS);
1111     ok(retval == -1, "expected GetMouseMovePointsEx to fail, got %d\n", retval);
1112     ok(ERROR_INVALID_PARAMETER == GetLastError(),
1113        "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError());
1114
1115 #undef BUFLIM
1116 #undef MYERROR
1117 }
1118
1119 static void test_key_map(void)
1120 {
1121     HKL kl = GetKeyboardLayout(0);
1122     UINT kL, kR, s, sL;
1123     int i;
1124     static const UINT numpad_collisions[][2] = {
1125         { VK_NUMPAD0, VK_INSERT },
1126         { VK_NUMPAD1, VK_END },
1127         { VK_NUMPAD2, VK_DOWN },
1128         { VK_NUMPAD3, VK_NEXT },
1129         { VK_NUMPAD4, VK_LEFT },
1130         { VK_NUMPAD6, VK_RIGHT },
1131         { VK_NUMPAD7, VK_HOME },
1132         { VK_NUMPAD8, VK_UP },
1133         { VK_NUMPAD9, VK_PRIOR },
1134     };
1135
1136     s  = MapVirtualKeyEx(VK_SHIFT,  MAPVK_VK_TO_VSC, kl);
1137     ok(s != 0, "MapVirtualKeyEx(VK_SHIFT) should return non-zero\n");
1138     sL = MapVirtualKeyEx(VK_LSHIFT, MAPVK_VK_TO_VSC, kl);
1139     ok(s == sL, "%x != %x\n", s, sL);
1140
1141     kL = MapVirtualKeyEx(0x2a, MAPVK_VSC_TO_VK, kl);
1142     ok(kL == VK_SHIFT, "Scan code -> vKey = %x (not VK_SHIFT)\n", kL);
1143     kR = MapVirtualKeyEx(0x36, MAPVK_VSC_TO_VK, kl);
1144     ok(kR == VK_SHIFT, "Scan code -> vKey = %x (not VK_SHIFT)\n", kR);
1145
1146     kL = MapVirtualKeyEx(0x2a, MAPVK_VSC_TO_VK_EX, kl);
1147     ok(kL == VK_LSHIFT, "Scan code -> vKey = %x (not VK_LSHIFT)\n", kL);
1148     kR = MapVirtualKeyEx(0x36, MAPVK_VSC_TO_VK_EX, kl);
1149     ok(kR == VK_RSHIFT, "Scan code -> vKey = %x (not VK_RSHIFT)\n", kR);
1150
1151     /* test that MAPVK_VSC_TO_VK prefers the non-numpad vkey if there's ambiguity */
1152     for (i = 0; i < sizeof(numpad_collisions)/sizeof(numpad_collisions[0]); i++)
1153     {
1154         UINT numpad_scan = MapVirtualKeyEx(numpad_collisions[i][0],  MAPVK_VK_TO_VSC, kl);
1155         UINT other_scan  = MapVirtualKeyEx(numpad_collisions[i][1],  MAPVK_VK_TO_VSC, kl);
1156
1157         /* do they really collide for this layout? */
1158         if (numpad_scan && other_scan == numpad_scan)
1159         {
1160             UINT vkey = MapVirtualKeyEx(numpad_scan, MAPVK_VSC_TO_VK, kl);
1161             ok(vkey != numpad_collisions[i][0],
1162                "Got numpad vKey %x for scan code %x when there was another choice\n",
1163                vkey, numpad_scan);
1164         }
1165     }
1166 }
1167
1168 START_TEST(input)
1169 {
1170     init_function_pointers();
1171
1172     if (!pSendInput)
1173         skip("SendInput is not available\n");
1174     else
1175         test_Input_whitebox();
1176
1177     test_Input_blackbox();
1178     test_keynames();
1179     test_mouse_ll_hook();
1180     test_key_map();
1181
1182     if(pGetMouseMovePointsEx)
1183         test_GetMouseMovePointsEx();
1184     else
1185         skip("GetMouseMovePointsEx is not available\n");
1186 }